oops.UndoRedo.Framework
1.0.2
dotnet add package oops.UndoRedo.Framework --version 1.0.2
NuGet\Install-Package oops.UndoRedo.Framework -Version 1.0.2
<PackageReference Include="oops.UndoRedo.Framework" Version="1.0.2" />
paket add oops.UndoRedo.Framework --version 1.0.2
#r "nuget: oops.UndoRedo.Framework, 1.0.2"
// Install oops.UndoRedo.Framework as a Cake Addin #addin nuget:?package=oops.UndoRedo.Framework&version=1.0.2 // Install oops.UndoRedo.Framework as a Cake Tool #tool nuget:?package=oops.UndoRedo.Framework&version=1.0.2
oops
First open source, cross-platform Do/Undo/Redo Framework that works in a user-driver application that's worth a dime. Available on nuget here
Premise
It is common for many user-facing applications to want some kind of undo/redo functionality for a user-driven action. This framework should be cross-platform, configurable, and loose in its structure to allow for unforeseen usage requirements.
Definition and Goals
The difficulty in creating a general purpose undo/redo framework lies in tackling (at least) two core issues:
Being able to define a range of actions from a user as undoable, or making every single thing the user does undoable
- Making every character typed undoable, for example, can result in a klunky user experience, but hey it's a start.
- The ideal framework would allow the code to aggregate changes together
- oops allows either configuration or somewhere in-between. It's all about the scope of the Accumulator you 'new' up and use.
Observing changes in all objects in the system in an ordered fashion for N unknown objects
For example, there may be 50 objects that are created and/or modified for any given user action and it would be tedious (at best) or impossible (for any real-world use-case) to funnel all changes into one undo action for the user by explicitly telling each object at the time of the change how to do it, and then how to undo it. oops manages all of this for you by observing any and all property changes in a ViewModel (derived from TrackableViewModel) and any collection changes (TrackableCollection - list, stack, queue - or TrackableDictionary), and accumulating them into one undo action either globally or locally.
Why distinguish between global and local changes? Glad you asked.
Presume you have an application with a main window/form/page of some sort. Anything that goes on in the app will observed and coalesced into undo actions through a menu-driven undo/redo system (like the back and forward arrows we're all used to).
Now, presume you have a dialog that pops up that does some undoable stuff, and also pops a different dialog that also does some undoable stuff. Depending on the project manager's whimsy, all of those actions from both dialogs should be aggregated into one undoable action for the user, or maybe there should be two. Either scenario (and really any scenario I can think of) is available through oops with the option to create singleton/global scopes or local scopes, either of which can go onto the global undo stack.
Components to support Undo/Redo for any ViewModel or Collection change
TrackableCollection - a beefed up version of ObservableCollection:
- Takes care of context switching to the UI thread for you (if it is set up to fire on the UI thread)
- Seamlessly tracks all operations and makes them undoable - this can be turned on/off at any time
- Fires CollectionChanged and PropertyChanged events like the UI needs, in the right order, on the right thread, and synchronously with the caller's thread - this can be turned off at any time
- Handles concurrency just fine (hammer it from multiple threads and it'll all get sorted out)
- Works as a Stack<> or a Queue<> as well (because sometimes you need those bound to your UI)
- Highly performant (is that still a word?), as much as the UI will allow
TrackableDictionary - super-charged version of Dictionary
- Fires CollectionChanged and PropertyChanged events like the UI needs, in the right order, on the right thread, and synchronously with the caller's thread
- Handles concurrency just fine (hammer it from multiple threads and it'll all get sorted out)
- Takes care of context switching to the UI thread for you (if it is set up to fire on the UI thread)
- Seamlessly tracks all operations and makes them undoable - this can be turned off at any time
- Highly performant, as much as the UI will allow if it is bound to the UI
ConcurrentList
- A highly performant List<> that handles concurrency just fine
- Has lots of helper methods like RemoveAndGetIndex, ReplaceAll, etc. that are inherently thread-safe
- Also works as a Stack<> or a Queue<>
Accumulator
- Records all actions for undoing them later
- Can be used a Singleton for application-scoped actions, or as an instance for locally-scoped actions (like in a dialog)
AccumulatorManager
- Manages the Undo and Redo stacks of accumulators
- Auto-records redo operations for any undo, and vice-versa
TrackableViewModel
- Auto-tracks all changes to its properties in its referenced Accumulator (either the Singleton or a local one)
- Changes are tracked by using custom Get<>/Set<> methods for any property change
TrackableScope
- Utility class to easily create an Accumulator and push it to the AccumulatorManager on Dispose
- All changes within its
using(new TrackableScope("testing"))
block are aggregated into one Undo action for the User
Demo app
- See the oops.Demo WPF application
Show me some ❤️ and star this repo to support my project
Created & Maintained By
Brent Bulla (@outbred) (Insta)
If you found this project helpful or you learned something from the source code and want to thank me, consider buying me a cup of <img src="https://vignette.wikia.nocookie.net/logopedia/images/a/ad/Dr._Pepper_1958.jpg/revision/latest?cb=20100924201743" height="25em" /> - PayPal
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net5.0 was computed. net5.0-windows was computed. net6.0 was computed. net6.0-android was computed. net6.0-ios was computed. net6.0-maccatalyst was computed. net6.0-macos was computed. net6.0-tvos was computed. net6.0-windows was computed. net7.0 was computed. net7.0-android was computed. net7.0-ios was computed. net7.0-maccatalyst was computed. net7.0-macos was computed. net7.0-tvos was computed. net7.0-windows was computed. net8.0 was computed. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. net9.0 was computed. net9.0-android was computed. net9.0-browser was computed. net9.0-ios was computed. net9.0-maccatalyst was computed. net9.0-macos was computed. net9.0-tvos was computed. net9.0-windows was computed. |
.NET Core | netcoreapp2.0 was computed. netcoreapp2.1 was computed. netcoreapp2.2 was computed. netcoreapp3.0 was computed. netcoreapp3.1 was computed. |
.NET Standard | netstandard2.0 is compatible. netstandard2.1 was computed. |
.NET Framework | net461 was computed. net462 was computed. net463 was computed. net47 was computed. net471 was computed. net472 was computed. net48 was computed. net481 was computed. |
MonoAndroid | monoandroid was computed. |
MonoMac | monomac was computed. |
MonoTouch | monotouch was computed. |
Tizen | tizen40 was computed. tizen60 was computed. |
Xamarin.iOS | xamarinios was computed. |
Xamarin.Mac | xamarinmac was computed. |
Xamarin.TVOS | xamarintvos was computed. |
Xamarin.WatchOS | xamarinwatchos was computed. |
-
.NETStandard 2.0
- No dependencies.
NuGet packages
This package is not used by any NuGet packages.
GitHub repositories
This package is not used by any popular GitHub repositories.
Added an icon