]> git.lyx.org Git - lyx.git/blobdiff - src/BufferView.cpp
Fix text direction issue for InsetInfo in RTL context
[lyx.git] / src / BufferView.cpp
index 27176b350a48f677b68d5a804bd3d561af0f2ecf..a3d080438125dc0679dea450f8973855e377c779 100644 (file)
@@ -230,7 +230,8 @@ enum ScreenUpdateStrategy {
 
 struct BufferView::Private
 {
-       Private(BufferView & bv) : update_strategy_(FullScreenUpdate),
+       Private(BufferView & bv) :
+               update_strategy_(FullScreenUpdate),
                update_flags_(Update::Force),
                wh_(0), cursor_(bv),
                anchor_pit_(0), anchor_ypos_(0),
@@ -238,7 +239,8 @@ struct BufferView::Private
                last_inset_(0), clickable_inset_(false),
                mouse_position_cache_(),
                bookmark_edit_position_(-1), gui_(0),
-               horiz_scroll_offset_(0), repaint_caret_row_(false)
+               horiz_scroll_offset_(0),
+               caret_ascent_(0), caret_descent_(0)
        {
                xsel_cache_.set = false;
        }
@@ -318,11 +320,11 @@ struct BufferView::Private
        /// at previous draw event
        CursorSlice last_row_slice_;
 
-       /// a slice pointing to where the cursor has been drawn after the current
-       /// draw() call.
-       CursorSlice caret_slice_;
-       /// indicates whether the caret slice needs to be repainted in this draw() run.
-       bool repaint_caret_row_;
+       // The vertical size of the blinking caret. Only used for math
+       // Using it for text could be bad when undo restores the cursor
+       // current font, since the caret size could become wrong.
+       int caret_ascent_;
+       int caret_descent_;
 };
 
 
@@ -475,7 +477,8 @@ string flagsAsString(Update::flags flags)
        return string((flags & Update::FitCursor) ? "FitCursor " : "")
                + ((flags & Update::Force) ? "Force " : "")
                + ((flags & Update::ForceDraw) ? "ForceDraw " : "")
-               + ((flags & Update::SinglePar) ? "SinglePar " : "");
+               + ((flags & Update::SinglePar) ? "SinglePar " : "")
+               + ((flags & Update::Decoration) ? "Decoration " : "");
 }
 
 }
@@ -501,6 +504,16 @@ void BufferView::processUpdateFlags(Update::flags flags)
                updateMetrics(flags);
        }
 
+       // Detect whether we can only repaint a single paragraph.
+       // We handle this before FitCursor because the later will require
+       // correct metrics at cursor position.
+       if (!(flags & Update::ForceDraw)) {
+               if (singleParUpdate())
+                       flags = flags | Update::SinglePar;
+               else
+                       updateMetrics(flags);
+       }
+
        // Then make sure that the screen contains the cursor if needed
        if (flags & Update::FitCursor) {
                if (needsFitCursor()) {
@@ -511,14 +524,6 @@ void BufferView::processUpdateFlags(Update::flags flags)
                flags = flags & ~Update::FitCursor;
        }
 
-       // Finally detect whether we can only repaint a single paragraph
-       if (!(flags & Update::ForceDraw)) {
-               if (singleParUpdate())
-                       flags = flags | Update::SinglePar;
-               else
-                       updateMetrics(flags);
-       }
-
        // Add flags to the the update flags. These will be reset to None
        // after the redraw is actually done
        d->update_flags_ = d->update_flags_ | flags;
@@ -1152,7 +1157,7 @@ bool BufferView::getStatus(FuncRequest const & cmd, FuncStatus & flag)
                break;
 
        case LFUN_GRAPHICS_UNIFY:
-               flag.setEnabled(cur.selection());
+               flag.setEnabled(cur.countInsetsInSelection(GRAPHICS_CODE)>1);
                break;
 
        case LFUN_WORD_FINDADV: {
@@ -1390,14 +1395,14 @@ void BufferView::dispatch(FuncRequest const & cmd, DispatchResult & dr)
                // We need to find out if the bibliography information
                // has changed. See bug #11055.
                // So these should not be references...
-               LayoutModuleList const engines = buffer().params().citeEngine();
+               string const engine = buffer().params().citeEngine();
                CiteEngineType const enginetype = buffer().params().citeEngineType();
                if (!cur.textUndo())
                        dr.setMessage(_("No further undo information"));
                else {
                        dr.screenUpdate(Update::Force | Update::FitCursor);
                        dr.forceBufferUpdate();
-                       if (buffer().params().citeEngine() != engines ||
+                       if (buffer().params().citeEngine() != engine ||
                            buffer().params().citeEngineType() != enginetype)
                                buffer().invalidateCiteLabels();
                }
@@ -1410,14 +1415,14 @@ void BufferView::dispatch(FuncRequest const & cmd, DispatchResult & dr)
                // We need to find out if the bibliography information
                // has changed. See bug #11055.
                // So these should not be references...
-               LayoutModuleList const engines = buffer().params().citeEngine();
+               string const engine = buffer().params().citeEngine();
                CiteEngineType const enginetype = buffer().params().citeEngineType();
                if (!cur.textRedo())
                        dr.setMessage(_("No further redo information"));
                else {
                        dr.screenUpdate(Update::Force | Update::FitCursor);
                        dr.forceBufferUpdate();
-                       if (buffer().params().citeEngine() != engines ||
+                       if (buffer().params().citeEngine() != engine ||
                            buffer().params().citeEngineType() != enginetype)
                                buffer().invalidateCiteLabels();
                }
@@ -2988,20 +2993,25 @@ bool BufferView::paragraphVisible(DocIterator const & dit) const
 }
 
 
+void BufferView::setCaretAscentDescent(int asc, int des)
+{
+       d->caret_ascent_ = asc;
+       d->caret_descent_ = des;
+}
+
+
 void BufferView::caretPosAndHeight(Point & p, int & h) const
 {
+       int asc, des;
        Cursor const & cur = cursor();
-       Font const font = cur.real_current_font;
-       frontend::FontMetrics const & fm = theFontMetrics(font);
-       int asc = fm.maxAscent();
-       int des = fm.maxDescent();
-       // If the cursor is in mathed and it has cached metrics, reduce
-       // the height to fit the inner cell (mathed cells are tight
-       // vertically).
-       if (cur.inMathed() && coordCache().getArrays().hasDim(&cur.cell())) {
-               Dimension const dim = cur.cell().dimension(*this);
-               asc = min(asc, dim.asc);
-               des = min(des, dim.des);
+       if (cur.inMathed()) {
+               asc = d->caret_ascent_;
+               des = d->caret_descent_;
+       } else {
+               Font const font = cur.real_current_font;
+               frontend::FontMetrics const & fm = theFontMetrics(font);
+               asc = fm.maxAscent();
+               des = fm.maxDescent();
        }
        h = asc + des;
        p = getPos(cur);
@@ -3069,29 +3079,6 @@ void BufferView::setCurrentRowSlice(CursorSlice const & rowSlice)
 }
 
 
-namespace {
-
-bool sliceInRow(CursorSlice const & cs, Text const * text, Row const & row)
-{
-       /* The normal case is the last line. The previous line takes care
-        * of empty rows (e.g. empty paragraphs). Cursor boundary issues
-        * are taken care of when setting caret_slice_ in
-        * BufferView::draw.
-        */
-       return !cs.empty() && cs.text() == text && cs.pit() == row.pit()
-              && ((row.pos() == row.endpos() && row.pos() == cs.pos())
-                 || (row.pos() <= cs.pos() && cs.pos() < row.endpos()));
-}
-
-}
-
-
-bool BufferView::needRepaint(Text const * text, Row const & row) const
-{
-       return d->repaint_caret_row_ && sliceInRow(d->caret_slice_, text, row);
-}
-
-
 void BufferView::checkCursorScrollOffset()
 {
        CursorSlice rowSlice = d->cursor_.bottom();
@@ -3163,16 +3150,6 @@ void BufferView::draw(frontend::Painter & pain, bool paint_caret)
        int const y = tm.first().second->position();
        PainterInfo pi(this, pain);
 
-       /**  A repaint of the previous caret row is needed if there is
-        *  caret painted on screen and either
-        *   1/ a new caret has to be painted at a place different from
-        *      the existing one;
-        *   2/ there is no need for a caret anymore.
-        */
-       d->repaint_caret_row_ = !d->caret_slice_.empty() &&
-               ((paint_caret && d->cursor_.top() != d->caret_slice_)
-                || ! paint_caret);
-
        // Check whether the row where the cursor lives needs to be scrolled.
        // Update the drawing strategy if needed.
        checkCursorScrollOffset();
@@ -3188,7 +3165,7 @@ void BufferView::draw(frontend::Painter & pain, bool paint_caret)
                if (pain.isNull()) {
                        pi.full_repaint = true;
                        tm.draw(pi, 0, y);
-               } else if (d->repaint_caret_row_) {
+               } else {
                        pi.full_repaint = false;
                        tm.draw(pi, 0, y);
                }
@@ -3262,15 +3239,15 @@ void BufferView::draw(frontend::Painter & pain, bool paint_caret)
                d->update_flags_ = Update::None;
        }
 
-       // Remember what has just been done for the next draw() step
+       // If a caret has to be painted, mark its text row as dirty to
+       //make sure that it will be repainted on next redraw.
+       /* FIXME: investigate whether this can be avoided when the cursor did not
+        * move at all
+        */
        if (paint_caret) {
-               d->caret_slice_ = d->cursor_.top();
-               if (d->caret_slice_.pos() > 0
-                   && (d->cursor_.boundary()
-                       || d->caret_slice_.pos() == d->caret_slice_.lastpos()))
-                       --d->caret_slice_.pos();
-       } else
-               d->caret_slice_ = CursorSlice();
+               Row const & caret_row = d->cursor_.textRow();
+               caret_row.changed(true);
+       }
 }