iOS使用UICollectionView實(shí)現(xiàn)拖拽移動(dòng)單元格
本文實(shí)例為大家分享了iOS開發(fā)UICollectionView拖拽移動(dòng)單元格的具體代碼,供大家參考,具體內(nèi)容如下
一.介紹
iOS9提供API實(shí)現(xiàn)單元格排序呢功能,使用UICollectionView及其代理方法.iOS9之后有自帶方法可以實(shí)現(xiàn)該效果,只需添加長(zhǎng)按手勢(shì),實(shí)現(xiàn)手勢(shì)方法和調(diào)用iOS9的API交換數(shù)據(jù),iOS9之前需要自己寫方法實(shí)現(xiàn)這效果,除了要添加長(zhǎng)按手勢(shì),這里還需要利用截圖替換原理,手動(dòng)計(jì)算移動(dòng)位置來(lái)處理視圖交換和數(shù)據(jù)交換.
二.方法和步驟
1.創(chuàng)建工程項(xiàng)目和視圖控制器,如下圖
2.聲明對(duì)象和設(shè)置代理和數(shù)據(jù)源代理
@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,并添加長(zhǎng)按手勢(shì),在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]; ? ?? ? ? // 添加長(zhǎng)按手勢(shì) ? ? UILongPressGestureRecognizer *longPress = [[UILongPressGestureRecognizer alloc] initWithTarget:self action:@selector(handlelongGesture:)]; ? ? [collectionView addGestureRecognizer:longPress];
4.實(shí)例化數(shù)據(jù)源,(50個(gè)隨機(jī)顏色,透明度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.實(shí)現(xiàn)UICollectionView的UICollectionViewDataSource的兩個(gè)必須實(shí)現(xiàn)的方法
#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.重點(diǎn)來(lái)了,實(shí)現(xiàn)長(zhǎng)按手勢(shì)方法
#pragma mark - 長(zhǎng)按手勢(shì) - (void)handlelongGesture:(UILongPressGestureRecognizer *)longPress { ? ? if ([[[UIDevice currentDevice] systemVersion] floatValue] < 9.0) { ? ? ? ? [self action:longPress]; ? ? } else { ? ? ? ? [self iOS9_Action:longPress]; ? ? } }
7.iOS9之后的實(shí)現(xiàn)
#pragma mark - iOS9 之后的方法 - (BOOL)collectionView:(UICollectionView *)collectionView canMoveItemAtIndexPath:(NSIndexPath *)indexPath { ? ? // 返回YES允許row移動(dòng) ? ? return YES; } ? - (void)collectionView:(UICollectionView *)collectionView moveItemAtIndexPath:(NSIndexPath *)sourceIndexPath toIndexPath:(NSIndexPath *)destinationIndexPath { ? ? //取出移動(dòng)row數(shù)據(jù) ? ? id color = self.dataArr[sourceIndexPath.row]; ? ? //從數(shù)據(jù)源中移除該數(shù)據(jù) ? ? [self.dataArr removeObject:color]; ? ? //將數(shù)據(jù)插入到數(shù)據(jù)源中的目標(biāo)位置 ? ? [self.dataArr insertObject:color atIndex:destinationIndexPath.row]; } ? - (void)iOS9_Action:(UILongPressGestureRecognizer *)longPress { ? ? switch (longPress.state) { ? ? ? ? case UIGestureRecognizerStateBegan: ? ? ? ? { //手勢(shì)開始 ? ? ? ? ? ? //判斷手勢(shì)落點(diǎn)位置是否在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方法 移動(dòng)cell ? ? ? ? ? ? [self.collectionView beginInteractiveMovementForItemAtIndexPath:indexPath]; ? ? ? ? } ? ? ? ? ? ? break; ? ? ? ? case UIGestureRecognizerStateChanged: ? ? ? ? { // 手勢(shì)改變 ? ? ? ? ? ? // iOS9方法 移動(dòng)過(guò)程中隨時(shí)更新cell位置 ? ? ? ? ? ? [self.collectionView updateInteractiveMovementTargetPosition:[longPress locationInView:self.collectionView]]; ? ? ? ? } ? ? ? ? ? ? break; ? ? ? ? case UIGestureRecognizerStateEnded: ? ? ? ? { // 手勢(shì)結(jié)束 ? ? ? ? ? ? // iOS9方法 移動(dòng)結(jié)束后關(guān)閉cell移動(dòng) ? ? ? ? ? ? [self.collectionView endInteractiveMovement]; ? ? ? ? } ? ? ? ? ? ? break; ? ? ? ? default: //手勢(shì)其他狀態(tài) ? ? ? ? ? ? [self.collectionView cancelInteractiveMovement]; ? ? ? ? ? ? break; ? ? } }
8.iOS9之前的實(shí)現(xiàn)
#pragma mark - iOS9 之前的方法 - (void)action:(UILongPressGestureRecognizer *)longPress { ? ? switch (longPress.state) { ? ? ? ? case UIGestureRecognizerStateBegan: ? ? ? ? { // 手勢(shì)開始 ? ? ? ? ? ? //判斷手勢(shì)落點(diǎn)位置是否在row上 ? ? ? ? ? ? NSIndexPath *indexPath = [self.collectionView indexPathForItemAtPoint:[longPress locationInView:self.collectionView]]; ? ? ? ? ? ? self.oldIndexPath = indexPath; ? ? ? ? ? ? if (indexPath == nil) { ? ? ? ? ? ? ? ? break; ? ? ? ? ? ? } ? ? ? ? ? ? UICollectionViewCell *cell = [self.collectionView cellForItemAtIndexPath:indexPath]; ? ? ? ? ? ? // 使用系統(tǒng)的截圖功能,得到cell的截圖視圖 ? ? ? ? ? ? UIView *snapshotView = [cell snapshotViewAfterScreenUpdates:NO]; ? ? ? ? ? ? snapshotView.frame = cell.frame; ? ? ? ? ? ? [self.view addSubview:self.snapshotView = snapshotView]; ? ? ? ? ? ? // 截圖后隱藏當(dāng)前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: ? ? ? ? { // 手勢(shì)改變 ? ? ? ? ? ? //當(dāng)前手指位置 截圖視圖位置隨著手指移動(dòng)而移動(dòng) ? ? ? ? ? ? CGPoint currentPoint = [longPress locationInView:self.collectionView]; ? ? ? ? ? ? self.snapshotView.center = currentPoint; ? ? ? ? ? ? // 計(jì)算截圖視圖和哪個(gè)可見cell相交 ? ? ? ? ? ? for (UICollectionViewCell *cell in self.collectionView.visibleCells) { ? ? ? ? ? ? ? ? // 當(dāng)前隱藏的cell就不需要交換了,直接continue ? ? ? ? ? ? ? ? if ([self.collectionView indexPathForCell:cell] == self.oldIndexPath) { ? ? ? ? ? ? ? ? ? ? continue; ? ? ? ? ? ? ? ? } ? ? ? ? ? ? ? ? // 計(jì)算中心距 ? ? ? ? ? ? ? ? CGFloat space = sqrtf(pow(self.snapshotView.center.x - cell.center.x, 2) + powf(self.snapshotView.center.y - cell.center.y, 2)); ? ? ? ? ? ? ? ? // 如果相交一半就移動(dòng) ? ? ? ? ? ? ? ? if (space <= self.snapshotView.bounds.size.width / 2) { ? ? ? ? ? ? ? ? ? ? self.moveIndexPath = [self.collectionView indexPathForCell:cell]; ? ? ? ? ? ? ? ? ? ? //移動(dòng) 會(huì)調(diào)用willMoveToIndexPath方法更新數(shù)據(jù)源 ? ? ? ? ? ? ? ? ? ? [self.collectionView moveItemAtIndexPath:self.oldIndexPath toIndexPath:self.moveIndexPath]; ? ? ? ? ? ? ? ? ? ? //設(shè)置移動(dòng)后的起始indexPath ? ? ? ? ? ? ? ? ? ? self.oldIndexPath = self.moveIndexPath; ? ? ? ? ? ? ? ? ? ? break; ? ? ? ? ? ? ? ? } ? ? ? ? ? ? } ? ? ? ? } ? ? ? ? ? ? break; ? ? ? ? default: ? ? ? ? { // 手勢(shì)結(jié)束和其他狀態(tài) ? ? ? ? ? ? UICollectionViewCell *cell = [self.collectionView cellForItemAtIndexPath:self.oldIndexPath]; ? ? ? ? ? ? // 結(jié)束動(dòng)畫過(guò)程中停止交互,防止出問(wèn)題 ? ? ? ? ? ? self.collectionView.userInteractionEnabled = NO; ? ? ? ? ? ? // 給截圖視圖一個(gè)動(dòng)畫移動(dòng)到隱藏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);
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
IOS手勢(shì)操作(拖動(dòng)、捏合、旋轉(zhuǎn)、點(diǎn)按、長(zhǎng)按、輕掃、自定義)
這篇文章主要介紹了IOS手勢(shì)操作(拖動(dòng)、捏合、旋轉(zhuǎn)、點(diǎn)按、長(zhǎng)按、輕掃、自定義),需要的朋友可以參考下2015-07-07解決Alamofire庫(kù)在iOS7下設(shè)置Head無(wú)效的問(wèn)題
本文主要介紹Alamofire庫(kù)在iOS下設(shè)置Head,這里通過(guò)代碼實(shí)例解決不同版本的IOS系統(tǒng)出現(xiàn)的問(wèn)題,有需要的小伙伴可以參考下2016-07-07iOS實(shí)現(xiàn)實(shí)時(shí)檢測(cè)網(wǎng)絡(luò)狀態(tài)的示例代碼
網(wǎng)絡(luò)連接狀態(tài)檢測(cè)對(duì)于我們的iOS開發(fā)來(lái)說(shuō)是一個(gè)非常通用的需求。下面這篇文章主要就給大家介紹了關(guān)于利用iOS實(shí)現(xiàn)實(shí)時(shí)檢測(cè)網(wǎng)絡(luò)狀態(tài)的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起看看吧。2017-07-07實(shí)例講解iOS應(yīng)用開發(fā)中UIPickerView滾動(dòng)選擇欄的用法
這篇文章主要介紹了iOS應(yīng)用開發(fā)中UIPickerView滾動(dòng)選擇欄的用法,示例代碼基于傳統(tǒng)的Objective-C,需要的朋友可以參考下2016-04-04iOS schem與Universal Link 調(diào)試時(shí)踩坑解決記錄
這篇文章主要為大家介紹了iOS schem與Universal Link 調(diào)試時(shí)踩坑解決記錄,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-01-01IOS百度地圖導(dǎo)航開發(fā)功能實(shí)現(xiàn)簡(jiǎn)述
百度地圖導(dǎo)航非常實(shí)用,那么基于代碼是如何實(shí)現(xiàn)的呢,下面通過(guò)本文給大家介紹IOS百度地圖導(dǎo)航開發(fā)功能實(shí)現(xiàn)簡(jiǎn)述,需要的朋友可以參考下本文2016-03-03iOS開發(fā)中蘋果輸入手機(jī)號(hào)變用戶的名字
今天我們的用戶輸入手機(jī)號(hào)之后變成了用戶的名字,沒辦法獲取驗(yàn)證碼,因?yàn)槭謾C(jī)格式不對(duì)。下面通過(guò)本文給大家分享開發(fā)中蘋果輸入手機(jī)號(hào)變用戶的名字,需要的朋友可以參考下2017-05-05iOS 實(shí)現(xiàn)類似QQ分組樣式的兩種方式
這篇文章主要介紹了iOS 實(shí)現(xiàn)類似QQ分組樣式的兩種方式,思路很簡(jiǎn)單,對(duì)模型數(shù)據(jù)操作或則控制界面顯示,需要的朋友可以參考下2017-07-07