使用过Gmail或者163邮箱的同学会经常看到,当对邮件进行一些操作时会出现一个类似Toast的提示(大致意思是:操作已经完成,是否撤销)如下图所示:
当点击撤销时,之前执行的操作能够被还原,这种设计对于用户的误操作是一个非常棒的补救方案。
之前听过一句有关交互设计的话说的非常好,不要在用户每做一步操作时弹出Alert让用户选择”确定”或者”取消”,更好的做法是执行操作,然后让用户能够Undo。
其实Undo是很老的技术,在编辑器中无处不在,只是在API设计中使用的还比较少。最近一直在使用Node开发,因此想使用Node实现一下API的Undo。
首先要明确几点:
通过上边的方案对比,我发现Plan A更简单、灵活,因此决定实现Plan A。
使用过Koa的通许都知道,koa的中间件非常强大(类似Java web开发中的拦截器),它能够拦截所有请求并执行一些逻辑,例如计算API请求到响应时长等,这里我们就可以使用这个特性将需要Undo的API延迟执行。
首先我们要设置Undo的超时时间,以及那些API需要Undo:
var apis = (options || {}).apis;
var expired = (options || {}).expired || 3000;
还要明确当前访问API的用户:
/**
* `x-identify-key` is used to identify the user of this request,
* one user can not undo another`s request.
*/
var user = context.header['x-identify-key'];
然后延迟执行API逻辑:
var undo = yield delayNext(user, expired, context);
如果用户没有调用Undo接口,则执行逻辑,否则返回’undo’:
if (!undo) { return yield next; }
this.body = 'undo';
如果用户调用Undo接口,移除延迟执行的逻辑;调用其他接口则立即执行延迟的逻辑:
clearTimeout(undoObj.timeoutId);
if (path === '/undo' && method === 'POST') {
undoObj.delayFn.call(undoObj.context, true);
context.body = 'done';
return;
} else if (undoObj.delayFn) {
undoObj.delayFn.call(undoObj.context, false);
}
具体实现逻辑大概就这些,完整代码如下:
/**
* Store users' undo context
*
* @type {Object}
*/
var undos = {};
/**
* Expose `undo`
*
* @param {Object} options Config object for undo
* @example
* {
* expired: 3000
* }
*/
module.exports = function (options) {
var apis = (options || {}).apis;
var expired = (options || {}).expired || 3000;
return function* (next) {
var context = this;
var path = context.path;
var needUndo = false;
if (apis && Array.isArray(apis) && apis.length) {
needUndo = apis.filter(function (api) {
return path === api;
}).length;
}
if (!needUndo && path !== '/undo') { return yield next; }
var method = context.method;
/**
* Can not undo get request.
*/
if (method === 'GET') { return yield next; }
/**
* 'x-identify-key' is used to identify the user of this request,
* one user can not undo another's request.
*/
var user = context.header['x-identify-key'];
if (!user) { return yield next; }
var undoObj = undos[user];
if (undoObj) {
clearTimeout(undoObj.timeoutId);
if (path === '/undo' && method === 'POST') {
undoObj.delayFn.call(undoObj.context, true);
context.body = 'done';
return;
} else if (undoObj.delayFn) {
undoObj.delayFn.call(undoObj.context, false);
}
}
var undo = yield delayNext(user, expired, context);
if (!undo) { return yield next; }
this.body = 'undo';
};
};
/**
* Block the logic for specified ms.
*
* @param {String} user The user's identity
* @param {String} expired The expired ms
* @param {Object} context The koa context object
* @api private
*/
function delayNext(user, expired, context) {
return function (callback) {
var delayFn = function (undo) {
delete undos[user];
callback(null, undo);
};
var timeoutId = setTimeout(delayFn, expired);
undos[user] = {
timeoutId: timeoutId,
delayFn: delayFn,
context: context
};
};
}
目前此项目托管在Github上,https://github.com/sweetvvck/koa-undo,koa-undo具体使用方法项目主页有详细介绍,感兴趣的同学欢迎提Issue、PR;同时koa-undo也发布到了Npm上,https://www.npmjs.com/package/koa-undo ,欢迎大家使用。