1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
3 * This file is part of the LibreOffice project.
5 * This Source Code Form is subject to the terms of the Mozilla Public
6 * License, v. 2.0. If a copy of the MPL was not distributed with this
7 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
9 * This file incorporates work covered by the following license notice:
11 * Licensed to the Apache Software Foundation (ASF) under one or more
12 * contributor license agreements. See the NOTICE file distributed
13 * with this work for additional information regarding copyright
14 * ownership. The ASF licenses this file to you under the Apache
15 * License, Version 2.0 (the "License"); you may not use this file
16 * except in compliance with the License. You may obtain a copy of
17 * the License at http://www.apache.org/licenses/LICENSE-2.0 .
19 #ifndef INCLUDED_SVL_UNDO_HXX
20 #define INCLUDED_SVL_UNDO_HXX
22 #include <svl/svldllapi.h>
23 #include <rtl/ustring.hxx>
24 #include <tools/datetime.hxx>
25 #include <o3tl/strong_int.hxx>
30 typedef o3tl::strong_int
<sal_Int32
, struct ViewShellIdTag
> ViewShellId
;
31 typedef o3tl::strong_int
<int, struct ViewShellDocIdTag
> ViewShellDocId
;
33 typedef struct _xmlTextWriter
* xmlTextWriterPtr
;
35 class SVL_DLLPUBLIC
SAL_LOPLUGIN_ANNOTATE("crosscast") SfxRepeatTarget
38 virtual ~SfxRepeatTarget() = 0;
42 class SVL_DLLPUBLIC SfxUndoContext
46 * Don't undo the top undo action, but an earlier one. It's the caller's responsibility to
47 * ensure that the earlier undo action is independent from the following ones.
49 virtual size_t GetUndoOffset() { return 0; }
51 virtual ~SfxUndoContext() = 0;
55 class SVL_DLLPUBLIC SfxUndoAction
59 virtual ~SfxUndoAction() COVERITY_NOEXCEPT_FALSE
;
62 virtual void UndoWithContext( SfxUndoContext
& i_context
);
64 virtual void RedoWithContext( SfxUndoContext
& i_context
);
65 virtual void Repeat(SfxRepeatTarget
&);
66 virtual bool CanRepeat(SfxRepeatTarget
&) const;
68 virtual bool Merge( SfxUndoAction
*pNextAction
);
70 virtual OUString
GetComment() const;
71 virtual OUString
GetRepeatComment(SfxRepeatTarget
&) const;
72 /// ID of the view shell that created this undo action.
73 virtual ViewShellId
GetViewShellId() const;
74 /// Timestamp when this undo item was created.
75 const DateTime
& GetDateTime() const;
76 virtual void dumpAsXml(xmlTextWriterPtr pWriter
) const;
79 SfxUndoAction( const SfxUndoAction
& ) = delete;
80 SfxUndoAction
& operator=( const SfxUndoAction
& ) = delete;
86 /// is a mark on the Undo stack
87 typedef sal_Int32 UndoStackMark
;
88 #define MARK_INVALID ::std::numeric_limits< UndoStackMark >::max()
90 struct MarkedUndoAction
92 std::unique_ptr
<SfxUndoAction
> pAction
;
93 ::std::vector
< UndoStackMark
> aMarks
;
95 MarkedUndoAction(std::unique_ptr
<SfxUndoAction
> p
) : pAction(std::move(p
)) {}
98 /** do not make use of these implementation details, unless you
99 really really have to! */
100 struct SVL_DLLPUBLIC SfxUndoArray
102 std::vector
<MarkedUndoAction
> maUndoActions
;
103 size_t nMaxUndoActions
;
104 size_t nCurUndoAction
;
105 SfxUndoArray
*pFatherUndoArray
;
107 SfxUndoArray(size_t nMax
=0) :
108 nMaxUndoActions(nMax
), nCurUndoAction(0), pFatherUndoArray(nullptr) {}
109 virtual ~SfxUndoArray();
111 SfxUndoArray
& operator=( SfxUndoArray
const & ) = delete; // MSVC2017 workaround
112 SfxUndoArray( SfxUndoArray
const & ) = delete; // MSVC2017 workaround
114 SfxUndoAction
* GetUndoAction(size_t idx
) { return maUndoActions
[idx
].pAction
.get(); }
115 std::unique_ptr
<SfxUndoAction
> Remove(int idx
);
116 void Remove( size_t i_pos
, size_t i_count
);
117 void Insert( std::unique_ptr
<SfxUndoAction
> i_action
, size_t i_pos
);
121 /** do not make use of these implementation details, unless you
122 really really have to! */
123 class SVL_DLLPUBLIC SfxListUndoAction final
: public SfxUndoAction
, public SfxUndoArray
127 UndoAction to composite multiple Undos in one UndoAction.
128 These actions are used by SfxUndomanager. With < SfxUndoManager::EnterListAction >
129 you can go one composite level down and with < SfxUndoManager::LeaveListAction > up again.
130 Redo and Undo work element wise on SfxListUndoActions.
134 std::unique_ptr
<Impl
> mpImpl
;
139 const OUString
&rComment
, const OUString
& rRepeatComment
, sal_uInt16 nId
, ViewShellId nViewShellId
, SfxUndoArray
*pFather
);
140 virtual ~SfxListUndoAction() override
;
142 virtual void Undo() override
;
143 virtual void UndoWithContext( SfxUndoContext
& i_context
) override
;
144 virtual void Redo() override
;
145 virtual void RedoWithContext( SfxUndoContext
& i_context
) override
;
146 virtual void Repeat(SfxRepeatTarget
&) override
;
147 virtual bool CanRepeat(SfxRepeatTarget
&) const override
;
149 virtual bool Merge( SfxUndoAction
*pNextAction
) override
;
151 virtual OUString
GetComment() const override
;
152 /// See SfxUndoAction::GetViewShellId().
153 ViewShellId
GetViewShellId() const override
;
154 virtual OUString
GetRepeatComment(SfxRepeatTarget
&) const override
;
155 sal_uInt16
GetId() const;
157 void SetComment(const OUString
& rComment
);
158 void dumpAsXml(xmlTextWriterPtr pWriter
) const override
;
162 /** is a callback interface for notifications about state changes of an SfxUndoManager
164 class SAL_NO_VTABLE SfxUndoListener
167 virtual void actionUndone( const OUString
& i_actionComment
) = 0;
168 virtual void actionRedone( const OUString
& i_actionComment
) = 0;
169 virtual void undoActionAdded( const OUString
& i_actionComment
) = 0;
170 virtual void cleared() = 0;
171 virtual void clearedRedo() = 0;
172 virtual void resetAll() = 0;
173 virtual void listActionEntered( const OUString
& i_comment
) = 0;
174 virtual void listActionLeft( const OUString
& i_comment
) = 0;
175 virtual void listActionCancelled() = 0;
178 ~SfxUndoListener() {}
182 namespace svl::undo::impl
184 class UndoManagerGuard
;
188 struct SfxUndoManager_Data
;
189 class SVL_DLLPUBLIC SfxUndoManager
191 std::unique_ptr
< SfxUndoManager_Data
>
194 static bool const CurrentLevel
= true;
195 static bool const TopLevel
= false;
197 SfxUndoManager( size_t nMaxUndoActionCount
= 20 );
198 virtual ~SfxUndoManager();
200 void SetMaxUndoActionCount( size_t nMaxUndoActionCount
);
201 size_t GetMaxUndoActionCount() const;
202 virtual void AddUndoAction( std::unique_ptr
<SfxUndoAction
> pAction
, bool bTryMerg
=false );
203 virtual size_t GetUndoActionCount( bool const i_currentLevel
= CurrentLevel
) const;
204 OUString
GetUndoActionComment( size_t nNo
=0, bool const i_currentLevel
= CurrentLevel
) const;
205 SfxUndoAction
* GetUndoAction( size_t nNo
=0 ) const;
206 /// Get info about all undo actions (comment, view shell id, etc.)
207 OUString
GetUndoActionsInfo() const;
208 virtual size_t GetRedoActionCount( bool const i_currentLevel
= CurrentLevel
) const;
209 OUString
GetRedoActionComment( size_t nNo
=0, bool const i_currentLevel
= CurrentLevel
) const;
210 SfxUndoAction
* GetRedoAction(size_t nNo
= 0) const;
211 /// Get info about all redo actions (comment, view shell id, etc.)
212 OUString
GetRedoActionsInfo() const;
215 /** Clears both the Redo and the Undo stack.
216 Will assert and bail out when called while within a list action (<member>IsInListAction</member>).
218 virtual void Clear();
219 /** Clears the Redo stack.
220 Will assert and bail out when called while within a list action (<member>IsInListAction</member>).
222 virtual void ClearRedo();
223 /** leaves any possible open list action (<member>IsInListAction</member>), and clears both the Undo and the
226 Effectively, calling this method is equivalent to <code>while ( IsInListAction() ) LeaveListAction();</code>,
227 followed by <code>Clear()</code>. The only difference to this calling sequence is that Reset is an
228 atomic operation, also resulting in only one notification.
231 /** determines whether an Undo or Redo is currently running
233 bool IsDoing() const;
234 size_t GetRepeatActionCount() const;
235 OUString
GetRepeatActionComment( SfxRepeatTarget
&rTarget
) const;
236 bool Repeat( SfxRepeatTarget
&rTarget
);
237 bool CanRepeat( SfxRepeatTarget
&rTarget
) const;
238 virtual void EnterListAction(const OUString
&rComment
, const OUString
& rRepeatComment
, sal_uInt16 nId
, ViewShellId nViewShellId
);
239 /** Leaves the list action entered with EnterListAction
240 @return the number of the sub actions in the list which has just been left. Note that in case no such
241 actions exist, the list action does not contribute to the Undo stack, but is silently removed.
243 size_t LeaveListAction();
245 /** Leaves the list action entered with EnterListAction, and forcefully merges the previous
246 action on the stack into the newly created list action.
248 Say you have an Undo action A on the stack, then call EnterListAction, followed by one or more calls to
249 AddUndoAction, followed by a call to LeaveAndMergeListAction. In opposite to LeaveListAction, your Undo
250 stack will now still contain one undo action: the newly created list action, whose first child is the
251 original A, whose other children are those you added via AddUndoAction, and whose comment is the same as
254 Effectively, this means that all actions added between EnterListAction and LeaveAndMergeListAction are
255 hidden from the user.
257 @return the number of the sub actions in the list which has just been left. Note that in case no such
258 actions exist, the list action does not contribute to the Undo stack, but is silently removed.
260 size_t LeaveAndMergeListAction();
261 /// determines whether we're within a ListAction context, i.e. a LeaveListAction/LeaveAndMergeListAction call is pending
262 bool IsInListAction() const;
263 /// Determines how many nested list actions are currently open
264 size_t GetListActionDepth() const;
265 /** Clears the redo stack and removes the top undo action */
266 void RemoveLastUndoAction();
267 /** enables (true) or disables (false) recording of undo actions
269 If undo actions are added while undo is disabled, they are deleted.
270 Disabling undo does not clear the current undo buffer!
272 Multiple calls to <code>EnableUndo</code> are not cumulative. That is, calling <code>EnableUndo( false )</code>
273 twice, and then calling <code>EnableUndo( true )</code> means that Undo is enable afterwards.
275 void EnableUndo( bool bEnable
);
276 /// returns true if undo is currently enabled.
277 /// This returns false if undo was disabled using EnableUndo( false ) and
278 /// also during the runtime of the Undo() and Redo() methods.
279 bool IsUndoEnabled() const;
280 /// Adds a new listener to be notified about changes in the UndoManager's state
281 void AddUndoListener( SfxUndoListener
& i_listener
);
282 void RemoveUndoListener( SfxUndoListener
& i_listener
);
283 bool IsEmptyActions() const;
286 /** marks the current top-level element of the Undo stack, and returns a unique ID for it
288 UndoStackMark
MarkTopUndoAction();
290 /** removes a mark given by its ID.
291 After the call, the mark ID is invalid.
293 void RemoveMark( UndoStackMark
const i_mark
);
295 /** determines whether the top action on the Undo stack has a given mark
297 bool HasTopUndoActionMark( UndoStackMark
const i_mark
);
299 /** removes the oldest Undo actions from the stack
301 void RemoveOldestUndoAction();
303 void dumpAsXml(xmlTextWriterPtr pWriter
) const;
306 bool UndoWithContext( SfxUndoContext
& i_context
);
307 bool RedoWithContext( SfxUndoContext
& i_context
);
309 void ImplClearRedo_NoLock( bool const i_currentLevel
);
311 /** clears all undo actions on the current level, plus all undo actions on superordinate levels,
312 as soon as those levels are reached.
314 If no list action is active currently, i.e. we're on the top level already, this method is equivalent to
317 Otherwise, the Undo actions on the current level are removed. Upon leaving the current list action, all
318 undo actions on the then-current level are removed, too. This is continued until the top level is reached.
320 void ClearAllLevels();
321 virtual void EmptyActionsChanged();
324 size_t ImplLeaveListAction( const bool i_merge
, ::svl::undo::impl::UndoManagerGuard
& i_guard
);
325 bool ImplAddUndoAction_NoNotify( std::unique_ptr
<SfxUndoAction
> pAction
, bool bTryMerge
, bool bClearRedo
, ::svl::undo::impl::UndoManagerGuard
& i_guard
);
326 void ImplClearRedo( ::svl::undo::impl::UndoManagerGuard
& i_guard
, bool const i_currentLevel
);
327 void ImplClearUndo( ::svl::undo::impl::UndoManagerGuard
& i_guard
);
328 void ImplClearCurrentLevel_NoNotify( ::svl::undo::impl::UndoManagerGuard
& i_guard
);
329 size_t ImplGetRedoActionCount_Lock( bool const i_currentLevel
= CurrentLevel
) const;
330 bool ImplIsUndoEnabled_Lock() const;
331 bool ImplIsInListAction_Lock() const;
332 void ImplEnableUndo_Lock( bool const i_enable
);
334 bool ImplUndo( SfxUndoContext
* i_contextOrNull
);
335 bool ImplRedo( SfxUndoContext
* i_contextOrNull
);
336 void ImplCheckEmptyActions();
337 inline bool ImplIsEmptyActions() const;
339 friend class ::svl::undo::impl::LockGuard
;
344 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */