ctrl+z的JAVA实现,借助了命令模式(command pattern)
前些天学习<<JAVA与模式>>,到命令模式时,随带给了一个CTRL+Z案例的实现,想来学习编程这么久,CTRL+Z还没有认真实现过。
因此,借助JAVA与模式里面的源代码,自己对撤销和回退进行了实现(JAVA与模式书中代码有部分问题)。此次用到了命令模式,因为有界面,有按钮,有接收者,有发送者。
以下是类图,只为方便,未考虑UML细节规范。


package undoRedo; import java.awt.Frame;
import java.awt.event.*; public class MainFrame extends Frame { private static UndoableTextArea text;
private static ItsukyuQuatation panel; public MainFrame(String title)
{
super(title);
} public static void main(String[] args)
{ //构造可撤消的textArea
text = new UndoableTextArea("Your text here");
//构造panel
panel = new ItsukyuQuatation(text);
//构造frame
MainFrame frame = new MainFrame("测试undo和redo");
//增加窗体关闭事件
frame.addWindowListener(new WindowAdapter()
{
public void windowClosing(WindowEvent e)
{
System.exit(0);
}
});
frame.add(panel);
frame.setSize(300,300);
frame.setVisible(true);
}
}
ItsukyuQuatation类:面板类,里面包括undo和redo两个按钮,类似于命令模式中的摇控器。实现了ActionListener接口。
package undoRedo; import java.awt.BorderLayout;
import java.awt.Panel;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener; public class ItsukyuQuatation extends Panel implements ActionListener { private final Command undo;
private final Command redo; public ItsukyuQuatation(UndoableTextArea text)
{
this.setLayout(new BorderLayout());
Panel toolbar = new Panel();
undo = new UndoCommand(text);
redo = new RedoCommand(text);
undo.addActionListener(this);
redo.addActionListener(this);
toolbar.add(undo);
toolbar.add(redo); this.add(toolbar,"North");
this.add(text,"Center"); } public void actionPerformed(ActionEvent e) {
// TODO Auto-generated method stub
Command cmd = (Command)e.getSource();
cmd.execute();
} }
UndoableTextArea类:类似于命令模式中的电视机类。实现了StateEditable接口,因此该类具有可撤销和重做的功能。重载的storeState和restoreState即是把文本中的内容放到hashTable里去。
该类是最核心的类,实现ctrl+z的原理简单说就是UndoManger.edits是一个Vector动态数组,对每次的按键动作,都会存储一份当前的文本内容到数据。当点击undo按钮,
数组索引(UnoManager类的indexOfNextAdd变量)前移一位,取得前一状态的文本内容。点击redo按钮,数组索引后移一位,取后一状态的文本内容。但是各状态内容均未删或修改。
initUndoable()方法,如果用keyPressed事件,那么最近的状态未加到undoManager.edits列里。因此选用keyReleased事件。如果监听valueChanged事件,那么代码就
和JAVA与模式一书中一样,只能实现最近一次的编辑撤消。
package undoRedo; import java.awt.TextArea;
import java.awt.event.*;
import java.util.Hashtable;
import javax.swing.undo.*; import javax.swing.undo.StateEditable; public class UndoableTextArea extends TextArea implements StateEditable { private UndoManager undoManager;
private final static String KEY_STATE="UndoableTextAreaKey";
private StateEdit currentEdit;
boolean textChanged = false; public UndoableTextArea()
{
super();
initUndoable();
}
public UndoableTextArea(String string)
{
super(string);
initUndoable();
} //存储状态
public void storeState(Hashtable<Object, Object> hashTable) {
// TODO Auto-generated method stub
hashTable.put(KEY_STATE, this.getText());
} //还原状态
public void restoreState(Hashtable<?, ?> hashTable) {
// TODO Auto-generated method stub
Object data = hashTable.get(KEY_STATE);
if(data!=null)
{
this.setText((String)data);
}
} //撤销方法
public boolean undo()
{
try
{
undoManager.undo();
System.out.println("undo=" + undoManager.toString());
return true;
}catch(CannotUndoException e)
{
System.out.println("Can't undo");
}
return false;
} //重做方法
public boolean redo()
{
try
{
undoManager.redo();
System.out.println("redo=" + undoManager.toString());
return true;
}catch(CannotRedoException e)
{
System.out.println("Can't redo");
}
return false;
} private void initUndoable()
{
this.undoManager = new UndoManager(); this.currentEdit = new StateEdit(this); this.addKeyListener(new KeyAdapter()
{
/*
*
* 如果用keyPressed事件,那么最近的状态未加到undoManager.edits列里。
* 因此选用keyReleased事件。
*
public void keyPressed(KeyEvent event)
{
if(event.isActionKey())
{
//takeSnapshot();
}else
{
textChanged = true;
takeSnapshot();
}
}
*/
public void keyReleased(KeyEvent event)
{
if(event.isActionKey())
{
//takeSnapshot();
}else
{
textChanged = true;
takeSnapshot();
}
}
}); this.addFocusListener(new FocusAdapter()
{
public void focusLost(FocusEvent event)
{
//takeSnapshot();
}
}); } private void takeSnapshot()
{
if(textChanged)
{
this.currentEdit.end(); this.undoManager.addEdit(this.currentEdit); this.currentEdit = new StateEdit(this); System.out.println("takeSnapshot=" + undoManager.toString()); textChanged = false;
}
} }
Command抽象类:定义按钮,抽象方法 execute()
UndoCommand,RedoCommand均是Command类的实现,类似于命令模式中的具体按钮,调用接收者的方法进行实现。
package undoRedo;
import java.awt.Button;
public abstract class Command extends Button {
public Command(String caption)
{
super(caption);
}
public abstract void execute();
}
package undoRedo;
public class UndoCommand extends Command {
UndoableTextArea text;
public UndoCommand(UndoableTextArea text)
{
super("Undo");
this.text = text;
}
@Override
public void execute() {
// TODO Auto-generated method stub
text.undo();
}
}
package undoRedo;
public class RedoCommand extends Command {
UndoableTextArea text;
public RedoCommand(UndoableTextArea text)
{
super("Redo");
this.text = text;
}
@Override
public void execute() {
// TODO Auto-generated method stub
text.redo();
}
}
ctrl+z的JAVA实现,借助了命令模式(command pattern)的更多相关文章
- 设计模式 - 命令模式(command pattern) 多命令 具体解释
命令模式(command pattern) 多命令 具体解释 本文地址: http://blog.csdn.net/caroline_wendy 參考命令模式: http://blog.csdn.ne ...
- 设计模式 - 命令模式(command pattern) 具体解释
命令模式(command pattern) 详细解释 本文地址: http://blog.csdn.net/caroline_wendy 命令模式(command pattern) : 将请求封装成对 ...
- 设计模式 - 命令模式(command pattern) 宏命令(macro command) 具体解释
命令模式(command pattern) 宏命令(macro command) 具体解释 本文地址: http://blog.csdn.net/caroline_wendy 參考: 命名模式(撤销) ...
- 乐在其中设计模式(C#) - 命令模式(Command Pattern)
原文:乐在其中设计模式(C#) - 命令模式(Command Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 命令模式(Command Pattern) 作者:webabcd ...
- 设计模式 - 命令模式(command pattern) 撤销(undo) 具体解释
命令模式(command pattern) 撤销(undo) 详细解释 本文地址: http://blog.csdn.net/caroline_wendy 參考命令模式: http://blog.cs ...
- 二十四种设计模式:命令模式(Command Pattern)
命令模式(Command Pattern) 介绍将一个请求封装为一个对象,从而使你可用不同的请求对客户进行参数化:对请求排队或记录请求日志,以及支持可取消的操作. 示例有一个Message实体类,某个 ...
- 设计模式-15命令模式(Command Pattern)
1.模式动机 在软件设计中,我们经常需要向某些对象发送请求,但是并不知道请求的接收者是谁,也不知道被请求的操作是哪个,我们只需在程序运行时指定具体的请求接收者即可,此时,可以使用命令模式来进行设计,使 ...
- 《JAVA设计模式》之命令模式(Command)
在阎宏博士的<JAVA与模式>一书中开头是这样描述命令(Command)模式的: 命令模式属于对象的行为模式.命令模式又称为行动(Action)模式或交易(Transaction)模式. ...
- 设计模式----行为型模式之命令模式(Command Pattern)
下面来自head first设计模式的命令模式一章节. 定义 将"请求"封装成对象,以便使用不同的请求.队列或者日志来参数化其他对象.命令模式也支持可撤销的操作. 类图 注: 1. ...
- 设计模式(六):控制台中的“命令模式”(Command Pattern)
今天的博客中就来系统的整理一下“命令模式”.说到命令模式,我就想起了控制台(Console)中的命令.无论是Windows操作系统(cmd.exe)还是Linux操作系统(命令行式shell(Comm ...
随机推荐
- lua在MacOS系统上的安装方法
lua是一种非常小巧的脚本语言,由标准C编写而成,可以很方便的调用c/c++或者被c/c++.另外相关的还有一个luaJIT,是lua在某些平台上的编译器. 我们在这里只安装lua. 1.检测电脑上是 ...
- Python学习_列表推导和Lambda表达式
列表推导目的是减少将一个列表转换为另一个列表时所需编写的代码量,其功能也能用列表迭代完成 1.根据要求创建列表threes_and_fives(列表值包括1到15中能够被3或者5正常的数) three ...
- java中抽象类与接口的区别
1.abstract class 在 Java 语言中表示的是一种继承关系,一个类只能使用一次继承关系.但是,一个类却可以实现多个interface. 2.在abstract class 中可以有自己 ...
- .net 反射访问私有变量和私有方法
以下为本次实践代码: using System; using System.Collections.Generic; using System.ComponentModel; using System ...
- AppStore IPv6-only审核被拒原因分析及解决方案-b
自2016年6月1日起,苹果要求所有提交App Store的iOS应用必须支持IPv6-only环境,背景也是众所周知的,IPv4地址已基本分配完毕,同时IPv6比IPv4也更加高效,向IPv6过渡是 ...
- sharepoint 2013 安装配置PowerView
安装sharepoint 2013 网络上有很多说明.这里列出两个实例: 1.说得比较详细,并提供了下载连接:http://www.sqlant.com/2012/10/sharepoint-2013 ...
- js 模块化
http://kb.cnblogs.com/page/132461/ http://www.adequatelygood.com/JavaScript-Module-Pattern-In-Depth. ...
- 【Xamarin挖墙脚系列:Xamarin开发环境配置需求】
原文:[Xamarin挖墙脚系列:Xamarin开发环境配置需求] 前言 因为操作的全是大块头,加大你们的内存,CPU网上飙.... 卤煮的机器配置 最近的版本部署包,百度云离线下载:版本:Xama ...
- 《鸟哥的Linux私房菜》读书笔记四
1.Linux的目录配置以『树状目录』来配置,至於磁碟分割槽(partition)则需要与树状目录相配合! 请问,在预设的情况下,在安装的时候系统会要求你一定要分割出来的两个Partition为何? ...
- Android 如何把一个 RelativeLayout或ImageView背景设为透明
在项目中,需要把RelativeLayout 和 ImageView背景设置为透明,怎么实现呢?这里主要通过代码,请参阅以下关键代码: public ImageView imgDetail; pri ...