我的编程空间,编程开发者的网络收藏夹
学习永远不晚

iOS开发UICollectionView实现拖拽效果

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

iOS开发UICollectionView实现拖拽效果

一.介绍

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

二.方法和步骤

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

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


@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);


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

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

iOS开发UICollectionView实现拖拽效果

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

下载Word文档

猜你喜欢

iOS开发UICollectionView实现拖拽效果

一.介绍 iOS9提供API实现单元格排序功能,使用UICollectionView及其代理方法。iOS9之后有自带方法可以实现该效果,只需添加长按手势,实现手势方法和调用iOS9的API交换数据,iOS9之前需要自己写方法实现这效果,除了
2022-05-23

iOS UICollectionView实现卡片效果

现在使用卡片效果的app很多,之前公司让实现一种卡片效果,就写了一篇关于实现卡片的文章。文章最后附有demo 实现上我选择了使用UICollectionView ;用UICollectionViewFlowLayout来定制样式;下面看看具
2022-05-28

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

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

iOS实现拖拽View跟随手指浮动效果

本文实例为大家分享了iOS实现拖拽View跟随手指浮动的具体代码,供大家参考,具体内容如下 效果图:1.自定义要跟随手指浮动的那个View// // OrangeView.m // 拖拽View跟随手指浮动 // // Created by
2022-05-31

拖拽插件sortable.js如何实现el-table表格拖拽效果

这篇文章将为大家详细讲解有关拖拽插件sortable.js如何实现el-table表格拖拽效果,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。问题描述Sortable.js是一款优秀的js拖拽库,因为是原生
2023-06-29

iOS开发实现抽屉效果

这篇文章主要为大家详细介绍了iOS开发实现抽屉效果,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
2022-11-13

编程热搜

  • Android:VolumeShaper
    VolumeShaper(支持版本改一下,minsdkversion:26,android8.0(api26)进一步学习对声音的编辑,可以让音频的声音有变化的播放 VolumeShaper.Configuration的三个参数 durati
    Android:VolumeShaper
  • Android崩溃异常捕获方法
    开发中最让人头疼的是应用突然爆炸,然后跳回到桌面。而且我们常常不知道这种状况会何时出现,在应用调试阶段还好,还可以通过调试工具的日志查看错误出现在哪里。但平时使用的时候给你闹崩溃,那你就欲哭无泪了。 那么今天主要讲一下如何去捕捉系统出现的U
    Android崩溃异常捕获方法
  • android开发教程之获取power_profile.xml文件的方法(android运行时能耗值)
    系统的设置–>电池–>使用情况中,统计的能耗的使用情况也是以power_profile.xml的value作为基础参数的1、我的手机中power_profile.xml的内容: HTC t328w代码如下:
    android开发教程之获取power_profile.xml文件的方法(android运行时能耗值)
  • Android SQLite数据库基本操作方法
    程序的最主要的功能在于对数据进行操作,通过对数据进行操作来实现某个功能。而数据库就是很重要的一个方面的,Android中内置了小巧轻便,功能却很强的一个数据库–SQLite数据库。那么就来看一下在Android程序中怎么去操作SQLite数
    Android SQLite数据库基本操作方法
  • ubuntu21.04怎么创建桌面快捷图标?ubuntu软件放到桌面的技巧
    工作的时候为了方便直接打开编辑文件,一些常用的软件或者文件我们会放在桌面,但是在ubuntu20.04下直接直接拖拽文件到桌面根本没有效果,在进入桌面后发现软件列表中的软件只能收藏到面板,无法复制到桌面使用,不知道为什么会这样,似乎并不是很
    ubuntu21.04怎么创建桌面快捷图标?ubuntu软件放到桌面的技巧
  • android获取当前手机号示例程序
    代码如下: public String getLocalNumber() { TelephonyManager tManager =
    android获取当前手机号示例程序
  • Android音视频开发(三)TextureView
    简介 TextureView与SurfaceView类似,可用于显示视频或OpenGL场景。 与SurfaceView的区别 SurfaceView不能使用变换和缩放等操作,不能叠加(Overlay)两个SurfaceView。 Textu
    Android音视频开发(三)TextureView
  • android获取屏幕高度和宽度的实现方法
    本文实例讲述了android获取屏幕高度和宽度的实现方法。分享给大家供大家参考。具体分析如下: 我们需要获取Android手机或Pad的屏幕的物理尺寸,以便于界面的设计或是其他功能的实现。下面就介绍讲一讲如何获取屏幕的物理尺寸 下面的代码即
    android获取屏幕高度和宽度的实现方法
  • Android自定义popupwindow实例代码
    先来看看效果图:一、布局
  • Android第一次实验
    一、实验原理 1.1实验目标 编程实现用户名与密码的存储与调用。 1.2实验要求 设计用户登录界面、登录成功界面、用户注册界面,用户注册时,将其用户名、密码保存到SharedPreference中,登录时输入用户名、密码,读取SharedP
    Android第一次实验

目录