iOS 列表拖拽cell排序

先看效果:

拖拽cell排序

一、 UITableView有自带的cell拖拽排序功能,但是有个缺点,拖拽平移的手势只能在cell的最右侧,拖拽按钮上的图标也不方便更换,在此不在细说;

二、使用UICollectionView实现cell拖拽排序

视频中的效果是使用UICollectionView实现的;

UICollectionView部分的代码不再写了,就是常规的实现;以下只把拖拽手势实现的逻辑贴出来,仅供参考:

1、在listView: UICollectionView!上添加PanGesture手势:

复制代码
        self.listView.register(UINib(nibName: "RecordListCVCell", bundle: nil), forCellWithReuseIdentifier: "RecordListCVCell")
        self.listView.contentInset = UIEdgeInsets(top: 0, left: 0, bottom: 0, right: 0)
        let layout = UICollectionViewFlowLayout()
        layout.scrollDirection = .vertical
        layout.itemSize = CGSize(width: ScreenWidth - 32, height: 55)
        self.listView.collectionViewLayout = layout
        self.listView.reloadData()
        
        panGesture = UIPanGestureRecognizer(target: self, action: #selector(handlePanGesture(_:)))
        panGesture!.isEnabled = true
        panGesture!.delegate = self
        self.listView.addGestureRecognizer(panGesture!)片

2、在UIGestureRecognizerDelegate代理方法中限制手势的作用范围:

这一步主要是防止 panGesture 与 listView滚动冲突

复制代码
//限制拖拽手势的作用范围:仅左侧60pt宽度区域有效
    func gestureRecognizerShouldBegin(_ gestureRecognizer: UIGestureRecognizer) -> Bool {
        guard gestureRecognizer == panGesture,
              let pan = gestureRecognizer as? UIPanGestureRecognizer else {
            return true
        }
        // 1. 获取触摸点坐标
        let touchPoint = pan.location(in: listView)
        // 2. 区域限制:仅左侧60pt宽度区域有效
        guard touchPoint.x < 60 else {
            return false // 超出区域不响应
        }
        return true
    }

3、cell的平移操作:

复制代码
//平移
    @objc func handlePanGesture(_ panPress: UIPanGestureRecognizer) {
        switch panPress.state {
        case .began:
            // 手势作用的位置
            if let selectIndexPath = listView.indexPathForItem(at: panPress.location(in: listView)) {
                // 找到当前的 cell
                if let _ = listView.cellForItem(at: selectIndexPath) as? RecordListCVCell {
                    // 开始移动
                    listView.beginInteractiveMovementForItem(at: selectIndexPath)
                }
            }
            break
        case .changed:
            // 更新移动的位置:只移动Y,移动时cell的中心点X会跑到0点位置,所以要 x = (ScreenWidth - 32) / 2.0  纠正过来
            listView.updateInteractiveMovementTargetPosition(CGPoint(x: (ScreenWidth - 32) / 2.0, y: panPress.location(in: panPress.view).y))
            break
        case .ended:
            // 结束移动
            listView.endInteractiveMovement()
            break
        default:
            listView.cancelInteractiveMovement()
            break
        }
    }

4、在UICollectionViewDataSource代理中实现数据源的重新排序:

复制代码
func collectionView(_ collectionView: UICollectionView, moveItemAt sourceIndexPath: IndexPath, to destinationIndexPath: IndexPath){
        //拖拽排序:更新数据源顺序
        let element = dataList.remove(at: sourceIndexPath.row)
        dataList.insert(element, at: destinationIndexPath.row)
        self.listView.reloadData()
    }
相关推荐
wjm0410061 天前
ios内存管理
ios·objective-c·swift·客户端开发
黑科技iOS上架1 天前
ios应用被封号后再次上架很难么?
经验分享·ios
柚鸥ASO优化1 天前
一篇讲透安卓ASO!开发者千万别只盯着iOS了
android·ios·aso优化
黑科技iOS上架1 天前
Swift Package Manager包管理工具的优缺点
经验分享·ios
大熊猫侯佩1 天前
Swift 6.4 的 Ref / MutableRef 大揭秘:给值类型开一扇“安全的小窗”
ios·swift·编程语言
黑科技iOS上架1 天前
没有mac电脑如何借助windows系统上传ipa到App Store
经验分享·ios
Layer1 天前
从 WWDC 26 空间重构(Spatial Reframing)再看端侧 2D 转 3D 的技术演进
ios·aigc
Cutecat_2 天前
视频字幕处理工具横向:提取模式 vs 编辑模式,该如何选择
android·前端·ios·语音识别
大熊猫侯佩2 天前
WWDC26 SwiftUI 进化之路:砸碎黑盒,彻底迎来开发自由!
ios·swiftui·swift
游戏开发爱好者82 天前
iPhone真机调试有哪些方法?一次定位推送权限问题时整理出来的几种方案
ide·vscode·ios·objective-c·个人开发·swift·敏捷流程