关于3DTouch中Peek与Pop的正确使用

前段时间花了点时间去研究了3DTouch这一块,咱们就来聊一下,看看怎么在app中集成3DTouch中peek pop。 虽说iOS9出了这个功能感觉不太实用,但是作为一个开发者,我觉得还是有必要去了解一下,弄好了还能让自己的app装一回逼。那下面简单介绍一下peek和pop分别是什么。

   peek:当你用力按下屏幕按到一定程度时,系统会弹出一个预览视图,这个过程就称之为peek。

   pop:再用力按下去就会展开到预览视图的控制器,这过程就是pop。

     废话不多说,直接上代码

1、在viewDidLoad方法中注册预览代理

// 注意这是ios9的API,需要做一下版本控制,否则运行到9以下的会崩。
#ifdef __IPHONE_9_0
if(KC_IS_IOS9) {
// 判断设备是否支持3dTouch
if(self.traitCollection.forceTouchCapability == UIForceTouchCapabilityAvailable) {
         [selfregisterForPreviewingWithDelegate:self sourceView:self.view];
     }
}
#endif

2、注册完成后控制器需要遵守<UIViewControllerPreviewingDelegate>此协议,那我们来看看协议内定义的方法。

NS_CLASS_AVAILABLE_IOS(9_0)@protocolUIViewControllerPreviewingDelegate 

// If you return nil, a preview presentation will not be performed
// peek过程会调用此方法,返回需要预览的控制器(即需要跳转到下一级的控制器)
- (nullableUIViewController*)previewingContext:(id)previewingContext viewControllerForLocation:(CGPoint)locationNS_AVAILABLE_IOS(9_0); 

// pop过程会调用此方法,执行跳转
- (void)previewingContext:(id)previewingContext commitViewController:(UIViewController*)viewControllerToCommitNS_AVAILABLE_IOS(9_0); 

@end

协议中两个方法是require的,所以控制器必须要实现这两个方法,那该在这两个方法干些什么呢?我们来实现协议中的方法吧。

#pragma mark -UIViewControllerPreviewingDelegate
// Peek
- (UIViewController*)previewingContext:(id)context viewControllerForLocation:(CGPoint) point

{
   /*
        context: 执行peek的上下文对象。
        point: 按压位置再souceView上的点,可以理解为手指在屏幕上的按压点。
   */
  // 1、获取sourceView 即注册时传入的sourceView,一般为控制器的view
     UIView*sourceView = [context sourceView];
    // 然后判断按压点是否在某个控件的frame内,假设我们有某个自定义控件为self.locationView
/*
    这里有个涉及到坐标系转换的细节问题
    如果self.locationView不是sourceView的直接子控件,那么我们需要把point转换到self.locationView的父控件的坐标系中,代码如下:
    point = [self.locationView.superView convertPoint:point fromView:sourceView];
    若缺少这段代码,你会发现按压位置错乱的bug。
*/
// 2、如果self.locationView.frame不包含这个点就直接return,不做任何操作
    if(!CGRectContainsPoint(self.locationView.frame, point)) return nil;
// 能来到这里,即触摸点在self.locationView上,那么
// 3、设置sourceRect,这个souceRect就是当你按压时候浮起来的那个矩形区域
CGRect sourceRect = self.locationView.frame;
/*
如果self.locationView不是sourceView的直接子控件,这里同时也涉及到坐标系转换
sourceRect = [self convertRect:self.locationView.frame toView:sourceView];
*/
[context setSourceRect:sourceRect];
// 4、然后创建需要跳转的目标控制器返回就OK了
MyLocationViewController*locationVC = [MyLocationViewController new]; 
// 目标控制器有参数也需要传入参数
locationVC.locationModel=self.locationModel;
return locationVC;
}

// pop
- (void)previewingContext:(id)previewingContext commitViewController:(UIViewController*)viewControllerToCommit {

// 这里十分简单,只需要直接show一下就OK了
     [self showViewController:viewController ToCommitsender:self];
}

3、peek,pop这么简单就实现了,下面我们再来看看怎么在tableView中使用

- (UIViewController*)previewingContext:(id)context viewControllerForLocation:(CGPoint) point
{
// 1、获取sourceView
UIView*sourceView = [context sourceView];
/**通过坐标点获取 indexPath */
/*
同样如果sourceView != self.tableView的话,也需要转换坐标系
point = [self.tableView convertPoint:point fromView:sourceView];
*/
NSIndexPath*indexPath = [self.tableView indexPathForRowAtPoint:point];
// 如果indexPath为nil,则直接返回nil
if(!indexPath) return nil;
/**获得当前cell */
HCQBusinessCell*cell = [self.tableView cellForRowAtIndexPath:indexPath];
// 设置sourceRect
CGRect sourceRect = cell.frame;
/*
这里同时也涉及到坐标系转换
sourceRect = [self.tableView convertRect:cell.frame toView:sourceView];
*/
[context setSourceRect:sourceRect];
HCQMechantDetailViewController*detailVC = [HCQMechantDetailViewController new];
detailVC.mechantModel= cell.mechantModel;
returndetailVC;
}

4、至此,在tableView上Peek,pop也集成完毕,至于collectionView也是类似,我就不多说了。然而我在网上看到一些文章,他们是把注册放到cellForRow方法中,类似这样子:

- (UITableViewCell*)tableView:(UITableView*)tableView cellForRowAtIndexPath:(NSIndexPath*)indexPath
{
HCQBusinessCell*cell = [tableViewdequeueReusableCellWithIdentifier:HCQBusinessCellReuseID];
cell.mechantModel=self.mechantModels[indexPath.row];
// 3DTouch peek..pop
if(KC_IS_IOS9) {
    if(self.traitCollection.forceTouchCapability == UIForceTouchCapabilityAvailable) {
        [self registerForPreviewingWithDelegate:self sourceView:cell];
    }
}
returncell;
}

我只想说:大家千万不要这样搞,代理只需要注册一次,重复注册就会有问题了。有兴趣的可以自行尝试,但我真不建议你这样做。好了,先说这么多吧,快点为你的app集成3DTouch,开展你的装B之旅吧~~

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 201,784评论 5 474
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,745评论 2 378
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 148,702评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,229评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,245评论 5 363
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,376评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,798评论 3 393
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,471评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,655评论 1 295
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,485评论 2 318
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,535评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,235评论 3 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,793评论 3 304
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,863评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,096评论 1 258
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,654评论 2 348
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,233评论 2 341

推荐阅读更多精彩内容