欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

使用iOS控件UICollectionView生成可拖动的桌面的实例

程序员文章站 2023-12-21 11:31:28
一个app受欢迎的程度,一方面来源于它本身为用户提供便捷的功能,另一方面则来源于它的ui。ui是用户体验重要的组成部分,构成ui的的元素恰恰离不开那些看似独立的控件。在开发...

一个app受欢迎的程度,一方面来源于它本身为用户提供便捷的功能,另一方面则来源于它的ui。ui是用户体验重要的组成部分,构成ui的的元素恰恰离不开那些看似独立的控件。在开发的过程中,大家对uitableview应该很熟悉吧!确实uitableview在处理数据显示方面有着很强大的功能,例如网红们使用的微博,微信社交软件的聊天界面等等,这种流式布局使用uitableview简直最合适不过了;但毕竟uitableview不是万能的,当需要显示横纵向的数据时它就显得捉襟见肘了,虽然这也难不倒我们程序猿但是何必要大费周章的去定义复杂的cell呢!uicollectionview就是专门用来应付这种布局的,使用uicollectionview可以给我们带来以下几点优势:1.可以高度的定制内容展示的样式 2.高效的管理大量的数据。

首先给大家看一下这个demo的效果图:

使用iOS控件UICollectionView生成可拖动的桌面的实例

ios设备不知道现在有没有可以屏幕录制的app,这样我就可以把操作的动作用gif图片po上来了,大家如果有推荐可以告诉我哈!关于拖动,长按图片后就可以将图片拖到你想要的位置上,另外的图片则会依次排序,很顺畅的体验。

在使用uicollectionview的时后,我们的类需要实现这些协议:uicollectionviewdatasource,uicollectionviewdelegateflowlayout,

uicollectionviewdelegate,uigesturerecognizerdelegate。

uicollectionviewdatasource:和我们在uitableview中所需要实现的uitableviewdatasource是一个道理,它里面包括以下这些api:

@protocol uicollectionviewdatasource <nsobject> 
@required 
 
- (nsinteger)collectionview:(uicollectionview *)collectionview numberofitemsinsection:(nsinteger)section; 
 
// the cell that is returned must be retrieved from a call to -dequeuereusablecellwithreuseidentifier:forindexpath: 
- (uicollectionviewcell *)collectionview:(uicollectionview *)collectionview cellforitematindexpath:(nsindexpath *)indexpath; 
 
@optional 
 
- (nsinteger)numberofsectionsincollectionview:(uicollectionview *)collectionview; 
 
// the view that is returned must be retrieved from a call to -dequeuereusablesupplementaryviewofkind:withreuseidentifier:forindexpath: 
- (uicollectionreusableview *)collectionview:(uicollectionview *)collectionview viewforsupplementaryelementofkind:(nsstring *)kind atindexpath:(nsindexpath *)indexpath; 
 
- (bool)collectionview:(uicollectionview *)collectionview canmoveitematindexpath:(nsindexpath *)indexpath ns_available_ios(9_0); 
- (void)collectionview:(uicollectionview *)collectionview moveitematindexpath:(nsindexpath *)sourceindexpath toindexpath:(nsindexpath*)destinationindexpath ns_available_ios(9_0); 
 
@end 

uicollectionviewdelegateflowlayout:uicollectionviewflowlayout是一个专门用来管理collectionview布局的类,可以通过实现以下函数来调整我们界面的样式:

@protocol uicollectionviewdelegateflowlayout <uicollectionviewdelegate> 
@optional 
 
- (cgsize)collectionview:(uicollectionview *)collectionview layout:(uicollectionviewlayout*)collectionviewlayout sizeforitematindexpath:(nsindexpath *)indexpath; 
- (uiedgeinsets)collectionview:(uicollectionview *)collectionview layout:(uicollectionviewlayout*)collectionviewlayout insetforsectionatindex:(nsinteger)section; 
- (cgfloat)collectionview:(uicollectionview *)collectionview layout:(uicollectionviewlayout*)collectionviewlayout minimumlinespacingforsectionatindex:(nsinteger)section; 
- (cgfloat)collectionview:(uicollectionview *)collectionview layout:(uicollectionviewlayout*)collectionviewlayout minimuminteritemspacingforsectionatindex:(nsinteger)section; 
- (cgsize)collectionview:(uicollectionview *)collectionview layout:(uicollectionviewlayout*)collectionviewlayout referencesizeforheaderinsection:(nsinteger)section; 
- (cgsize)collectionview:(uicollectionview *)collectionview layout:(uicollectionviewlayout*)collectionviewlayout referencesizeforfooterinsection:(nsinteger)section; 
 
@end 

uicollectionviewdelegate:和uitableviewdelegate一样,这里就把它协议里面的的函数po出来了,通过重写里面的函数,我们可以实现cell的点击与拖动。

uigesturerecognizerdelegate:由于我们还有一个拖动的功能,所以需要实现用户手势的协议。

好了,基础的概念讲了,现在我们就开始动手实现他吧!老样子直接看源码:

- (void)viewdidload { 
  [super viewdidload]; 
  // do any additional setup after loading the view, typically from a nib. 
   
  //设置背景色 
  [self.view setbackgroundcolor:[uicolor colorwithpatternimage:[uiimage imagenamed:@"back.jpg"]]]; 
  //设置数据源 
  self.datasource = [[nsmutablearray alloc] initwithobjects: 
            @"1.png",@"2.png",@"3.png", @"4.png", @"5.png", @"6.png", @"7.png", @"8.png", @"9.png", @"10.png", @"11.jpg", @"12.jpg", 
            @"13.jpg", @"14.jpg", @"15.jpg", @"16.png", @"17.png", @"18.png", @"19.png", @"20.png", nil nil]; 
  //初始化布局 
  self.flow = [[uicollectionviewflowlayout alloc] init]; 
  self.collect = [[uicollectionview alloc] initwithframe:cgrectzero collectionviewlayout:self.flow]; 
  [self.collect setbackgroundcolor:[uicolor clearcolor]]; 
   
  //注册cell 这一步必须要实现 
  [self.collect registerclass:[customcollectioncell class] forcellwithreuseidentifier:@"customcell"]; 
   
  self.collect.delegate = self; 
  self.collect.datasource = self; 
   
  [self.collect setframe:self.view.bounds]; 
   
  //添加长按手势 
  self.longpressgesturerecognizer = [[uilongpressgesturerecognizer alloc] init]; 
  [self.longpressgesturerecognizer addtarget:self action:@selector(handlelongpressrecognizer:)]; 
  [self.collect addgesturerecognizer:self.longpressgesturerecognizer]; 
   
  [self.view addsubview:self.collect]; 
} 

在viewdidload函数中,初始化了一个uicollectionviewflowlayout布局,并且需要配合uicollectionview来使用,两者加一起来使用才能“完美”,想比uitableview 中cell使用的不同,在uicollectionview中必须先对cell进行注册(registerclass),不然会在运行过程中报错。如何使排列的图片可以拖动呢,在这里我为uicollectionview添加了一个长按手势uilongpressgesturerecognizer。当我们长按时会触发handlelongpressrecognizer,代码如下:

- (void)handlelongpressrecognizer:(uilongpressgesturerecognizer *)gesture{ 
  switch (gesture.state) { 
    case uigesturerecognizerstatebegan:{ 
      nsindexpath *path = [self.collect indexpathforitematpoint:[gesture locationinview:gesture.view]]; 
      if(path == nil){ 
        break; 
      } 
       
      [self.collect begininteractivemovementforitematindexpath:path]; 
    } 
      break; 
    case uigesturerecognizerstatechanged: 
      [self.collect updateinteractivemovementtargetposition:[gesture locationinview:gesture.view]]; 
      break; 
    case uigesturerecognizerstateended: 
      [self.collect endinteractivemovement]; 
      break; 
    default: 
      [self.collect cancelinteractivemovement]; 
      break; 
  } 
} 

好看的界面才能抓住用户的心,这里我自定义了cell继承自uicollectionviewcell,cell中展示的图片会根据自身图片的大小进行按比例缩放,这样我们的桌面看上去就会有横版图片与竖版图片,如果不自定义的话就都是方方正正的九宫格,还是花点心思自定义一下显示效果吧!customcollectioncell的的代码如下:

#import "customcollectioncell.h" 
 
@implementation customcollectioncell 
@synthesize imagev = _imagev; 
@synthesize labelv = _labelv; 
@synthesize boundview = _boundview; 
 
- (id)initwithframe:(cgrect) frame{ 
  self = [super initwithframe:frame]; 
  //init attributes 
  if(self){ 
    [self setbackgroundcolor:[uicolor clearcolor]]; 
    self.imagev = [[uiimageview alloc] initwithframe:cgrectzero]; 
    self.labelv = [[uilabel alloc] initwithframe:cgrectzero]; 
    self.boundview = [[uiview alloc] initwithframe:cgrectzero]; 
    [self.boundview setbackgroundcolor:[uicolor whitecolor]]; 
    [self.boundview addsubview:self.imagev]; 
    [self addsubview:self.boundview]; 
    [self addsubview:self.labelv]; 
  } 
   
  return self; 
} 
 
- (void)setimagewithtext:(uiimage *)image text:(nsstring *)text{ 
  if(!image){ 
    return; 
  } 
   
  cgfloat imgwidth = image.size.width; 
  cgfloat imgheight = image.size.height; 
  cgfloat iconwidth = 0.0; 
  cgfloat iconheight = 0.0; 
   
  if(imgwidth > imgheight){ 
    iconheight = roundf(((self.frame.size.width - 16)*imgheight)/imgwidth); 
    iconwidth = self.frame.size.width - 16; 
    [self.boundview setframe:cgrectmake(0, self.frame.size.height - iconheight - 16, iconwidth + 16, iconheight + 16)]; 
    [self.imagev setframe:cgrectmake(8, 8, iconwidth, iconheight)]; 
  }else{ 
    iconwidth = roundf(((self.frame.size.width - 16) *imgwidth)/imgheight); 
    iconheight = self.frame.size.height - 16; 
    [self.boundview setframe:cgrectmake(roundf((self.frame.size.width - iconwidth)/2), 0, iconwidth + 16, iconheight + 16)]; 
    [self.imagev setframe:cgrectmake(8, 8, iconwidth, iconheight)]; 
  } 
   
  [self.imagev setimage:image]; 
} 
 
@end 

以上这些就是构成该demo的重要组成部分了,uicollectionview还有很多的要素在这里面没有讲到,往后我还会再研究这个控件更加高级的的使用,本篇就好比是餐前的开胃小菜吧,希望大家喜欢。附上这个例子的源码:

@implementation viewcontroller 
@synthesize datasource = _datasource; 
@synthesize longpressgesturerecognizer = _longpressgesturerecognizer; 
@synthesize flow = _flow; 
@synthesize collect = _collect; 
 
- (void)viewdidload { 
  [super viewdidload]; 
  // do any additional setup after loading the view, typically from a nib. 
   
  //设置背景色 
  [self.view setbackgroundcolor:[uicolor colorwithpatternimage:[uiimage imagenamed:@"back.jpg"]]]; 
  //设置数据源 
  self.datasource = [[nsmutablearray alloc] initwithobjects: 
            @"1.png",@"2.png",@"3.png", @"4.png", @"5.png", @"6.png", @"7.png", @"8.png", @"9.png", @"10.png", @"11.jpg", @"12.jpg", 
            @"13.jpg", @"14.jpg", @"15.jpg", @"16.png", @"17.png", @"18.png", @"19.png", @"20.png", nil nil]; 
  //初始化布局 
  self.flow = [[uicollectionviewflowlayout alloc] init]; 
  self.collect = [[uicollectionview alloc] initwithframe:cgrectzero collectionviewlayout:self.flow]; 
  [self.collect setbackgroundcolor:[uicolor clearcolor]]; 
   
  //注册cell 这一步必须要实现 
  [self.collect registerclass:[customcollectioncell class] forcellwithreuseidentifier:@"customcell"]; 
   
  self.collect.delegate = self; 
  self.collect.datasource = self; 
   
  [self.collect setframe:self.view.bounds]; 
   
  //添加长按手势 
  self.longpressgesturerecognizer = [[uilongpressgesturerecognizer alloc] init]; 
  [self.longpressgesturerecognizer addtarget:self action:@selector(handlelongpressrecognizer:)]; 
  [self.collect addgesturerecognizer:self.longpressgesturerecognizer]; 
   
  [self.view addsubview:self.collect]; 
} 
 
- (void)didreceivememorywarning { 
  [super didreceivememorywarning]; 
  // dispose of any resources that can be recreated. 
} 
 
- (void)handlelongpressrecognizer:(uilongpressgesturerecognizer *)gesture{ 
  switch (gesture.state) { 
    case uigesturerecognizerstatebegan:{ 
      nsindexpath *path = [self.collect indexpathforitematpoint:[gesture locationinview:gesture.view]]; 
      if(path == nil){ 
        break; 
      } 
       
      [self.collect begininteractivemovementforitematindexpath:path]; 
    } 
      break; 
    case uigesturerecognizerstatechanged: 
      [self.collect updateinteractivemovementtargetposition:[gesture locationinview:gesture.view]]; 
      break; 
    case uigesturerecognizerstateended: 
      [self.collect endinteractivemovement]; 
      break; 
    default: 
      [self.collect cancelinteractivemovement]; 
      break; 
  } 
} 
 
#pragma mark uicollectionviewdatasource 
- (nsinteger)collectionview:(uicollectionview *)collectionview numberofitemsinsection:(nsinteger)section{ 
  return self.datasource.count; 
} 
 
- (nsinteger)numberofsectionsincollectionview:(uicollectionview *)collectionview{ 
  return 1; 
} 
 
- (uicollectionviewcell *)collectionview:(uicollectionview *)collectionview cellforitematindexpath:(nsindexpath *)indexpath{ 
  customcollectioncell * cell = (customcollectioncell *)[collectionview dequeuereusablecellwithreuseidentifier:@"customcell" forindexpath:indexpath]; 
  uiimage *image = [uiimage imagenamed:[self.datasource objectatindex:indexpath.row]]; 
  [cell setimagewithtext:image text:@""]; 
   
  return cell; 
} 
 
- (bool)collectionview:(uicollectionview *)collectionview canmoveitematindexpath:(nsindexpath *)indexpath{ 
  return yes; 
} 
 
- (void)collectionview:(uicollectionview *)collectionview moveitematindexpath:(nsindexpath *)sourceindexpath toindexpath:(nsindexpath*)destinationindexpath{ 
   
  id item = [self.datasource objectatindex:sourceindexpath.item]; 
  [self.datasource removeobject:item]; 
   
  [self.datasource insertobject:item atindex:destinationindexpath.item]; 
} 
 
#pragma mark uicollectionviewdelegate 
- (void)collectionview:(uicollectionview *)collectionview didselectitematindexpath:(nsindexpath *)indexpath{ 
  testviewcontroller *test = [[testviewcontroller alloc] initwithnibname:@"testviewcontroller" bundle:nil]; 
  nsstring *imagename = [self.datasource objectatindex:indexpath.row]; 
  test.imagename = imagename; 
   
  [self.navigationcontroller pushviewcontroller:test animated:yes]; 
} 
 
- (bool)collectionview:(uicollectionview *)collectionview shouldhighlightitematindexpath:(nsindexpath *)indexpath{ 
  return yes; 
} 
 
//选中放大效果 
- (void)collectionview:(uicollectionview *)collectionview didhighlightitematindexpath:(nsindexpath *)indexpath{ 
//  customcollectioncell * cell = (customcollectioncell *)[collectionview cellforitematindexpath:indexpath]; 
//   
//  [uiview animatewithduration:1 animations:^{ 
//    cell.transform = cgaffinetransformmakescale(2.0f, 2.0f); 
//  }]; 
} 
 
//缩小效果 
- (void)collectionview:(uicollectionview *)collectionview didunhighlightitematindexpath:(nsindexpath *)indexpath{ 
//  customcollectioncell * cell = (customcollectioncell *)[collectionview cellforitematindexpath:indexpath]; 
//   
//  [uiview animatewithduration:1 animations:^{ 
//    cell.transform = cgaffinetransformmakescale(1.0f, 1.0f); 
//  }]; 
} 
 
 
#pragma mark uicollectionviewdelegateflowlayout 
- (cgsize)collectionview:(uicollectionview *)collectionview layout:(uicollectionviewlayout*)collectionviewlayout sizeforitematindexpath:(nsindexpath *)indexpath{ 
   
  return cgsizemake(100, 100); 
} 
 
/* 
 * 上左下右间距 
 */ 
- (uiedgeinsets)collectionview:(uicollectionview *)collectionview layout:(uicollectionviewlayout*)collectionviewlayout insetforsectionatindex:(nsinteger)section{ 
   
  return uiedgeinsetsmake(15, 15, 15, 15); 
} 
 
/* 
 * item space 
 */ 
- (cgfloat)collectionview:(uicollectionview *)collectionview layout:(uicollectionviewlayout*)collectionviewlayout minimuminteritemspacingforsectionatindex:(nsinteger)section{ 
   
  return 8; 
} 
 
/* 
 * 行距 20 
 */ 
- (cgfloat)collectionview:(uicollectionview *)collectionview layout:(uicollectionviewlayout*)collectionviewlayout minimumlinespacingforsectionatindex:(nsinteger)section{ 
   
  return 10; 
} 

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

上一篇:

下一篇: