iOS9之前collectionView拖拽重排的实现

最近因需要实现一个标签类的 demo, 需要支持拖拽重排, 首先想到的便是 collectionView, 并且很快就在 API 文档中找到了类似的接口:

- (BOOL)beginInteractiveMovementForItemAtIndexPath:(NSIndexPath *)indexPath NS_AVAILABLE_IOS(9_0);
- (void)updateInteractiveMovementTargetPosition:(CGPoint)targetPosition NS_AVAILABLE_IOS(9_0);
- (void)endInteractiveMovement NS_AVAILABLE_IOS(9_0);
- (void)cancelInteractiveMovement NS_AVAILABLE_IOS(9_0);

用系统提供的这几个接口很快便能实现我们的需求, 代码见 gitbub地址.
但是你有没有发现上面的方法后面跟着的NS_AVAILABLE_IOS(9_0), iOS9才开始支持, 这就尴尬了.
那iOS9之前怎么手动实现这个需求呢, 于是边分析边开始列出实现步骤:

  1. 长按 cell 进入拖拽模式
  2. 隐藏当前 cell, 并添加一个当前 cell 的截图到 collectionView 上
  3. 随手势移动截图, 当截图的center 进入到另一个 cell2 的范围内的时候, 将隐藏的 cell 移动到cell2的位置, 并移动 cell2及cell2之后的 cell
  4. 若继续移动, 则执行(3)
  5. 若松开手指结束移动, 则移除掉截图, 并显示之前隐藏的 cell

过程理顺了, 那么接下来便是代码的实现了.

1. 长按 cell 进入拖拽模式

我这里直接给 collectionView添加了一个长按手势

UILongPressGestureRecognizer *longG = [[UILongPressGestureRecognizer alloc]initWithTarget:self action:@selector(longPress:)];
longG.minimumPressDuration = 0.2f;
[self.collectionView addGestureRecognizer:longG];

- (void)longPress:(UILongPressGestureRecognizer *)lpGesture
{
    switch(lpGesture.state) {
        case UIGestureRecognizerStateBegan:
        {
            [self collectionViewCellDidBeginChange:lpGesture];
            break;
        }
        case UIGestureRecognizerStateChanged:
        {
            [self collectionViewCellDidChange:lpGesture];
            break;
        }
        case UIGestureRecognizerStateEnded:
        {
            [self collectionViewCellDidEndChange:lpGesture];
            break;
        }
        default:
            if (AVAILABLE_IOS_9_0)
            {
                [self.collectionView cancelInteractiveMovement];
            }
            break;
    }
}
2. 隐藏当前 cell, 并添加一个当前 cell 的截图到 collectionView 上
- (void)collectionViewCellDidBeginChange:(UILongPressGestureRecognizer *)lpGesture
{
    self.beginIndexPath = [self.collectionView indexPathForItemAtPoint:[lpGesture locationInView:self.collectionView]];
    self.beginCell = (CollectionViewCell *)[self.collectionView cellForItemAtIndexPath:self.beginIndexPath];
    
    if (AVAILABLE_IOS_9_0)
    {
        [self.collectionView beginInteractiveMovementForItemAtIndexPath:self.beginIndexPath];
    }
    else
    {
        // 截图  并隐藏原 cell
        self.tempView = [self.beginCell snapshotViewAfterScreenUpdates:YES];
        self.tempView.frame = self.beginCell.frame;
        [self.collectionView addSubview:self.tempView];
        
        self.beginPoint = [lpGesture locationInView:self.collectionView];
        self.beginCell.hidden = YES;
    }
}

3. 移动截图
- (void)collectionViewCellDidChange:(UILongPressGestureRecognizer *)lpGesture
{
    CGPoint targetPosion = [lpGesture locationInView:lpGesture.view];
    
    if (AVAILABLE_IOS_9_0)
    {
        [self.collectionView updateInteractiveMovementTargetPosition:targetPosion];
    }
    else
    {
        CGFloat tranX = [lpGesture locationOfTouch:0 inView:self.collectionView].x - self.beginPoint.x;
        CGFloat tranY = [lpGesture locationOfTouch:0 inView:self.collectionView].y - self.beginPoint.y;
        
        // 设置截图视图位置
        self.tempView.center = CGPointApplyAffineTransform(self.tempView.center, CGAffineTransformMakeTranslation(tranX, tranY));
        self.beginPoint = [lpGesture locationOfTouch:0 inView:_collectionView];
        
        // 计算截图视图和哪个cell相交
        for (UICollectionViewCell *cell in [_collectionView visibleCells])
        {
            //跳过隐藏的cell
            if ([_collectionView indexPathForCell:cell] == self.beginIndexPath)
            {
                continue;
            }
            //计算中心距
            CGFloat space = sqrtf(pow(self.tempView.center.x - cell.center.x, 2) + powf(self.tempView.center.y - cell.center.y, 2));
            
            //如果相交一半且两个视图Y的绝对值小于高度的一半就移动
            if (space <= self.tempView.bounds.size.width * 0.5 && (fabs(self.tempView.center.y - cell.center.y) <= self.tempView.bounds.size.height * 0.5))
            {
                NSIndexPath *nextIndexPath = [_collectionView indexPathForCell:cell];
                
                [self updateDataWithSourceIndexPath:self.beginIndexPath toIndexPath:nextIndexPath];
                
                [_collectionView moveItemAtIndexPath:self.beginIndexPath toIndexPath:nextIndexPath];
                
                //设置移动后的起始indexPath
                self.beginIndexPath = nextIndexPath;
                
                break;
            }
        }
    }
}

数据源的改动

- (void)updateDataWithSourceIndexPath:(NSIndexPath *)sourceIndexPath toIndexPath:(NSIndexPath *)destinationIndexPath
{
    if (sourceIndexPath.section == destinationIndexPath.section) // 同一个 section 间移动
    {
        if (destinationIndexPath.item > sourceIndexPath.item) {
            for (NSUInteger i = sourceIndexPath.item; i < destinationIndexPath.item ; i ++) {
                [self.dataArray[sourceIndexPath.section] exchangeObjectAtIndex:i withObjectAtIndex:i + 1];
            }
        }else{
            for (NSUInteger i = sourceIndexPath.item; i > destinationIndexPath.item ; i --) {
                [self.dataArray[sourceIndexPath.section] exchangeObjectAtIndex:i withObjectAtIndex:i - 1];
            }
        }
    }
    else // 不同 section 之间移动 ,  删除 sourceData, 插入到 destinationData
    {
        NSMutableArray *sourceArr = self.dataArray[sourceIndexPath.section];
        NSMutableArray *destinationArr = self.dataArray[destinationIndexPath.section];
        
        id obj = [sourceArr objectAtIndex:sourceIndexPath.row];
        
        [sourceArr removeObjectAtIndex:sourceIndexPath.row];
        
        [destinationArr insertObject:obj atIndex:destinationIndexPath.row];
    }
    
}
4. 若继续移动, 则执行(3)
5. 若松开手指结束移动, 则移除掉截图, 并显示之前隐藏的 cell
- (void)collectionViewCellDidEndChange:(UILongPressGestureRecognizer *)lpGesture
{
    if (AVAILABLE_IOS_9_0)
    {
        [self.collectionView endInteractiveMovement];
    }
    else
    {
        [UIView animateWithDuration:0.25 animations:^{
            
            self.tempView.center = self.beginCell.center;
            
        }completion:^(BOOL finished) {
            
            [self.tempView removeFromSuperview];
            self.beginCell.hidden = NO;
        }];
    }
}

好了, 到此就完成了 iOS9以下的适配, 实现起来也不是很复杂, 主要的是实现之前把思路理顺, 那实现起来就不难了.

详细的代码见gitbub地址.如果有收获,欢迎 star!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,547评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,399评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,428评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,599评论 1 274
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,612评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,577评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,941评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,603评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,852评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,605评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,693评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,375评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,955评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,936评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,172评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,970评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,414评论 2 342

推荐阅读更多精彩内容