]> git.lyx.org Git - lyx.git/blob - src/Cursor.cpp
Now we can restore the old (and better) behavior of descriptions.
[lyx.git] / src / Cursor.cpp
1 /**
2  * \file Cursor.cpp
3  * This file is part of LyX, the document processor.
4  * Licence details can be found in the file COPYING.
5  *
6  * \author Alejandro Aguilar Sierra
7  * \author Alfredo Braunstein
8  * \author Dov Feldstern
9  * \author André Pönitz
10  * \author Stefan Schimanski
11  *
12  * Full author contact details are available in file CREDITS.
13  */
14
15 #include <config.h>
16
17 #include "Bidi.h"
18 #include "Buffer.h"
19 #include "BufferView.h"
20 #include "CoordCache.h"
21 #include "Cursor.h"
22 #include "CutAndPaste.h"
23 #include "DispatchResult.h"
24 #include "Encoding.h"
25 #include "Font.h"
26 #include "FuncCode.h"
27 #include "FuncRequest.h"
28 #include "Language.h"
29 #include "LyXFunc.h" // only for setMessage()
30 #include "LyXRC.h"
31 #include "paragraph_funcs.h"
32 #include "Paragraph.h"
33 #include "ParIterator.h"
34 #include "Row.h"
35 #include "Text.h"
36 #include "TextMetrics.h"
37 #include "TocBackend.h"
38
39 #include "support/lassert.h"
40 #include "support/debug.h"
41 #include "support/docstream.h"
42
43 #include "insets/InsetTabular.h"
44 #include "insets/InsetText.h"
45
46 #include "mathed/InsetMath.h"
47 #include "mathed/InsetMathBrace.h"
48 #include "mathed/InsetMathScript.h"
49 #include "mathed/MacroTable.h"
50 #include "mathed/MathData.h"
51 #include "mathed/MathMacro.h"
52
53 #include <boost/bind.hpp>
54
55 #include <sstream>
56 #include <limits>
57 #include <map>
58
59 using namespace std;
60
61 namespace lyx {
62
63 namespace {
64
65 bool positionable(DocIterator const & cursor, DocIterator const & anchor)
66 {
67         // avoid deeper nested insets when selecting
68         if (cursor.depth() > anchor.depth())
69                 return false;
70
71         // anchor might be deeper, should have same path then
72         for (size_t i = 0; i < cursor.depth(); ++i)
73                 if (&cursor[i].inset() != &anchor[i].inset())
74                         return false;
75
76         // position should be ok.
77         return true;
78 }
79
80
81 // Find position closest to (x, y) in cell given by iter.
82 // Used only in mathed
83 DocIterator bruteFind2(Cursor const & c, int x, int y)
84 {
85         double best_dist = numeric_limits<double>::max();
86
87         DocIterator result;
88
89         DocIterator it = c;
90         it.top().pos() = 0;
91         DocIterator et = c;
92         et.top().pos() = et.top().asInsetMath()->cell(et.top().idx()).size();
93         for (size_t i = 0;; ++i) {
94                 int xo;
95                 int yo;
96                 Inset const * inset = &it.inset();
97                 map<Inset const *, Geometry> const & data =
98                         c.bv().coordCache().getInsets().getData();
99                 map<Inset const *, Geometry>::const_iterator I = data.find(inset);
100
101                 // FIXME: in the case where the inset is not in the cache, this
102                 // means that no part of it is visible on screen. In this case
103                 // we don't do elaborate search and we just return the forwarded
104                 // DocIterator at its beginning.
105                 if (I == data.end()) {
106                         it.top().pos() = 0;
107                         return it;
108                 }
109
110                 Point o = I->second.pos;
111                 inset->cursorPos(c.bv(), it.top(), c.boundary(), xo, yo);
112                 // Convert to absolute
113                 xo += o.x_;
114                 yo += o.y_;
115                 double d = (x - xo) * (x - xo) + (y - yo) * (y - yo);
116                 // '<=' in order to take the last possible position
117                 // this is important for clicking behind \sum in e.g. '\sum_i a'
118                 LYXERR(Debug::DEBUG, "i: " << i << " d: " << d
119                         << " best: " << best_dist);
120                 if (d <= best_dist) {
121                         best_dist = d;
122                         result = it;
123                 }
124                 if (it == et)
125                         break;
126                 it.forwardPos();
127         }
128         return result;
129 }
130
131
132 /*
133 /// moves position closest to (x, y) in given box
134 bool bruteFind(Cursor & cursor,
135         int x, int y, int xlow, int xhigh, int ylow, int yhigh)
136 {
137         LASSERT(!cursor.empty(), return false);
138         Inset & inset = cursor[0].inset();
139         BufferView & bv = cursor.bv();
140
141         CoordCache::InnerParPosCache const & cache =
142                 bv.coordCache().getParPos().find(cursor.bottom().text())->second;
143         // Get an iterator on the first paragraph in the cache
144         DocIterator it(inset);
145         it.push_back(CursorSlice(inset));
146         it.pit() = cache.begin()->first;
147         // Get an iterator after the last paragraph in the cache
148         DocIterator et(inset);
149         et.push_back(CursorSlice(inset));
150         et.pit() = boost::prior(cache.end())->first;
151         if (et.pit() >= et.lastpit())
152                 et = doc_iterator_end(inset);
153         else
154                 ++et.pit();
155
156         double best_dist = numeric_limits<double>::max();;
157         DocIterator best_cursor = et;
158
159         for ( ; it != et; it.forwardPos(true)) {
160                 // avoid invalid nesting when selecting
161                 if (!cursor.selection() || positionable(it, cursor.anchor_)) {
162                         Point p = bv.getPos(it, false);
163                         int xo = p.x_;
164                         int yo = p.y_;
165                         if (xlow <= xo && xo <= xhigh && ylow <= yo && yo <= yhigh) {
166                                 double const dx = xo - x;
167                                 double const dy = yo - y;
168                                 double const d = dx * dx + dy * dy;
169                                 // '<=' in order to take the last possible position
170                                 // this is important for clicking behind \sum in e.g. '\sum_i a'
171                                 if (d <= best_dist) {
172                                         //      lyxerr << "*" << endl;
173                                         best_dist   = d;
174                                         best_cursor = it;
175                                 }
176                         }
177                 }
178         }
179
180         if (best_cursor != et) {
181                 cursor.setCursor(best_cursor);
182                 return true;
183         }
184
185         return false;
186 }
187 */
188
189
190 /// moves position closest to (x, y) in given box
191 bool bruteFind3(Cursor & cur, int x, int y, bool up)
192 {
193         BufferView & bv = cur.bv();
194         int ylow  = up ? 0 : y + 1;
195         int yhigh = up ? y - 1 : bv.workHeight();
196         int xlow = 0;
197         int xhigh = bv.workWidth();
198
199 // FIXME: bit more work needed to get 'from' and 'to' right.
200         pit_type from = cur.bottom().pit();
201         //pit_type to = cur.bottom().pit();
202         //lyxerr << "Pit start: " << from << endl;
203
204         //lyxerr << "bruteFind3: x: " << x << " y: " << y
205         //      << " xlow: " << xlow << " xhigh: " << xhigh
206         //      << " ylow: " << ylow << " yhigh: " << yhigh
207         //      << endl;
208         DocIterator it = doc_iterator_begin(cur.buffer());
209         it.pit() = from;
210         DocIterator et = doc_iterator_end(cur.buffer());
211
212         double best_dist = numeric_limits<double>::max();
213         DocIterator best_cursor = et;
214
215         for ( ; it != et; it.forwardPos()) {
216                 // avoid invalid nesting when selecting
217                 if (bv.cursorStatus(it) == CUR_INSIDE
218                                 && (!cur.selection() || positionable(it, cur.anchor_))) {
219                         Point p = bv.getPos(it, false);
220                         int xo = p.x_;
221                         int yo = p.y_;
222                         if (xlow <= xo && xo <= xhigh && ylow <= yo && yo <= yhigh) {
223                                 double const dx = xo - x;
224                                 double const dy = yo - y;
225                                 double const d = dx * dx + dy * dy;
226                                 //lyxerr << "itx: " << xo << " ity: " << yo << " d: " << d
227                                 //      << " dx: " << dx << " dy: " << dy
228                                 //      << " idx: " << it.idx() << " pos: " << it.pos()
229                                 //      << " it:\n" << it
230                                 //      << endl;
231                                 // '<=' in order to take the last possible position
232                                 // this is important for clicking behind \sum in e.g. '\sum_i a'
233                                 if (d <= best_dist) {
234                                         //lyxerr << "*" << endl;
235                                         best_dist   = d;
236                                         best_cursor = it;
237                                 }
238                         }
239                 }
240         }
241
242         //lyxerr << "best_dist: " << best_dist << " cur:\n" << best_cursor << endl;
243         if (best_cursor == et)
244                 return false;
245         cur.setCursor(best_cursor);
246         return true;
247 }
248
249 docstring parbreak(Paragraph const & par)
250 {
251         odocstringstream os;
252         os << '\n';
253         // only add blank line if we're not in an ERT or Listings inset
254         if (par.ownerCode() != ERT_CODE && par.ownerCode() != LISTINGS_CODE)
255                 os << '\n';
256         return os.str();
257 }
258
259 } // namespace anon
260
261
262 // be careful: this is called from the bv's constructor, too, so
263 // bv functions are not yet available!
264 Cursor::Cursor(BufferView & bv)
265         : DocIterator(&bv.buffer()), bv_(&bv), anchor_(),
266           x_target_(-1), textTargetOffset_(0),
267           selection_(false), mark_(false), logicalpos_(false),
268           current_font(inherit_font)
269 {}
270
271
272 void Cursor::reset(Inset & inset)
273 {
274         clear();
275         push_back(CursorSlice(inset));
276         anchor_ = doc_iterator_begin(&inset.buffer(), &inset);
277         anchor_.clear();
278         clearTargetX();
279         selection_ = false;
280         mark_ = false;
281 }
282
283
284 // this (intentionally) does neither touch anchor nor selection status
285 void Cursor::setCursor(DocIterator const & cur)
286 {
287         DocIterator::operator=(cur);
288 }
289
290
291 void Cursor::dispatch(FuncRequest const & cmd0)
292 {
293         LYXERR(Debug::DEBUG, "cmd: " << cmd0 << '\n' << *this);
294         if (empty())
295                 return;
296
297         fixIfBroken();
298         FuncRequest cmd = cmd0;
299         Cursor safe = *this;
300         
301         // store some values to be used inside of the handlers
302         beforeDispatchCursor_ = *this;
303         for (; depth(); pop(), boundary(false)) {
304                 LYXERR(Debug::DEBUG, "Cursor::dispatch: cmd: "
305                         << cmd0 << endl << *this);
306                 LASSERT(pos() <= lastpos(), /**/);
307                 LASSERT(idx() <= lastidx(), /**/);
308                 LASSERT(pit() <= lastpit(), /**/);
309
310                 // The common case is 'LFUN handled, need update', so make the
311                 // LFUN handler's life easier by assuming this as default value.
312                 // The handler can reset the update and val flags if necessary.
313                 disp_.update(Update::FitCursor | Update::Force);
314                 disp_.dispatched(true);
315                 inset().dispatch(*this, cmd);
316                 if (disp_.dispatched())
317                         break;
318         }
319         
320         // it completely to get a 'bomb early' behaviour in case this
321         // object will be used again.
322         if (!disp_.dispatched()) {
323                 LYXERR(Debug::DEBUG, "RESTORING OLD CURSOR!");
324                 // We might have invalidated the cursor when removing an empty
325                 // paragraph while the cursor could not be moved out the inset
326                 // while we initially thought we could. This might happen when
327                 // a multiline inset becomes an inline inset when the second 
328                 // paragraph is removed.
329                 if (safe.pit() > safe.lastpit()) {
330                         safe.pit() = safe.lastpit();
331                         safe.pos() = safe.lastpos();
332                 }
333                 operator=(safe);
334                 disp_.update(Update::None);
335                 disp_.dispatched(false);
336         } else {
337                 // restore the previous one because nested Cursor::dispatch calls
338                 // are possible which would change it
339                 beforeDispatchCursor_ = safe.beforeDispatchCursor_;
340         }
341 }
342
343
344 DispatchResult Cursor::result() const
345 {
346         return disp_;
347 }
348
349
350 BufferView & Cursor::bv() const
351 {
352         LASSERT(bv_, /**/);
353         return *bv_;
354 }
355
356
357 void Cursor::pop()
358 {
359         LASSERT(depth() >= 1, /**/);
360         pop_back();
361 }
362
363
364 void Cursor::push(Inset & p)
365 {
366         push_back(CursorSlice(p));
367         p.setBuffer(*buffer());
368 }
369
370
371 void Cursor::pushBackward(Inset & p)
372 {
373         LASSERT(!empty(), return);
374         //lyxerr << "Entering inset " << t << " front" << endl;
375         push(p);
376         p.idxFirst(*this);
377 }
378
379
380 bool Cursor::popBackward()
381 {
382         LASSERT(!empty(), return false);
383         if (depth() == 1)
384                 return false;
385         pop();
386         return true;
387 }
388
389
390 bool Cursor::popForward()
391 {
392         LASSERT(!empty(), return false);
393         //lyxerr << "Leaving inset from in back" << endl;
394         const pos_type lp = (depth() > 1) ? (*this)[depth() - 2].lastpos() : 0;
395         if (depth() == 1)
396                 return false;
397         pop();
398         pos() += lastpos() - lp + 1;
399         return true;
400 }
401
402
403 int Cursor::currentMode()
404 {
405         LASSERT(!empty(), /**/);
406         for (int i = depth() - 1; i >= 0; --i) {
407                 int res = operator[](i).inset().currentMode();
408                 if (res != Inset::UNDECIDED_MODE)
409                         return res;
410         }
411         return Inset::TEXT_MODE;
412 }
413
414
415 void Cursor::getPos(int & x, int & y) const
416 {
417         Point p = bv().getPos(*this, boundary());
418         x = p.x_;
419         y = p.y_;
420 }
421
422
423 Row const & Cursor::textRow() const
424 {
425         CursorSlice const & cs = innerTextSlice();
426         ParagraphMetrics const & pm = bv().parMetrics(cs.text(), cs.pit());
427         LASSERT(!pm.rows().empty(), /**/);
428         return pm.getRow(pos(), boundary());
429 }
430
431
432 void Cursor::resetAnchor()
433 {
434         anchor_ = *this;
435 }
436
437
438 void Cursor::setCursorToAnchor()
439 {
440         if (selection())
441                 setCursor(anchor_);
442 }
443
444
445 bool Cursor::posBackward()
446 {
447         if (pos() == 0)
448                 return false;
449         --pos();
450         return true;
451 }
452
453
454 bool Cursor::posForward()
455 {
456         if (pos() == lastpos())
457                 return false;
458         ++pos();
459         return true;
460 }
461
462
463 bool Cursor::posVisRight(bool skip_inset)
464 {
465         Cursor new_cur = *this; // where we will move to
466         pos_type left_pos; // position visually left of current cursor
467         pos_type right_pos; // position visually right of current cursor
468         bool new_pos_is_RTL; // is new position we're moving to RTL?
469
470         getSurroundingPos(left_pos, right_pos);
471
472         LYXERR(Debug::RTL, left_pos <<"|"<< right_pos << " (pos: "<< pos() <<")");
473
474         // Are we at an inset?
475         new_cur.pos() = right_pos;
476         new_cur.boundary(false);
477         if (!skip_inset &&
478                 text()->checkAndActivateInsetVisual(new_cur, right_pos >= pos(), false)) {
479                 // we actually move the cursor at the end of this function, for now
480                 // we just keep track of the new position in new_cur...
481                 LYXERR(Debug::RTL, "entering inset at: " << new_cur.pos());
482         }
483
484         // Are we already at rightmost pos in row?
485         else if (text()->empty() || right_pos == -1) {
486                 
487                 new_cur = *this;
488                 if (!new_cur.posVisToNewRow(false)) {
489                         LYXERR(Debug::RTL, "not moving!");
490                         return false;
491                 }
492                 
493                 // we actually move the cursor at the end of this function, for now 
494                 // just keep track of the new position in new_cur...
495                 LYXERR(Debug::RTL, "right edge, moving: " << int(new_cur.pit()) << "," 
496                         << int(new_cur.pos()) << "," << (new_cur.boundary() ? 1 : 0));
497
498         }
499         // normal movement to the right
500         else {
501                 new_cur = *this;
502                 // Recall, if the cursor is at position 'x', that means *before* 
503                 // the character at position 'x'. In RTL, "before" means "to the 
504                 // right of", in LTR, "to the left of". So currently our situation
505                 // is this: the position to our right is 'right_pos' (i.e., we're 
506                 // currently to the left of 'right_pos'). In order to move to the 
507                 // right, it depends whether or not the character at 'right_pos' is RTL.
508                 new_pos_is_RTL = paragraph().getFontSettings(
509                         buffer()->params(), right_pos).isVisibleRightToLeft();
510                 // If the character at 'right_pos' *is* LTR, then in order to move to
511                 // the right of it, we need to be *after* 'right_pos', i.e., move to
512                 // position 'right_pos' + 1.
513                 if (!new_pos_is_RTL) {
514                         new_cur.pos() = right_pos + 1;
515                         // set the boundary to true in two situations:
516                         if (
517                         // 1. if new_pos is now lastpos, and we're in an RTL paragraph
518                         // (this means that we're moving right to the end of an LTR chunk
519                         // which is at the end of an RTL paragraph);
520                                 (new_cur.pos() == lastpos()
521                                  && paragraph().isRTL(buffer()->params()))
522                         // 2. if the position *after* right_pos is RTL (we want to be 
523                         // *after* right_pos, not before right_pos + 1!)
524                                 || paragraph().getFontSettings(buffer()->params(),
525                                                 new_cur.pos()).isVisibleRightToLeft()
526                         )
527                                 new_cur.boundary(true);
528                         else // set the boundary to false
529                                 new_cur.boundary(false);
530                 }
531                 // Otherwise (if the character at position 'right_pos' is RTL), then
532                 // moving to the right of it is as easy as setting the new position
533                 // to 'right_pos'.
534                 else {
535                         new_cur.pos() = right_pos;
536                         new_cur.boundary(false);
537                 }
538         
539         }
540
541         bool moved = (new_cur.pos() != pos()
542                                   || new_cur.pit() != pit()
543                                   || new_cur.boundary() != boundary()
544                                   || &new_cur.inset() != &inset());
545         
546         if (moved) {
547                 LYXERR(Debug::RTL, "moving to: " << new_cur.pos() 
548                         << (new_cur.boundary() ? " (boundary)" : ""));
549                 *this = new_cur;
550         }
551
552         return moved;
553 }
554
555
556 bool Cursor::posVisLeft(bool skip_inset)
557 {
558         Cursor new_cur = *this; // where we will move to
559         pos_type left_pos; // position visually left of current cursor
560         pos_type right_pos; // position visually right of current cursor
561         bool new_pos_is_RTL; // is new position we're moving to RTL?
562
563         getSurroundingPos(left_pos, right_pos);
564
565         LYXERR(Debug::RTL, left_pos <<"|"<< right_pos << " (pos: "<< pos() <<")");
566
567         // Are we at an inset?
568         new_cur.pos() = left_pos;
569         new_cur.boundary(false);
570         if (!skip_inset && 
571                 text()->checkAndActivateInsetVisual(new_cur, left_pos >= pos(), true)) {
572                 // we actually move the cursor at the end of this function, for now 
573                 // we just keep track of the new position in new_cur...
574                 LYXERR(Debug::RTL, "entering inset at: " << new_cur.pos());
575         }
576
577         // Are we already at leftmost pos in row?
578         else if (text()->empty() || left_pos == -1) {
579                 
580                 new_cur = *this;
581                 if (!new_cur.posVisToNewRow(true)) {
582                         LYXERR(Debug::RTL, "not moving!");
583                         return false;
584                 }
585                 
586                 // we actually move the cursor at the end of this function, for now 
587                 // just keep track of the new position in new_cur...
588                 LYXERR(Debug::RTL, "left edge, moving: " << int(new_cur.pit()) << "," 
589                         << int(new_cur.pos()) << "," << (new_cur.boundary() ? 1 : 0));
590
591         }
592         // normal movement to the left
593         else {
594                 new_cur = *this;
595                 // Recall, if the cursor is at position 'x', that means *before* 
596                 // the character at position 'x'. In RTL, "before" means "to the 
597                 // right of", in LTR, "to the left of". So currently our situation
598                 // is this: the position to our left is 'left_pos' (i.e., we're 
599                 // currently to the right of 'left_pos'). In order to move to the 
600                 // left, it depends whether or not the character at 'left_pos' is RTL.
601                 new_pos_is_RTL = paragraph().getFontSettings(
602                         buffer()->params(), left_pos).isVisibleRightToLeft();
603                 // If the character at 'left_pos' *is* RTL, then in order to move to
604                 // the left of it, we need to be *after* 'left_pos', i.e., move to
605                 // position 'left_pos' + 1.
606                 if (new_pos_is_RTL) {
607                         new_cur.pos() = left_pos + 1;
608                         // set the boundary to true in two situations:
609                         if (
610                         // 1. if new_pos is now lastpos and we're in an LTR paragraph
611                         // (this means that we're moving left to the end of an RTL chunk
612                         // which is at the end of an LTR paragraph);
613                                 (new_cur.pos() == lastpos()
614                                  && !paragraph().isRTL(buffer()->params()))
615                         // 2. if the position *after* left_pos is not RTL (we want to be 
616                         // *after* left_pos, not before left_pos + 1!)
617                                 || !paragraph().getFontSettings(buffer()->params(),
618                                                 new_cur.pos()).isVisibleRightToLeft()
619                         )
620                                 new_cur.boundary(true);
621                         else // set the boundary to false
622                                 new_cur.boundary(false);
623                 }
624                 // Otherwise (if the character at position 'left_pos' is LTR), then
625                 // moving to the left of it is as easy as setting the new position
626                 // to 'left_pos'.
627                 else {
628                         new_cur.pos() = left_pos;
629                         new_cur.boundary(false);
630                 }
631         
632         }
633
634         bool moved = (new_cur.pos() != pos() 
635                                   || new_cur.pit() != pit()
636                                   || new_cur.boundary() != boundary());
637
638         if (moved) {
639                 LYXERR(Debug::RTL, "moving to: " << new_cur.pos() 
640                         << (new_cur.boundary() ? " (boundary)" : ""));
641                 *this = new_cur;
642         }
643                 
644         return moved;
645 }
646
647
648 void Cursor::getSurroundingPos(pos_type & left_pos, pos_type & right_pos)
649 {
650         // preparing bidi tables
651         Paragraph const & par = paragraph();
652         Buffer const & buf = *buffer();
653         Row const & row = textRow();
654         Bidi bidi;
655         bidi.computeTables(par, buf, row);
656
657         LYXERR(Debug::RTL, "bidi: " << row.pos() << "--" << row.endpos());
658
659         // The cursor is painted *before* the character at pos(), or, if 'boundary'
660         // is true, *after* the character at (pos() - 1). So we already have one
661         // known position around the cursor:
662         pos_type const known_pos = boundary() && pos() > 0 ? pos() - 1 : pos();
663         
664         // edge case: if we're at the end of the paragraph, things are a little 
665         // different (because lastpos is a position which does not really "exist" 
666         // --- there's no character there yet).
667         if (known_pos == lastpos()) {
668                 if (par.isRTL(buf.params())) {
669                         left_pos = -1;
670                         right_pos = bidi.vis2log(row.pos());
671                 } else { 
672                         // LTR paragraph
673                         right_pos = -1;
674                         left_pos = bidi.vis2log(row.endpos() - 1);
675                 }
676                 return;
677         }
678         
679         // Whether 'known_pos' is to the left or to the right of the cursor depends
680         // on whether it is an RTL or LTR character...
681         bool const cur_is_RTL = 
682                 par.getFontSettings(buf.params(), known_pos).isVisibleRightToLeft();
683         // ... in the following manner:
684         // For an RTL character, "before" means "to the right" and "after" means
685         // "to the left"; and for LTR, it's the reverse. So, 'known_pos' is to the
686         // right of the cursor if (RTL && boundary) or (!RTL && !boundary):
687         bool const known_pos_on_right = cur_is_RTL == boundary();
688
689         // So we now know one of the positions surrounding the cursor. Let's 
690         // determine the other one:     
691         if (known_pos_on_right) {
692                 right_pos = known_pos;
693                 // *visual* position of 'left_pos':
694                 pos_type v_left_pos = bidi.log2vis(right_pos) - 1;
695                 // If the position we just identified as 'left_pos' is a "skipped 
696                 // separator" (a separator which is at the logical end of a row,
697                 // except for the last row in a paragraph; such separators are not
698                 // painted, so they "are not really there"; note that in bidi text,
699                 // such a separator could appear visually in the middle of a row),
700                 // set 'left_pos' to the *next* position to the left.
701                 if (bidi.inRange(v_left_pos) 
702                                 && bidi.vis2log(v_left_pos) + 1 == row.endpos() 
703                                 && row.endpos() < lastpos()
704                                 && par.isSeparator(bidi.vis2log(v_left_pos)))
705                         --v_left_pos;
706
707                 // calculate the logical position of 'left_pos', if in row
708                 if (!bidi.inRange(v_left_pos))
709                         left_pos = -1;
710                 else
711                         left_pos = bidi.vis2log(v_left_pos);
712                 // If the position we identified as 'right_pos' is a "skipped 
713                 // separator", set 'right_pos' to the *next* position to the right.
714                 if (right_pos + 1 == row.endpos() && row.endpos() < lastpos() 
715                                 && par.isSeparator(right_pos)) {
716                         pos_type const v_right_pos = bidi.log2vis(right_pos) + 1;
717                         if (!bidi.inRange(v_right_pos))
718                                 right_pos = -1;
719                         else
720                                 right_pos = bidi.vis2log(v_right_pos);
721                 }
722         } else { 
723                 // known_pos is on the left
724                 left_pos = known_pos;
725                 // *visual* position of 'right_pos'
726                 pos_type v_right_pos = bidi.log2vis(left_pos) + 1;
727                 // If the position we just identified as 'right_pos' is a "skipped 
728                 // separator", set 'right_pos' to the *next* position to the right.
729                 if (bidi.inRange(v_right_pos) 
730                                 && bidi.vis2log(v_right_pos) + 1 == row.endpos() 
731                                 && row.endpos() < lastpos()
732                                 && par.isSeparator(bidi.vis2log(v_right_pos)))
733                         ++v_right_pos;
734
735                 // calculate the logical position of 'right_pos', if in row
736                 if (!bidi.inRange(v_right_pos)) 
737                         right_pos = -1;
738                 else
739                         right_pos = bidi.vis2log(v_right_pos);
740                 // If the position we identified as 'left_pos' is a "skipped 
741                 // separator", set 'left_pos' to the *next* position to the left.
742                 if (left_pos + 1 == row.endpos() && row.endpos() < lastpos() 
743                                 && par.isSeparator(left_pos)) {
744                         pos_type const v_left_pos = bidi.log2vis(left_pos) - 1;
745                         if (!bidi.inRange(v_left_pos))
746                                 left_pos = -1;
747                         else
748                                 left_pos = bidi.vis2log(v_left_pos);
749                 }
750         }
751         return;
752 }
753
754
755 bool Cursor::posVisToNewRow(bool movingLeft)
756 {
757         Paragraph const & par = paragraph();
758         Buffer const & buf = *buffer();
759         Row const & row = textRow();
760         bool par_is_LTR = !par.isRTL(buf.params());
761
762         // Inside a table, determining whether to move to the next or previous row
763         // should be done based on the table's direction. 
764         int s = depth() - 1;
765         if (s >= 1 && (*this)[s].inset().asInsetTabular()) {
766                 par_is_LTR = !(*this)[s].inset().asInsetTabular()->isRightToLeft(*this);
767                 LYXERR(Debug::RTL, "Inside table! par_is_LTR=" << (par_is_LTR ? 1 : 0));
768         }
769         
770         // if moving left in an LTR paragraph or moving right in an RTL one, 
771         // move to previous row
772         if (par_is_LTR == movingLeft) {
773                 if (row.pos() == 0) { // we're at first row in paragraph
774                         if (pit() == 0) // no previous paragraph! don't move
775                                 return false;
776                         // move to last pos in previous par
777                         --pit();
778                         pos() = lastpos();
779                         boundary(false);
780                 } else { // move to previous row in this par
781                         pos() = row.pos() - 1; // this is guaranteed to be in previous row
782                         boundary(false);
783                 }
784         }
785         // if moving left in an RTL paragraph or moving right in an LTR one, 
786         // move to next row
787         else {
788                 if (row.endpos() == lastpos()) { // we're at last row in paragraph
789                         if (pit() == lastpit()) // last paragraph! don't move
790                                 return false;
791                         // move to first row in next par
792                         ++pit();
793                         pos() = 0;
794                         boundary(false);
795                 } else { // move to next row in this par
796                         pos() = row.endpos();
797                         boundary(false);
798                 }
799         }
800         
801         // make sure we're at left-/right-most pos in new row
802         posVisToRowExtremity(!movingLeft);
803
804         return true;
805 }
806
807
808 void Cursor::posVisToRowExtremity(bool left)  
809 {
810         // prepare bidi tables
811         Paragraph const & par = paragraph();
812         Buffer const & buf = *buffer();
813         Row const & row = textRow();
814         Bidi bidi;
815         bidi.computeTables(par, buf, row);
816
817         LYXERR(Debug::RTL, "entering extremity: " << pit() << "," << pos() << ","
818                 << (boundary() ? 1 : 0));
819
820         if (left) { // move to leftmost position
821                 // if this is an RTL paragraph, and we're at the last row in the
822                 // paragraph, move to lastpos
823                 if (par.isRTL(buf.params()) && row.endpos() == lastpos())
824                         pos() = lastpos();
825                 else {
826                         pos() = bidi.vis2log(row.pos());
827
828                         // Moving to the leftmost position in the row, the cursor should
829                         // normally be placed to the *left* of the leftmost position.
830                         // A very common exception, though, is if the leftmost character 
831                         // also happens to be the separator at the (logical) end of the row
832                         // --- in this case, the separator is positioned beyond the left 
833                         // margin, and we don't want to move the cursor there (moving to 
834                         // the left of the separator is equivalent to moving to the next
835                         // line). So, in this case we actually want to place the cursor 
836                         // to the *right* of the leftmost position (the separator). 
837                         // Another exception is if we're moving to the logically last 
838                         // position in the row, which is *not* a separator: this means
839                         // that the entire row has no separators (if there were any, the 
840                         // row would have been broken there); and therefore in this case
841                         // we also move to the *right* of the last position (this indicates
842                         // to the user that there is no space after this position, and is 
843                         // consistent with the behavior in the middle of a row --- moving
844                         // right or left moves to the next/previous character; if we were
845                         // to move to the *left* of this position, that would simulate 
846                         // a separator which is not really there!). 
847                         // Finally, there is an exception to the previous exception: if 
848                         // this non-separator-but-last-position-in-row is an inset, then
849                         // we *do* want to stay to the left of it anyway: this is the 
850                         // "boundary" which we simulate at insets.
851                         // Another exception is when row.endpos() is 0.
852                         
853                         // do we want to be to the right of pos?
854                         // as explained above, if at last pos in row, stay to the right
855                         bool const right_of_pos = row.endpos() > 0
856                                 && pos() == row.endpos() - 1 && !par.isInset(pos());
857
858                         // Now we know if we want to be to the left or to the right of pos,
859                         // let's make sure we are where we want to be.
860                         bool const new_pos_is_RTL = 
861                                 par.getFontSettings(buf.params(), pos()).isVisibleRightToLeft();
862
863                         if (new_pos_is_RTL != right_of_pos) {
864                                 ++pos();
865                                 boundary(true);
866                         }
867                 }
868         } else { 
869                 // move to rightmost position
870                 // if this is an LTR paragraph, and we're at the last row in the
871                 // paragraph, move to lastpos
872                 if (!par.isRTL(buf.params()) && row.endpos() == lastpos())
873                         pos() = lastpos();
874                 else {
875                         pos() = row.endpos() > 0 ? bidi.vis2log(row.endpos() - 1) : 0;
876
877                         // Moving to the rightmost position in the row, the cursor should
878                         // normally be placed to the *right* of the rightmost position.
879                         // A very common exception, though, is if the rightmost character 
880                         // also happens to be the separator at the (logical) end of the row
881                         // --- in this case, the separator is positioned beyond the right 
882                         // margin, and we don't want to move the cursor there (moving to 
883                         // the right of the separator is equivalent to moving to the next
884                         // line). So, in this case we actually want to place the cursor 
885                         // to the *left* of the rightmost position (the separator). 
886                         // Another exception is if we're moving to the logically last 
887                         // position in the row, which is *not* a separator: this means
888                         // that the entire row has no separators (if there were any, the 
889                         // row would have been broken there); and therefore in this case
890                         // we also move to the *left* of the last position (this indicates
891                         // to the user that there is no space after this position, and is 
892                         // consistent with the behavior in the middle of a row --- moving
893                         // right or left moves to the next/previous character; if we were
894                         // to move to the *right* of this position, that would simulate 
895                         // a separator which is not really there!). 
896                         // Finally, there is an exception to the previous exception: if 
897                         // this non-separator-but-last-position-in-row is an inset, then
898                         // we *do* want to stay to the right of it anyway: this is the 
899                         // "boundary" which we simulate at insets.
900                         // Another exception is when row.endpos() is 0.
901                         
902                         // do we want to be to the left of pos?
903                         // as explained above, if at last pos in row, stay to the left,
904                         // unless the last position is the same as the first.
905                         bool const left_of_pos = row.endpos() > 0
906                                 && pos() == row.endpos() - 1 && !par.isInset(pos());
907
908                         // Now we know if we want to be to the left or to the right of pos,
909                         // let's make sure we are where we want to be.
910                         bool const new_pos_is_RTL = 
911                                 par.getFontSettings(buf.params(), pos()).isVisibleRightToLeft();
912
913                         if (new_pos_is_RTL == left_of_pos) {
914                                 ++pos();
915                                 boundary(true);
916                         }
917                 }
918         }
919         LYXERR(Debug::RTL, "leaving extremity: " << pit() << "," << pos() << ","
920                 << (boundary() ? 1 : 0));
921 }
922
923
924 CursorSlice Cursor::anchor() const
925 {
926         LASSERT(anchor_.depth() >= depth(), /**/);
927         CursorSlice normal = anchor_[depth() - 1];
928         if (depth() < anchor_.depth() && top() <= normal) {
929                 // anchor is behind cursor -> move anchor behind the inset
930                 ++normal.pos();
931         }
932         return normal;
933 }
934
935
936 CursorSlice Cursor::selBegin() const
937 {
938         if (!selection())
939                 return top();
940         return anchor() < top() ? anchor() : top();
941 }
942
943
944 CursorSlice Cursor::selEnd() const
945 {
946         if (!selection())
947                 return top();
948         return anchor() > top() ? anchor() : top();
949 }
950
951
952 DocIterator Cursor::selectionBegin() const
953 {
954         if (!selection())
955                 return *this;
956
957         DocIterator di;
958         // FIXME: This is a work-around for the problem that
959         // CursorSlice doesn't keep track of the boundary.
960         if (anchor() == top())
961                 di = anchor_.boundary() > boundary() ? anchor_ : *this;
962         else
963                 di = anchor() < top() ? anchor_ : *this;
964         di.resize(depth());
965         return di;
966 }
967
968
969 DocIterator Cursor::selectionEnd() const
970 {
971         if (!selection())
972                 return *this;
973
974         DocIterator di;
975         // FIXME: This is a work-around for the problem that
976         // CursorSlice doesn't keep track of the boundary.
977         if (anchor() == top())
978                 di = anchor_.boundary() < boundary() ? anchor_ : *this;
979         else
980                 di = anchor() > top() ? anchor_ : *this;
981
982         if (di.depth() > depth()) {
983                 di.resize(depth());
984                 ++di.pos();
985         }
986         return di;
987 }
988
989
990 void Cursor::setSelection()
991 {
992         setSelection(true);
993         // A selection with no contents is not a selection
994         // FIXME: doesnt look ok
995         if (idx() == anchor().idx() && 
996             pit() == anchor().pit() && 
997             pos() == anchor().pos())
998                 setSelection(false);
999 }
1000
1001
1002 void Cursor::setSelection(DocIterator const & where, int n)
1003 {
1004         setCursor(where);
1005         setSelection(true);
1006         anchor_ = where;
1007         pos() += n;
1008 }
1009
1010
1011 void Cursor::clearSelection()
1012 {
1013         setSelection(false);
1014         setMark(false);
1015         resetAnchor();
1016 }
1017
1018
1019 void Cursor::setTargetX(int x)
1020 {
1021         x_target_ = x;
1022         textTargetOffset_ = 0;
1023 }
1024
1025
1026 int Cursor::x_target() const
1027 {
1028         return x_target_;
1029 }
1030
1031
1032 void Cursor::clearTargetX()
1033 {
1034         x_target_ = -1;
1035         textTargetOffset_ = 0;
1036 }
1037
1038
1039 void Cursor::updateTextTargetOffset()
1040 {
1041         int x;
1042         int y;
1043         getPos(x, y);
1044         textTargetOffset_ = x - x_target_;
1045 }
1046
1047
1048 void Cursor::info(odocstream & os) const
1049 {
1050         for (int i = 1, n = depth(); i < n; ++i) {
1051                 operator[](i).inset().infoize(os);
1052                 os << "  ";
1053         }
1054         if (pos() != 0) {
1055                 Inset const * inset = prevInset();
1056                 // prevInset() can return 0 in certain case.
1057                 if (inset)
1058                         prevInset()->infoize2(os);
1059         }
1060         // overwite old message
1061         os << "                    ";
1062 }
1063
1064
1065 bool Cursor::selHandle(bool sel)
1066 {
1067         //lyxerr << "Cursor::selHandle" << endl;
1068         if (mark())
1069                 sel = true;
1070         if (sel == selection())
1071                 return false;
1072
1073         if (!sel)
1074                 cap::saveSelection(*this);
1075
1076         resetAnchor();
1077         setSelection(sel);
1078         return true;
1079 }
1080
1081
1082 ostream & operator<<(ostream & os, Cursor const & cur)
1083 {
1084         os << "\n cursor:                                | anchor:\n";
1085         for (size_t i = 0, n = cur.depth(); i != n; ++i) {
1086                 os << " " << cur[i] << " | ";
1087                 if (i < cur.anchor_.depth())
1088                         os << cur.anchor_[i];
1089                 else
1090                         os << "-------------------------------";
1091                 os << "\n";
1092         }
1093         for (size_t i = cur.depth(), n = cur.anchor_.depth(); i < n; ++i) {
1094                 os << "------------------------------- | " << cur.anchor_[i] << "\n";
1095         }
1096         os << " selection: " << cur.selection_
1097            << " x_target: " << cur.x_target_ << endl;
1098         return os;
1099 }
1100
1101
1102 LyXErr & operator<<(LyXErr & os, Cursor const & cur)
1103 {
1104         os.stream() << cur;
1105         return os;
1106 }
1107
1108
1109 } // namespace lyx
1110
1111
1112 ///////////////////////////////////////////////////////////////////
1113 //
1114 // FIXME: Look here
1115 // The part below is the non-integrated rest of the original math
1116 // cursor. This should be either generalized for texted or moved
1117 // back to mathed (in most cases to InsetMathNest).
1118 //
1119 ///////////////////////////////////////////////////////////////////
1120
1121 #include "mathed/InsetMathChar.h"
1122 #include "mathed/InsetMathGrid.h"
1123 #include "mathed/InsetMathScript.h"
1124 #include "mathed/InsetMathUnknown.h"
1125 #include "mathed/MathFactory.h"
1126 #include "mathed/MathStream.h"
1127 #include "mathed/MathSupport.h"
1128
1129
1130 namespace lyx {
1131
1132 //#define FILEDEBUG 1
1133
1134
1135 bool Cursor::isInside(Inset const * p) const
1136 {
1137         for (size_t i = 0; i != depth(); ++i)
1138                 if (&operator[](i).inset() == p)
1139                         return true;
1140         return false;
1141 }
1142
1143
1144 void Cursor::leaveInset(Inset const & inset)
1145 {
1146         for (size_t i = 0; i != depth(); ++i) {
1147                 if (&operator[](i).inset() == &inset) {
1148                         resize(i);
1149                         return;
1150                 }
1151         }
1152 }
1153
1154
1155 bool Cursor::openable(MathAtom const & t) const
1156 {
1157         if (!t->isActive())
1158                 return false;
1159
1160         if (t->lock())
1161                 return false;
1162
1163         if (!selection())
1164                 return true;
1165
1166         // we can't move into anything new during selection
1167         if (depth() >= anchor_.depth())
1168                 return false;
1169         if (t.nucleus() != &anchor_[depth()].inset())
1170                 return false;
1171
1172         return true;
1173 }
1174
1175
1176 void Cursor::setScreenPos(int x, int /*y*/)
1177 {
1178         setTargetX(x);
1179         //bruteFind(*this, x, y, 0, bv().workWidth(), 0, bv().workHeight());
1180 }
1181
1182
1183
1184 void Cursor::plainErase()
1185 {
1186         cell().erase(pos());
1187 }
1188
1189
1190 void Cursor::markInsert()
1191 {
1192         insert(char_type(0));
1193 }
1194
1195
1196 void Cursor::markErase()
1197 {
1198         cell().erase(pos());
1199 }
1200
1201
1202 void Cursor::plainInsert(MathAtom const & t)
1203 {
1204         cell().insert(pos(), t);
1205         ++pos();
1206         inset().setBuffer(bv_->buffer());
1207         inset().initView();
1208 }
1209
1210
1211 void Cursor::insert(docstring const & str)
1212 {
1213         for_each(str.begin(), str.end(),
1214                  boost::bind(static_cast<void(Cursor::*)(char_type)>
1215                              (&Cursor::insert), this, _1));
1216 }
1217
1218
1219 void Cursor::insert(char_type c)
1220 {
1221         //lyxerr << "Cursor::insert char '" << c << "'" << endl;
1222         LASSERT(!empty(), /**/);
1223         if (inMathed()) {
1224                 cap::selClearOrDel(*this);
1225                 insert(new InsetMathChar(c));
1226         } else {
1227                 text()->insertChar(*this, c);
1228         }
1229 }
1230
1231
1232 void Cursor::insert(MathAtom const & t)
1233 {
1234         //lyxerr << "Cursor::insert MathAtom '" << t << "'" << endl;
1235         macroModeClose();
1236         cap::selClearOrDel(*this);
1237         plainInsert(t);
1238 }
1239
1240
1241 void Cursor::insert(Inset * inset0)
1242 {
1243         LASSERT(inset0, /**/);
1244         if (inMathed())
1245                 insert(MathAtom(inset0));
1246         else {
1247                 text()->insertInset(*this, inset0);
1248                 inset0->setBuffer(bv_->buffer());
1249                 inset0->initView();
1250         }
1251 }
1252
1253
1254 void Cursor::niceInsert(docstring const & t, Parse::flags f)
1255 {
1256         MathData ar;
1257         asArray(t, ar, f);
1258         if (ar.size() == 1)
1259                 niceInsert(ar[0]);
1260         else
1261                 insert(ar);
1262 }
1263
1264
1265 void Cursor::niceInsert(MathAtom const & t)
1266 {
1267         macroModeClose();
1268         docstring const safe = cap::grabAndEraseSelection(*this);
1269         plainInsert(t);
1270         // enter the new inset and move the contents of the selection if possible
1271         if (t->isActive()) {
1272                 posBackward();
1273                 // be careful here: don't use 'pushBackward(t)' as this we need to
1274                 // push the clone, not the original
1275                 pushBackward(*nextInset());
1276                 // We may not use niceInsert here (recursion)
1277                 MathData ar;
1278                 asArray(safe, ar);
1279                 insert(ar);
1280         }
1281 }
1282
1283
1284 void Cursor::insert(MathData const & ar)
1285 {
1286         macroModeClose();
1287         if (selection())
1288                 cap::eraseSelection(*this);
1289         cell().insert(pos(), ar);
1290         pos() += ar.size();
1291 }
1292
1293
1294 bool Cursor::backspace()
1295 {
1296         autocorrect() = false;
1297
1298         if (selection()) {
1299                 cap::eraseSelection(*this);
1300                 return true;
1301         }
1302
1303         if (pos() == 0) {
1304                 // If empty cell, and not part of a big cell
1305                 if (lastpos() == 0 && inset().nargs() == 1) {
1306                         popBackward();
1307                         // Directly delete empty cell: [|[]] => [|]
1308                         if (inMathed()) {
1309                                 plainErase();
1310                                 resetAnchor();
1311                                 return true;
1312                         }
1313                         // [|], can not delete from inside
1314                         return false;
1315                 } else {
1316                         if (inMathed())
1317                                 pullArg();
1318                         else
1319                                 popBackward();
1320                         return true;
1321                 }
1322         }
1323
1324         if (inMacroMode()) {
1325                 InsetMathUnknown * p = activeMacro();
1326                 if (p->name().size() > 1) {
1327                         p->setName(p->name().substr(0, p->name().size() - 1));
1328                         return true;
1329                 }
1330         }
1331
1332         if (pos() != 0 && prevAtom()->nargs() > 0) {
1333                 // let's require two backspaces for 'big stuff' and
1334                 // highlight on the first
1335                 resetAnchor();
1336                 setSelection(true);
1337                 --pos();
1338         } else {
1339                 --pos();
1340                 plainErase();
1341         }
1342         return true;
1343 }
1344
1345
1346 bool Cursor::erase()
1347 {
1348         autocorrect() = false;
1349         if (inMacroMode())
1350                 return true;
1351
1352         if (selection()) {
1353                 cap::eraseSelection(*this);
1354                 return true;
1355         }
1356
1357         // delete empty cells if possible
1358         if (pos() == lastpos() && inset().idxDelete(idx()))
1359                 return true;
1360
1361         // special behaviour when in last position of cell
1362         if (pos() == lastpos()) {
1363                 bool one_cell = inset().nargs() == 1;
1364                 if (one_cell && lastpos() == 0) {
1365                         popBackward();
1366                         // Directly delete empty cell: [|[]] => [|]
1367                         if (inMathed()) {
1368                                 plainErase();
1369                                 resetAnchor();
1370                                 return true;
1371                         }
1372                         // [|], can not delete from inside
1373                         return false;
1374                 }
1375                 // remove markup
1376                 if (!one_cell)
1377                         inset().idxGlue(idx());
1378                 return true;
1379         }
1380
1381         // 'clever' UI hack: only erase large items if previously slected
1382         if (pos() != lastpos() && nextAtom()->nargs() > 0) {
1383                 resetAnchor();
1384                 setSelection(true);
1385                 ++pos();
1386         } else {
1387                 plainErase();
1388         }
1389
1390         return true;
1391 }
1392
1393
1394 bool Cursor::up()
1395 {
1396         macroModeClose();
1397         DocIterator save = *this;
1398         FuncRequest cmd(selection() ? LFUN_UP_SELECT : LFUN_UP, docstring());
1399         this->dispatch(cmd);
1400         if (disp_.dispatched())
1401                 return true;
1402         setCursor(save);
1403         autocorrect() = false;
1404         return false;
1405 }
1406
1407
1408 bool Cursor::down()
1409 {
1410         macroModeClose();
1411         DocIterator save = *this;
1412         FuncRequest cmd(selection() ? LFUN_DOWN_SELECT : LFUN_DOWN, docstring());
1413         this->dispatch(cmd);
1414         if (disp_.dispatched())
1415                 return true;
1416         setCursor(save);
1417         autocorrect() = false;
1418         return false;
1419 }
1420
1421
1422 bool Cursor::macroModeClose()
1423 {
1424         if (!inMacroMode())
1425                 return false;
1426         InsetMathUnknown * p = activeMacro();
1427         p->finalize();
1428         MathData selection;
1429         asArray(p->selection(), selection);
1430         docstring const s = p->name();
1431         --pos();
1432         cell().erase(pos());
1433
1434         // do nothing if the macro name is empty
1435         if (s == "\\")
1436                 return false;
1437
1438         // trigger updates of macros, at least, if no full
1439         // updates take place anyway
1440         updateFlags(Update::Force);
1441
1442         docstring const name = s.substr(1);
1443         InsetMathNest * const in = inset().asInsetMath()->asNestInset();
1444         if (in && in->interpretString(*this, s))
1445                 return true;
1446         MathAtom atom = createInsetMath(name);
1447
1448         // try to put argument into macro, if we just inserted a macro
1449         bool macroArg = false;
1450         MathMacro * atomAsMacro = atom.nucleus()->asMacro();
1451         if (atomAsMacro) {
1452                 // macros here are still unfolded (in init mode in fact). So
1453                 // we have to resolve the macro here manually and check its arity
1454                 // to put the selection behind it if arity > 0.
1455                 MacroData const * data = buffer()->getMacro(atomAsMacro->name());
1456                 if (selection.size() > 0 && data && data->numargs() - data->optionals() > 0) {
1457                         macroArg = true;
1458                         atomAsMacro->setDisplayMode(MathMacro::DISPLAY_INTERACTIVE_INIT, 1);
1459                 } else
1460                         // non-greedy case. Do not touch the arguments behind
1461                         atomAsMacro->setDisplayMode(MathMacro::DISPLAY_INTERACTIVE_INIT, 0);
1462         }
1463
1464         // insert remembered selection into first argument of a non-macro
1465         else if (atom.nucleus()->nargs() > 0)
1466                 atom.nucleus()->cell(0).append(selection);
1467         
1468         plainInsert(atom);
1469
1470         // finally put the macro argument behind, if needed
1471         if (macroArg) {
1472                 if (selection.size() > 1)
1473                         plainInsert(MathAtom(new InsetMathBrace(selection)));
1474                 else
1475                         insert(selection);
1476         }
1477         
1478         return true;
1479 }
1480
1481
1482 docstring Cursor::macroName()
1483 {
1484         return inMacroMode() ? activeMacro()->name() : docstring();
1485 }
1486
1487
1488 void Cursor::handleNest(MathAtom const & a, int c)
1489 {
1490         //lyxerr << "Cursor::handleNest: " << c << endl;
1491         MathAtom t = a;
1492         asArray(cap::grabAndEraseSelection(*this), t.nucleus()->cell(c));
1493         insert(t);
1494         posBackward();
1495         pushBackward(*nextInset());
1496 }
1497
1498
1499 int Cursor::targetX() const
1500 {
1501         if (x_target() != -1)
1502                 return x_target();
1503         int x = 0;
1504         int y = 0;
1505         getPos(x, y);
1506         return x;
1507 }
1508
1509
1510 int Cursor::textTargetOffset() const
1511 {
1512         return textTargetOffset_;
1513 }
1514
1515
1516 void Cursor::setTargetX()
1517 {
1518         int x;
1519         int y;
1520         getPos(x, y);
1521         setTargetX(x);
1522 }
1523
1524
1525 bool Cursor::inMacroMode() const
1526 {
1527         if (!inMathed())
1528                 return false;
1529         if (pos() == 0)
1530                 return false;
1531         InsetMathUnknown const * p = prevAtom()->asUnknownInset();
1532         return p && !p->final();
1533 }
1534
1535
1536 InsetMathUnknown * Cursor::activeMacro()
1537 {
1538         return inMacroMode() ? prevAtom().nucleus()->asUnknownInset() : 0;
1539 }
1540
1541
1542 InsetMathUnknown const * Cursor::activeMacro() const
1543 {
1544         return inMacroMode() ? prevAtom().nucleus()->asUnknownInset() : 0;
1545 }
1546
1547
1548 void Cursor::pullArg()
1549 {
1550         // FIXME: Look here
1551         MathData ar = cell();
1552         if (popBackward() && inMathed()) {
1553                 plainErase();
1554                 cell().insert(pos(), ar);
1555                 resetAnchor();
1556         } else {
1557                 //formula()->mutateToText();
1558         }
1559 }
1560
1561
1562 void Cursor::touch()
1563 {
1564         // FIXME: look here
1565 #if 0
1566         DocIterator::const_iterator it = begin();
1567         DocIterator::const_iterator et = end();
1568         for ( ; it != et; ++it)
1569                 it->cell().touch();
1570 #endif
1571 }
1572
1573
1574 void Cursor::normalize()
1575 {
1576         if (idx() > lastidx()) {
1577                 lyxerr << "this should not really happen - 1: "
1578                        << idx() << ' ' << nargs()
1579                        << " in: " << &inset() << endl;
1580                 idx() = lastidx();
1581         }
1582
1583         if (pos() > lastpos()) {
1584                 lyxerr << "this should not really happen - 2: "
1585                         << pos() << ' ' << lastpos() <<  " in idx: " << idx()
1586                        << " in atom: '";
1587                 odocstringstream os;
1588                 WriteStream wi(os, false, true, false);
1589                 inset().asInsetMath()->write(wi);
1590                 lyxerr << to_utf8(os.str()) << endl;
1591                 pos() = lastpos();
1592         }
1593 }
1594
1595
1596 bool Cursor::upDownInMath(bool up)
1597 {
1598         // Be warned: The 'logic' implemented in this function is highly
1599         // fragile. A distance of one pixel or a '<' vs '<=' _really
1600         // matters. So fiddle around with it only if you think you know
1601         // what you are doing!
1602         int xo = 0;
1603         int yo = 0;
1604         getPos(xo, yo);
1605         xo = theLyXFunc().cursorBeforeDispatchX();
1606         
1607         // check if we had something else in mind, if not, this is the future
1608         // target
1609         if (x_target_ == -1)
1610                 setTargetX(xo);
1611         else if (inset().asInsetText() && xo - textTargetOffset() != x_target()) {
1612                 // In text mode inside the line (not left or right) possibly set a new target_x,
1613                 // but only if we are somewhere else than the previous target-offset.
1614                 
1615                 // We want to keep the x-target on subsequent up/down movements
1616                 // that cross beyond the end of short lines. Thus a special
1617                 // handling when the cursor is at the end of line: Use the new
1618                 // x-target only if the old one was before the end of line
1619                 // or the old one was after the beginning of the line
1620                 bool inRTL = isWithinRtlParagraph(*this);
1621                 bool left;
1622                 bool right;
1623                 if (inRTL) {
1624                         left = pos() == textRow().endpos();
1625                         right = pos() == textRow().pos();
1626                 } else {
1627                         left = pos() == textRow().pos();
1628                         right = pos() == textRow().endpos();
1629                 }
1630                 if ((!left && !right) ||
1631                                 (left && !right && xo < x_target_) ||
1632                                 (!left && right && x_target_ < xo))
1633                         setTargetX(xo);
1634                 else
1635                         xo = targetX();
1636         } else
1637                 xo = targetX();
1638
1639         // try neigbouring script insets
1640         Cursor old = *this;
1641         if (inMathed() && !selection()) {
1642                 // try left
1643                 if (pos() != 0) {
1644                         InsetMathScript const * p = prevAtom()->asScriptInset();
1645                         if (p && p->has(up)) {
1646                                 --pos();
1647                                 push(*const_cast<InsetMathScript*>(p));
1648                                 idx() = p->idxOfScript(up);
1649                                 pos() = lastpos();
1650                                 
1651                                 // we went in the right direction? Otherwise don't jump into the script
1652                                 int x;
1653                                 int y;
1654                                 getPos(x, y);
1655                                 int oy = theLyXFunc().cursorBeforeDispatchY();
1656                                 if ((!up && y <= oy) ||
1657                                                 (up && y >= oy))
1658                                         operator=(old);
1659                                 else
1660                                         return true;
1661                         }
1662                 }
1663                 
1664                 // try right
1665                 if (pos() != lastpos()) {
1666                         InsetMathScript const * p = nextAtom()->asScriptInset();
1667                         if (p && p->has(up)) {
1668                                 push(*const_cast<InsetMathScript*>(p));
1669                                 idx() = p->idxOfScript(up);
1670                                 pos() = 0;
1671                                 
1672                                 // we went in the right direction? Otherwise don't jump into the script
1673                                 int x;
1674                                 int y;
1675                                 getPos(x, y);
1676                                 int oy = theLyXFunc().cursorBeforeDispatchY();
1677                                 if ((!up && y <= oy) ||
1678                                                 (up && y >= oy))
1679                                         operator=(old);
1680                                 else
1681                                         return true;
1682                         }
1683                 }
1684         }
1685                 
1686         // try to find an inset that knows better then we,
1687         if (inset().idxUpDown(*this, up)) {
1688                 //lyxerr << "idxUpDown triggered" << endl;
1689                 // try to find best position within this inset
1690                 if (!selection())
1691                         setCursor(bruteFind2(*this, xo, yo));
1692                 return true;
1693         }
1694         
1695         // any improvement going just out of inset?
1696         if (popBackward() && inMathed()) {
1697                 //lyxerr << "updown: popBackward succeeded" << endl;
1698                 int xnew;
1699                 int ynew;
1700                 int yold = theLyXFunc().cursorBeforeDispatchY();
1701                 getPos(xnew, ynew);
1702                 if (up ? ynew < yold : ynew > yold)
1703                         return true;
1704         }
1705         
1706         // no success, we are probably at the document top or bottom
1707         operator=(old);
1708         return false;
1709 }
1710
1711
1712 bool Cursor::atFirstOrLastRow(bool up)
1713 {
1714         TextMetrics const & tm = bv_->textMetrics(text());
1715         ParagraphMetrics const & pm = tm.parMetrics(pit());
1716         
1717         int row;
1718         if (pos() && boundary())
1719                 row = pm.pos2row(pos() - 1);
1720         else
1721                 row = pm.pos2row(pos());
1722         
1723         if (up) {
1724                 if (pit() == 0 && row == 0)
1725                         return true;
1726         } else {
1727                 if (pit() + 1 >= int(text()->paragraphs().size()) &&
1728                                 row + 1 >= int(pm.rows().size()))
1729                         return true;
1730         }
1731         return false;
1732 }
1733
1734 bool Cursor::upDownInText(bool up, bool & updateNeeded)
1735 {
1736         LASSERT(text(), /**/);
1737
1738         // where are we?
1739         int xo = 0;
1740         int yo = 0;
1741         getPos(xo, yo);
1742         xo = theLyXFunc().cursorBeforeDispatchX();
1743
1744         // update the targetX - this is here before the "return false"
1745         // to set a new target which can be used by InsetTexts above
1746         // if we cannot move up/down inside this inset anymore
1747         if (x_target_ == -1)
1748                 setTargetX(xo);
1749         else if (xo - textTargetOffset() != x_target() &&
1750                                          depth() == beforeDispatchCursor_.depth()) {
1751                 // In text mode inside the line (not left or right) possibly set a new target_x,
1752                 // but only if we are somewhere else than the previous target-offset.
1753                 
1754                 // We want to keep the x-target on subsequent up/down movements
1755                 // that cross beyond the end of short lines. Thus a special
1756                 // handling when the cursor is at the end of line: Use the new
1757                 // x-target only if the old one was before the end of line
1758                 // or the old one was after the beginning of the line
1759                 bool inRTL = isWithinRtlParagraph(*this);
1760                 bool left;
1761                 bool right;
1762                 if (inRTL) {
1763                         left = pos() == textRow().endpos();
1764                         right = pos() == textRow().pos();
1765                 } else {
1766                         left = pos() == textRow().pos();
1767                         right = pos() == textRow().endpos();
1768                 }
1769                 if ((!left && !right) ||
1770                                 (left && !right && xo < x_target_) ||
1771                                 (!left && right && x_target_ < xo))
1772                         setTargetX(xo);
1773                 else
1774                         xo = targetX();
1775         } else
1776                 xo = targetX();
1777                 
1778         // first get the current line
1779         TextMetrics & tm = bv_->textMetrics(text());
1780         ParagraphMetrics const & pm = tm.parMetrics(pit());
1781         int row;
1782         if (pos() && boundary())
1783                 row = pm.pos2row(pos() - 1);
1784         else
1785                 row = pm.pos2row(pos());
1786                 
1787         if (atFirstOrLastRow(up)) {
1788                 // Is there a place for the cursor to go ? If yes, we
1789                 // can execute the DEPM, otherwise we should keep the
1790                 // paragraph to host the cursor.
1791                 Cursor dummy = *this;
1792                 bool valid_destination = false;
1793                 for(; dummy.depth(); dummy.pop())
1794                         if (!dummy.atFirstOrLastRow(up)) {
1795                                 valid_destination = true;
1796                                 break;
1797                         }
1798
1799                 // will a next dispatch follow and if there is a new 
1800                 // dispatch will it move the cursor out ?
1801                 if (depth() > 1 && valid_destination) {
1802                         // The cursor hasn't changed yet. This happens when
1803                         // you e.g. move out of an inset. And to give the 
1804                         // DEPM the possibility of doing something we must
1805                         // provide it with two different cursors. (Lgb, vfr)
1806                         dummy = *this;
1807                         dummy.pos() = dummy.pos() == 0 ? dummy.lastpos() : 0;
1808                         dummy.pit() = dummy.pit() == 0 ? dummy.lastpit() : 0;
1809
1810                         updateNeeded |= bv().checkDepm(dummy, *this);
1811                         updateTextTargetOffset();
1812                 }
1813                 return false;
1814         }
1815
1816         // with and without selection are handled differently
1817         if (!selection()) {
1818                 int yo = bv().getPos(*this, boundary()).y_;
1819                 Cursor old = *this;
1820                 // To next/previous row
1821                 if (up)
1822                         tm.editXY(*this, xo, yo - textRow().ascent() - 1);
1823                 else
1824                         tm.editXY(*this, xo, yo + textRow().descent() + 1);
1825                 clearSelection();
1826                 
1827                 // This happens when you move out of an inset.  
1828                 // And to give the DEPM the possibility of doing  
1829                 // something we must provide it with two different  
1830                 // cursors. (Lgb)  
1831                 Cursor dummy = *this;
1832                 if (dummy == old)
1833                         ++dummy.pos();
1834                 if (bv().checkDepm(dummy, old)) {
1835                         updateNeeded = true;
1836                         // Make sure that cur gets back whatever happened to dummy(Lgb) 
1837                         operator=(dummy);
1838                 }
1839         } else {
1840                 // if there is a selection, we stay out of any inset, and just jump to the right position:
1841                 Cursor old = *this;
1842                 if (up) {
1843                         if (row > 0) {
1844                                 top().pos() = min(tm.x2pos(pit(), row - 1, xo), top().lastpos());
1845                         } else if (pit() > 0) {
1846                                 --pit();
1847                                 TextMetrics & tm = bv_->textMetrics(text());
1848                                 if (!tm.contains(pit()))
1849                                         tm.newParMetricsUp();
1850                                 ParagraphMetrics const & pmcur = tm.parMetrics(pit());
1851                                 top().pos() = min(tm.x2pos(pit(), pmcur.rows().size() - 1, xo), top().lastpos());
1852                         }
1853                 } else {
1854                         if (row + 1 < int(pm.rows().size())) {
1855                                 top().pos() = min(tm.x2pos(pit(), row + 1, xo), top().lastpos());
1856                         } else if (pit() + 1 < int(text()->paragraphs().size())) {
1857                                 ++pit();
1858                                 TextMetrics & tm = bv_->textMetrics(text());
1859                                 if (!tm.contains(pit()))
1860                                         tm.newParMetricsDown();
1861                                 top().pos() = min(tm.x2pos(pit(), 0, xo), top().lastpos());
1862                         }
1863                 }
1864
1865                 updateNeeded |= bv().checkDepm(*this, old);
1866         }
1867
1868         updateTextTargetOffset();
1869         return true;
1870 }       
1871
1872
1873 void Cursor::handleFont(string const & font)
1874 {
1875         LYXERR(Debug::DEBUG, font);
1876         docstring safe;
1877         if (selection()) {
1878                 macroModeClose();
1879                 safe = cap::grabAndEraseSelection(*this);
1880         }
1881
1882         recordUndoInset();
1883
1884         if (lastpos() != 0) {
1885                 // something left in the cell
1886                 if (pos() == 0) {
1887                         // cursor in first position
1888                         popBackward();
1889                 } else if (pos() == lastpos()) {
1890                         // cursor in last position
1891                         popForward();
1892                 } else {
1893                         // cursor in between. split cell
1894                         MathData::iterator bt = cell().begin();
1895                         MathAtom at = createInsetMath(from_utf8(font));
1896                         at.nucleus()->cell(0) = MathData(bt, bt + pos());
1897                         cell().erase(bt, bt + pos());
1898                         popBackward();
1899                         plainInsert(at);
1900                 }
1901         } else {
1902                 // nothing left in the cell
1903                 popBackward();
1904                 plainErase();
1905                 resetAnchor();
1906         }
1907         insert(safe);
1908 }
1909
1910
1911 void Cursor::message(docstring const & msg) const
1912 {
1913         theLyXFunc().setMessage(msg);
1914 }
1915
1916
1917 void Cursor::errorMessage(docstring const & msg) const
1918 {
1919         theLyXFunc().setErrorMessage(msg);
1920 }
1921
1922
1923 docstring Cursor::selectionAsString(bool with_label) const
1924 {
1925         if (!selection())
1926                 return docstring();
1927
1928         int const label = with_label
1929                 ? AS_STR_LABEL | AS_STR_INSETS : AS_STR_INSETS;
1930
1931         if (inTexted()) {
1932                 idx_type const startidx = selBegin().idx();
1933                 idx_type const endidx = selEnd().idx();
1934                 if (startidx != endidx) {
1935                         // multicell selection
1936                         InsetTabular * table = inset().asInsetTabular();
1937                         LASSERT(table, return docstring());
1938                         return table->asString(startidx, endidx);
1939                 }
1940                 
1941                 ParagraphList const & pars = text()->paragraphs();
1942
1943                 pit_type const startpit = selBegin().pit();
1944                 pit_type const endpit = selEnd().pit();
1945                 size_t const startpos = selBegin().pos();
1946                 size_t const endpos = selEnd().pos();
1947
1948                 if (startpit == endpit)
1949                         return pars[startpit].asString(startpos, endpos, label);
1950
1951                 // First paragraph in selection
1952                 docstring result = pars[startpit].
1953                         asString(startpos, pars[startpit].size(), label)
1954                                  + parbreak(pars[startpit]);
1955
1956                 // The paragraphs in between (if any)
1957                 for (pit_type pit = startpit + 1; pit != endpit; ++pit) {
1958                         Paragraph const & par = pars[pit];
1959                         result += par.asString(0, par.size(), label)
1960                                   + parbreak(pars[pit]);
1961                 }
1962
1963                 // Last paragraph in selection
1964                 result += pars[endpit].asString(0, endpos, label);
1965
1966                 return result;
1967         }
1968
1969         if (inMathed())
1970                 return cap::grabSelection(*this);
1971
1972         return docstring();
1973 }
1974
1975
1976 docstring Cursor::currentState() const
1977 {
1978         if (inMathed()) {
1979                 odocstringstream os;
1980                 info(os);
1981                 return os.str();
1982         }
1983
1984         if (inTexted())
1985                 return text()->currentState(*this);
1986
1987         return docstring();
1988 }
1989
1990
1991 docstring Cursor::getPossibleLabel() const
1992 {
1993         return inMathed() ? from_ascii("eq:") : text()->getPossibleLabel(*this);
1994 }
1995
1996
1997 Encoding const * Cursor::getEncoding() const
1998 {
1999         if (empty())
2000                 return 0;
2001         CursorSlice const & sl = innerTextSlice();
2002         Text const & text = *sl.text();
2003         Font font = text.getPar(sl.pit()).getFont(
2004                 bv().buffer().params(), sl.pos(), outerFont(sl.pit(), text.paragraphs()));
2005         return font.language()->encoding();
2006 }
2007
2008
2009 void Cursor::undispatched()
2010 {
2011         disp_.dispatched(false);
2012 }
2013
2014
2015 void Cursor::dispatched()
2016 {
2017         disp_.dispatched(true);
2018 }
2019
2020
2021 void Cursor::updateFlags(Update::flags f)
2022 {
2023         disp_.update(f);
2024 }
2025
2026
2027 void Cursor::noUpdate()
2028 {
2029         disp_.update(Update::None);
2030 }
2031
2032
2033 Font Cursor::getFont() const
2034 {
2035         // The logic here should more or less match to the Cursor::setCurrentFont
2036         // logic, i.e. the cursor height should give a hint what will happen
2037         // if a character is entered.
2038         
2039         // HACK. far from being perfect...
2040
2041         CursorSlice const & sl = innerTextSlice();
2042         Text const & text = *sl.text();
2043         Paragraph const & par = text.getPar(sl.pit());
2044         
2045         // on boundary, so we are really at the character before
2046         pos_type pos = sl.pos();
2047         if (pos > 0 && boundary())
2048                 --pos;
2049         
2050         // on space? Take the font before (only for RTL boundary stay)
2051         if (pos > 0) {
2052                 TextMetrics const & tm = bv().textMetrics(&text);
2053                 if (pos == sl.lastpos()
2054                         || (par.isSeparator(pos) 
2055                         && !tm.isRTLBoundary(sl.pit(), pos)))
2056                         --pos;
2057         }
2058         
2059         // get font at the position
2060         Font font = par.getFont(buffer()->params(), pos,
2061                 outerFont(sl.pit(), text.paragraphs()));
2062
2063         return font;
2064 }
2065
2066
2067 bool Cursor::fixIfBroken()
2068 {
2069         if (DocIterator::fixIfBroken()) {
2070                         clearSelection();
2071                         return true;
2072         }
2073         return false;
2074 }
2075
2076
2077 bool notifyCursorLeavesOrEnters(Cursor const & old, Cursor & cur)
2078 {
2079         // find inset in common
2080         size_type i;
2081         for (i = 0; i < old.depth() && i < cur.depth(); ++i) {
2082                 if (&old[i].inset() != &cur[i].inset())
2083                         break;
2084         }
2085
2086         // update words if we just moved to another paragraph
2087         if (i == old.depth() && i == cur.depth()
2088             && !cur.buffer()->isClean()
2089             && cur.inTexted() && old.inTexted()
2090             && cur.pit() != old.pit()) {
2091                 old.paragraph().updateWords();
2092         }
2093
2094         // notify everything on top of the common part in old cursor,
2095         // but stop if the inset claims the cursor to be invalid now
2096         for (size_type j = i; j < old.depth(); ++j) {
2097                 Cursor insetPos = old;
2098                 insetPos.cutOff(j);
2099                 if (old[j].inset().notifyCursorLeaves(insetPos, cur))
2100                         return true;
2101         }
2102
2103         // notify everything on top of the common part in new cursor,
2104         // but stop if the inset claims the cursor to be invalid now
2105         for (; i < cur.depth(); ++i) {
2106                 if (cur[i].inset().notifyCursorEnters(cur))
2107                         return true;
2108         }
2109         
2110         return false;
2111 }
2112
2113
2114 void Cursor::setCurrentFont()
2115 {
2116         CursorSlice const & cs = innerTextSlice();
2117         Paragraph const & par = cs.paragraph();
2118         pos_type cpit = cs.pit();
2119         pos_type cpos = cs.pos();
2120         Text const & ctext = *cs.text();
2121         TextMetrics const & tm = bv().textMetrics(&ctext);
2122
2123         // are we behind previous char in fact? -> go to that char
2124         if (cpos > 0 && boundary())
2125                 --cpos;
2126
2127         // find position to take the font from
2128         if (cpos != 0) {
2129                 // paragraph end? -> font of last char
2130                 if (cpos == lastpos())
2131                         --cpos;
2132                 // on space? -> look at the words in front of space
2133                 else if (cpos > 0 && par.isSeparator(cpos))     {
2134                         // abc| def -> font of c
2135                         // abc |[WERBEH], i.e. boundary==true -> font of c
2136                         // abc [WERBEH]| def, font of the space
2137                         if (!tm.isRTLBoundary(cpit, cpos))
2138                                 --cpos;
2139                 }
2140         }
2141
2142         // get font
2143         BufferParams const & bufparams = buffer()->params();
2144         current_font = par.getFontSettings(bufparams, cpos);
2145         real_current_font = tm.displayFont(cpit, cpos);
2146
2147         // special case for paragraph end
2148         if (cs.pos() == lastpos()
2149             && tm.isRTLBoundary(cpit, cs.pos())
2150             && !boundary()) {
2151                 Language const * lang = par.getParLanguage(bufparams);
2152                 current_font.setLanguage(lang);
2153                 current_font.fontInfo().setNumber(FONT_OFF);
2154                 real_current_font.setLanguage(lang);
2155                 real_current_font.fontInfo().setNumber(FONT_OFF);
2156         }
2157 }
2158
2159
2160 bool Cursor::textUndo()
2161 {
2162         DocIterator dit = *this;
2163         // Undo::textUndo() will modify dit.
2164         if (!buffer()->undo().textUndo(dit))
2165                 return false;
2166         // Set cursor
2167         setCursor(dit);
2168         clearSelection();
2169         fixIfBroken();
2170         return true;
2171 }
2172
2173
2174 bool Cursor::textRedo()
2175 {
2176         DocIterator dit = *this;
2177         // Undo::textRedo() will modify dit.
2178         if (!buffer()->undo().textRedo(dit))
2179                 return false;
2180         // Set cursor
2181         setCursor(dit);
2182         clearSelection();
2183         fixIfBroken();
2184         return true;
2185 }
2186
2187
2188 void Cursor::finishUndo() const
2189 {
2190         buffer()->undo().finishUndo();
2191 }
2192
2193
2194 void Cursor::beginUndoGroup() const
2195 {
2196         buffer()->undo().beginUndoGroup();
2197 }
2198
2199
2200 void Cursor::endUndoGroup() const
2201 {
2202         buffer()->undo().endUndoGroup();
2203 }
2204
2205
2206 void Cursor::recordUndo(UndoKind kind, pit_type from, pit_type to) const
2207 {
2208         buffer()->undo().recordUndo(*this, kind, from, to);
2209 }
2210
2211
2212 void Cursor::recordUndo(UndoKind kind, pit_type from) const
2213 {
2214         buffer()->undo().recordUndo(*this, kind, from);
2215 }
2216
2217
2218 void Cursor::recordUndo(UndoKind kind) const
2219 {
2220         buffer()->undo().recordUndo(*this, kind);
2221 }
2222
2223
2224 void Cursor::recordUndoInset(UndoKind kind) const
2225 {
2226         buffer()->undo().recordUndoInset(*this, kind);
2227 }
2228
2229
2230 void Cursor::recordUndoFullDocument() const
2231 {
2232         buffer()->undo().recordUndoFullDocument(*this);
2233 }
2234
2235
2236 void Cursor::recordUndoSelection() const
2237 {
2238         if (inMathed()) {
2239                 if (cap::multipleCellsSelected(*this))
2240                         recordUndoInset();
2241                 else
2242                         recordUndo();
2243         } else {
2244                 buffer()->undo().recordUndo(*this, ATOMIC_UNDO,
2245                         selBegin().pit(), selEnd().pit());
2246         }
2247 }
2248
2249
2250 void Cursor::checkBufferStructure()
2251 {
2252         Buffer const * master = buffer()->masterBuffer();
2253         master->tocBackend().updateItem(*this);
2254         if (master != buffer() && !master->hasGuiDelegate())
2255                 // In case the master has no gui associated with it, 
2256                 // the TocItem is not updated (part of bug 5699).
2257                 buffer()->tocBackend().updateItem(*this);
2258 }
2259
2260
2261 } // namespace lyx