Bug 5173: Undo Grouping

http://bugzilla.lyx.org/show_bug.cgi?id=5173

	* Undo.cpp (beginUndoGroup, endUndoGroup): new methods.
	(UndoElement): add group_id member
	(UndoElementStack): when removing old undo entries, make sure to
	remove whole groups.
	(Undo::Private): add group_id and group_level (for nesting) members.
	(doTextUndoOrRedo): new method. Apply _one_ UndoElement.
	(textUndoOrRedo): call doTextUndoOrRedo on each element of the current
	group.
	(recordUndoFullDocument): put inside an undo group

	* Cursor.cpp (beginUndoGroup, endUndoGroup): new methods.

	* LyXFunc.cpp (dispatch): add calls to (begin|end)UndoGroup.

	* lyxfind.cpp (replaceAll): use several recordUndo instead
	of one recordUndoFullDocument.


git-svn-id: svn://svn.lyx.org/lyx/lyx-devel/trunk@26178 a592a061-630c-0410-9148-cb99ea01b6c8
This commit is contained in:
Jean-Marc Lasgouttes 2008-08-15 19:24:56 +00:00
parent 608c46b080
commit 17dd645862
6 changed files with 124 additions and 32 deletions

View File

@ -2117,6 +2117,18 @@ void Cursor::finishUndo() const
} }
void Cursor::beginUndoGroup() const
{
bv_->buffer().undo().beginUndoGroup();
}
void Cursor::endUndoGroup() const
{
bv_->buffer().undo().endUndoGroup();
}
void Cursor::recordUndo(UndoKind kind, pit_type from, pit_type to) const void Cursor::recordUndo(UndoKind kind, pit_type from, pit_type to) const
{ {
bv_->buffer().undo().recordUndo(*this, kind, from, to); bv_->buffer().undo().recordUndo(*this, kind, from, to);

View File

@ -242,6 +242,12 @@ public:
/// makes sure the next operation will be stored /// makes sure the next operation will be stored
void finishUndo() const; void finishUndo() const;
/// open a new group of undo operations. Groups can be nested.
void beginUndoGroup() const;
/// end the current undo group
void endUndoGroup() const;
/// The general case: prepare undo for an arbitrary range. /// The general case: prepare undo for an arbitrary range.
void recordUndo(UndoKind kind, pit_type from, pit_type to) const; void recordUndo(UndoKind kind, pit_type from, pit_type to) const;

View File

@ -1569,10 +1569,17 @@ void LyXFunc::dispatch(FuncRequest const & cmd)
} }
LASSERT(lyx_view_->view(), /**/); LASSERT(lyx_view_->view(), /**/);
// Start an undo group. Normally, all the code
// above only invoked recordUndoFullDocument,
// which does not really require a group.
view()->cursor().beginUndoGroup();
// Let the current BufferView dispatch its own actions. // Let the current BufferView dispatch its own actions.
if (view()->dispatch(cmd)) { if (view()->dispatch(cmd)) {
// The BufferView took care of its own updates if needed. // The BufferView took care of its own updates if needed.
updateFlags = Update::None; updateFlags = Update::None;
view()->cursor().endUndoGroup();
break; break;
} }
@ -1582,6 +1589,10 @@ void LyXFunc::dispatch(FuncRequest const & cmd)
cursorPosBeforeDispatchY_); cursorPosBeforeDispatchY_);
view()->cursor().dispatch(cmd); view()->cursor().dispatch(cmd);
// we assume here that the buffer view has not
// changed since the beginUndoGroup.
view()->cursor().endUndoGroup();
// notify insets we just left // notify insets we just left
if (view()->cursor() != old) { if (view()->cursor() != old) {
old.fixIfBroken(); old.fixIfBroken();

View File

@ -71,9 +71,9 @@ struct UndoElement
StableDocIterator const & cel, StableDocIterator const & cel,
pit_type fro, pit_type en, ParagraphList * pl, pit_type fro, pit_type en, ParagraphList * pl,
MathData * ar, BufferParams const & bp, MathData * ar, BufferParams const & bp,
bool ifb) : bool ifb, size_t gid) :
kind(kin), cursor(cur), cell(cel), from(fro), end(en), kind(kin), cursor(cur), cell(cel), from(fro), end(en),
pars(pl), array(ar), bparams(0), isFullBuffer(ifb) pars(pl), array(ar), bparams(0), isFullBuffer(ifb), group_id(gid)
{ {
if (isFullBuffer) if (isFullBuffer)
bparams = new BufferParams(bp); bparams = new BufferParams(bp);
@ -91,6 +91,7 @@ struct UndoElement
bparams = ue.isFullBuffer bparams = ue.isFullBuffer
? new BufferParams(*ue.bparams) : ue.bparams; ? new BufferParams(*ue.bparams) : ue.bparams;
isFullBuffer = ue.isFullBuffer; isFullBuffer = ue.isFullBuffer;
group_id = ue.group_id;
} }
/// ///
~UndoElement() ~UndoElement()
@ -116,6 +117,8 @@ struct UndoElement
BufferParams const * bparams; BufferParams const * bparams;
/// Only used in case of full backups /// Only used in case of full backups
bool isFullBuffer; bool isFullBuffer;
/// the element's group id
size_t group_id;
private: private:
/// Protect construction /// Protect construction
UndoElement(); UndoElement();
@ -148,13 +151,17 @@ public:
c_.clear(); c_.clear();
} }
/// Push an item on to the stack, deleting the /// Push an item on to the stack, deleting the bottom group on
/// bottom item on overflow. /// overflow.
void push(UndoElement const & v) { void push(UndoElement const & v) {
c_.push_front(v); c_.push_front(v);
if (c_.size() > limit_) if (c_.size() > limit_) {
// remove a whole group at once.
const size_t gid = c_.back().group_id;
while (!c_.empty() && c_.back().group_id == gid)
c_.pop_back(); c_.pop_back();
} }
}
private: private:
/// Internal contents. /// Internal contents.
@ -166,10 +173,14 @@ private:
struct Undo::Private struct Undo::Private
{ {
Private(Buffer & buffer) : buffer_(buffer), undo_finished_(true) {} Private(Buffer & buffer) : buffer_(buffer), undo_finished_(true),
group_id(0), group_level(0) {}
// Returns false if no undo possible. // Do one undo/redo step
void doTextUndoOrRedo(DocIterator & cur, UndoElementStack & stack, UndoElementStack & otherStack);
// Apply one undo/redo group. Returns false if no undo possible.
bool textUndoOrRedo(DocIterator & cur, bool isUndoOperation); bool textUndoOrRedo(DocIterator & cur, bool isUndoOperation);
/// ///
void doRecordUndo(UndoKind kind, void doRecordUndo(UndoKind kind,
DocIterator const & cell, DocIterator const & cell,
@ -177,8 +188,7 @@ struct Undo::Private
pit_type last_pit, pit_type last_pit,
DocIterator const & cur, DocIterator const & cur,
bool isFullBuffer, bool isFullBuffer,
bool isUndoOperation); UndoElementStack & stack);
/// ///
void recordUndo(UndoKind kind, void recordUndo(UndoKind kind,
DocIterator const & cur, DocIterator const & cur,
@ -194,6 +204,11 @@ struct Undo::Private
/// The flag used by Undo::finishUndo(). /// The flag used by Undo::finishUndo().
bool undo_finished_; bool undo_finished_;
/// Current group Id.
size_t group_id;
/// Current group nesting nevel.
size_t group_level;
}; };
@ -246,8 +261,13 @@ void Undo::Private::doRecordUndo(UndoKind kind,
pit_type first_pit, pit_type last_pit, pit_type first_pit, pit_type last_pit,
DocIterator const & cur, DocIterator const & cur,
bool isFullBuffer, bool isFullBuffer,
bool isUndoOperation) UndoElementStack & stack)
{ {
if (!group_level) {
LYXERR0("There is no group open (creating one)");
++group_id;
}
if (first_pit > last_pit) if (first_pit > last_pit)
swap(first_pit, last_pit); swap(first_pit, last_pit);
@ -256,7 +276,6 @@ void Undo::Private::doRecordUndo(UndoKind kind,
// we want combine 'similar' non-ATOMIC undo recordings to one. // we want combine 'similar' non-ATOMIC undo recordings to one.
pit_type from = first_pit; pit_type from = first_pit;
pit_type end = cell.lastpit() - last_pit; pit_type end = cell.lastpit() - last_pit;
UndoElementStack & stack = isUndoOperation ? undostack_ : redostack_;
if (!undo_finished_ if (!undo_finished_
&& kind != ATOMIC_UNDO && kind != ATOMIC_UNDO
&& !stack.empty() && !stack.empty()
@ -266,9 +285,10 @@ void Undo::Private::doRecordUndo(UndoKind kind,
&& stack.top().end == end) && stack.top().end == end)
return; return;
LYXERR(Debug::UNDO, "Create undo element of group " << group_id);
// create the position information of the Undo entry // create the position information of the Undo entry
UndoElement undo(kind, cur, cell, from, end, 0, 0, UndoElement undo(kind, cur, cell, from, end, 0, 0,
buffer_.params(), isFullBuffer); buffer_.params(), isFullBuffer, group_id);
// fill in the real data to be saved // fill in the real data to be saved
if (cell.inMathed()) { if (cell.inMathed()) {
@ -304,7 +324,7 @@ void Undo::Private::recordUndo(UndoKind kind, DocIterator const & cur,
LASSERT(last_pit <= cur.lastpit(), /**/); LASSERT(last_pit <= cur.lastpit(), /**/);
doRecordUndo(kind, cur, first_pit, last_pit, cur, doRecordUndo(kind, cur, first_pit, last_pit, cur,
false, true); false, undostack_);
undo_finished_ = false; undo_finished_ = false;
redostack_.clear(); redostack_.clear();
@ -314,20 +334,11 @@ void Undo::Private::recordUndo(UndoKind kind, DocIterator const & cur,
} }
bool Undo::Private::textUndoOrRedo(DocIterator & cur, bool isUndoOperation) void Undo::Private::doTextUndoOrRedo(DocIterator & cur, UndoElementStack & stack, UndoElementStack & otherstack)
{ {
undo_finished_ = true;
UndoElementStack & stack = isUndoOperation ? undostack_ : redostack_;
if (stack.empty())
// Nothing to do.
return false;
UndoElementStack & otherstack = isUndoOperation ? redostack_ : undostack_;
// Adjust undo stack and get hold of current undo data. // Adjust undo stack and get hold of current undo data.
UndoElement & undo = stack.top(); UndoElement & undo = stack.top();
LYXERR(Debug::UNDO, "Undo element of group " << undo.group_id);
// We'll pop the stack only when we're done with this element. So do NOT // We'll pop the stack only when we're done with this element. So do NOT
// try to return early. // try to return early.
@ -336,7 +347,7 @@ bool Undo::Private::textUndoOrRedo(DocIterator & cur, bool isUndoOperation)
doRecordUndo(ATOMIC_UNDO, cell_dit, doRecordUndo(ATOMIC_UNDO, cell_dit,
undo.from, cell_dit.lastpit() - undo.end, cur, undo.from, cell_dit.lastpit() - undo.end, cur,
undo.isFullBuffer, !isUndoOperation); undo.isFullBuffer, otherstack);
// This does the actual undo/redo. // This does the actual undo/redo.
//LYXERR0("undo, performing: " << undo); //LYXERR0("undo, performing: " << undo);
@ -393,9 +404,27 @@ bool Undo::Private::textUndoOrRedo(DocIterator & cur, bool isUndoOperation)
cur = undo.cursor.asDocIterator(&buffer_.inset()); cur = undo.cursor.asDocIterator(&buffer_.inset());
// Now that we're done with undo, we pop it off the stack. // Now that we're done with undo, we pop it off the stack.
stack.pop(); stack.pop();
}
bool Undo::Private::textUndoOrRedo(DocIterator & cur, bool isUndoOperation)
{
undo_finished_ = true;
UndoElementStack & stack = isUndoOperation ? undostack_ : redostack_;
if (stack.empty())
// Nothing to do.
return false;
UndoElementStack & otherstack = isUndoOperation ? redostack_ : undostack_;
const size_t gid = stack.top().group_id;
while (!stack.empty() && stack.top().group_id == gid)
doTextUndoOrRedo(cur, stack, otherstack);
// Addapt the new material to current buffer. // Addapt the new material to current buffer.
updateLabels(buffer_); updateLabels(buffer_);
undo_finished_ = true;
return true; return true;
} }
@ -419,6 +448,30 @@ bool Undo::textRedo(DocIterator & cur)
} }
void Undo::beginUndoGroup()
{
if (d->group_level == 0) {
// create a new group
++d->group_id;
LYXERR(Debug::UNDO, "+++++++Creating new group " << d->group_id);
}
++d->group_level;
}
void Undo::endUndoGroup()
{
if (d->group_level == 0)
LYXERR0("There is no undo group to end here");
--d->group_level;
if (d->group_level == 0) {
// real end of the group
LYXERR(Debug::UNDO, "-------End of group " << d->group_id);
}
}
void Undo::recordUndo(DocIterator const & cur, UndoKind kind) void Undo::recordUndo(DocIterator const & cur, UndoKind kind)
{ {
d->recordUndo(kind, cur, cur.pit(), cur.pit()); d->recordUndo(kind, cur, cur.pit(), cur.pit());
@ -429,7 +482,7 @@ void Undo::recordUndoInset(DocIterator const & cur, UndoKind kind)
{ {
DocIterator c = cur; DocIterator c = cur;
c.pop_back(); c.pop_back();
d->doRecordUndo(kind, c, c.pit(), c.pit(), cur, false, true); d->doRecordUndo(kind, c, c.pit(), c.pit(), cur, false, d->undostack_);
} }
@ -448,14 +501,18 @@ void Undo::recordUndo(DocIterator const & cur, UndoKind kind,
void Undo::recordUndoFullDocument(DocIterator const & cur) void Undo::recordUndoFullDocument(DocIterator const & cur)
{ {
// This one may happen outside of the main undo group, so we
// put it in its own subgroup to avoid complaints.
beginUndoGroup();
d->doRecordUndo( d->doRecordUndo(
ATOMIC_UNDO, ATOMIC_UNDO,
doc_iterator_begin(d->buffer_.inset()), doc_iterator_begin(d->buffer_.inset()),
0, d->buffer_.paragraphs().size() - 1, 0, d->buffer_.paragraphs().size() - 1,
cur, cur,
true, true,
true d->undostack_
); );
endUndoGroup();
} }

View File

@ -79,6 +79,12 @@ public:
/// ///
bool hasRedoStack() const; bool hasRedoStack() const;
/// open a new group of undo operations. Groups can be nested.
void beginUndoGroup();
/// end the current undo group
void endUndoGroup();
/// The general case: prepare undo for an arbitrary range. /// The general case: prepare undo for an arbitrary range.
void recordUndo(DocIterator const & cur, UndoKind kind, void recordUndo(DocIterator const & cur, UndoKind kind,
pit_type from, pit_type to); pit_type from, pit_type to);

View File

@ -148,19 +148,19 @@ int replaceAll(BufferView * bv,
if (!searchAllowed(bv, searchstr) || buf.isReadonly()) if (!searchAllowed(bv, searchstr) || buf.isReadonly())
return 0; return 0;
bv->cursor().recordUndoFullDocument();
MatchString const match(searchstr, cs, mw); MatchString const match(searchstr, cs, mw);
int num = 0; int num = 0;
int const rsize = replacestr.size(); int const rsize = replacestr.size();
int const ssize = searchstr.size(); int const ssize = searchstr.size();
DocIterator cur = doc_iterator_begin(buf.inset()); Cursor cur(*bv);
cur.setCursor(doc_iterator_begin(buf.inset()));
while (findForward(cur, match, false)) { while (findForward(cur, match, false)) {
pos_type pos = cur.pos(); pos_type pos = cur.pos();
Font const font Font const font
= cur.paragraph().getFontSettings(buf.params(), pos); = cur.paragraph().getFontSettings(buf.params(), pos);
cur.recordUndo();
int striked = ssize - cur.paragraph().eraseChars(pos, pos + ssize, int striked = ssize - cur.paragraph().eraseChars(pos, pos + ssize,
buf.params().trackChanges); buf.params().trackChanges);
cur.paragraph().insert(pos, replacestr, font, cur.paragraph().insert(pos, replacestr, font,