随着iPhone 6s and 6s Plus的发布,苹果介绍了全新的手机交互方式:重按手势。你应该知道,这个特性其实早已应用在苹果手表和MacBook产品中,名字叫Force Touch。它给用户交互添加了全新的维度!


或许你对Force Touch为什么被重命名为3D touch感到奇怪。在克雷格·费德里吉(CraigFederighi,苹果工程师、高级副总裁)介绍这个新事物时(他自己也为名字感到困惑),大家就热烈讨论起来了。Force Touch这个名字有什么问题呢?太多星球大战的梗?


但是它们之间其实是有区别的!3D Touch能感应压力的大小,而Force Touch只可以检测到发生了重按。


虽然这个改变看起来并不十分重要,它允许开发者获得按压iPhone的压力明确值。这个实例app叫做 Gravity ,使得你的iPhone变为一个可以显示数字等级的Force Touch.虽然由于不明原因,苹果拒绝这样做,但是这个想法是有趣的,演示了3D Touch是如何工作的,让我们开始这个app吧!


Let’s Get Started

首先,我们下载项目的模板。基本上,这只是一个空的 Single View iPhone应用。我创建了app的界面(UILabels 和 a UIImage),并且连接了IBOutlets到ViewController.swift中。


ios 压测 苹果手机测压力_3D


我们设计的app十分简单:一个包含两个label的view controller,其中一个是标题,另一个显示iPhone的压力百分比。


我们开始编码吧!在iPhone 6s 和 6s Plus,UITouch对象有两个CGFloat类型的属性,分别是force和maximumPossibleForce。force代表触摸的压力的大小,1.0表示正常的触摸。MaximumPossibleForce表示一次触摸的最大压力。


当用户触摸屏幕,touchesBegan会被调用,然后是touchesMoved(如果用户在屏幕上移动它的手指),再然后就要根据实际情况,可能是touchedCancelled或者TouchesEnded被调用。根据我们的需求,只有touchesMoved方法才是我们需要的。TouchesMoved有两个参数:touches和event。touches是一个存储了UITouch对象的Set对象(无顺序集合),touches中至少含有一个UITouch对象,但是谨慎起见,强烈建议先判断 touches.first是否为nil,在ViewController.swift中插入下面的方法:

override func touchesMoved(touches: Set<UITouch>, withEvent event: UIEvent?) {
    if let touch = touches.first {
        if #available(iOS 9.0, *) {
            if traitCollection.forceTouchCapability == UIForceTouchCapability.Available {
                // 3D Touch capable
            }
        }
    }
}


在if子句中,我们先判断设备是否支持3D Touch。如果你只是凭兴趣玩一下,这部分是可选的。如果你要将app发布到AppStore,一定要执行判断,因为iPhone6等老设备是不支持3D Touch的。


注意我同时检测了设备是否在iOS9.0及更新的版本中运行。我使用了Swift2.0的#available语句。如果你想进一步学习Swift 2.0的新特性,我建议你阅读这篇博客。如果你项目的deployment target是9.0或者以上版本,版本检测也是可选的。


为了获得压力百分比,只需要简单地将压力除以可能达到的最大压力(touch.maximumPossibleForce),然后更新label的文字:

override func touchesMoved(touches: Set<UITouch>, withEvent event: UIEvent?) {
    if let touch = touches.first {
        if #available(iOS 9.0, *) {
            if traitCollection.forceTouchCapability == UIForceTouchCapability.Available {
                // 3D Touch capable
                let force = touch.force/touch.maximumPossibleForce
                forceLabel.text = "\(force)% force"
            }
        }
    }
}



如果你在iPhone 6s/6s Plus上运行项目,当你按压屏幕时,会显示压力百分比。然而你可能希望以“克”为单位来显示。根据Ryan McLeod的测试,感器的计量范围的最大值是 385g(约等于3.8牛)。通过简单的计算,你可以将 %压力转化为克。我们只需要将压力百分比乘以385。对于超过385的数据,我们只要改变label的文字,告诉用户:“385+ 克”。





现在,更新代码片段:

override func touchesMoved(touches: Set<UITouch>, withEvent event: UIEvent?) {        
    if let touch = touches.first {
        if #available(iOS 9.0, *) {
            if traitCollection.forceTouchCapability == UIForceTouchCapability.Available {
                if touch.force >= touch.maximumPossibleForce {
                    forceLabel.text = "385+ grams"
                } else {
                    let force = touch.force/touch.maximumPossibleForce
                    let grams = force * 385
                    let roundGrams = Int(grams)
                    forceLabel.text = "\(roundGrams) grams"
                }
            }
        }
    }
}



太酷了!你已经完成了一个数字压力app。


ios 压测 苹果手机测压力_压力_02


现在,当手指移开屏幕后,app并没有复位,你可以实现touchesEnded方法来将label复位。

override func touchesEnded(touches: Set<UITouch>, withEvent event: UIEvent?) {
    forceLabel.text = "0 gram"
}


Home Screen Quick Actions

3D Touch的另一个用法是在Home页的Quick Actions(快速操作)。Quick actions 是用户直接跳转到app某些页面的一个快捷方式。简单地重压app的图标,你可以看见一个快捷菜单。在3D Touch的介绍文档,展示了Twitter 、Instagram,以及一些其它苹果app使用了这一特性。

ios 压测 苹果手机测压力_ios 压测_03


我们就为我们的app添加Quick Action,来以蓝色的背景打开它而不是现在的白色。为了添加Quick Actions,打开项目的info.plist 文件。在文件中,添加一个’UIApplicationShortcutItems’数组。数组的每一个元素都是一个dictionary,而dictionary包含一个Quick Action的一些属性:


在这个数组中,我们定义了4个属性来配置 “OpenBlue” Quick Action。你的 info.plist应该是这样子的:


ios 压测 苹果手机测压力_3D_04


说明一下,我使用’$(PRODUCT_BUNDLE_IDENTIFIER)’而不是 ‘com.appcoda.Scale’或者其它bundle ID,是为了安全着想:如果我将Bundle ID改为’General’,会影响整个项目,每个用到Bundle ID的地方都需要修改。在info.plist文件中,你会发现Bundle Identifier key也是使用‘$(PRODUCT_BUNDLE_IDENTIFIER)’来描述项目的Bundle ID。


最后需要做的是,当用户启动时,实现Quick action。在AppDelegate.swift中的performActionForShortcutItem方法中处理快捷事件。当触发了Quick Action,这个方法会被调用。所以需要在此方法中处理快捷事件。



func application(application: UIApplication, performActionForShortcutItem shortcutItem: UIApplicationShortcutItem, completionHandler: (Bool) -> Void) {
    
    // Handle quick actions
    completionHandler(handleQuickAction(shortcutItem))
    
}



你需要根据quick action的成功或失败,调用completionHandler,并且传递一个布尔值。这里我们创建一个单独的函数handleQuickAction来处理快捷事件。推荐使用UIApplicationShortcutItemType枚举,来表示 Quick Action的事件。像下面这样声明枚举并且实现handleQuickAction方法:

enum Shortcut: String {
    case openBlue = "OpenBlue"
}
 
func handleQuickAction(shortcutItem: UIApplicationShortcutItem) -> Bool {
    
    var quickActionHandled = false
    let type = shortcutItem.type.componentsSeparatedByString(".").last!
    if let shortcutType = Shortcut.init(rawValue: type) {
        switch shortcutType {
        case .openBlue:
            self.window?.backgroundColor = UIColor(red: 151.0/255.0, green: 187.0/255.0, blue: 255.0/255.0, alpha: 1.0)
            quickActionHandled = true
        }
    }
    
    return quickActionHandled
}



这一切都十分简单。如果你现在quick actions使用运行app,它的背景是蓝色的!





ios 压测 苹果手机测压力_压力_05




One Thing to Keep in Mind

但是你还需要记住一件事,使用Quick Action正常启动app和从后台打开是有区别的。当一个app正常启动willFinishLaunchingWithOptions和didFinishLaunchingWithOptions方法会被调用,但从后台打开时,只会执行performActionForShortcutItem方法。

ios 压测 苹果手机测压力_touch_06



在didFinishLaunchingWithOptions方法中,我们有一行代码来设置背景色为白色。在直接点击图标启动app时,这行代码会被执行。



func application(application: UIApplication, didFinishLaunchingWithOptions launchOptions:
    [NSObject: AnyObject]?) -> Bool {
    // Override point for customization after application launch.
    self.window?.backgroundColor = UIColor.whiteColor()
        
    return true
}


这就是问题所在:当你通过quick action启动时,willFinish,didFinish,然后是performActionForShortcutItem会被执行,所以背景是由白色变为蓝色。很明显,你并不希望在通过快捷启动app启动时,将背景设为白色。



为了解决这个问题,我们需要在didFinishLaunchingWithOptions方法中执行判断:


func application(application: UIApplication, didFinishLaunchingWithOptions launchOptions:
    [NSObject: AnyObject]?) -> Bool {
    print("didFinishLaunchingWithOptions called")
    var isLaunchedFromQuickAction = false
   
    // Check if it's launched from Quick Action
    if let shortcutItem = launchOptions?[UIApplicationLaunchOptionsShortcutItemKey] as? UIApplicationShortcutItem {
        
        isLaunchedFromQuickAction = true
        // Handle the sortcutItem
        handleQuickAction(shortcutItem)
    } else {
        self.window?.backgroundColor = UIColor.whiteColor()
    }
    
    // Return false if the app was launched from a shortcut, so performAction... will not be called.
    return !isLaunchedFromQuickAction
}


你可以通过UIApplicationLaunchOptionsShortcutItemKey来确定app是否是通过快速启动打开的,然后决定是否调用handleQuickAction来改变背景颜色。


因为我们已经在didFinishLaunchingWithOptions中处理快捷启动,我们不需要再调用performActionForShortcutItem来处理。所以,最后我们返回一个false值,告诉系统不需要执行performActionForShortcutItem方法。


你可以测试你的app了,快捷启动应该可以完美运行!




Conclusion


3D Touch是一个好用且有意思的功能,但你需要知道,还不是所有设备都支持3D Touch。


你阅读完这篇博文后,你应该能在你的app中添加Quick Actions,知道怎么检测按压力度了。你可以下载完整的源码作为参考。(晚点上传Objective-c版本)