Android:基于EditText实现撤销和重做机制

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/DrkCore/article/details/53440392

【转载请注明出处】
笔者:DrkCore (http://blog.csdn.net/DrkCore)
原文链接:(http://blog.csdn.net/drkcore/article/details/53440392)



一、 场景描述和思路分析

说到撤销和重做想必大家脑海中浮现的一定是Ctrl+Z、Ctrl+Y这两个快捷键,平常生产开发的时候也少不了要和这两个按键打交道。作为一个开发者笔者自然对其中的实现方法感到好奇,想必阅读此文的你也是一样的。

如果你稍微懂点数据结构并且有着基础的封装思想的话,大体都能想到一些思路:

将用户操作抽象成一个接口,接口包含undo()和redo()两个方法,并用栈来记录操作的顺序,通过出入栈和调用两个方法来处理撤销和重做的逻辑。

涉及到撤销和重做的大部分都是需要用户编辑的功能,如果你想在Android上基于EditText开发出一个文本编辑器的话,那么按照这个思路一步步实现肯定是没有问题的。

我们发现文本的编辑操作其实可以简化为插入、删除。用户选中文本后粘贴的操作,也就是替换,可以分解为删除选中文本后插入粘贴板内容。

接下来只要记录下输入和删除的操作就可以保存用户的操作了,这里我们可以使用EditText提供了TextWatcher用于监听文本变化。

接下来请看代码实现。

二、 代码实现

首先我们需要实现编辑操作类,代码如下:

class EditOperation implements Parcelable, Serializable {

    //原始内容,通常是被删除的部分
    private String src;
    private int srcStart;
    private int srcEnd;

    //目标内容,通常是输入的部分
    private String dst;
    private int dstStart;
    private int dstEnd;

    EditOperation setSrc(CharSequence src, int srcStart, int srcEnd) {
        this.src = src != null ? src.toString() : "";
        this.srcStart = srcStart;
        this.srcEnd = srcEnd;
        return this;
    }

    EditOperation setDst(CharSequence dst, int dstStart, int dstEnd) {
        this.dst = dst != null ? dst.toString() : "";
        this.dstStart = dstStart;
        this.dstEnd = dstEnd;
        return this;
    }

    void undo(EditText text) {
        Editable editable = text.getText();

        int idx = -1;
        if (dstEnd > 0) {//删除目标内容
            editable.delete(dstStart, dstEnd);

            if (src == null) {
                idx = dstStart;
            }
        }
        if (src != null) {//插入原始内容
            editable.insert(srcStart, src);
            idx = srcStart + src.length();
        }
        if (idx >= 0) {//恢复光标位置
            text.setSelection(idx);
        }
    }

    void redo(EditText text) {
        Editable editable = text.getText();

        int idx = -1;
        if (srcEnd > 0) {//删除原始内容
            editable.delete(srcStart, srcEnd);
            if (dst == null) {
                idx = srcStart;
            }
        }
        if (dst != null) {//插入目标内容
            editable.insert(dstStart, dst);
            idx = dstStart + dst.length();
        }
        if (idx >= 0) {//恢复光标位置
            text.setSelection(idx);
        }
    }
}     

之后我们要在用户编辑文本的时候生成对应的EditOperation实例,也就是实现TextWatcher

public class OperationManager implements TextWatcher {

    private EditOperation opt;

    //启用开关,用于过滤撤销/重做时的编辑操作
    private boolean enable = true;

    OperationManager disable() {
        enable = false;
        return this;
    }

    OperationManager enable() {
        enable = true;
        return this;
    }

    @Override
    public void beforeTextChanged(CharSequence s, int start, int count, int after) {
        if (count > 0) {
            int end = start + count;
            if (enable) {
                if (opt == null) {
                    opt = new EditOperation();
                }
                //记录原始内容
                opt.setSrc(s.subSequence(start, end), start, end);
            }
        }
    }

    @Override
    public void onTextChanged(CharSequence s, int start, int before, int count) {
        if (count > 0) {
            int end = start + count;
            if (enable) {
                if (opt == null) {
                    opt = new EditOperation();
                }
                //记录目标内容
                opt.setDst(s.subSequence(start, end), start, end);
            }
        }
    }

    @Override
    public void afterTextChanged(Editable s) {
        if (enable && opt != null) {
            if (!redoOpts.isEmpty()) {//重做栈不空时用户又编辑了文本,视为抛弃重做栈
                redoOpts.clear();
            }
            //将操作入栈
            undoOpts.push(opt);
        }
        opt = null;
    }

    //使用LinkedList代替栈
    private final LinkedList<EditOperation> undoOpts = new LinkedList<>();
    private final LinkedList<EditOperation> redoOpts = new LinkedList<>();

}

之后的撤销重做就很简单了:

    public boolean undo() {
        if (canUndo()) {
            EditOperation undoOpt = undoOpts.pop();

            //屏蔽撤销产生的事件
            disable();
            undoOpt.undo(editText);
            enable();

            //填入重做栈
            redoOpts.push(undoOpt);
            return true;
        }
        return false;
    }

    public boolean redo() {
        if (canRedo()) {
            EditOperation redoOpt = redoOpts.pop();

            //屏蔽重做产生的事件
            disable();
            redoOpt.redo(editText);
            enable();

            //填入撤销
            undoOpts.push(redoOpt);
            return true;
        }
        return false;
    }

最后实现效果如图:
撤销重做

源代码请移步我的GitHub

猜你喜欢

转载自blog.csdn.net/DrkCore/article/details/53440392