mirror of
https://git.lyx.org/repos/lyx.git
synced 2025-01-11 19:14:51 +00:00
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:
parent
608c46b080
commit
17dd645862
@ -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);
|
||||||
|
@ -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;
|
||||||
|
|
||||||
|
@ -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();
|
||||||
|
113
src/Undo.cpp
113
src/Undo.cpp
@ -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();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -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);
|
||||||
|
@ -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,
|
||||||
|
Loading…
Reference in New Issue
Block a user