广告
返回顶部
首页 > 资讯 > 精选 >iOS怎么使用UICollectionView实现拖拽移动单元格
  • 734
分享到

iOS怎么使用UICollectionView实现拖拽移动单元格

2023-06-30 10:06:44 734人浏览 独家记忆
摘要

这篇“iOS怎么使用UICollectionView实现拖拽移动单元格”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“iOS怎

这篇“iOS怎么使用UICollectionView实现拖拽移动单元格”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“iOS怎么使用UICollectionView实现拖拽移动单元格”文章吧。

一.介绍

iOS9提供api实现单元格排序呢功能,使用UICollectionView及其代理方法.iOS9之后有自带方法可以实现该效果,只需添加长按手势,实现手势方法和调用iOS9的API交换数据,iOS9之前需要自己写方法实现这效果,除了要添加长按手势,这里还需要利用截图替换原理,手动计算移动位置来处理视图交换和数据交换.

二.方法和步骤

创建工程项目和视图控制器,如下图

iOS怎么使用UICollectionView实现拖拽移动单元格

声明对象和设置代理和数据源代理

@interface ViewController ()<UICollectionViewDelegate,UICollectionViewDataSource,UICollectionViewDelegateFlowLayout> @property (nonatomic, strong) NSMutableArray *dataArr;@property (nonatomic, strong) UICollectionView *collectionView;@property (nonatomic, strong) NSIndexPath *oldIndexPath;@property (nonatomic, strong) UIView *snapshotView;@property (nonatomic, strong) NSIndexPath *moveIndexPath; @end

初始化UICollectionView,并添加长按手势,在viewDidLoad中初始化

CGFloat SCREEN_WIDTH = self.view.frame.size.width;    UICollectionViewFlowLayout *flowLayout = [[UICollectionViewFlowLayout alloc] init];    flowLayout.itemSize = CGSizeMake((SCREEN_WIDTH-40.0)/3, (SCREEN_WIDTH-40.0)/3);    UICollectionView *collectionView = [[UICollectionView alloc] initWithFrame:CGRectMake(0, 50.0, SCREEN_WIDTH, (SCREEN_WIDTH-40.0)/3+20.0) collectionViewLayout:flowLayout];    collectionView.dataSource = self;    collectionView.delegate = self;    collectionView.backgroundColor = [UIColor whiteColor];    [collectionView reGISterClass:[UICollectionViewCell class] forCellWithReuseIdentifier:@"uicollectionviewcell"];    [self.view addSubview:self.collectionView = collectionView];        // 添加长按手势    UILongPressGestureRecognizer *longPress = [[UILongPressGestureRecognizer alloc] initWithTarget:self action:@selector(handlelongGesture:)];    [collectionView addGestureRecognizer:longPress];

实例化数据源,(50个随机颜色,透明度0.8),在viewDidLoad中初始化

self.dataArr = [[NSMutableArray alloc] init];for (NSInteger index = 0; index < 50; index ++) {        CGFloat hue = (arc4random()%256/256.0); //0.0 到 1.0        CGFloat saturation = (arc4random()%128/256.0)+0.5; //0.5 到 1.0        CGFloat brightness = (arc4random()%128/256.0)+0.5; //0.5 到 1.0        UIColor *color = [UIColor colorWithHue:hue saturation:saturation brightness:brightness alpha:0.5];        [self.dataArr addObject:color];    }

实现UICollectionView的UICollectionViewDataSource的两个必须实现的方法

#pragma mark - UICollectionViewDataSource- (NSInteger)collectionView:(UICollectionView *)collectionView numberOfItemsInSection:(NSInteger)section{    return self.dataArr.count;} - (UICollectionViewCell *)collectionView:(UICollectionView *)collectionView cellForItemAtIndexPath:(NSIndexPath *)indexPath{    UICollectionViewCell *cell = [collectionView dequeueReusableCellWithReuseIdentifier:@"uicollectionviewcell" forIndexPath:indexPath];    cell.backgroundColor = self.dataArr[indexPath.row];    return cell;}

重点来了,实现长按手势方法

#pragma mark - 长按手势- (void)handlelongGesture:(UILongPressGestureRecognizer *)longPress{    if ([[[UIDevice currentDevice] systemVersion] floatValue] < 9.0) {        [self action:longPress];    } else {        [self iOS9_Action:longPress];    }}

iOS9之后的实现

#pragma mark - iOS9 之后的方法- (BOOL)collectionView:(UICollectionView *)collectionView canMoveItemAtIndexPath:(NSIndexPath *)indexPath{    // 返回YES允许row移动    return YES;} - (void)collectionView:(UICollectionView *)collectionView moveItemAtIndexPath:(NSIndexPath *)sourceIndexPath toIndexPath:(NSIndexPath *)destinationIndexPath{    //取出移动row数据    id color = self.dataArr[sourceIndexPath.row];    //从数据源中移除该数据    [self.dataArr removeObject:color];    //将数据插入到数据源中的目标位置    [self.dataArr insertObject:color atIndex:destinationIndexPath.row];} - (void)iOS9_Action:(UILongPressGestureRecognizer *)longPress{    switch (longPress.state) {        case UIGestureRecognizerStateBegan:        { //手势开始            //判断手势落点位置是否在row上            NSIndexPath *indexPath = [self.collectionView indexPathForItemAtPoint:[longPress locationInView:self.collectionView]];            if (indexPath == nil) {                break;            }            UICollectionViewCell *cell = [self.collectionView cellForItemAtIndexPath:indexPath];            [self.view bringSubviewToFront:cell];            //iOS9方法 移动cell            [self.collectionView beginInteractiveMovementForItemAtIndexPath:indexPath];        }            break;        case UIGestureRecognizerStateChanged:        { // 手势改变            // iOS9方法 移动过程中随时更新cell位置            [self.collectionView updateInteractiveMovementTargetPosition:[longPress locationInView:self.collectionView]];        }            break;        case UIGestureRecognizerStateEnded:        { // 手势结束            // iOS9方法 移动结束后关闭cell移动            [self.collectionView endInteractiveMovement];        }            break;        default: //手势其他状态            [self.collectionView cancelInteractiveMovement];            break;    }}

iOS9之前的实现

#pragma mark - iOS9 之前的方法- (void)action:(UILongPressGestureRecognizer *)longPress{    switch (longPress.state) {        case UIGestureRecognizerStateBegan:        { // 手势开始            //判断手势落点位置是否在row上            NSIndexPath *indexPath = [self.collectionView indexPathForItemAtPoint:[longPress locationInView:self.collectionView]];            self.oldIndexPath = indexPath;            if (indexPath == nil) {                break;            }            UICollectionViewCell *cell = [self.collectionView cellForItemAtIndexPath:indexPath];            // 使用系统的截图功能,得到cell的截图视图            UIView *snapshotView = [cell snapshotViewAfterScreenUpdates:NO];            snapshotView.frame = cell.frame;            [self.view addSubview:self.snapshotView = snapshotView];            // 截图后隐藏当前cell            cell.hidden = YES;                        CGPoint currentPoint = [longPress locationInView:self.collectionView];            [UIView animateWithDuration:0.25 animations:^{                snapshotView.transfORM = CGAffineTransformMakeScale(1.05, 1.05);                snapshotView.center = currentPoint;            }];        }            break;        case UIGestureRecognizerStateChanged:        { // 手势改变            //当前手指位置 截图视图位置随着手指移动而移动            CGPoint currentPoint = [longPress locationInView:self.collectionView];            self.snapshotView.center = currentPoint;            // 计算截图视图和哪个可见cell相交            for (UICollectionViewCell *cell in self.collectionView.visibleCells) {                // 当前隐藏的cell就不需要交换了,直接continue                if ([self.collectionView indexPathForCell:cell] == self.oldIndexPath) {                    continue;                }                // 计算中心距                CGFloat space = sqrtf(pow(self.snapshotView.center.x - cell.center.x, 2) + powf(self.snapshotView.center.y - cell.center.y, 2));                // 如果相交一半就移动                if (space <= self.snapshotView.bounds.size.width / 2) {                    self.moveIndexPath = [self.collectionView indexPathForCell:cell];                    //移动 会调用willMoveToIndexPath方法更新数据源                    [self.collectionView moveItemAtIndexPath:self.oldIndexPath toIndexPath:self.moveIndexPath];                    //设置移动后的起始indexPath                    self.oldIndexPath = self.moveIndexPath;                    break;                }            }        }            break;        default:        { // 手势结束和其他状态            UICollectionViewCell *cell = [self.collectionView cellForItemAtIndexPath:self.oldIndexPath];            // 结束动画过程中停止交互,防止出问题            self.collectionView.userInteractionEnabled = NO;            // 给截图视图一个动画移动到隐藏cell的新位置            [UIView animateWithDuration:0.25 animations:^{                self.snapshotView.center = cell.center;                self.snapshotView.transform = CGAffineTransformMakeScale(1.0, 1.0);            } completion:^(BOOL finished) {                // 移除截图视图,显示隐藏的cell并开始交互                [self.snapshotView removeFromSuperview];                cell.hidden = NO;                self.collectionView.userInteractionEnabled = YES;            }];        }            break;    }}

三.iOS9之后添加的API如下

// Support for reordering- (BOOL)beginInteractiveMovementForItemAtIndexPath:(NSIndexPath *)indexPath NS_AVaiLABLE_IOS(9_0); // returns NO if reordering was prevented from beginning - otherwise YES- (void)updateInteractiveMovementTargetPosition:(CGPoint)targetPosition NS_AVAILABLE_IOS(9_0);- (void)endInteractiveMovement NS_AVAILABLE_IOS(9_0);- (void)cancelInteractiveMovement NS_AVAILABLE_IOS(9_0);

以上就是关于“iOS怎么使用UICollectionView实现拖拽移动单元格”这篇文章的内容,相信大家都有了一定的了解,希望小编分享的内容对大家有帮助,若想了解更多相关的知识内容,请关注编程网精选频道。

--结束END--

本文标题: iOS怎么使用UICollectionView实现拖拽移动单元格

本文链接: https://www.lsjlt.com/news/328257.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • iOS使用UICollectionView实现拖拽移动单元格
    目录一.介绍二.方法和步骤三.iOS9之后添加的API本文实例为大家分享了iOS开发UICollectionView拖拽移动单元格的具体代码,供大家参考,具体内容如下 一.介绍 iO...
    99+
    2022-11-13
  • iOS怎么使用UICollectionView实现拖拽移动单元格
    这篇“iOS怎么使用UICollectionView实现拖拽移动单元格”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“iOS怎...
    99+
    2023-06-30
  • 怎么使用Vue3实现一个飘逸元素拖拽功能
    这篇文章主要介绍了怎么使用Vue3实现一个飘逸元素拖拽功能的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇怎么使用Vue3实现一个飘逸元素拖拽功能文章都会有所收获,下面我们一起来看看吧。进入正题元素拖拽是一个比较...
    99+
    2023-07-06
  • layui怎么实现鼠标移动到单元格上显示数据
    这篇文章主要介绍layui怎么实现鼠标移动到单元格上显示数据,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!如下所示:{ field : 'operNm&#...
    99+
    2022-10-19
  • 使用Python怎么操作Excel实现自动分组合并单元格
    这篇文章主要介绍了使用Python怎么操作Excel实现自动分组合并单元格,编程网小编觉得不错,现在分享给大家,也给大家做个参考,一起跟随编程网小编来看看吧!df.to_excel('test.xlsx',index=Fal...
    99+
    2023-06-06
  • 怎么使用CSS实现鼠标移动控制页面元素效果
    这篇文章主要介绍怎么使用CSS实现鼠标移动控制页面元素效果,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!映射鼠标位置或实现拖拽效果,我们可以在 JavaScript 中做到这一点。但实际上,在CSS中有更加简洁的方法...
    99+
    2023-06-14
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作