iOS开发UICollectionView实现拖拽效果

一.介绍

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

二.方法和步骤

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

iOS开发UICollectionView实现拖拽效果

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

@interface ViewController ()<UICollectionViewDelegate,UICollectionViewDataSource,UICollectionViewDelegateFlowLayout>

@property (nonatomic,strong) NSMutableArray *dataArr;
@property (nonatomic,strong) UICollectionView *collectionView;
/**之前选中cell的NSIndexPath*/
@property (nonatomic,strong) NSIndexPath *oldIndexPath;
/**单元格的截图*/
@property (nonatomic,strong) UIView *snapshotView;
/**之前选中cell的NSIndexPath*/
@property (nonatomic,strong) NSIndexPath *moveIndexPath;

@end

3.初始化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];

4.实例化数据源,(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];
  }

5.实现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;
}

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

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

7.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;
  }
}

8.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);

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


当我们远离最新的 iOS 16 更新版本时,我们听到了困扰 Apple 最新软件的错误和性能问题。
欧版/美版 特别说一下,美版选错了 可能会永久丧失4G,不过只有5%的概率会遇到选择运营商界面且部分必须连接到iTunes才可以激活
一般在接外包的时候, 通常第三方需要安装你的app进行测试(这时候你的app肯定是还没传到app store之前)。
前言为了让更多的人永远记住12月13日,各大厂都在这一天将应用变灰了。那么接下来我们看一下Flutter是如何实现的。Flutter中实现整个App变为灰色在Flutter中实现整个App变为灰色是非常简单的,只需要在最外层的控件上包裹ColorFiltered,用法如下:ColorFiltered(颜色过滤器)看名字就知道是增加颜色滤镜效果的,ColorFiltered( colorFilter:ColorFilter.mode(Colors.grey, BlendMode.
flutter升级/版本切换
(1)在C++11标准时,open函数的文件路径可以传char指针也可以传string指针,而在C++98标准,open函数的文件路径只能传char指针;(2)open函数的第二个参数是打开文件的模式,从函数定义可以看出,如果调用open函数时省略mode模式参数,则默认按照可读可写(ios_base:in | ios_base::out)的方式打开;(3)打开文件时的mode的模式是从内存的角度来定义的,比如:in表示可读,就是从文件读数据往内存读写;out表示可写,就是把内存数据写到文件中;
文章目录方法一:分别将图片和文字置灰UIImage转成灰度图UIColor转成灰度颜色方法二:给App整体添加灰色滤镜参考App页面置灰,本质是将彩色图像转换为灰度图像,本文提供两种方法实现,一种是App整体置灰,一种是单个页面置灰,可结合具体的业务场景使用。方法一:分别将图片和文字置灰一般情况下,App页面的颜色深度是24bit,也就是RGB各8bit;如果算上Alpha通道的话就是32bit,RGBA(或者ARGB)各8bit。灰度图像的颜色深度是8bit,这8bit表示的颜色不是彩色,而是256
领导让调研下黑(灰)白化实现方案,自己调研了两天,根据网上资料,做下记录只是学习过程中的记录,还是写作者牛逼
让学前端不再害怕英语单词(二),通过本文,可以对css,js和es6的单词进行了在逻辑上和联想上的记忆,让初学者更快的上手前端代码
用Python送你一颗跳动的爱心
在uni-app项目中实现人脸识别,既使用uni-app中的live-pusher开启摄像头,创建直播推流。通过快照截取和压缩图片,以base64格式发往后端。
商户APP调用微信提供的SDK调用微信支付模块,商户APP会跳转到微信中完成支付,支付完后跳回到商户APP内,最后展示支付结果。CSDN前端领域优质创作者,资深前端开发工程师,专注前端开发,在CSDN总结工作中遇到的问题或者问题解决方法以及对新技术的分享,欢迎咨询交流,共同学习。),验证通过打开选择支付方式弹窗页面,选择微信支付或者支付宝支付;4.可取消支付,放弃支付会返回会员页面,页面提示支付取消;2.判断支付方式,如果是1,则是微信支付方式。1.判断是否在微信内支付,需要在微信外支付。
Mac命令行修改ipa并重新签名打包
首先在 iOS 设备中打开开发者模式。位于:设置 - 隐私&安全 - 开发者模式(需重启)
一 现象导入MBProgressHUD显示信息时,出现如下异常现象Undefined symbols for architecture x86_64: "_OBJC_CLASS_$_MBProgressHUD", referenced from: objc-class-ref in ViewController.old: symbol(s) not found for architecture x86_64clang: error: linker command failed wit
Profiles >> 加号添加 >> Distribution >> "App Store" >> 选择 2.1 创建的App ID >> 选择绑定 2.3 的发布证书(.cer)>> 输入描述文件名称 >> Generate 生成描述文件 >> Download。Certificates >> 加号添加 >> "App Store and Ad Hoc" >> “Choose File...” >> 选择上一步生成的证书请求文件 >> Continue >> Download。
今天有需求,要实现的功能大致如下:在安卓和ios端实现分享功能可以分享链接,图片,文字,视频,文件,等欢迎大佬多多来给萌新指正,欢迎大家来共同探讨。如果各位看官觉得文章有点点帮助,跪求各位给点个“一键三连”,谢啦~声明:本博文章若非特殊注明皆为原创原文链接。