第十二章——子类化UITableViewCell【译】

UITableView 显示一个 UITableViewCell 对象的列表。 对于许多应用程序,具有 textLabeldetailTextLabelimageView 的基本 cell 就足够了。 但是,当您需要具有更多细节或不同布局的单元格时,可以将 UITableViewCell 子类化。

在本章中,您将创建一个名为 ItemCellUITableViewCell 子类,将更有效地显示 Item 实例。 每个 cell 都将显示 Item 的名称,美元值 和 序列号(图12.1)。

图12.1 具有子类化的表视图 cell 的 Homepwner

您可以通过将子视图添加到其 contentView 来自定义 UITableViewCell 子类的外观。 将子视图添加到 contentView 而不是直接到 cell 本身很重要,因为 cell 会在某些时候调整其 contentView 的大小。 例如,当表视图进入编辑模式时,contentView 将自己调整大小,为编辑控件腾出空间(图12.2)。 如果您将子视图直接添加到 UITableViewCell,则编辑控件会掩盖子视图。 进入编辑模式时,cell 不能调整其大小(它必须保持表视图的宽度),但是 contentView 可以。

图12.2 标准和编辑模式下的表视图 cell 布局

创建 ItemCell

创建一个名为 ItemCell 的新 Swift 文件。 在 ItemCell.swift 中,将 ItemCell 定义为 UITableViewCell 子类。

import Foundation
import UIKit

class ItemCell: UITableViewCell {

}

最简单的配置 UITableViewCell 子类的方法是通过故事板。 在第10章中,您看到表视图控制器的故事板具有 Prototype Cells 部分。 这是您将为 ItemCell 制作内容的地方。

打开 Main.storyboard 并在文档大纲中选择 UITableViewCell。 打开属性检查器,将 Style 更改为 Custom,并将 Identifier 更改为 ItemCell

现在打开它的身份检查器,图标


。 在 Class 字段中,输入 ItemCell(图12.3)。

图12.3更改 cell 的类

将 prototype cell 的高度更改为约 65 点。 您可以在画布上或通过选择表视图 cell 并从其大小检查器更改 Row Height 来进行更改。

ItemCell 将显示三个文本元素,因此将三个 UILabel 对象拖动到单元格上。 配置如图12.4所示。 将底部标签的文字设为稍微较小的灰色字体。 确保标签一点也不重叠。

图12.4 ItemCell 的布局

对这三个标签添加约束如下。

  1. 选择左上角的标签并打开 Add New Constraints 菜单。 选择顶部和左侧支柱,然后单击 Add 2 Constraints
  2. 您希望左下角的标签始终与左上角的标签对齐。 右键从左下角的标签拖动到左上角的标签,然后选择 Leading
  3. 选中左下方的标签,打开 Add New Constraints 菜单,选择底部的支柱,然后单击 Add 1 Constraint
  4. 选择正确的标签,然后右击从右侧的控件拖动到其父视图。 选择 Trailing Space to Container MarginCenter Vertically in Container
  5. 选择左下角的标签并打开其尺寸检查器。 查找 Vertical Content Hugging Priority 并将其降低到 250.将 Vertical Content Compression Resistance Priority 降低到749.您将在第13章中了解这些自动布局属性的功能。
  6. 您的边框可能显示在错误的位置,因此请选中三个标签,然后单击 Update Frames 按钮。

暴露 ItemCell的属性

要通过 ItemsViewController 来配置 TableView(_:cellForRowAt :) 中的 ItemCell 的内容,cell 必须有暴露这三个标签的属性。 这些属性将通过 Main.storyboard 中的 outlet 连接来设置。

那么下一步就是在 ItemCell 上为每个子视图创建和连接 outlet。

打开 ItemCell.swift 并为 outlet 添加三个属性。

import UIKit

class ItemCell: UITableViewCell {

  @IBOutlet var nameLabel: UILabel!
  @IBOutlet var serialNumberLabel: UILabel!
  @IBOutlet var valueLabel: UILabel!

}

你将连接三个视图的 outlet 到 ItemCell。 当在书中较早连接 outlet 时,您可以从故事板中的视图控制器拖动到相应的视图。 但连接 ItemCell 的 outlet 不是控制器上的 outlet。 它们是另一个视图中的 outlet:自定义的 UITableViewCell 子类。

因此,要连接 ItemCell 的 outlet,您将要连接到 ItemCell

打开 Main.storyboard。 右击文档轮廓中的 ItemCell,并使三个 outlet 连接如图12.5所示。

图12.5连接 outlet

使用 ItemCell

让我们在屏幕上看到你的自定义单元格。 在 ItemViewControllertableView(_:cellForRowAt :) 方法中,您将在表中的每一行显示一个 ItemCell 的实例。

现在您正在使用自定义 UITableViewCell 子类,表视图需要知道每行的高度。 有几种方法可以实现此目的,但最简单的方法是将表视图的 rowHeight 属性设置为常量值。 本章稍后将会看到另一种方式。

打开 ItemsViewController.swift 并更新 viewDidLoad() 来设置表视图 cell 的高度。

override func viewDidLoad() {
  super.viewDidLoad()

  // Get the height of the status bar
  let statusBarHeight = UIApplication.shared.statusBarFrame.height

  let insets = UIEdgeInsets(top: statusBarHeight, left: 0, bottom: 0, right: 0)
  tableView.contentInset = insets
  tableView.scrollIndicatorInsets = insets

  tableView.rowHeight = 65
}

请注意,您已经使用表格视图(使用故事板中的 prototype cell)注册了 ItemCell,您可以要求表格视图将具有标识符 “ItemCell” 的 cell 列出。

在 ItemsViewController.swift 中,修改 tableView(_:cellForRowAt :)

override func tableView(_ tableView: UITableView, cellForRowAt indexPath: NSIndexPath) -> UITableViewCell {
  // Get a new or recycled cell

  let cell = tableView.dequeueReusableCell(withIdentifier: "UITableViewCell",for: indexPath)

  let cell = tableView.dequeueReusableCell(withIdentifier: "ItemCell", for: indexPath) as! ItemCell

  // Set the text on the cell with the description of the item
  // that is at the nth index of items, where n = row this cell
  // will appear in on the tableview
  let item = itemStore.allItems[indexPath.row]

  //cell.textLabel?.text = item.name
  //cell.detailTextLabel?.text = "$\(item.valueInDollars)"

  // Configure the cell with the Item
  cell.nameLabel.text = item.name
  cell.serialNumberLabel.text = item.serialNumber
  cell.valueLabel.text = "$\(item.valueInDollars)"

  return cell
}

首先,更新重用标识符以反映您的新子类。 该方法结束时的代码是相当明显的——对于单元格上的每个标签,将其 text 设置为相应的 Item 的某些属性。

构建并运行应用程序。 现在,新的 cell 加载了它们的标签,并填充了每个 Item 的值。

动态单元格高度

目前,cell 的固定高度为 65 点。 更好的是根据 cell 的内容来修改它的高度。 这样,如果内容更改,表视图 cell 的高度可以自动更改。

您可以通过自动布局实现这一目标,就像您已经猜到的那样。 UITableViewCell 需要具有垂直约束,以确定 cell 的高度。 目前,ItemCell 没有足够的限制。 您需要在两个左侧标签之间添加一个约束,这些约束修正了它们之间的垂直间距。

打开 Main.storyboard。 右击从 nameLabel 拖动到 serialNumberLabel 并选择 Vertical Spacing

现在打开 ItemsViewController.swift 并更新 viewDidLoad() 来告诉表视图它应该基于约束来计算 cell 高度。

override func viewDidLoad() {
  super.viewDidLoad()

  // Get the height of the status bar
  let statusBarHeight = UIApplication.shared.statusBarFrame.height

  let insets = UIEdgeInsets(top: statusBarHeight, left: 0, bottom: 0, right: 0)
  tableView.contentInset = insets
  tableView.scrollIndicatorInsets = insets

  //tableView.rowHeight = 65
  tableView.rowHeight = UITableViewAutomaticDimension
  tableView.estimatedRowHeight = 65
}

UITableViewAutomaticDimensionrowHeight 的默认值,所以这其实没有必要添加,这里是为了让你理解。 在表视图中设置 estimateRowHeight 属性可以提高性能。设置此属性可以节省某些性能成本,直到用户开始滚动而不是表视图加载时才请求每个 cell 的高度。

构建并运行应用程序。 该应用程序将看起来像以前一样。 在下一节中,您将了解一种称为 动态类型(Dynamic Type) 的技术,该技术适用于自动调整大小的表视图单元格。

动态类型

创建一个能吸引每一个人的界面可能是很难的。 有些人喜欢更紧凑的界面,因此他们可以一次看到更多的信息。 其他人可能希望能够一目了然地轻松查看信息,也许他们的视力较差。 总之:人们有不同的需求。 良好的开发人员力求使应用程序能够满足这些需求。

动态类型(Dynamic Type) 是一种通过提供专门设计的文本样式来帮助实现这一目标的技术,这些文本样式被优化为可读性。 用户可以从 Apple 的 设置(Settings) 应用程序中选择七种首选文本大小之一(另外还有一些可扩展性部分中的其他更大的尺寸),支持动态类型的应用程序的字体可以适当地缩放。 在本节中,您将更新 ItemCell 以支持动态类型。 图12.6 显示了以最小和最大用户可选动态类型大小呈现的应用程序。

图12.6支持动态类型的ItemCell

动态类型系统以文本样式为中心。 当为给定的文本样式请求字体时,系统将考虑与文本样式相关联的用户首选文本大小以返回适当配置的字体。 图12.7显示了不同的文字样式。

图12.7 文字样式

打开 Main.storyboard。 让我们更新标签以使用文本样式,而不是固定的字体。 选择 nameLabelvalueLabel 并打开属性检查器。 点击 Font 右边的文本图标。 对于 Font,选择 Text Styles - Body (图12.8)。 对 serialNumberLabel 重复相同的步骤,选择 Caption 1 文本样式。

图12.8 更改文本样式

现在我们来更改首选的字体大小。 您可以通过 设置(Settings) 应用程序来执行

构建并运行应用程序。 从模拟器的 Hardware 菜单中,选择 Home。 接下来,在模拟器的主屏幕上,打开 Settings 应用程序。 选择 General,然后选择 Accessibility,然后选择 Larger Text。 (在实际的设备上,也可以通过 Display & BrightnessText SizeSettings 中访问此菜单。)将滑块一直向左拖动以将字体大小设置为最小值(图12.9)。

图12.9 文字大小设置

构建并运行应用程序。 (如果您切换回应用程序,使用任务切换器或通过主屏幕,将不会看到更改,您将在下一节中进行修复。)将一些 item 添加到表视图中,您将看到新的较小的字体尺寸。

响应用户更改

当用户更改首选文本大小并返回到应用程序时,表视图将被重新加载。 不幸的是,标签不会知道新的首选文本大小。 要解决这个问题,您需要将标签根据内容大小的更改而自动调整。

打开 ItemCell.swift 并覆盖 awakeFromNib() 使标签自动调整。

override func awakeFromNib() {
  super.awakeFromNib()

  nameLabel.adjustsFontForContentSizeCategory = true
  serialNumberLabel.adjustsFontForContentSizeCategory = true
  valueLabel.adjustsFontForContentSizeCategory = true
}

从一个归档文件中加载之后,方法 awakeFromNib() 会被一个对象所调用,在这种情况下,它就是故事板文件。 到这个方法被调用的时候,所有的 outlet 都有值,可以被使用。

构建并运行应用程序并添加几行。 进入 Settings,将首选阅读尺寸更改为最大尺寸。 与以前不同,现在可以通过打开任务切换器或通过主屏幕切换回 Homepwner,并且表视图将更新以反映新的首选文本大小。

青铜挑战:cell 颜色

如果值小于50,则更新 ItemCell 以将 valueInDollars 显示为绿色,如果值大于或等于 50,则显示为红色。

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

推荐阅读更多精彩内容