Undo

在电脑上,我们一般想要撤销之前的操作的话,是通过按下快捷键 Command + Z 来实现的。而在iOS中,其实有时候我们也需要在app中添加这样的Undo 功能,实现起来也没有很复杂。类似于 UITextField 和 UITextView 这样的类,已经实现了 Undo 功能。你只需要在app中的某个区域添加这个功能按钮。

Undo Manager


Undo功能是由一个NSUndoManager实例来提供的,这个实例管理了包含所有可撤销操作的栈,还有一个包含了可重复操作的栈。当用户选择撤销衣蛾操作时,位于Undo 栈的栈顶位置的操作将会弹出,恢复到之前的状态,然后把这个弹出的操作压入Redo栈的栈顶。

为了演示这个功能,下面的例子中,用户可以拖动一个小的矩形区域。MyView是UIView的子类,添加了一个UIPanGestureRecognizer 手势识别,让用户可以拖动:

- (void) dragging: (UIPanGestureRecognizer*) p {
    if (p.state == UIGestureRecognizerStateBegan ||
            p.state == UIGestureRecognizerStateChanged) {
        CGPoint delta = [p translationInView: self.superview];
    CGPoint c = self.center;
        c.x += delta.x; c.y += delta.y;
        self.center = c;
        [p setTranslation: CGPointZero inView: self.superview];
    } 
}

为了让这个拖动可以撤销,我们需要一个NSUndoManager 实例。我们保存这个实例到MyView的undoer 属性里面。

有两种方式可以注册一个动作为可撤销的。

1、 一种是调用 NSUndoManager 的 registerUndoWithTarget:selector:object: 方法。稍后,只要NSUndoManager收到了undo 消息,它就是调用这个 target 指定的 selector 方法。

下面我们修改一下 dragging: 方法,不直接设置 self.center,而是调用一个下面的方法:

[self setCenterUndoably: [NSValue valueWithCGPoint:c]];

在这个方法里面,设置self.center属性:

- (void) setCenterUndoably: (NSValue*) newCenter {
    [self.undoer registerUndoWithTarget:self
        selector:@selector(setCenterUndoably:)
          object:[NSValue valueWithCGPoint:self.center]];
    self.center = [newCenter CGPointValue];
}

上面的做法不仅可以让我们操作可撤销,还可以让这个操作可重复。为什么,因为在NSUndoManager里面,有一条规则,就是如果当NSUndoManager正在撤销操作时,又收到registerUndoWithTarget:selector:object: 消息,那么就会把这个目标动作信息放在 Redo 栈中,而不是放在Undo 栈中。

目前为止,一切都还OK。但是我们每次在 dragging: 方法被调用时,都会添加一个单一的对象到Undo 栈中。而且这个dragging:方法是每次用户拖动都会被重复调用,这不是我们想要的。我们想要的是,每次撤销,都是撤销一个完整的拖动手势,而不是一点点的位置撤销。所以我们修改一下dragging: 方法,通过撤销组来实现:

- (void) dragging: (UIPanGestureRecognizer*) p {
    if (p.state == UIGestureRecognizerStateBegan)
        [self.undoer beginUndoGrouping];
    if (p.state == UIGestureRecognizerStateBegan ||
            p.state == UIGestureRecognizerStateChanged) {
        CGPoint delta = [p translationInView: self.superview];
        CGPoint c = self.center;
        c.x += delta.x; c.y += delta.y;
        [self setCenterUndoably: [NSValue valueWithCGPoint:c]];
        [p setTranslation: CGPointZero inView: self.superview];
    }
    if (p.state == UIGestureRecognizerStateEnded || 
            p.state == UIGestureRecognizerStateCancelled)
        [self.undoer endUndoGrouping];
}

接下来,我们让撤销操作执行动画:

- (void) setCenterUndoably: (NSValue*) newCenter {
    [self.undoer registerUndoWithTarget:self
        selector:@selector(setCenterUndoably:)
          object:[NSValue valueWithCGPoint:self.center]];
    if (self.undoer.isUndoing || self.undoer.isRedoing) { // animate
        UIViewAnimationOptions opt =
            UIViewAnimationOptionBeginFromCurrentState;
        [UIView animateWithDuration:0.4 delay:0.1 options:opt animations:^{
            self.center = [newCenter CGPointValue];
        } completion:nil];
    } else { // just do it
        self.center = [newCenter CGPointValue];
    } 
}

2、 第二种方法是,调用prepareWithInvocationTarget:

这种方式,可以传递任意类型的数据和任意数量的参数,我们把下面的代码:

[self.undoer registerUndoWithTarget:self selector:@selector(setCenterUndoably:) 
                             object:[NSValue valueWithCGPoint:self.center]];

替换成:

[[self.undoer prepareWithInvocationTarget:self]
    setCenterUndoably: [NSValue valueWithCGPoint:self.center]];

内部是通过 NSInvocation来实现稍后发送信息给指定的目标。

那么,我们就不需要把CGPoint数据封装成NSNumber对象了,我们修改:

- (void) setCenterUndoably: (CGPoint) newCenter {
    [[self.undoer prepareWithInvocationTarget:self]
        setCenterUndoably: self.center];
    if (self.undoer.isUndoing || self.undoer.isRedoing) { // animate
        UIViewAnimationOptions opt =
            UIViewAnimationOptionBeginFromCurrentState;
        [UIView animateWithDuration:0.4 delay:0.1 options:opt animations:^{
            self.center = newCenter;
        } completion:nil];
    } else { // just do it
        self.center = newCenter;
    }
}
- (void) dragging: (UIPanGestureRecognizer*) p {
    [self becomeFirstResponder];
    if (p.state == UIGestureRecognizerStateBegan)
        [self.undoer beginUndoGrouping];
    if (p.state == UIGestureRecognizerStateBegan ||
            p.state == UIGestureRecognizerStateChanged) {
        CGPoint delta = [p translationInView: self.superview];
        CGPoint c = self.center;
        c.x += delta.x; c.y += delta.y;
        [self setCenterUndoably: c];
        [p setTranslation: CGPointZero inView: self.superview];
    }
    if (p.state == UIGestureRecognizerStateEnded ||
            p.state == UIGestureRecognizerStateCancelled)
        [self.undoer endUndoGrouping];
}

Undo 界面


默认情况下,你的应用支持 shake-to-edit。意味着,当用户摇动设备时,会调出 undo/redo 界面。如果你没有明确设置UIApplication 的 applicationSupportsShakeToEdit 属性为NO,那么当用户摇动设备时,应用会沿着响应者链,第一响应者开始,查找该响应者是否继承了 undoManager 属性,返回一个实际的 NSUndoManager实例。如果找到一个,应用会调出undo/redo 界面,允许用户与NSUndoManager交互。

下面让我们视图可以成为第一响应者,然后在用户拖动结束或者取消时,让该视图成为第一响应者:

- (BOOL) canBecomeFirstResponder {
    return YES;
}
- (void) dragging: (UIPanGestureRecognizer*) p {
    // ... the rest as before ...
    if (p.state == UIGestureRecognizerStateEnded ||
            p.state == UIGestureRecognizerStateCancelled) {
        [self.undoer endUndoGrouping];
        [self becomeFirstResponder];
    } 
}

然后,让shake-to-edit 工作:

- (NSUndoManager*) undoManager {
    return self.undoer;
}

为了让弹出视图的button功能更加清晰,我们可以添加一个动作名称,用来显示在界面上:

[[self.undoer prepareWithInvocationTarget:self]
    setCenterUndoably: self.center];
[self.undoer setActionName: @"Move"];
// ... and so on ...

最终的效果如下图:

我们还可以使用长按弹出菜单:

- (void) longPress: (UIGestureRecognizer*) g {
    if (g.state == UIGestureRecognizerStateBegan) {
        UIMenuController *m = [UIMenuController sharedMenuController];
        [m setTargetRect:self.bounds inView:self];
        UIMenuItem *mi1 =
            [[UIMenuItem alloc] initWithTitle:[self.undoer undoMenuItemTitle]
                                       action:@selector(undo:)];
        UIMenuItem *mi2 =
            [[UIMenuItem alloc] initWithTitle:[self.undoer redoMenuItemTitle]
                                       action:@selector(redo:)];
        [m setMenuItems:@[mi1, mi2]];
        [m setMenuVisible:YES animated:YES];
    }
}
- (BOOL)canPerformAction:(SEL)action withSender:(id)sender {
    if (action == @selector(undo:))
        return [self.undoer canUndo];
    if (action == @selector(redo:))
        return [self.undoer canRedo];
    return [super canPerformAction:action withSender:sender];
}
- (void) undo: (id) dummy {
    [self.undoer undo];
}
- (void) redo: (id) dummy {
    [self.undoer redo];
}
最近的文章

UIViewController

通用自动视图 Generic Automatic ViewUIViewController 创建时,如果没有重写 loadView方法,那么UIViewController会默认在loadView方法里为我们创建一个通用的视图,类型就是UIView,然后帮我们赋值给self.view 属性:- (void) loadView { UIView* v = [UIView new]; self.view = v;}在 viewDidLoad 方法里,使用这个 self.view 是...…

iOS继续阅读
更早的文章

iCloud

iCloud一旦你的app通过UIDocument进行操作,那么iCloud服务会让你更满意。你只需要两步接入iCloud服务:注册使用iCloud的资格在苹果开发者网站,注册你的App ID ,勾选启用iCloud,并生成对应的provisioning profile文件,下载下来双击:然后在项目工程中,勾选使用iCloud服务:获取一个兼容iCloud的目录在你app启动时,在后台调用 NSFileManager 的 URLForUbiquityContainerIdentifier:...…

iOS继续阅读