复制、剪切、和粘贴操作(二) <- iOS文本编程指南
复制和剪切选中内容
当用户点击编辑菜单的复制或者剪切命令,系统调用响应对象的copy:或者cut:方法。通常第一响应者(你的自定义视图)实现这些方法,但是如果第一响应者没有实现,消息会以通常的方式进入响应者链。注意,UIResponderStandardEditActions非正式协议声明了这些方法。
注意:因为UIResponderStandardEditActions是非正式协议,在app中的所有类都可以实现它的方法。但是为了充分利用响应者链的默认行为,实现这些方法类应该继承UIResponder并被安装到响应者链中去。
为了响应cut:或 paster:消息,你可以将所选择的对象或数据以尽可能多的不同的方式写入剪贴板。该操作涉及到以下步骤(假设是单一的剪贴板项目):
- 从所选内容中,识别或获取对象或该对象响应的二进制数据。
二进制对象必须封装在NSData对象中,如果你打算把对象的其他类型写入到剪贴板中,它必须是属性列表对象,也就是下面类之一的对象:NSString、NSArray、NSDictionary、NSDate、NSNumber、或者NSURL。(更多属性列表对象的内容,参见Property List Programming Guide。) - 如果有可能,生成对象或数据的一个或更多的表示法。
例如,如果在之前的步骤中创建了一个UIImage对象来表示选中的图片,你可以使用UIImageKPEGRepresentation和UIImagePNGRepresentation函数来把该图片转换成不同的表示法。 - 获取一个剪贴板对象。
在多数情况下是通用剪贴板,你可以通过generalPasteboard类方法得到。 - 分配合适的UTI给每个要写入剪贴板项目数据的表示法。
参见“剪贴板概念”中对这一部分的讨论。 - 把数据写入每个表示法类型的第一个剪贴板项目:
- 要想写入数据对象,发送setData:forPasteboardType:消息给剪贴板对象。
- 要想写入属性列表对象,发送setValue:forPasteboardType:消息给剪贴板对象。
- 如果命令是Cut(cut:方法),从app数据模型移除所选表示的对象,并更新视图。
代码清单5-1展示了copy:和cut:方法的实现。cut:方法调用copy:方法,然后从视图和数据模型中移除选中的对象。注意,copy:方法归档一个自定义对象,来获取一个NSData对象,该对象可以在setData:forPasteboardType:中传递给剪贴板。
代码清单 5-1 复制和剪切操作
- (void)copy:(id)sender {
UIPasteboard *gpBoard = [UIPasteboard generalPasteboard];
ColorTile *theTile = [self colorTileForOrigin:currentSelection];
if (theTile) {
NSData *tileData = [NSKeyedArchiver archivedDataWithRootObject:theTile];
if (tileData)
[gpBoard setData:tileData forPasteboardType:ColorTileUTI];
}
}
- (void)cut:(id)sender {
[self copy:sender];
ColorTile *theTile = [self colorTileForOrigin:currentSelection];
if (theTile) {
CGPoint tilePoint = theTile.tileOrigin;
[tiles removeObject:theTile];
CGRect tileRect = [self rectFromOrigin:tilePoint inset:TILE_INSET];
[self setNeedsDisplayInRect:tileRect];
}
}
粘贴所选内容
当用户点击菜单中的粘贴命令时,系统调用响应对象的paste:方法。通常第一响应者(你的自定义视图)实现该方法,但是如果第一响应者没有实现它,该消息会以通常的方式进入响应者链。paste:方法通过UIResponderStandardEditActions非正式协议声明。
响应paste:消息,你从应用支持的表示发中读取剪贴板对象。然后,你添加要粘贴的对象到app的数据密性中,并在用户指定的位置的视图中显示新的对象。这个操作会调用如下步骤(假设是单一的剪贴板项目):
- 获取剪贴板对象。
在大多数情况下是通用剪贴板,你可以通过generalPasteboard类方法得到。 - 验证第一个剪贴板项目包含某种表示法的数据,该表示法是app可以通过调用containsPasteboardTypes:方法或pasteboardTypes方法可以处理的,然后验证返回的类型的数组。
注意,你应该已经在实现canPerformAction:withSender:中执行了本操作。 - 如果剪贴板的第一个项目包含app可以处理的数据,调用下面方法之一来读取它:
- dataForPasteboardType: ,如果读取的数据被封装在NSData对象中。
- valueForPasteboardType:,如果读取的数据被封装在属性列表对象中(参见“复制和剪切所选内容”)。
- 添加对象到app的数据模型。
- 在用户界面指定的位置显示对象的表示法。
代码清单5-2是一个实现paste:方法的例子。它是cut:和copy:方法的反过程。自定义视图首先看通用剪贴板是否持有它的自定义的数据表示法;如果它有,它就从剪贴板中读取数据,把它添加到app的数据模型中,并为重新绘制标记自身部分(当前选中的内容)。
代码清单 5-2 粘贴数据到所选处
- (void)paste:(id)sender {
UIPasteboard *gpBoard = [UIPasteboard generalPasteboard];
NSArray *pbType = [NSArray arrayWithObject:ColorTileUTI];
ColorTile *theTile = [self colorTileForOrigin:currentSelection];
if (theTile == nil && [gpBoard containsPasteboardTypes:pbType]) {
NSData *tileData = [gpBoard dataForPasteboardType:ColorTileUTI];
ColorTile *theTile = (ColorTile *)[NSKeyedUnarchiver unarchiveObjectWithData:tileData];
if (theTile) {
theTile.tileOrigin = self.currentSelection;
[tiles addObject:theTile];
CGRect tileRect = [self rectFromOrigin:currentSelection inset:TILE_INSET];
[self setNeedsDisplayInRect:tileRect];
}
}
}
编辑操作
当你实现 cut:, copy: 或 paste: 命令返回时,编辑菜单自动隐藏。如果你愿意,你可以通过编程的方式让它们可见。更多信息,参见Dismissing the Edit Menu。a