http://www.itnose.net/news/155/6429874


2016-01-03 19:46

这是一个完全依靠手势的操作TodoList的演示,功能上左划删除,右划完成任务,拖拽调整顺序,捏合张开插入。

项目源码: https://github.com/luan-ma/ClearStyleDemo.Swift

初始化

TDCTodoItem.swift 定义模型对象

TDCTodoListController.swift 继承自UITableViewController, 演示UITableView操作

varitems=[
TDCTodoItem(text:"Feedthecat"),TDCTodoItem(text:"Buyeggs"),TDCTodoItem(text:"PackbagsforWWDC"),TDCTodoItem(text:"Ruletheweb"),TDCTodoItem(text:"BuyanewiPhone"),TDCTodoItem(text:"Findmissingsocks"),TDCTodoItem(text:"Writeanewtutorial"),TDCTodoItem(text:"MasterObjective-C"),TDCTodoItem(text:"Rememberyourweddinganniversary!"),TDCTodoItem(text:"Drinklessbeer"),TDCTodoItem(text:"Learntodraw"),TDCTodoItem(text:"Takethecartothegarage"),TDCTodoItem(text:"SellthingsoneBay"),TDCTodoItem(text:"Learntojuggle"),TDCTodoItem(text:"Giveup")
]

overridefuncviewDidLoad(){
super.viewDidLoad()

//捏合手势
letpinch=UIPinchGestureRecognizer(target:self,action:"handlePinch:")
//长按拖拽
letlongPress=UILongPressGestureRecognizer(target:self,action:"handleLongPress:")

tableView.addGestureRecognizer(pinch)
tableView.addGestureRecognizer(longPress)
}


左划删除、右划完成

在每一个Cell添加滑动手势(Pan)。处理划动距离,超过宽度1/3就为有效操作,左划为删除操作,右划为完成操作。

布局使用AutoLayout,中间内容区的限制条件是宽度等于容器宽度、高度等于容器高度、垂直中对齐、水平中对齐,而平移操作实际上就是操作水平中对齐的距离值。

TDCTodoItemCell.swift关键代码如下

手势判断

//如果是划动手势,仅支持左右划动;如果是其它手势,则有父类负责
overridefuncgestureRecognizerShouldBegin(gestureRecognizer:UIGestureRecognizer)->Bool{
ifletpanGesture=gestureRecognizeras?UIPanGestureRecognizer{
lettranslation=panGesture.translationInView(self.superview)
returnfabs(translation.x)>fabs(translation.y)
}else{
returnsuper.gestureRecognizerShouldBegin(gestureRecognizer)
}
}

手势操作

varonDelete:((TDCTodoItemCell)->Void)?
varonComplete:((TDCTodoItemCell)->Void)?

privatevardeleteOnDragrelease:Bool=false
privatevarcompleteOnDragrelease:Bool=false

//划动平移的实际AutoLayout中的水平中对齐的距离
@IBOutletweakvarcenterConstraint:NSLayoutConstraint!
privatevaroriginConstant:CGFloat=0

funchandlePan(panGesture:UIPanGestureRecognizer){
switchpanGesture.state{
case.Began:
originConstant=centerConstraint.constant
case.Changed:
lettranslation=panGesture.translationInView(self)
centerConstraint.constant=translation.x

//划动移动1/3宽度为有效划动
letfinished=fabs(translation.x)>CGRectGetWidth(bounds)/3
iftranslation.x<originConstant{//右划
iffinished{
deleteOnDragrelease=true
rightLabel.textColor=UIColor.redColor()
}else{
deleteOnDragrelease=false
rightLabel.textColor=UIColor.whiteColor()
}
}else{//左划
iffinished{
completeOnDragrelease=true
leftLabel.textColor=UIColor.greenColor()
}else{
completeOnDragrelease=false
leftLabel.textColor=UIColor.whiteColor()
}
}
case.Ended:
centerConstraint.constant=originConstant

ifdeleteOnDragrelease{
deleteOnDragrelease=false
ifletonDelete=onDelete{
onDelete(self)
}
}

ifcompleteOnDragrelease{
completeOnDragrelease=false
ifletonComplete=onComplete{
onComplete(self)
}
}
default:
break
}
}

TDCTodoListController.swift中执行删除操作

/*
//简单删除
funcdeletetoDoItem(indexPath:NSIndexPath){
tableView.beginUpdates()
items.removeAtIndex(indexPath.row)
tableView.deleteRowsAtIndexPaths([indexPath],withRowAnimation:.Automatic)
tableView.endUpdates()
}
*/

//视觉效果更漂亮的删除
funcdeletetoDoItem(indexPath:NSIndexPath){
letitem=items.removeAtIndex(indexPath.row)
varanimationEnabled=false
letlastCell=tableView.visibleCells.last
vardelay:NSTimeInterval=0
forcellintableView.visibleCells{
letcell=cellas!TDCTodoItemCell
ifanimationEnabled{
UIView.animateWithDuration(0.25,delay:delay,options:.CurveEaseInOut,animations:{()->Voidin
cell.frame=CGRectOffset(cell.frame,-CGRectGetHeight(cell.frame))
},completion:{(completed)->Voidin
ifcell==lastCell{
self.tableView.reloadData()
}
})
delay+=0.03
}

ifcell.todoItem==item{
animationEnabled=true
cell.hidden=true
}
}
}


拖拽排序

长按选中某Cell,截图此Cell生成UIImageView,然后隐藏此Cell(hidden=true),随手指移动拖拽UIImageView,每次拖拽到一个Cell上的时候,交换当前Cell和隐藏Cell位置。效果如下

截图UIView生成一个新的UIImageVIew

funcsnapView(view:UIView)->UIImageView{
UIGraphicsBeginImageContextWithOptions(view.bounds.size,false,0)
view.layer.renderInContext(UIGraphicsGetCurrentContext()!)
letimage=UIGraphicsGetimageFromCurrentimageContext()
UIGraphicsEndImageContext()

letsnapShot=UIImageView(image:image)
snapShot.layer.masksToBounds=false;
snapShot.layer.cornerRadius=0;
snapShot.layer.shadowOffset=CGSizeMake(-5.0,0.0);
snapShot.layer.shadowOpacity=0.4;
snapShot.layer.shadowRadius=5;
snapShot.frame=view.frame
returnsnapShot
}

拖拽操作代码,详细操作参考注释

privatevarsourceIndexPath:NSIndexPath?
privatevarsnapView:UIView?

funchandleLongPress(longPress:UILongPressGestureRecognizer){
letpoint=longPress.locationInView(tableView)

ifletindexPath=tableView.indexPathForRowAtPoint(point){
switchlongPress.state{
case.Began:
ifletcell=tableView.cellForRowAtIndexPath(indexPath){
sourceIndexPath=indexPath
letsnapView=self.snapView(cell)
snapView.alpha=0

self.snapView=snapView

tableView.addSubview(snapView)

UIView.animateWithDuration(0.25,animations:{
//选中Cell跳出放大效果
snapView.alpha=0.95
snapView.center=CGPointMake(cell.center.x,point.y)
snapView.transform=CGAffineTransformMakeScale(1.05,1.05)

cell.alpha=0
},completion:{(completed)->Voidin
cell.hidden=true
cell.alpha=1
})
}else{
sourceIndexPath=nil
snapView=nil
break
}
case.Changed:
ifletsnapView=snapView{
//截图随手指上下移动
snapView.center=CGPointMake(snapView.center.x,point.y)
}

//如果手指移动到一个新的Cell上面,隐藏Cell跟此Cell交换位置
ifletfromIndexPath=sourceIndexPath{
iffromIndexPath!=indexPath{
tableView.beginUpdates()
lettemp=items[indexPath.row]
items[indexPath.row]=items[fromIndexPath.row]
items[fromIndexPath.row]=temp
tableView.moveRowAtIndexPath(fromIndexPath,toIndexPath:indexPath)
tableView.endUpdates()
sourceIndexPath=indexPath
}
}

//手指移动到屏幕顶端或底部,UITableView自动滚动
letstep:CGFloat=64
ifletparentView=tableView.superview{
letparentPos=tableView.convertPoint(point,toView:parentView)
ifparentPos.y>parentView.bounds.height-step{
varoffset=tableView.contentOffset
offset.y+=(parentPos.y-parentView.bounds.height+step)
ifoffset.y>tableView.contentSize.height-tableView.bounds.height{
offset.y=tableView.contentSize.height-tableView.bounds.height
}
tableView.setContentOffset(offset,animated:false)
}elseifparentPos.y<=step{
varoffset=tableView.contentOffset
offset.y-=(step-parentPos.y)
ifoffset.y<0{
offset.y=0
}
tableView.setContentOffset(offset,animated:false)
}
}
default:
ifletsnapView=snapView,letfromIndexPath=sourceIndexPath,letcell=tableView.cellForRowAtIndexPath(fromIndexPath){
cell.alpha=0
cell.hidden=false

//长按移动结束,隐藏的Cell恢复显示,删除截图
UIView.animateWithDuration(0.25,animations:{()->Voidin
snapView.center=cell.center
snapView.alpha=0

cell.alpha=1
},completion:{[uNownedself](completed)->Voidin
snapView.removeFromSuperview()
self.snapView=nil
self.sourceIndexPath=nil

self.tableView.performSelector("reloadData",withObject:nil,afterDelay:0.5)
})
}
}
}
}


捏合张开插入

通过捏合手势中两个触点获取两个相邻的Cell,通过修改UIView.transform属性移动屏幕上的Cell位置。

获取Pinch两个手指坐标的工具方法

funcpointsOfPinch(pinch:UIPinchGestureRecognizer)->(CGPoint,CGPoint){
ifpinch.numberOftouches()>1{
letpoint1=pinch.locationOfTouch(0,inView:tableView)
letpoint2=pinch.locationOfTouch(1,inView:tableView)
ifpoint1.y<=point2.y{
return(point1,point2)
}else{
return(point2,point1)
}
}else{
letpoint=pinch.locationOfTouch(0,inView:tableView)
return(point,point)
}
}

捏合张开操作代码,详情请参考代码和注释

//插入点
privatevarpinchIndexPath:NSIndexPath?
//临时代理视图
privatevarplacheHolderCell:TDCPlaceHolderView?
//两触点的起始位置
privatevarsourcePoints:(upperPoint:CGPoint,downPoint:CGPoint)?
//可以插入操作的标志
privatevarpinchInsertEnabled=false

funchandlePinch(pinch:UIPinchGestureRecognizer){
switchpinch.state{
case.Began:
pinchBegan(pinch)
case.Changed:
pinchChanged(pinch)
default:
pinchEnd(pinch)
}
}

funcpinchBegan(pinch:UIPinchGestureRecognizer){
pinchIndexPath=nil
sourcePoints=nil
pinchInsertEnabled=false

let(upperPoint,downPoint)=pointsOfPinch(pinch)
ifletupperIndexPath=tableView.indexPathForRowAtPoint(upperPoint),letdownIndexPath=tableView.indexPathForRowAtPoint(downPoint){
ifdownIndexPath.row-upperIndexPath.row==1{
letupperCell=tableView.cellForRowAtIndexPath(upperIndexPath)!
letplacheHolder=NSBundle.mainBundle().loadNibNamed("TDCPlaceHolderView",owner:tableView,options:nil).firstas!TDCPlaceHolderView
placheHolder.frame=CGRectOffset(upperCell.frame,CGRectGetHeight(upperCell.frame)/2)
tableView.insertSubview(placheHolder,atIndex:0)

sourcePoints=(upperPoint,downPoint)
pinchIndexPath=upperIndexPath
placheHolderCell=placheHolder
}
}
}

funcpinchChanged(pinch:UIPinchGestureRecognizer){
ifletpinchIndexPath=pinchIndexPath,letoriginPoints=sourcePoints,letplacheHolderCell=placheHolderCell{
letpoints=pointsOfPinch(pinch)

letupperdistance=points.0.y-originPoints.upperPoint.y
letdowndistance=originPoints.downPoint.y-points.1.y
letdistance=-min(0,min(upperdistance,downdistance))
NSLog("distance=\(distance)")

//移动两边的Cell
forcellintableView.visibleCells{
letindexPath=tableView.indexPathForCell(cell)!
ifindexPath.row<=pinchIndexPath.row{
cell.transform=CGAffineTransformMakeTranslation(0,-distance)
}else{
cell.transform=CGAffineTransformMakeTranslation(0,distance)
}
}

//插入的Cell变形
letscaleY=min(64,fabs(distance)*2)/CGFloat(64)
placheHolderCell.transform=CGAffineTransformMakeScale(1,scaleY)

placheHolderCell.lblTitle.text=scaleY<=0.5?"张开双指插入新项目":"松手可以插入新项目"

//张开超过一个Cell高度时,执行插入操作
pinchInsertEnabled=scaleY>=1
}
}

funcpinchEnd(pinch:UIPinchGestureRecognizer){
ifletpinchIndexPath=pinchIndexPath,letplacheHolderCell=placheHolderCell{
placheHolderCell.transform=CGAffineTransformIdentity
placheHolderCell.removeFromSuperview()
self.placheHolderCell=nil

ifpinchInsertEnabled{
//恢复各Cell的transform
forcellinself.tableView.visibleCells{
cell.transform=CGAffineTransformIdentity
}

//插入操作
letindex=pinchIndexPath.row+1
items.insert(TDCTodoItem(text:""),atIndex:index)
tableView.reloadData()

//弹出键盘
letcell=tableView.cellForRowAtIndexPath(NSIndexPath(forRow:index,inSection:0))as!TDCTodoItemCell
cell.txtField.becomeFirstResponder()
}else{
//放弃插入,恢复原位置
UIView.animateWithDuration(0.25,delay:0,animations:{[uNownedself]()->Voidin
forcellinself.tableView.visibleCells{
cell.transform=CGAffineTransformIdentity
}
},completion:{[uNownedself](completed)->Voidin
self.tableView.reloadData()
})
}
}

sourcePoints=nil
pinchIndexPath=nil
pinchInsertEnabled=false
}


参考

1. https://github.com/ColinEberhardt/iOS-ClearStyle

2. http://blog.csdn.net/u013604612/article/details/43884039

Swift语言开发:仿Clear手势操作(拖拽、划动、捏合)UITableView的更多相关文章

  1. 移动HTML5前端框架—MUI的使用

    这篇文章主要介绍了移动HTML5前端框架—MUI的使用的相关资料,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧

  2. HTML5 weui使用笔记

    这篇文章主要介绍了HTML5 weui使用笔记,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧

  3. HTML5单页面手势滑屏切换原理分析

    H5单页面手势滑屏切换是采用HTML5 触摸事件(Touch) 和 CSS3动画(Transform,Transition)来实现的,下面通过本文简单说一下其实现原理和主要思路,感兴趣的朋友一起看看吧

  4. h5使用canvas画布实现手势解锁

    这篇文章主要介绍了h5使用canvas画布实现手势解锁的相关资料,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧

  5. Html5写一个简单的俄罗斯方块小游戏

    这篇文章主要介绍了基于Html5写一个简单的俄罗斯方块小游戏,本文通过图文并茂的形式给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友参考下吧

  6. ios – UITableView和Cell Reuse

    这是我的CustomCell类的init方法解决方法如果没有要显示的图像,则必须清除图像视图:

  7. ios – fetchedResultsController.fetchedObjects.count = 0但它充满了对象

    我正在使用相当标准的fetchedResultsController实现来输出tableView.在-viewDidLoad的最后,我正在进行第一次调用:这是我的fetchedResultsController:我的tableView方法:所以,问题是:在_fetchedResultsController.fetchedobjects.count的日志中等于0,但在视觉上tableView充满了对

  8. ios – UITableView在滚动时阻止重新加载

    或者你能想象一个防止这种行为的好方法吗?解决方法抱歉,我没有足够的声誉来添加评论,因此在单独的答案中回答您的上一个问题.-performSelector:withObject:afterDelay:延迟为0.0秒不会立即执行给定的选择器,而是在当前的RunloopCycle结束后和给定的延迟之后执行它.-performSelector:withObject:添加到当前Runloop循环中并执行.这与直接调用该方法相同.因此,使用-performSelector:withObject:afterDelay:

  9. ios – 在Swift中通过标记访问UITableViewCell内部的不同视图

    我正在尝试使用swift为iOS8制作应用程序.这里的目标是制作一种新闻源.此Feed显示来自用户的帖子,其遵循特定模式.我想过使用UITableView,其中每个单元格都遵循自定义布局.当我尝试访问其中的文本标签时出现问题.我尝试通过它的标签访问它,但是当我这样做时,整个应用程序崩溃了.报告的错误是“Swift动态转换失败”,我使用以下代码访问视图:难道我做错了什么?解决方法我认为问题是标签0.所有视图都是默认值0.所以尝试另一个标签值.

  10. ios – 如何实现`prepareForReuse`?

    解决方法尝试将此添加到您的MGSwipeTableCell.m:

随机推荐

  1. Swift UITextField,UITextView,UISegmentedControl,UISwitch

    下面我们通过一个demo来简单的实现下这些控件的功能.首先,我们拖将这几个控件拖到storyboard,并关联上相应的属性和动作.如图:关联上属性和动作后,看看实现的代码:

  2. swift UISlider,UIStepper

    我们用两个label来显示slider和stepper的值.再用张图片来显示改变stepper值的效果.首先,这三个控件需要全局变量声明如下然后,我们对所有的控件做个简单的布局:最后,当slider的值改变时,我们用一个label来显示值的变化,同样,用另一个label来显示stepper值的变化,并改变图片的大小:实现效果如下:

  3. preferredFontForTextStyle字体设置之更改

    即:

  4. Swift没有异常处理,遇到功能性错误怎么办?

    本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容,请发送邮件至dio@foxmail.com举报,一经查实,本站将立刻删除。

  5. 字典实战和UIKit初探

    ios中数组和字典的应用Applicationschedule类别子项类别名称优先级数据包contactsentertainment接触UIKit学习用Swift调用CocoaTouchimportUIKitletcolors=[]varbackView=UIView(frame:CGRectMake(0.0,0.0,320.0,CGFloat(colors.count*50)))backView

  6. swift语言IOS8开发战记21 Core Data2

    上一话中我们简单地介绍了一些coredata的基本知识,这一话我们通过编程来实现coredata的使用。还记得我们在coredata中定义的那个Model么,上面这段代码会加载这个Model。定义完方法之后,我们对coredata的准备都已经完成了。最后强调一点,coredata并不是数据库,它只是一个框架,协助我们进行数据库操作,它并不关心我们把数据存到哪里。

  7. swift语言IOS8开发战记22 Core Data3

    上一话我们定义了与coredata有关的变量和方法,做足了准备工作,这一话我们来试试能不能成功。首先打开上一话中生成的Info类,在其中引用头文件的地方添加一个@objc,不然后面会报错,我也不知道为什么。

  8. swift实战小程序1天气预报

    在有一定swift基础的情况下,让我们来做一些小程序练练手,今天来试试做一个简单地天气预报。然后在btnpressed方法中依旧增加loadWeather方法.在loadWeather方法中加上信息的显示语句:运行一下看看效果,如图:虽然显示出来了,但是我们的text是可编辑状态的,在storyboard中勾选Editable,再次运行:大功告成,而且现在每次单击按钮,就会重新请求天气情况,大家也来试试吧。

  9. 【iOS学习01】swift ? and !  的学习

    如果不初始化就会报错。

  10. swift语言IOS8开发战记23 Core Data4

    接着我们需要把我们的Rest类变成一个被coredata管理的类,点开Rest类,作如下修改:关键字@NSManaged的作用是与实体中对应的属性通信,BinaryData对应的类型是NSData,CoreData没有布尔属性,只能用0和1来区分。进行如下操作,输入类名:建立好之后因为我们之前写的代码有些地方并不适用于coredata,所以编译器会报错,现在来一一解决。

返回
顶部