纯javascript对撤销和重写(undo、redo)的完美实现,适用于任何页面元素操作

摘要:
最近项目组开发一个报表设计器,需要用到撤销和重写的功能,这样用户就能方便的看到历史操作。内存,这是目前采取的实现,与后台无任何关联!!varstack=newUndo.Stack();每次操作完后,将命令入栈。撤销回退还有一个注意点是:每次新的操作来了之后,redo(重写)都必须清空,这是目前所有软件实现undo、redo一致的行为。

最近项目组开发一个报表设计器,需要用到撤销和重写的功能,这样用户就能方便的看到历史操作。
不知道大家看过java的命令模式没有,命令模式在英文里也叫undo,在javascript设计模式这本书里里就是这样子说的,虽然有好几个英文名称。
具体思路是每个对应页面的操作,譬如对表格的操作,在js里都是一个命令对象,我们暂且叫Undo.Command,Undo.Command里都有undo和redo的自定义实现,并且每个
CommandObj里都存储了操作的对象的属性,以方便在undo和redo里对其操作。

UpCommand = Undo.Command.extend({
constructor: function(li) {
this.li = li;//存储操作对象属性
},
execute: function() {
},

//撤销
undo: function() {
this.li.insertAfter(this.li.next());
},

//重写

redo:function(){

this.li.insertBefore(this.li.prev());

}
})

这些命令保存在哪里?

内存,这是目前采取的实现,与后台无任何关联!!

必须定义一个全局命令堆栈。var stack = new Undo.Stack();

每次操作完后,将命令入栈。

但是如果对堆栈数量不做限制,大家可以想象到时候浏览器会是什么情况,所以必须对堆栈数量限制,先进先出。

撤销回退还有一个注意点是 : 每次新的操作来了之后(不包括对undo、redo按钮的操作),redo(重写)都必须清空,这是目前所有软件实现undo、redo一致的行为。

还有一个注意点是,命令的对象越小越好,这样内存中的对象属性就比较少,但是实现undo、redo的细节就会繁琐,我们现在的报表设计器不会只是回退一个表的细微操作,这样的代码复杂度是很大的。类似表格的很多对象都是保存在工作区的page上的,所以当前堆栈中保存的是page。刷新page即可。

(function() {

var ctor = function(){};

var inherits = function(parent, protoProps) {
var child;

if (protoProps && protoProps.hasOwnProperty('constructor')) {
child = protoProps.constructor;
} else {
child = function(){ return parent.apply(this, arguments); };
}

ctor.prototype = parent.prototype;
child.prototype = new ctor();
if (protoProps) extend(child.prototype, protoProps);
child.prototype.constructor = child;
child.__super__ = parent.prototype;
return child;
};

function extend(target, ref) {
var name, value;
for ( name in ref ) {
value = ref[name];
if (value !== undefined) {
target[ name ] = value;
}
}
return target;
};

var Undo;
if (typeof exports !== 'undefined') {
Undo = exports;
} else {
Undo = this.Undo = {};
}

Undo.Stack = function() {
this.commands = [];
this.stackPosition = -1;
this.savePosition = -1;
};

extend(Undo.Stack.prototype, {
execute: function(command) {

this._clearRedo();
command.execute();

//必须对堆栈数量进行限制,自己去实现吧
this.commands.push(command);
this.stackPosition++;
this.changed();
},
undo: function() {
this.commands[this.stackPosition].undo();
this.stackPosition--;
this.changed();
},
canUndo: function() {
return this.stackPosition >= 0;
},
redo: function() {
this.stackPosition++;
this.commands[this.stackPosition].redo();
this.changed();
},
canRedo: function() {
return this.stackPosition < this.commands.length - 1;
},
save: function() {
this.savePosition = this.stackPosition;
this.changed();
},
dirty: function() {
return this.stackPosition != this.savePosition;
},
_clearRedo: function() {
this.commands = this.commands.slice(0, this.stackPosition + 1);
},
changed: function() {
}
});

Undo.Command = function(name) {
this.name = name;
}

var up = new Error("override me!");

extend(Undo.Command.prototype, {
execute: function() {
throw up;
},
undo: function() {
throw up;
},
redo: function() {
this.execute();
}
});

Undo.Command.extend = function(protoProps) {
var child = inherits(this, protoProps);
child.extend = Undo.Command.extend;
return child;
};
}).call(this);

使用方式:

var stack = new Undo.Stack(),

UpCommand = Undo.Command.extend({
constructor: function(li) {
this.li = li;
},
execute: function() {
this.li.insertBefore(this.li.prev());
},
undo: function() {
this.li.insertAfter(this.li.next());
}
}),
DownCommand = UpCommand.extend({
execute: UpCommand.prototype.undo,
undo: UpCommand.prototype.execute,
});

操作完后命令入堆栈:

stack.execute(new UpCommand($(this).parent()));

其实有了思想,大家剩下的自己实现吧,明白人还是明白人

技术交流群:55919698

免责声明:文章转载自《纯javascript对撤销和重写(undo、redo)的完美实现,适用于任何页面元素操作》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇使用TPU在PyTorch中实现ResNet50「杂谈」苏州吴中区买房有哪些选项?下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

随便看看

WinRAR 激活的小办法

WinRAR是一个强大的压缩文件管理工具。它可以备份数据,减少电子邮件附件的大小,解压缩从Internet下载的RAR、ZIP和其他压缩文件,并以RAR和ZIP格式创建压缩文件。如果您使用的正版WinRAR未激活,请将以下注册代码复制到新文档并将其重命名为rarreg。键,然后复制rarreg。键到WinRAR根目录以激活它。...

字符串解压缩类库(zip、GZIP、QuickLz、snappy、lzf、jzlib)介绍

它旨在提供高压缩速度和合理的压缩比=-1){out.write;}字节[]未压缩=输出。到字节数组();--返回提取字符串的字节数组。介绍使用预先选择的解压缩类库-GZIP压缩字符串=“这是一个用于测试的字符串”;ByteArrayOutputStreamout=新的ByteArray输出流();GZipOutputStreamgout=newGZipOut...

如何在linux下安装idea

[通过正式安装包安装]http://www.jetbrains.com/在官方网站上下载相应版本。终极旗舰社区版本,将其解压缩到本地对应目录,然后执行/idea.sh命令。安装后,可以在启动程序中找到创意图标。...

面试了一个 31岁的iOS开发者,思绪万千,30岁以上的程序员还有哪些出路?

前言之前HR给了我一份简历,刚看到简历的第一眼,31岁?31岁,iOS开发工程师,工作经历7年,5年左右都在外包公司,2年左右在创业公司。iOS开发工程师这块,还是很少遇到30岁以上的开发,正好,来了一个30岁的开发,说实话,对我来说,还是蛮期待的,希望对我有所启示。这样的过程持续了半个小时那么年过350岁的程序员还有出路吗?作为一个8年的iOS开发,而且几...

MySQL 字段类型占用空间

MySQL支持多种列类型:数值类型、日期/时间类型和字符串(字符)类型。)1或2个字节,取决于枚举值的个数SET(‘value1’,’value2’,…)1、2、3、4或者8个字节,取决于set成员的数目上表的M只是为了说明占用空间大小,在实际创建表中char、varchar,20指的是字符而不是字节;那么字符和字节的转换要看字符集,utf-8下,1字符=3...

如何设置Navicat的显示字体与字体大小?

方法/步骤打开Navicat点击菜单,再选择在界面,点击下的设置网格字体和大小设置编辑器字体和大小设置命令列界面字体和大小设置ER图表字体和大小,最后点击END...