1. 添加操作
- 添加一个按钮让tableView 进入编辑状态
- 设置cell在编辑状态下的类型为
UITableViewCellEditingStyleInsert
- 在提交操作里面(
tableView:commitEditingStyle:forRowAtIndexPath:
)刷新数据源及表视图(或单独的行)
点击cell前面的绿色小+ 会执行commitEditingStyle
方法中的操作
2. 删除操作
- 设置tableView 处于编辑状态(侧滑也可以出发删除操作)
[self.tableView setEditing:YES animated:YES];
- 设置tableView的编辑类型为
UITableViewCellEditingStyleDelete
- 在提交操作中删除数据
-(void)tableView:(UITableView *)tableView commitEditingStyle:(UITableViewCellEditingStyle)editingStyle forRowAtIndexPath:(NSIndexPath *)indexPath
{
[self deleteCellAtIndexPath:indexPath];
}
8.0 之后侧滑栏设置
- (nullable NSArray *)tableView:(UITableView *)tableView editActionsForRowAtIndexPath:(NSIndexPath *)indexPath
{
UITableViewRowAction *deleteAction = [UITableViewRowAction rowActionWithStyle:UITableViewRowActionStyleNormal title:@"删除" handler:^(UITableViewRowAction * _Nonnull action, NSIndexPath * _Nonnull indexPath) {
[self deleteCellAtIndexPath:indexPath];
}];
deleteAction.backgroundColor = [UIColor magentaColor];
NSArray *array = [NSArray arrayWithObjects:deleteAction, nil];
return array;
}
11.0 后侧滑栏设置
- (nullable UISwipeActionsConfiguration *)tableView:(UITableView *)tableView trailingSwipeActionsConfigurationForRowAtIndexPath:(NSIndexPath *)indexPath API_AVAILABLE(ios(11.0)) API_UNAVAILABLE(tvos)
{
UIContextualAction *topAction = [UIContextualAction contextualActionWithStyle:UIContextualActionStyleNormal title:@"置顶" handler:^(UIContextualAction * _Nonnull action, __kindof UIView * _Nonnull sourceView, void (^ _Nonnull completionHandler)(BOOL)) {
NSLog(@"置顶");
// 执行完回调后侧边栏是否收回
completionHandler(YES);
}];
UIContextualAction *deleteAction = [UIContextualAction contextualActionWithStyle:UIContextualActionStyleNormal title:@"删除" handler:^(UIContextualAction * _Nonnull action, __kindof UIView * _Nonnull sourceView, void (^ _Nonnull completionHandler)(BOOL)) {
[self deleteCellAtIndexPath:indexPath];
completionHandler (true);
}];
deleteAction.backgroundColor = [UIColor redColor];
UISwipeActionsConfiguration *condig = [UISwipeActionsConfiguration configurationWithActions:@[deleteAction,topAction]];
// 设置为NO,防止完整的滑动动作执行第一个action的回调
condig.performsFirstActionWithFullSwipe = NO;
return condig;
}
3. 选择操作
- 设置tableView 处于编辑状态(侧滑也可以出发删除操作)
[self.tableView setEditing:YES animated:YES];
- 设置tableView的编辑类型为
UITableViewCellEditingStyleDelete|UITableViewCellEditingStyleInsert
- 全选操作
for (int i = 0; i < self.dataArray.count; i++)
{
NSIndexPath *indexPath = [NSIndexPath indexPathForRow:i inSection:0];
[self.tableView selectRowAtIndexPath:indexPath animated:YES scrollPosition:UITableViewScrollPositionNone];
}
- 全不选
for (int i = 0; i < self.dataArray.count; i++)
{
NSIndexPath *indexPath = [NSIndexPath indexPathForRow:i inSection:0];
[self.tableView deselectRowAtIndexPath:indexPath animated:YES];
}
注: [self.tableView selectRowAtIndexPath:indexPath animated:YES scrollPosition:UITableViewScrollPositionNone]
方法并不会触发tableView的- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
代理方法。
4. 移动
- 设置tableView 处于编辑状态(侧滑也可以出发删除操作)
[self.tableView setEditing:YES animated:YES];
- 设置允许那些行移动
-(BOOL)tableView:(UITableView *)tableView canMoveRowAtIndexPath:(NSIndexPath *)indexPath
{
return YES;
}
- 交换数据
NSString *temp = _sectionZeroArray[sourceIndexPath.row];
_sectionZeroArray[sourceIndexPath.row] = _sectionZeroArray[destinationIndexPath.row];
_sectionZeroArray[destinationIndexPath.row] = temp;
- 对移动目标地址添加限制
例:设置同一个源cell和目标cell必须在同一section中才可移动
// proposedDestinationIndexPath为即将移动到的位置
- (NSIndexPath *)tableView:(UITableView *)tableView targetIndexPathForMoveFromRowAtIndexPath:(NSIndexPath *)sourceIndexPath toProposedIndexPath:(NSIndexPath *)proposedDestinationIndexPath
{
if(sourceIndexPath.section == proposedDestinationIndexPath.section)
{
return proposedDestinationIndexPath;
}
else
{
return sourceIndexPath;
}
}
5. 拖拽
- 设置tableView 的drag 和 drop delegate
self.tableView.dragDelegate = self;
self.tableView.dropDelegate = self;
- 设置可供拖拽的Item
- (nonnull NSArray *)tableView:(nonnull UITableView *)tableView itemsForBeginningDragSession:(nonnull id)session atIndexPath:(nonnull NSIndexPath *)indexPath {
NSItemProvider *itemProvider = [[NSItemProvider alloc] initWithObject:self.dataArray[indexPath.row]];
UIDragItem *item = [[UIDragItem alloc] initWithItemProvider:itemProvider];
self.dragIndexPath = indexPath;
return @[item];
}
- 返回拖拽预览参数
- (nullable UIDragPreviewParameters *)tableView:(UITableView *)tableView dragPreviewParametersForRowAtIndexPath:(NSIndexPath *)indexPath
{
UIDragPreviewParameters *parameters = [[UIDragPreviewParameters alloc] init];
CGRect rect = CGRectMake(0, 0, tableView.bounds.size.width, tableView.rowHeight);
parameters.visiblePath = [UIBezierPath bezierPathWithRoundedRect:rect cornerRadius:15];
return parameters;
}
- drop delegate
// 当用户开始初始化 drop 手势的时候会调用该方法
- (void)tableView:(UITableView *)tableView performDropWithCoordinator:(id)coordinator {
NSIndexPath *destinationIndexPath = coordinator.destinationIndexPath;
// 如果开始拖拽的 indexPath 和 要释放的目标 indexPath 一致,就不做处理
if (self.dragIndexPath.section == destinationIndexPath.section && self.dragIndexPath.row == destinationIndexPath.row) {
return;
}
[tableView performBatchUpdates:^{
// 目标 cell 换位置
id obj = self.dataArray[self.dragIndexPath.row];
[self.dataArray removeObjectAtIndex:self.dragIndexPath.row];
[self.dataArray insertObject:obj atIndex:destinationIndexPath.row];
[tableView deleteRowsAtIndexPaths:@[self.dragIndexPath] withRowAnimation:UITableViewRowAnimationFade];
[tableView insertRowsAtIndexPaths:@[destinationIndexPath] withRowAnimation:UITableViewRowAnimationFade];
} completion:^(BOOL finished) {
}];
}
// 该方法是提供释放方案的方法,虽然是optional,但是最好实现。
// 当跟踪 drop 行为在 tableView 空间坐标区域内部时会频繁调用
// 当drop手势在某个section末端的时候,传递的目标索引路径还不存在(此时 indexPath 等于 该 section 的行数),这时候会追加到该section 的末尾;
// 在某些情况下,目标索引路径可能为空(比如拖到一个没有cell的空白区域);
// 在某些情况下,你的建议可能不被系统所允许,此时系统将执行不同的建议,可以通过 -[session locationInView:] 做你自己的命中测试;
- (UITableViewDropProposal *)tableView:(UITableView *)tableView dropSessionDidUpdate:(id)session withDestinationIndexPath:(nullable NSIndexPath *)destinationIndexPath
{
UITableViewDropProposal *dropProposal;
// 如果是拖拽到另外一个app,localDragSession为nil,此时就要执行copy,通过这个属性判断是否是在当前app中释放,只有 iPad 才需要这个适配
if (session.localDragSession)
{
dropProposal = [[UITableViewDropProposal alloc] initWithDropOperation:UIDropOperationMove intent:UITableViewDropIntentInsertAtDestinationIndexPath];
}
else
{
dropProposal = [[UITableViewDropProposal alloc] initWithDropOperation:UIDropOperationCopy intent:UITableViewDropIntentInsertAtDestinationIndexPath];
}
return dropProposal;
}