Undo/Redo几种实现方式介绍(五)——两个需要讨论的问题

上述几种实现Undo/Redo的方式都不能避免的涉及到下面问题。
1)是否需要无限undo/redo
所谓无限undo/redo是指从用户开始编辑到结束编辑过程中的所有操作都被记录在案,用户可以通过undo/redo操作将窗口状态恢复到任意时刻。
前面给出的例子都不是无限的undo/redo,当用户进行下一步的编辑后,程序会将redo缓冲区清空。目前的主流软件都是类似的操作。在word还有一些其他软件中,当键入新的内容之后,redo操作只会重复你现在的操作。原先的都被清空掉了。
之所以无限undo/redo应用比较少,我认为原因有一下几点:a)太耗费内存,加入有个10M的图片,每次对这个图片操作之后都会在undo/redo记录中留下一份记录,那么内存会被耗费的很多,但是如果在每次记录用户操作的时候都把redo缓冲区清空的话,就会节省很多内存;b)很少有用户会一直进行undo/redo操作,如果说undo是用户之前操作错误,才会进行undo操作把错误操作撤销,那么redo就意味着用户认为他自己撤销操作的操作是错误的,这种错误发生的几率比较小,所以不需要一直保存redo内容。
2)是否记录所有的改动
在信息编辑窗口中,用户的操作虽然改变了某些属性值,但是这些属性值的变化并不会在信息编辑窗口中直接体现出来,也就是说用户改变了某些属性值,但是信息编辑窗口中没有什么变化。这时就需要考虑是否对这些属性的变化进行undo操作记录。
如果不记录,则会给用户误导,用户会认为在点击undo的时候会将这些属性值恢复为原始值,但是实际上并没有恢复。如果记录这种操作,当用户点击undo按钮的时候,信息编辑窗口界面并没有发生变化,这样会使用户产生疑惑:undo按钮到底有用没有,是不是时灵时不灵。
一种折中的方法是将undo记录分为明操作和隐藏操作,如果用户的操作使信息编辑窗口发生变化,则此操作是明操作,反之为隐藏操作,当用户进行undo/redo操作的时候,程序会一直undo或者redo到下一个明操作,期间夹杂的隐藏操作都自动执行undo/redo操作。
关于undo/redo操作就是这些内容了,当然除了这几种方法肯定还有会其他方式实现类似操作,如果以后发现其它实现undo/redo的方法我会继续添加。
在 Eclipse 中,可以使用 `org.eclipse.ui.actions.ActionFactory.UNDO` 和 `org.eclipse.ui.actions.ActionFactory.REDO` 来创建 UndoRedo 动作。然后,将这些动作关联到菜单栏 Edit 中的 UndoRedo 菜单项。 要在 `ViewPart` 中实现 Undo/Redo,您需要遵循以下步骤: 1. 在 `ViewPart` 类中添加以下字段: ```java private IUndoContext undoContext; private IUndoableOperation undoOperation; private IAction undoAction; private IAction redoAction; ``` 2. 在 `createPartControl` 方法中初始化这些字段: ```java public void createPartControl(Composite parent) { // 创建 Undo/Redo 上下文 undoContext = new ObjectUndoContext(new Object()); // 创建 Undo/Redo 动作 undoAction = ActionFactory.UNDO.create(getSite().getWorkbenchWindow()); undoAction.setText("Undo"); undoAction.setImageDescriptor(PlatformUI.getWorkbench().getSharedImages() .getImageDescriptor(ISharedImages.IMG_TOOL_UNDO)); undoAction.setEnabled(false); redoAction = ActionFactory.REDO.create(getSite().getWorkbenchWindow()); redoAction.setText("Redo"); redoAction.setImageDescriptor(PlatformUI.getWorkbench().getSharedImages() .getImageDescriptor(ISharedImages.IMG_TOOL_REDO)); redoAction.setEnabled(false); // 将 Undo/Redo 动作添加到菜单栏 Edit 中 IMenuManager menuMgr = getViewSite().getActionBars().getMenuManager(); menuMgr.add(new Separator()); menuMgr.add(undoAction); menuMgr.add(redoAction); } ``` 3. 在 `createPartControl` 方法中为 Undo/Redo 动作添加处理程序: ```java public void createPartControl(Composite parent) { // ... // 为 Undo/Redo 动作添加处理程序 undoAction.setHandler(new UndoActionHandler(getViewSite(), undoContext)); redoAction.setHandler(new RedoActionHandler(getViewSite(), undoContext)); } ``` 4. 在 `createPartControl` 方法中为 `parent` 控件创建上下文菜单,并将 Undo/Redo 动作添加到上下文菜单中: ```java public void createPartControl(Composite parent) { // ... // 为 parent 控件创建上下文菜单,并将 Undo/Redo 动作添加到上下文菜单中 MenuManager menuMgr = new MenuManager(); menuMgr.add(new Separator(IWorkbenchActionConstants.MB_ADDITIONS)); getSite().registerContextMenu(menuMgr, getSite().getSelectionProvider()); parent.setMenu(menuMgr.createContextMenu(parent)); menuMgr.addMenuListener(new IMenuListener() { public void menuAboutToShow(IMenuManager manager) { manager.add(new Separator()); manager.add(undoAction); manager.add(redoAction); } }); } ``` 5. 在 `ViewPart` 类中添加以下方法: ```java public void executeCommand(final IUndoableOperation operation) { // 如果操作不可撤销,则返回 if (!operation.canUndo()) { return; } // 如果当前有未完成的撤消操作,则合并操作 if (undoOperation != null) { undoOperation.addContext(operation.getContext()); undoOperation.add(operation); } else { // 否则,开始新的撤消操作 undoOperation = operation; } // 注册撤消操作 IOperationHistory operationHistory = OperationHistoryFactory.getOperationHistory(); operationHistory.add(undoOperation); // 更新 Undo/Redo 动作的状态 undoAction.setEnabled(undoOperation.canUndo()); redoAction.setEnabled(undoOperation.canRedo()); // 将操作添加到 Undo/Redo 上下文中 undoContext.addMatch(undoOperation); } public void undo() { // 撤消上一个操作 IOperationHistory operationHistory = OperationHistoryFactory.getOperationHistory(); operationHistory.undo(undoContext, null, null); } public void redo() { // 重做上一个操作 IOperationHistory operationHistory = OperationHistoryFactory.getOperationHistory(); operationHistory.redo(undoContext, null, null); } ``` 6. 在 `ViewPart` 类中添加以下代码,以便在视图关闭时清除 Undo/Redo 上下文中的操作: ```java public void dispose() { // 清除 Undo/Redo 上下文中的操作 IOperationHistory operationHistory = OperationHistoryFactory.getOperationHistory(); operationHistory.dispose(undoContext, true, true, true); super.dispose(); } ``` 现在,您可以在 `executeCommand` 方法中执行所有需要撤消/恢复支持的操作,并且在菜单栏 Edit 中的 Undo/Redo 菜单项中使用 Undo/Redo 动作来撤消/恢复操作。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值