標籤:style blog http color 使用 os
一個比較常見的改進使用者體驗的方案是用Redo/Undo來取代確認對話方塊,由於這個功能比較常用,本文簡單的給了一個在C#中通過Command模式實現Redo/Undo方案的例子,以供後續查詢。
class Program{ static void Main(string[] args) { var cmds = new CommandManager(); while (true) { var key = Console.ReadKey(true); if (key.KeyChar >= ‘0‘ && key.KeyChar <= ‘9‘) { cmds.DoNewCommand(key.KeyChar.ToString(), () => Console.WriteLine("process " + key.KeyChar), () => Console.WriteLine("redo " + key.KeyChar)); } else { if (key.Modifiers.HasFlag(ConsoleModifiers.Control) && (key.Key == ConsoleKey.Z)) cmds.UnDo(); else if (key.Modifiers.HasFlag(ConsoleModifiers.Control) && (key.Key == ConsoleKey.Y)) cmds.ReDo(); } } }}class CommandManager{ #region Command定義 public class Command { string name; Action action; Action unDoAction; internal Command(string name, Action action, Action unDoAction) { this.name = name; this.action = action; this.unDoAction = unDoAction; } internal void Do() { action(); } internal void UnDo() { unDoAction(); } public override string ToString() { return name.ToString(); } } #endregion public Stack<Command> ReDoActionStack { get; private set; } public Stack<Command> UnDoActionStack { get; private set; } public CommandManager() { ReDoActionStack = new Stack<Command>(); UnDoActionStack = new Stack<Command>(); } public void DoNewCommand(string name, Action action, Action unDoAction) { var cmd = new Command(name, action, unDoAction); UnDoActionStack.Push(cmd); ReDoActionStack.Clear(); cmd.Do(); } public void UnDo() { if (!CanUnDo) return; var cmd = UnDoActionStack.Pop(); ReDoActionStack.Push(cmd); cmd.UnDo(); } public void ReDo() { if (!CanReDo) return; var cmd = ReDoActionStack.Pop(); UnDoActionStack.Push(cmd); cmd.Do(); } public bool CanUnDo { get { return UnDoActionStack.Count != 0; } } public bool CanReDo { get { return ReDoActionStack.Count != 0; } } //public IEnumerable<Command> Actions { get { return ReDoActionStack.Reverse().Concat(UnDoActionStack); } }}
代碼
原理很簡單,通過Command模式把每一步操作封裝成一個可undo的命令(包含do和redo兩個操作)。並將每一步操作執行後用棧儲存起來,undo的時候就以此將Command依次出棧,並執行undo操作。(從某種意義上來說,redo就是undo操作的undo)
上面的代碼已經實現了基本的Undo/Redo功能,但實際使用的時候還是有一些細節需要考慮的:如undo或redo時失敗(拋異常)的處理等。由於這些細節方面的處理方式不盡相同,本文只是實現一個基本架構,以備後續使用時參考,並不想把它弄的過於複雜。
這種方式比較簡單,幾乎每種語言都可以輕易的寫出這種方式下的實現。但通過這種Command封裝的方式實現的也有一些限制,使用的時候需要注意:
- 每一步操作都需要封裝成command命令
- 每一步操作都是可逆的
- 當命令過多的時候需要考慮commandlist的記憶體佔用和命令查詢時的效能問題