iOS下的自定义键盘(译)

(本文来自《Custom Keyboard》
自定义键盘为那些希望体验更新颖的输入法或者需要用到iOS不支持的语言的用户,提供了替代系统键盘的备选。自定义键盘的核心功能很简单:响应按键、手势或其它输入事件,并提供转换后的文本字串,并将该字串插入到当前的光标位置。

开始阅读前
请先确认你需要开发的的确是系统范围的自定义键盘。如果只是希望在你的app内部提供可自定义的键盘,可以到《Custom Views for Data Input 》《Text Programming Guide for iOS》了解自定义输入视图输入辅助视图的相关内容,iOS SDK为此提供了更好的备选方案。

当用户选择了自定义键盘,该键盘即成为每个app的键盘。因此,你创建的键盘必须包含一些基本的功能。其中最重要的是,必须允许用户切换到其它键盘。

理解用户对键盘的预期

要理解用户对于自定义键盘的预期,可以系统键盘为标杆——它反应灵敏且高效。它不会用垃圾信息或请求打断用户输入。如果你提供了需要用户交互的功能,应该把它们放到键盘的app里,而不是键盘上。

iOS用户预期的键盘功能

每个自定义键盘都必须提供的iOS用户所预期的键盘功能是:切换到其它键盘的方法。在系统键盘中,有一个小地球的按键用来完成此功能。iOS 8也提供了专门的API用于切换到下一个键盘,可以参见《提供一种切换到其它键盘的方法》

系统键盘会根据当前文本输入对象的UIKeyboardType属性,展现与之匹配的键盘布局。如果当前的输入对象需要输入邮箱,系统键盘的句号建就会变化:长按会冒出一些顶级域名的后缀作为候选。你在设计自己的键盘布局时也应当考虑到当前的输入对象属性。

iOS用户还期望自动大写:在一个标准的文本输入区域,对于大小写敏感的语言来说,应当让句首的字母自动大写。

这类功能列出如下:

  • 对输入对象的属性考虑适当的键盘布局
  • 自动纠错和建议
  • 自动大写
  • 两个空格后自动添加句号
  • Caps lock键的支持
  • 键帽上的美观
  • 对于象形文字的多层转换

你可以自行决定是否实现这些功能;系统并没有为这些功能提供专用的API,在自己的输入法中提供这些功能可以让你的产品更有竞争力。

系统键盘中的哪些功能不适用于自定义键盘

自定义键盘不能访问在系统设置中的通用键盘设置数据(设置 > 通用 > 键盘),比如自动大写、使Caps Lock可用。自定义键盘也不能访问字典还原信息(设置 > 通用 > 还原 > 还原键盘词典)。要满足你用户的灵活性要求,你应该创建一个标准的设置bundle,这个话题在《偏好和设置编程指南》《实现iOS设置Bundle》中有讨论。这样你的自定义键盘的设置就会出现在系统设置的键盘区域。

还有一些文本输入对象,自定义键盘是不能在其上进行输入的。首先就是任何安全相关的文本输入对象。这类文本输入对象设置了其secureTextEntry属性为YES,在其上的输入内容将呈现为圆点。

当用户在密码框里输入时,系统会临时用系统键盘来替代自定义键盘。当用户在非密码框里输入时,自定义键盘又会恢复回来。

自定义键盘也无权在拨号输入的位置出现,例如通讯录的电话号码输入区域。对于这类输入对象,键盘是由运营商指定的一个数字/字母的小集合组成,并且具备如下属性:
UIKeyboardTypePhonePad
UIKeyboardTypeNamePhonePad

当用户点击拨号输入对象时,系统将临时用系统键盘替换掉你的自定义键盘。当用户再点击其它标准输入对象时,自定义键盘又会恢复回来。

app的开发者可以选择在app内部不使用自定义键盘。例如银行类app,或者必须遵守美国HIPAA隐私规则的app,可以这么干。这类app实现来自UIApplicationDelegate协议的application:shouldAllowExtensionPointIdentifier:方法,并返回NO,以达到使用系统键盘的效果。

由于自定义键盘只能绘制其UIInputViewController对象内的主视图,在它上面不能选择文字。选择文字是使用键盘的应用程序控制的。如果app提供了编辑菜单(如剪切、拷贝和粘贴),键盘是无权访问它的。自定义键盘不能提供在光标位置的自动inline纠错能力。

在iOS8.0下,如所有扩展app一样,自定义键盘不能访问麦克风,因此不能实现语音输入。

最后,显示插图不能超过键盘的主视图上边缘,系统键盘可以,但自定义键盘不行。如下图,可以发现自定义键盘和系统输入法的差别:
按键插图不能超越上边缘

自定义键盘API

本节将给出开发自定义键盘的快速入门。如下图,它展示了键盘运行过程中一些重要的对象,以及它们在开发流程中的的位置:


自定义键盘的基本结构

自定义键盘模板(在iOS“Application Extension”目标模板组)包含一个UIInputViewController的子类,它是你开发的键盘的主视图控制器。该模板包含键盘所必需的“下一个键盘”按钮的实现,它调用了UIInputViewController类的advanceToNextInputMode方法。如上图所示,可以在输入视图控制器的主视图(在其inputView属性)中添加子视图、控制器以及手势识别器等。对于其它类型的扩展应用,在目标上并不存在窗体,因此也就没有根视图控制器了。

在模板的Info.plist文件中有预先配置好的键盘所需要的最基本的值。参见其中的NSExtensionAttributes字典关键字,配置一个键盘的关键字在《配置自定义键盘的Info.plist文件》中有介绍。

默认,键盘不能访问网络,不能和它的app共享容器。如果要具备这种能力,必须要将Info.plist文件中RequestsOpenAccess的值置为YES。这需要扩展键盘的沙盒,在《设计用户信任》中有介绍相关内容。

一个输入视图控制器遵从各种与文本输入对象内容交互的协议:

[self.textDocumentProxy insertText:@"hello "]; // Inserts the string "hello " at the insertion point
[self.textDocumentProxy deleteBackward];       // Deletes the character to the left of the insertion point
[self.textDocumentProxy insertText:@"\n"];     // In a text view, inserts a newline character at the insertion point
NSString *precedingContext = self.textDocumentProxy.documentContextBeforeInput;
然后就可以删除你指定的文字区域了,比如单个字符还是空格后的所有字符。如果要按照语义执行删除,比如一个单词、句子、还是一个段落,可以使用[《 CFStringTokenizer Reference》](https://developer.apple.com/reference/corefoundation/cfstringtokenizer-rf8)中描述的函数,注意每个语种的语义规则是不同的。
  • 为了控制光标所在位置的操作,比如支持向前删除文字,需要调用UITextDocumentProxy协议中的adjustTextPositionByCharacterOffset:方法。比如向前删除一个字符,代码如下:
- (void) deleteForward {
    [self.textDocumentProxy adjustTextPositionByCharacterOffset: 1];
    [self.textDocumentProxy deleteBackward];
}
  • 通过实现UITextInputDelegate协议中的方法,可以响应当前输入文本对象的一些变化,比如内容变化以及用户触发的光标位置的变化。

为了展现与当前文本输入对象适配的键盘布局,需要参照该对象的UIKeyboardType属性,根据每种你的键盘所能支持的属性,变化布局内容。

在自定义键盘中,有两种方式来支持多语言:

  • 为每个语言创建一个键盘,每个键盘都作为向容器app添加的独立的Target
  • 创建一个多语言键盘,动态切换当前语言。可以使用UIInputViewController类的primaryLanguage属性来动态切换语言。

根据你要支持的语言数量以及你想提供的用户体验,你可以从上面选择最合适的方案。

每种自定义键盘(需要RequestsOpenAccess)都可以通过UILexicon类访问自动纠错的词典。通过使用该类,并结合你自己的词典设计,可以在用户输入过程中为他提供输入建议和自动纠错。UILexicon对象包含来自如下源的单词:

  • 来自用户通讯录的人名和姓
  • 在 设置 > 通用 > 键盘 > 快捷方式(文本替换) 列表
  • 通用词典

你可以使用自动布局来调整你的自定义键盘主视图的高度。默认情况下,自定义键盘会根据屏幕尺寸以及设备方向,和系统键盘的尺寸保持一致。自定义键盘的宽度通常与屏幕当前宽度一致。修改自定义键盘主视图的高度约束即可修改其高度。

下面的代码展示如何定义和添加约束:

CGFloat _expandedHeight = 500;
NSLayoutConstraint *_heightConstraint = 
    [NSLayoutConstraint constraintWithItem: self.view 
                                 attribute: NSLayoutAttributeHeight 
                                 relatedBy: NSLayoutRelationEqual 
                                    toItem: nil 
                                 attribute: NSLayoutAttributeNotAnAttribute 
                                multiplier: 0.0 
                                  constant: _expandedHeight];
[self.view addConstraint: _heightConstraint];

注意
在 iOS8.8下,你可以在主视图画到屏幕之后的任何时间里调整键盘高度。

自定义键盘的开发关键

自定义键盘开发有两个关键点:

  • 信任。 自定义键盘能访问用户输入的内容 ,因此在键盘和用户间建立信任非常关键。
  • “下一个键盘”键。 通过键盘界面必须能让用户能切换到下一个键盘。

为用户信任所做的设计

作为自定义键盘的开发者,你首先应当考虑的是如何建立和维护用户信任。你要理解隐私策略的最佳实践并知道如何实现它才能很好地践行。

注意
本节为你创建自定义键盘提供相关的开发手册,该手册要求尊重用户隐私。了解iOS编程要求,请阅读应用商店审核手册iOS人机交互手册iOS开发许可协议,请参见苹果的《应用审核支持》《支持用户隐私》《iOS应用编程指南》

对于键盘,如下三个方面对于建立和维护用户信任至关重要:
按键数据的安全。 用户希望他们的敲键会落在文档以及输入区域内,而不是上传到服务器或者用于其他不明目的。
最小化合理利用其它用户数据。 如果你的键盘还需要使用其他用户数据,例如定位服务或者通讯录,你有义务解释这给用户带来的好处是什么。
准确。把输入事件转换成文本要求精准,这本身虽然不是一个隐私话题,但他会影响到信任:每次文字转换需要体现出你的代码的精准。

在信任的开发设计过程中,首先考虑的是是否要获取open access权限。尽管开启了open access权限能给自定义键盘开发带来极大便利,但这也增加了你作为开发者的责任。下面是标准的open access的能力和隐私考虑:

Open Access 能力和限制 隐私考虑
Off(default) ·键盘可以执行所有基本键盘的职责

·可以访问通用词典以支持自动纠错和输入建议
·访问设置里的快捷短语
·不与containing应用共享容器
·不访问键盘容器以外的文件系统
·不访问键盘容器以外的文件系统
·不能直接或间接访问iCloud或游戏中心或应用内购买 | 用户了解按键仅仅被发送到当前使用键盘的应用里 |
| On | ·具备非联网自定义键盘的所有能力
·在用户许可情况下可以访问位置服务和通讯录
·键盘和containing app可以访问共享容器
·键盘可以为服务器侧处理过程发送按键或其他输入事件
·containing app自动纠错字典提供编辑界面
·通过containing app键盘可以使用iCloud来保证自动纠错词典和设置的更新
·通过containing app,键盘可以参与到游戏中心和应用内购买
·如果键盘支持移动设备管理(MDM),它可与被管理的应用共同工作 | ·用户了解键盘开发者会利用按键数据
·你必须遵守有联网能力的键盘开发手册iOS开发许可协议,可参见《应用审核支持》 |

如果你的自定义键盘不需要open access权限,系统确保敲键信息不会被发送给你的键盘以及别的地方。如果只想提供一般的键盘功能,请不要给键盘配备联网能力。由于有沙盒限制,不联网的键盘一定是满足苹果的数据隐私手册并能获得用户信任的。

开启open access权限(如上所述,可以在Info.plist文件中配置),能给你的开发带来更多可能性,同时也带来更多的责任。

注意
向应用商店提交一个open-access的键盘必须遵守苹果《应用审核支持》中的相关条款。

每一个与open access相关的功能都需要你履行相应的责任,应当最大限度地尊重用户数据,不得用于与用户输入无关的其他任何目的。下表列出了open access带来的好处以及开发者需承担的责任:

能力 用户利益示例 开发者责任
与containing app共享容器 为键盘的自动纠错词典管理UI界面 要考虑到自动纠错数据属于用户隐私。不要把他发到你的服务器,用作与输入无关的用途。
把按键数据发到你的服务器 通过开发者的计算资源可以提供更好的按键处理结果和输入预测 只有为用户提供更好的输入体验之用时,才能保存按键和语音数据
基于云的自动纠错词典 把人名、地名、热点新闻加入到自动纠错词典中 不要把用户身份与输入数据关联起来,不得将用户信息用作与输入体验无关的其他目的
通讯录 把人名、地名、电话号码添加到自动纠错词典中 不得讲通讯录用作与输入体验无关的其他目的
位置服务 将附近的地名添加到自动纠错词典中 不要在后台使用位置服务,不得将位置信息发送到你的服务器并用于与输入体验无关的其他目的

一个具有open-access权限的键盘和其containing app能将按键数据发送到服务器端,通过这些数据可以为用户提供更好的输入体验。如果你使用了这些能力,当不需要这些数据的时候,请及时在服务器端删除。参见上面的表格来履行你使用open-access权限中的义务。

提供切换到其他键盘的方法

系统键盘的小地球按键用于切换到其他键盘,如下所示:


系统键盘的小地球键

你的自定义键盘必须提供类似的机制能切换到其他键盘。

注意
要通过应用审核,必须在你的键盘上提供明显允许用户切换键盘的UI标识。

调用UIInputViewController类的advanceToNextInputMode方法可以切换到其他键盘。系统会选择下一个键盘,没有能获得键盘列表的API,也没有切换到指定键盘的API。

Xcode自定义键盘模板中就已经在下一个键盘按钮上具备了advanceToNextInputMode的功能。为了提供最好的用户体验,应当把你的下一个键盘按键放在靠近系统键盘的小地球键的位置。

开始自定义键盘的开发

本节中你将学习到如何创建自定义键盘,根据你的目标配置并在iOS模拟器或物理机上把它运行起来。你还将学习到一些替代系统键盘应谨记的UI要点。

使用Xcode自定义键盘模板

创建键盘及其containing app与其他扩展应用略有不同。本节将带你领略基本键盘的开发和运行。

在一个容器app中创建键盘,步骤如下:

  1. 在Xcode中选择File > New > Project > iOS > Application选择Single View Application模板。
  2. 点击Next
  3. 填写Project Name(如CKIme),点击Next
  4. 选择要保存的位置,点击Create。这样,你就有了一个空app,该app只能完成一个简单的操作,接下来它将承载键盘。在你提交到应用商店之前,你需要完成一些有用的功能。请到应用审核支持参考应用商店审核指南
  5. 选择File > New > Target > iOS > Application Extension选择Custom Keyboard Extension,点击Next
  6. 填写Product Name(如CKbd),点击Finish
  7. 确认ProjectEmbed in Application中都显示的是容器app的名字(CKIme),点击Finish。如果弹出Activate “CKbd” scheme提示让激活键盘工程,点击Activate

接下来你可以根据需要决定是否要自定义键盘的group name,它会出现在设置中的已购买键盘列表中。

自定义键盘group name,步骤如下:

  1. 在Xcode工程导航视图中,选择容器app的Info.plist文件,
  2. 在右侧plist编辑器中,鼠标hover到Bundle name上,点“+”按钮创建一行空属性。
  3. 在Key中填写Bundle display name,回车
  4. 双击该行的Value,填写你要自定义的键盘group name。
  5. 选择File > Save保存设置。

下表汇总了在容器app和键盘app的Info.plist文件中你可以配置的UI字符串:

iOSUI字符串 Info.plist关键字
· 在系统设置的已购键盘列表中的键盘group name 在容器app的Info.plist文件中的Bundle display name
· 系统设置中的键盘名称
· 键盘换列表中的键盘名称 在键盘app的Info.plist文件中的Bundle display name

现在你可以在iOS模拟器或真机上运行该键盘,看看它目前都具备什么行为和能力吧。

运行自定义键盘并将Xcode调试器attach到它上面

  1. 在Xcode,你的view controller实现中设置一个断点(比如可以断在viewDidLoad上)。
  2. 在Xcode工具栏确保当前活动的项目为键盘项目,并对应iOS模拟器或设备。
  3. 选择菜单Project > Run,或点击Build and then run the current scheme按钮(即播放按钮)。Xcode会提示选择host app。选择一个带有输入框的,比如通讯录或Safari。
  4. 点击Run
    Xcode将运行起你指定的host app。如果这是你第一次使用键盘扩展应用,需要现在设置中添加并启用键盘:
    1. Settings > General > Keyboard > Keyboards
    2. 点击Add New Keyboard...
    3. OTHER IPHONE KEYBOARDS中点击你刚刚创建的键盘
  5. 在iOS模拟器或真机上,调出你的自定义键盘。
    点击任意可输入区域,将显示出系统键盘。按住小地球,选择你的自定义键盘。
    此时你将看到自定义键盘,但是调试器尚未attach上来。一个从模板构建而来的极简键盘仅有一个Next Keyboard按钮,点击后切换回前一个键盘。
  6. 取消你的键盘(以便在第8步中你可以再次调出键盘以命中viewDidLoad断点)
  7. 在Xcode中,选择Debug > Attach to Process > By Process Identifier(PID) or Name
    在弹出对话框中,输入你的键盘扩展应用的名字(包含空格).默认就是该扩展应用在工程导航窗口里的group name。
  8. 点击Attach
    Xcode将显示出等待attach的调试器。
  9. 在任意能输入文字的app中调出键盘。
    当你的键盘主视图开始加载时,Xcode调试器将attache到你的键盘,并命中断点。

为自定义键盘配置Info.plist文件

自定义键盘的Info.plist文件允许静态定义键盘的现式特征,包括主要语言,以及是否需要open access权限。

打开Xcode并切换到自定义键盘的target。在工程导航栏选择Info.plist文件,按文本格式呈现如下:

<key>NSExtension</key>
<dict>
    <key>NSExtensionAttributes</key>
    <dict>
        <key>IsASCIICapable</key>
        <false/>
        <key>PrefersRightToLeft</key>
        <false/>
        <key>PrimaryLanguage</key>
        <string>en-US</string>
        <key>RequestsOpenAccess</key>
        <false/>
    </dict>
    <key>NSExtensionPointIdentifier</key>
    <string>com.apple.keyboard-service</string>
    <key>NSExtensionPrincipalClass</key>
    <string>KeyboardViewController</string>
</dict>

每个关键字在App Extension Keys中都有解释。可以使用字典NSExtensionAttributes中的关键字来描述你的自定义键盘的特征和需求,如下:

IsASCIICapable - 默认为NO的布尔值。用户键盘是否可以向文档中插入ASCII字串。如果要为UIKeyboardTypeASCIICapable属性的输入对象展现单独类型的键盘,需要将该值置为YES。

PrefersRightToLeft - 默认为NO的布尔值。是否为从右到左的语种设计的的自定义键盘。

PrimaryLanguage - 默认为en-US的字串。以<语种>-<区域>的形式描述键盘的主语言。可以到http://www.opensource.apple.com/source/CF/CF-476.14/CFLocaleIdentifier.c找到对应的语种和区域。

RequestsOpenAccess - 默认为NO的布尔值。是否需要比基础键盘更大的沙盒范围。把该值置为YES将需要完全访问权限,你的键盘将获得如下能力,每个能力都伴随有相应的权限:

  • 访问定位服务,通讯录数据库,相机,每个都需要用户允许
  • 与键盘的容器app共享容器数据,以便完成比如在容器app中管理用户词库的界面的功能
  • 通过网络发送按键、输入事件之类的数据供云端处理
  • 使用UIPasteboard
  • 播放音频,包括使用playInputClick方法播放按键音
  • 访问iCloud,可以用来根据用户身份同步比如键盘设置、自定义自动纠错词典
  • 通过容器app访问游戏中心和应用内购买
  • 如果你的键盘支持移动设备管理(MDM),可以与被管理的app无缝合作

当考虑是否将这些关键字设置为YES之前,一定要先阅读《用户信任设计》,这里描述了如何尊重和保护用户数据。

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