iOS 11:几点值得关注的 UIKit 改进

UIStackView

这个控件深受大家的喜爱,只有一个微调——但关键是,这正是它需要的微调。stack view 复杂但灵活——但考虑到它强大的能力和神奇的 AutoLayout,还是有一件事没有做好:已排列子视图间的可变间距。

在 iOS 11 中有所不同,只要一个新方法就可以轻松搞定:

let view1 = UIView()
let view2 = UIView()
let view3 = UIView()
let view4 = UIView()

let horizontalStackView = UIStackView(arrangedSubviews: [view1, view2, view3, view4])
horizontalStackView.spacing = 10

// view3 后面再多加 10 点间距
horizontalStackView.setCustomSpacing(10, after: view3)

之前我和 stack view 小伙伴已经一起遇到这个场景足够多次了,有点小生气,因为在实现 UIStackView 时,如果要应用一些 padding 的话,却发现只能全部拆下来或添加不符合直觉的”spacer“ view(这可能是 API 推出时的历史遗留问题)。

如果你需要让它产生动画或为其它 UI 考虑而腾出空间,它也可以随后被查询和参数化:

let currentPadding = horizontalStackView.customSpacing(after: view3)

UITableView

社区里曾经讨论是否要用 UITableViewFlowLayout 或类似的东西来替代 table view。在 iOS 11 中,Apple 已经重申,这两个控件是独立且分离的工具,应在开发者的大脑中占据同等地位。

对于新手来说,table view 现在默认你要开启自动行高,就不再需要这一步了:

tv.estimatedRowHeight = UITableViewAutomaticDimension?

这既是福音也是噩耗,它的确解决了很多令人头痛的问题,但也带来了一些问题(丢帧、content inset 计算、滑块解释性跳动等等)。

需要注意的是,如果你不需要这个行为——的确会有理由不需要,可以回溯到石器时代的 iOS 10,像这样

tv.estimatedRowHeight = 0

还有一种改进的方式来启用自定义操作,当用户在 cell 上向左向右滑时,或者更准确的说,从 leading 或 trailing 侧滑动时,这些 contextual 的 action 相对于目前 iOS 8 里增加的 UITableViewRowAction,是更为强大的 API:

let itemNameRow = 0
func tableView(_ tableView: UITableView, leadingSwipeActionsConfigurationForRowAt indexPath: IndexPath) -> UISwipeActionsConfiguration?
{
    if indexPath.row == itemNameRow
    {
        let editAction = UIContextualAction(style: .normal, title:  "Edit", handler: { (ac:UIContextualAction, view:UIView, success:(Bool) -> Void) in
             //做编辑

             //handler 将 context 重置为正常状态,true 是已完成的视觉指示
            success(true)
         })
        editAction.image = UIImage(named: "edit")
        editAction.backgroundColor = .purple
        let copyAction = UIContextualAction(style: .normal, title: "Copy", handler: { (ac:UIContextualAction, view:UIView, success:(Bool) -> Void) in
                 //做拷贝
                success(true)
        })
        return UISwipeActionsConfiguration(actions: [editAction, copyAction])
     }
    return nil
}

对于 trailing 动作,delegate 方法表现地完全相同。还有一个好的修改是,有一个属性可以用于当用户向左或向右滑到底时,发生”默认“滑动动作,就像”邮件“里删除邮件那样:

let contextualGroup = UISwipeActionsConfiguration(actions: [editAction, copyAction])
contextualGroup.performsFirstActionWithFullSwipe = true
return contextualGroup

这里的默认值是 true,所以如果你不想要就禁用它——尽管这好像打破了规则。

为了不过时,table view 还偷了它表弟提倡的方法,它现在也可以有 batch updates 了:

let tv = UITableView()
tv.performBatchUpdates({ () -> Void in
    tv.insertRows/deleteRows/insertSections/removeSections
}, completion:nil)

UIPasteConfiguration

这个改进小巧好用,最初在 ”What’s New in Cocoa Touch“ session 里引起了我的兴趣。既为了粘贴操作,也为了支持拖拽数据传递,现在每个 UIResponder 都有了粘贴配置(paste configuration)属性:

self.view.pasteConfiguration = UIPasteConfiguration()

这个类主要从粘贴操作或拖拽中检测传入的数据,然后进行过滤,通过提供的同一类型标识符(uniform type identifiers),只接受你指定想处理的东西:

//表示这个类已了解想要的 UTI
UIPasteConfiguration(forAccepting: UIImage.self)
//或者也可以在更细节的层面指定
UIPasteConfiguration(acceptableTypeIdentifiers:["public.video"])

还有,这些 UTI 是可修改的,如果 App 流程允许,可以即时更改它们:

let pasteConfig = UIPasteConfiguration(acceptableTypeIdentifiers: ["public.video"])
//带来更多数据
pasteConfig.addAcceptableTypeIdentifiers(["public.image, public.item"])
//或是添加一个已经采用 NSItemProviderReading 的实例
pasteConfig.addTypeIdentifiers(forAccepting: NSURL.self)

因为在 iOS 11 里所有 UIResponder 都符合UIPasteConfigurationSupporting,现在就可以轻松对系统或用户拖拽或粘贴给我们的结果进行处理:

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

推荐阅读更多精彩内容