20 View Model 架构:如何准备 UI 层的数据?
UI 是 App 的重要组成部分,因为所有 App 都必须呈现 UI,并接收用户的事件。为了让 UI 能正确显示,我们需要把 Model 数据进行转换。例如,当我们显示图片的时候,需要把字符串类型的 URL 转换成 iOS 所支持 URL 类型;当显示时间信息时,需要把 UTC 时间值转换成设备所在的时区。
不过存在一个问题,如果我们把所有类型转换的逻辑都放在 UI/View 层里面,作为 View 层的 View Controller 往往会变得越来越臃肿。 为了避免这一情况,我使用了 MVVM 模式和 RxSwift 来架构 Moments App。MVVM 模式的核心部分是 ViewModel 模块,主要用于把 Model 转换成 UI/View 层所需的数据。为了简化转换的工作,我使用了 RxSwift 的操作符(Operator)。
所以,在这一讲中,我会和你介绍下 ViewModel 模式是怎样工作的,以及如何使用 RxSwift 里常用的操作符。
ViewModel 模式的架构
首先我们以朋友圈功能为例,看看 ViewModel 模式的架构图。
View 模块负责呈现 UI,并接收用户的事件。在朋友圈功能中,MomentsTimelineViewController
负责呈现朋友圈的时间轴列表。为了正确显示该页面,我们需要为它准备好一些的数据,例如朋友的名字,朋友头像的 URL 等等,那些数据可以从 ViewModel 模块中读取。
ViewModel 模块是 MVVM 模式的核心,该模块由两个重要的协议所组成:ListViewModel
和ListItemViewModel
。其中ListViewModel
协议用于定义列表页面所需的 ViewModel,而ListItemViewModel
用于定义每一条列表项所需的 ViewModel。当他们需要读写数据时,会调用 Repository 模块。比如在朋友圈功能里面,它们都调用MoomentsRepoType
来读写数据。
ViewModel 模式的实现
有了上述的架构图,我们就可以看看 ViewModel 模块是怎样实现的。首先看一下ListViewModel
协议的定义。
protocol ListViewModel {
var hasContent: Observable<Bool> { get }
var hasError: BehaviorSubject<Bool> { get }
func trackScreenviews()
func loadItems() -> Observable<Void>
var listItems: BehaviorSubject<[SectionModel<String, ListItemViewModel>]> { get }
}
下面我们逐一介绍该协议的各个属性与方法。hasContent
属性用于通知 UI 是否有内容。例如,当 BFF 没有返回数据时,我们可以在页面上提示用户“目前还没有朋友圈信息,可以添加好友来查看更多的朋友圈信息”。
为了代码共享,我们为hasContent
属性提供了一个默认的实现,代码如下。
extension ListViewModel {
var hasContent: Observable<Bool> {
return listItems
.map(\.isEmpty)
.distinctUntilChanged()
.asObservable()
}
}
这个方法使用map
和distinctUntilChanged
操作符来把listItems
转换成 Bool 类型的hasContent
。其中map
用于提取listItems
里的数组并检查是否为空,distinctUntilChanged
用来保证只有在值发生改变时才发送新事件。
hasError
属性是一个BehaviorSubject
,其初始值为false
。它用于通知 UI 是否需要显示错误信息。
trackScreenviews()
方法用于发送用户行为数据。而loadItems() -> Observable<Void>
方法用于读取数据。
最后看一下listItems
属性。 该属性用于准备 TableView 所需的数据,其存放了类型为ListItemViewModel
的数据。ListItemViewModel
能为 TableView 的各个 Cell 提供所需数据。该协议只定义一个名为reuseIdentifier
的静态属性 ,如下所示。
protocol ListItemViewModel {
static var reuseIdentifier: String { get }
}
extension ListItemViewModel {
static var reuseIdentifier: String {
String(describing: self)
}
}
reuseIdentifier
属性作为 TableView Cell 的唯一标示,为了重用,我们通过协议扩展来为该属性提供一个默认的实现并把类型的名字作为字符串进行返回。
上述就是ListViewModel
协议的定义,接下来看它的实现结构体MomentsTimelineViewModel
。
由于MomentsTimelineViewModel
遵循了ListViewModel
协议,因此需要实现了该协议中listItems
和hasError
属性以及loadItems()
和trackScreenviews()
方法。我们首先看一下loadItems()
方法的实现。
func loadItems() -> Observable<Void> {
return momentsRepo.getMoments(userID: userID)
}
当 ViewModel 需要读取数据的时候,会调用 Repository 模块的组件,在朋友圈功能中,我们调用了MomentsRepoType
的getMoments()
方法来读取数据。
接着看看trackScreenviews()
方法的实现。在该方法里面,我们调用了TrackingRepoType
的trackScreenviews()
方法来发送用户的行为数据,具体实现如下。
func trackScreenviews() {
trackingRepo.trackScreenviews(ScreenviewsTrackingEvent(screenName: L10n.Tracking.momentsScreen, screenClass: String(describing: self)))
}
ViewModel 模块的一个核心功能,是把 Model 数据转换为用于 UI 呈现所需的 ViewModel 数据,我通过下面代码看它是怎样转换的。
func setupBindings() {
momentsRepo.momentsDetails
.map {
[UserProfileListItemViewModel(userDetails: $0.userDetails)]
+ $0.moments.map { MomentListItemViewModel(moment: $0) }
}
.subscribe(onNext: {
listItems.onNext([SectionModel(model: "", items: $0)])
}, onError: { _ in
hasError.onNext(true)
})
.disposed(by: disposeBag)
}
从代码中你可以发现,我们订阅了momentsRepo
的momentsDetails
属性,接收来自 Model 的数据更新。因为该属性的类型是MomentsDetails
,而 View 层用所需的数据类型为ListItemViewModel
。我们通过 map 操作符来进行类型转换,在转换成功后,调用listItems
的onNext()
方法把准备好的 ViewModel 数据发送给 UI。如果发生错误,就通过hasError
属性发送出错信息。
在 map 操作符的转换过程中,我们分别使用了UserProfileListItemViewModel
和MomentListItemViewModel
结构体来转换用户简介信息和朋友圈条目信息。这两个结构体都遵循了ListItemViewModel
协议。
接下来是它们的实现,首先看一下UserProfileListItemViewModel
。
struct UserProfileListItemViewModel: ListItemViewModel {
let name: String
let avatarURL: URL?
let backgroundImageURL: URL?
init(userDetails: MomentsDetails.UserDetails) {
name = userDetails.name
avatarURL = URL(string: userDetails.avatar)
backgroundImageURL = URL(string: userDetails.backgroundImage)
}
}
该结构体只包含了三个属性:name
、avatarURL
和backgroundImageURL
。
其中,由于name
属性的类型与MomentsDetails.UserDetails
中name
属性的类型都是字符串,我们只需要直接赋值就可以了。
而avatarURL
和backgroundImageURL
用于在 UI 上显示图片。因为 BFF 返回的 URL 值都是字符串类型,我们需要把字符串转换成URL
类型。所有的转换工作我都放在init(userDetails: MomentsDetails.UserDetails)
方法里面完成,我们只需要调用URL
的初始化函数即可。
接着看一下MomentListItemViewModel
结构体,它也是负责把 Model 的数据类型转换成用于 View 层显示 UI 的 ViewModel 数据。其转换的逻辑也封装在init()
方法中,我们一起看看该方法是如何工作的。
init(moment: MomentsDetails.Moment, now: Date = Date(), relativeDateTimeFormatter: RelativeDateTimeFormatterType = RelativeDateTimeFormatter()) {
userAvatarURL = URL(string: moment.userDetails.avatar)
userName = moment.userDetails.name
title = moment.title
if let firstPhoto = moment.photos.first {
photoURL = URL(string: firstPhoto)
} else {
photoURL = nil
}
var formatter = relativeDateTimeFormatter
formatter.unitsStyle = .full
if let timeInterval = TimeInterval(moment.createdDate) {
let createdDate = Date(timeIntervalSince1970: timeInterval)
postDateDescription = formatter.localizedString(for: createdDate, relativeTo: now)
} else {
postDateDescription = nil
}
}
userName
和title
属性都是字符串类型,只需要简单的赋值就可以了。而userAvatarURL
和photoURL
属性需要把字符串转换为URL
类型来呈现图片。
postDateDescription
属性相对复杂些,它的用途是显示一个相对的时间值,例如 “5 分钟前”“2 小时前”等。我们需要把朋友圈信息生成的时间与当前时间进行对比,然后根据手机上的语言配置来显示相对时间值。
RxSwift 操作符
ViewModel 的核心功能是把 Model 数据转换为用于 UI 呈现所需的数据。其实RxSwift 的操作符就是负责转换的,使用合适的操作符能帮我们减少代码量并提高生产力。因此我建议你把 RxSwift 所提供的所有操作符都看一遍,然后在实际工作再挑选合适的来满足业务需求。
在这里,我着重介绍下过滤操作符,转换操作符和合并操作符中常用的 filter、distinctUntilChanged、map 和 combineLatest 等用法。
过滤操作符
过滤操作符用于过滤事件,我们可以使用过滤操作符把订阅者不关心的事件给过滤掉。常用的过滤操作符有 filter 和 distinctUntilChanged。
filter操作符常用于通过规则过滤不需要的事件,例如在朋友圈功能里面,可以把发布时间早于一天前的信息过滤掉不显示。为了方便理解,我就以几个数字来解释下。如下所示,有 2、23、5、60、1、31,我想把小于 10 的数过滤掉,就可以通过 filter 设置过滤规则,然后打印出来的数字就是 23、 60、31。代码示例如下。
Observable.of(2, 23, 5, 60, 1, 31)
.filter { $0 > 10 }
.subscribe(onNext: {
print($0)
})
.disposed(by: disposeBag)
过滤操作符 filter 的效果
distinctUntilChanged用于把相同的事件过滤掉。如下面例子中的第二个 1 和第四个 2,使用distinctUntilChanged 就可以把它们给过滤掉,然后打印出 1、 2、 1。代码和图例如下所示。
Observable.of(1, 1, 2, 2, 1)
.distinctUntilChanged()
.subscribe(onNext: {
print($0)
})
.disposed(by: disposeBag)
过滤操作符 distinctUntilChanged 的效果
除了相同的事件,我们还可以使用操作符distinctUntilChanged过滤掉相同的状态,从而避免频繁更新 UI。例如,我们先使用本地缓存数据呈现 UI,然后发起网络请求。当请求成功以后可以把结果数据与缓存进行对比,如果数据一致就没必要再次更新 UI。
转换操作符
转换操作符非常实用,能帮助我们从一种数据类型转变成另外一种类型,例如我们可以把用于数据传输和存储的 Model 类型转换成用于 UI 呈现的 ViewModel 类型。在这里,我就以几个常用的转换操作符 map,compactMap 和 flapMap 来介绍下如何使用它们。
map是一个十分常用的操作符,可用于从一种类型转换成另外一种类型,例如下面的例子,我把数值类型转换成字符串。程序执行的时候会打印 "String: 1" 和 "String: 2"。代码和图例如下所示。
Observable.of(1, 2)
.map { "String: " + String($0) }
.subscribe(onNext: {
print($0)
})
.disposed(by: disposeBag)
转换操作符 map 的效果
compactMap常用于过滤掉值为nil
的操作符,你可以把 compactMap 理解为同时使用 filter 和 map 的两个操作符。filter 把nil
的值过滤掉,而 map 把非空的值进行转换。
例如下面的例子中,我把字符串的值转换为数值类型,并把转换不成功的值过滤掉。由于 "not-a-number" 不能转换成数值类型,因此被过滤掉了,执行的时候会打印 1 和 2。代码示例如下所示:
Observable.of("1", "not-a-number", "2")
.compactMap { Int($0) }
.subscribe(onNext: {
print($0)
})
.disposed(by: disposeBag)
转换操作符 compactMap 效果
flatMap用于把两层的 Observable 序列合并到一层。我们通过一个例子来解析到底怎样合并。
请看代码示例:
struct TemperatureSensor {
let temperature: Observable<Int>
}
let sensor1 = TemperatureSensor(temperature: Observable.of(21, 23))
let sensor2 = TemperatureSensor(temperature: Observable.of(22, 25))
Observable.of(sensor1, sensor2)
.flatMap { $0.temperature }
.subscribe(onNext: {
print($0)
})
.disposed(by: disposeBag)
在这个例子中,我定义一个叫作TemperatureSensor
的结构体,用来表示收集温度的传感器,该结构体包含了一个类型为Observable
的temperature
的属性。
假如天气站有多个这样的传感器,我们要把它们的温度信息合并到一个单独的 Observable 序列中方便统计,此时就可以使用 flatMap 来完成这项任务。
具体来说,我们在flatMap
方法的闭包里面返回temperature
属性,由于该属性是一个Observable
对象,因此flatMap
方法会把这些序列统一合并到一个单独的 Observable 序列里面,并打印出 21、23、22、25。
转换操作符 flatMap 的效果
合并操作符
合并操作符用于组装与合并多个 Observable 序列。我们通过 startWith,concat 和 merge 等几个常用的合并操作符,来看看它们是怎样运作的。
startWith可以使订阅者在接收到 Observable 序列的事件前,先收到传给 startWith 方法的事件。它的使用非常简单,例如在下面的例子中,我们把 3 和 4 传递给startWith
。那么在执行过程中,会先把 3 和 4 事件发送给订阅者,其运行效果为 3、4、1、2。代码示例如下:
Observable.of(1, 2)
.startWith(3, 4)
.subscribe(onNext: {
print($0)
})
.disposed(by: disposeBag)
合并操作符 startWith 效果
日常中我们可以通过startWith
方法,把加载事件插入网络数据事件之前,以此来保持 UI 状态的自动更新。
concat能把多个 Observable 序列按顺序合并在一起。例如,在下面的例子中我们合并了两个 Observable 序列,第一个包含 1 和 2,第二个包含 3 和 4,那么执行的时候会打印 1、2、3、4。代码示例如下。
Observable.of(1, 2)
.concat(Observable.of(3, 4))
.subscribe(onNext: {
print($0)
})
.disposed(by: disposeBag)
合并操作符 concat 效果
merge,常用于合并多个 Observable 序列的操作符,和 concat 不一样的地方是它能保持原来事件的顺序。我们可以通过一个例子来看看,它是怎样合并 Observable 序列的。代码示例如下:
let first = PublishSubject<Int>()
let second = PublishSubject<Int>()
Observable.of(first, second)
.merge()
.subscribe(onNext: {
print($0)
})
.disposed(by: disposeBag)
first.onNext(1)
first.onNext(2)
second.onNext(11)
first.onNext(3)
second.onNext(12)
second.onNext(13)
first.onNext(4)
我们调用merge
方法把两个 PublishSubject 合并在一起,然后不同的 PublishSubject 会分别发出不同的next
事件,订阅者根据事件发生的顺序来接收到相关事件。如下图所示,程序执行时会打印 1、2、11、3、12、13、4。
合并操作符 merge 的效果
combineLatest会把两个 Observable 序列里最后的事件合并起来,代码示例如下。
let first = PublishSubject<String>()
let second = PublishSubject<String>()
Observable.combineLatest(first, second) { $0 + $1 }
.subscribe(onNext: {
print($0)
})
.disposed(by: disposeBag)
first.onNext("1")
second.onNext("a")
first.onNext("2")
second.onNext("b")
second.onNext("c")
first.onNext("3")
first.onNext("4")
在程序执行过程中,当其中一个 PublishSubject 发出next
事件时,就会从另外一个 PublishSubject 取出其最后一个事件,然后调用combineLatest
方法的闭包,把这两个事件合并起来并通知订阅者。上述的例子在执行时会打印 1a、2a、2b、2c、3c、4c。
合并操作符 combineLatest
在实际开发中,combineLatest
方法非常实用。我们可以用它来监听多个 Observable 序列,然后组合起来统一更新状态。例如在一个登录页面里面,我们可以同时监听用户名和密码两个输入框,当它们同时有值的时候才激活登录按钮。
zip也能用于合并两个 Observable 序列,和 combineLatest 不一样的地方是, zip 只会把两个 Observable 序列的事件配对合并。就像两队小朋友,排在前头的手牵手来到一个新队列。一旦出来就不再留在原有队列了。
为了方便理解 zip 与 combineLatest 的区别,我在下面例子中也使用了一样的数据并保持事件发送的顺序。
let first = PublishSubject<String>()
let second = PublishSubject<String>()
Observable.zip(first, second) { $0 + $1 }
.subscribe(onNext: {
print($0)
})
.disposed(by: disposeBag)
first.onNext("1")
second.onNext("a")
first.onNext("2")
second.onNext("b")
second.onNext("c")
first.onNext("3")
first.onNext("4")
在上述的例子中,有两个 PublishSubject,其中first
发出 1、2、3、4,而second
发出 a、b、c。zip
方法会返回它们的合并事件 1a、2b、3c。由于first
所发出next("4")
事件没有在second
里面找到对应的事件,所以合并后的 Observable 序列只有三个事件。
合并操作符 zip 的效果
上面是常用的操作符,灵活使用它们,我们可以完成绝大部分的任务了。
总结
在这一讲中,我们介绍了 ViewModel 模式的架构与实现和 RxSwift 的操作符。有了 ViewModel,我们可以把业务逻辑从 View 层抽离出来,甚至把 View 层进行替换,例如把 UIKit 替换成 SwiftUI。而 UI 所需的数据,可以通过 ViewModel 模块把 Model 数据转换出来。至于转换工作,我们可以借助操作符来完成。
有关本讲操作符的例子代码,我都放在项目中的RxSwift Playground 文件里面,希望你能多练习,灵活运用。
RxSwift 为我们提供了 50 多个操作符,我建议你到 rxmarbles.com 或者到 App Store 下载 RxMarbles App,并在 App 中替换各种参数来观察执行的结果,这样能帮助你学会所有的操作符,在现实工作中能选择合适的操作符来简化大量的开发工作。
思考题
请问你会把所有逻辑都编写在 ViewController 里面吗?如果没有,使用了怎样模式与架构来解耦呢?能分享一下这方面的经验吗?
请把你的想法写到留言区哦,下一讲我将介绍如何开发统一并且灵活的 UI。
源码地址:
RxSwift Playground 文件地址:
https://github.com/lagoueduCol/iOS-linyongjian/blob/main/Playgrounds/RxSwiftPlayground.playground/Contents.swift ViewModel 协议定义的源码地址:https://github.com/lagoueduCol/iOS-linyongjian/tree/main/Moments/Moments/Foundations/ViewModels 朋友圈功能 ViewModel 实现的源码地址:
https://github.com/lagoueduCol/iOS-linyongjian/tree/main/Moments/Moments/Features/Moments/ViewModels
21 UI 层架构:如何开发统一并且灵活的 UI?
作为 iOS 开发者,我们每天都花大量的时间来开发和调试 UI,那有没有什么办法帮助我们把繁杂的 UI 开发工作简化成有章可循的步骤,从而提高开发的效率呢?在这一讲中,我就和你聊聊,如何架构和开发一套灵活的 UI 框架。
通用列表 UI 模块的架构与实现
列表 UI 是 App 最为常用的 UI 页面,它可以帮我们通过滚动的方式支持无限的内容。为了简化大量的重复性劳动,我在 Moments App 架构实现了一个通用的列表 UI 模块。下面是这个模块的架构图。
这个框架使用了UIViewController
和UITableView
来封装列表页面。其核心是BaseTableViewController
。BaseTableViewController
继承于BaseViewController
,而BaseViewController
继承自UIViewController
。
我们先看看BaseViewController
的具体实现,代码示例如下。
class BaseViewController: UIViewController {
lazy var disposeBag: DisposeBag = .init()
init() {
super.init(nibName: nil, bundle: nil)
}
@available(*, unavailable, message: "We don't support init view controller from a nib.")
override init(nibName nibNameOrNil: String?, bundle nibBundleOrNil: Bundle?) {
super.init(nibName: nibNameOrNil, bundle: nibBundleOrNil)
}
@available(*, unavailable, message: "We don't support init view controller from a nib.")
required init?(coder: NSCoder) {
fatalError(L10n.Development.fatalErrorInitCoderNotImplemented)
}
}
因为 Moments App 是使用纯代码的方式来编写 UI,所以BaseViewController
重写了init(nibName nibNameOrNil: String?, bundle nibBundleOrNil: Bundle?)
和init?(coder: NSCoder)
两个方法,并直接抛出异常。这样做使得所有继承BaseViewController
的子类,都没办法通过 Storyboard 或者 Xib 文件来生成 ViewController 的实例。因为我们使用 RxSwift,BaseViewController
还定义了一个disposeBag
属性来方便管理所有 Obervable 序列的订阅。
BaseTableViewController
继承了BaseViewController
,并使用UITableView
来封装一个通用的列表页面。我们一起看看它是怎样实现的。
在 MVVM 模式里,View 依赖于 ViewModel。作为 View 的BaseTableViewController
依赖于 ViewModel 层的ListViewModel
协议,这使得BaseTableViewController
只依赖于接口而不是具体的类型,从而提高了程序的可扩展性。
同时,BaseTableViewController
还定义了三个属性来显示 UI 控件:
tableView
属性用于显示一个 TableView;activityIndicatorView
属性用于显示俗称小菊花的加载器;errorLabel
用于显示出错信息的标签控件。
以下是属性定义的代码示例。
var viewModel: ListViewModel!
private let tableView: UITableView = configure(.init()) {
$0.translatesAutoresizingMaskIntoConstraints = false
$0.separatorStyle = .none
$0.rowHeight = UITableView.automaticDimension
$0.estimatedRowHeight = 100
$0.contentInsetAdjustmentBehavior = .never
$0.backgroundColor = UIColor.designKit.background
}
private let activityIndicatorView: UIActivityIndicatorView = configure(.init(style: .large)) {
$0.translatesAutoresizingMaskIntoConstraints = false
}
private let errorLabel: UILabel = configure(.init()) {
$0.translatesAutoresizingMaskIntoConstraints = false
$0.isHidden = true
$0.textColor = UIColor.designKit.primaryText
$0.text = L10n.MomentsList.errorMessage
}
为了方便初始化 UIKit 的控件,我写了一个公共的configure()
方法,具体代码如下:
func configure<T: AnyObject>(_ object: T, closure: (T) -> Void) -> T {
closure(object)
return object
}
有了该方法,我们就可以把所有初始化操作都放在一个闭包(Closure)里面,方便代码的维护。
接着我们看一下setupUI()
方法的代码实现。
func setupUI() {
view.backgroundColor = UIColor.designKit.background
tableViewCellsToRegister.forEach {
tableView.register($0.value, forCellReuseIdentifier: $0.key)
}
[tableView, activityIndicatorView, errorLabel].forEach {
view.addSubview($0)
}
}
该方法负责设置 UI 的样式,例如设置背景颜色,注册 TableView Cell 和添加子控件。
配置完 UI 的样式以后,下一步是配置自动布局的约束(Auto Layout Constraint)。当使用 UIKit 作为 View 层的时候,我推荐使用苹果公司所推荐的自动布局来排版 UI 页面。
自动布局能帮助我们支持不同分辨率和屏幕对比率的页面,而且苹果公司每年都在不断优化自动布局引擎的性能。不过,它也有一个缺点,那就是手写自动布局的约束代码会十分冗长,为此我使用一个名叫SnapKit 的库来进行简化。下面我们就通过setupConstraints()
的代码,来看看 SnapKit 的威力。
func setupConstraints() {
tableView.snp.makeConstraints {
$0.edges.equalToSuperview()
}
activityIndicatorView.snp.makeConstraints {
$0.center.equalToSuperview()
}
errorLabel.snp.makeConstraints {
$0.center.equalToSuperview()
}
}
如上述代码所示,当使用 SnapKit 来配置自动布局的约束时,我们需要调用它的扩展方法makeConstraints
,然后把所有约束的配置都放到闭包里面。在这里,我是通过edges.equalToSuperview()
把tableView
延伸到它的父组件(也就是BaseTableViewController
的view
)中,然后通过center.equalToSuperview()
方法把activityIndicatorView
和errorLabel
都分别居中。
假如不使用 SnapKit,要完成延伸tableView
的操作,就需要以下的代码。
NSLayoutConstraint.activate([
tableView.leadingAnchor.constraint(equalTo: view.leadingAnchor),
tableView.trailingAnchor.constraint(equalTo: view.trailingAnchor),
tableView.topAnchor.constraint(equalTo: view.topAnchor),
tableView.bottomAnchor.constraint(equalTo: view.bottomAnchor)
])
我们不得不分别配置每一个约束,并放进一个数组里面,然后传递给静态方法NSLayoutConstraint.activate
。 所以你看,使用 SnapKit 多么方便。
完成了 UI 的布局以后,我们看一下数据绑定。Moments App 使用了 RxSwift 把 ViewModel 层和 View 层进行绑定,绑定的代码在setupBindings()
函数里,具体如下。
func setupBindings() {
tableView.refreshControl = configure(UIRefreshControl()) {
let refreshControl = $0
$0.rx.controlEvent(.valueChanged)
.filter { refreshControl.isRefreshing }
.bind { [weak self] _ in self?.loadItems() }
.disposed(by: disposeBag)
}
let dataSource = RxTableViewSectionedReloadDataSource<SectionModel<String, ListItemViewModel>>(configureCell: { _, tableView, indexPath, item in
let cell = tableView.dequeueReusableCell(withIdentifier: String(describing: type(of: item)), for: indexPath)
(cell as? ListItemCell)?.update(with: item)
return cell
})
viewModel.listItems
.bind(to: tableView.rx.items(dataSource: dataSource))
.disposed(by: disposeBag)
viewModel.hasError
.map { !$0 }
.bind(to: errorLabel.rx.isHidden)
.disposed(by: disposeBag)
}
这个函数由三部分组成,第一部分是通过 RxSwift 和 RxCocoa ,把UIRefreshControl
控件里的isRefreshing
事件和loadItems()
函数绑定起来。当用户下拉刷新控件的时候会调用loadItems()
函数来刷新列表的数据。
第二部分是把 TableView Cell 控件与 ViewModel 的listItems
Subject 属性绑定起来,当listItems
发出新的事件时,我们会调用ListItemCell
的update(with viewModel: ListItemViewModel)
方法来更新 UI。经过了这一绑定,UI 就能随着 ViewModel 的数据变化而自动更新。
第三部分与第二部分类似,都是把 ViewModel 与 View 层的控件进行绑定。在这里,我们把 ViewModel 的hasError
Subject 属性绑定到errorLabel.rx.isHidden
属性来控制errorLabel
是否可见。
你可能注意到在errorLabel
后面有.rx
属性,这是 RxCocoa 为UILabel
控件所提供的一个扩展,它为isHidden
属性提供了响应式编程的功能。有了这一功能,它就可以与 ViewModel 的 Subject 属性进行绑定,从而实现自动更新。
数据绑定以后,我们一起看看loadItems()
函数的实现。
func loadItems() {
viewModel.hasError.onNext(false)
viewModel.loadItems()
.observeOn(MainScheduler.instance)
.do(onDispose: { [weak self] in
self?.activityIndicatorView.rx.isAnimating.onNext(false)
self?.tableView.refreshControl?.endRefreshing()
})
.map { false }
.startWith(true)
.distinctUntilChanged()
.bind(to: activityIndicatorView.rx.isAnimating)
.disposed(by: disposeBag)
}
loadItems()
方法用于加载数据。当我们第一次进入朋友圈页面的时候,或者用户下拉刷新控件的时候,就会调用该方法来重新加载数据。
该方法主要做两项工作,第一项是调用viewModel.hasError.onNext(false)
来更新 ViewModel 的hasError
属性, 它能让 UI 上的错误标签信息消失。
从代码中你可以看到,尽管我们想更新 UI 层的errorLabel
控件,却没有直接通过errorLabel.isHidden = true
的方式来更新,而是通过 ViewModel 的hasError
属性来完成。这是因为我要保证 View/UI 层都是由 ViewModel 驱动,通过单方向的数据流来减少 Bug ,从而提高代码的可维护性。
loadItems()
方法的第二项工作,是让 ViewModel 去加载数据并绑定到activityIndicatorView
控件的isAnimating
属性上。因为我们需要在主排程器上执行 UI 任务,因此调用了.observeOn(MainScheduler.instance)
,把所有任务都安排到主排程器上。
当 ViewModel 的loadItems()
方法开始执行的时候,先通过.startWith(true)
来让activityIndicatorView
启动动画效果。当 ViewModel 的loadItems()
方法返回数据时,把结果数据通过.map { false }
方法来返回false
,从而使得activityIndicatorView
停止动画效果。
假如用户在调用 ViewModel 的loadItems()
方法的过程中,退出列表页面,我们通过.do(onDispose:{})
方法来停止activityIndicatorView
和refreshControl
两个控件的刷新动画。
到此为止,我们已经知道BaseTableViewController
是如何通过 TableView 来实现列表 UI 的了。
为了显现不同的 TableView Cell,接下来我们了解下通用的 Cell 是如何实现的。
这部分由四个类型所组成,分别是ListItemCell
协议及其子结构体BaseTableViewCell
,以及ListItemView
协议及其子结构体BaseListItemView
。
ListItemCell
协议的定义非常简单,如下所示。
protocol ListItemCell: class {
func update(with viewModel: ListItemViewModel)
}
该协议只包含了一个update(with viewModel: ListItemViewModel)
方法来让其子类型根据ListItemViewModel
的数据进行更新。
其子类型BaseTableViewCell
的具体代码如下:
final class BaseTableViewCell<V: BaseListItemView>: UITableViewCell, ListItemCell {
private let view: V
override init(style: UITableViewCell.CellStyle, reuseIdentifier: String?) {
view = .init()
super.init(style: style, reuseIdentifier: reuseIdentifier)
selectionStyle = .none
contentView.addSubview(view)
view.snp.makeConstraints {
$0.edges.equalToSuperview()
}
}
required init?(coder: NSCoder) {
fatalError(L10n.Development.fatalErrorInitCoderNotImplemented)
}
func update(with viewModel: ListItemViewModel) {
view.update(with: viewModel)
}
}
BaseTableViewCell
是一个UITableViewCell
的子类,并遵循了ListItemCell
协议,因此它需要实现update(with viewModel: ListItemViewModel)
方法。在该方法里面,它直接调用view
属性的update(with viewModel: ListItemViewModel)
来更新BaseListItemView
组件的 UI。
那为什么我们不把所有 UI 子控件都直接写在 Cell 里,而使用一个额外的BaseListItemView
呢?因为这样做可以把BaseListItemView
复用到UICollectionView
等其他容器中。
接下来我们一起看看BaseListItemView
及其所遵循的ListItemView
协议的代码。
protocol ListItemView: class {
func update(with viewModel: ListItemViewModel)
}
class BaseListItemView: UIView, ListItemView {
lazy var disposeBag: DisposeBag = .init()
func update(with viewModel: ListItemViewModel) {
fatalError(L10n.Development.fatalErrorSubclassToImplement)
}
}
ListItemView
协议只定义了update(with viewModel: ListItemViewModel)
接口来通过 ViewModel 更新 UI。因为每个 UI 组件的布局与呈现都可能不一样,因此,BaseListItemView
在实现update(with viewModel: ListItemViewModel)
方法时,直接抛出了异常,这样能迫使其子类重写该方法。
上面就是通用列表 UI 模块的架构与实现,有了这一个框架,我们就能快速实现不同的列表页面,下面以朋友圈功能作为例子来看看如何实现一个朋友圈时间轴页面。
朋友圈时间轴页面的实现
首先我们一起看看朋友圈时间轴页面的架构图。
MomentsTimelineViewController
用于显示朋友圈时间轴页面,其具体代码如下。
final class MomentsTimelineViewController: BaseTableViewController {
override init() {
super.init()
viewModel = MomentsTimelineViewModel(userID: UserDataStore.current.userID)
}
override func viewDidAppear(_ animated: Bool) {
super.viewDidAppear(animated)
viewModel.trackScreenviews()
}
override var tableViewCellsToRegister: [String : UITableViewCell.Type] {
return [
UserProfileListItemViewModel.reuseIdentifier: BaseTableViewCell<UserProfileListItemView>.self,
MomentListItemViewModel.reuseIdentifier: BaseTableViewCell<MomentListItemView>.self
]
}
}
因为BaseViewController
已经封装好绝大部分的 UI 处理逻辑,作为子类,MomentsTimelineViewController
的实现变得非常简单,只需三部分。
首先是初始化viewModel
。因为BaseViewController
通过var viewModel: ListViewModel!
来定义viewModel
属性,作为子类的MomentsTimelineViewController
也必须初始化viewModel
属性,否则程序会崩溃。具体来说,我们只需创建一个MomentsTimelineViewModel
对象来完成初始化即可。它的实现我在上一讲已经介绍过了,你可以再留意下。
然后,我在func viewDidAppear(_ animated: Bool)
方法里面调用viewModel.trackScreenviews()
来让 ViewModel 发送用户行为数据。
为了帮BaseViewController
提供需要注册的 TableView Cell ,最后我重写了tableViewCellsToRegister
属性。该属性存放BaseTableViewCell
的实例。BaseTableViewCell
使用范型(generic)来存放BaseListItemView
的子类,这些子类包括UserProfileListItemView
和MomentListItemView
。
你可以从下图中看到它们所呈现的 UI 组件。
MomentsTimelineViewController
我们已介绍完毕了,下面咱们以UserProfileListItemView
为例,看一下开发子控件的步骤与实现。
UserProfileListItemView
用于显示用户自己的资料,例如用户名字,头像和背景图。因为有了通用和统一的 UI 开发框架,每次开发 UI 页面的步骤都是一致的,具体我分为以下几步完成:
- 初始化 UI 控件的属性;
- 配置 UI 控件的样式;
- 设置自动布局的约束;
- 重写
update(with viewModel: ListItemViewModel)
方法,根据 ViewModel 的数据来更新 UI。
先看一下初始化 UI 控件属性的代码。
private let backgroundImageView: UIImageView = configure(.init()) {
$0.translatesAutoresizingMaskIntoConstraints = false
$0.contentMode = .scaleAspectFill
$0.accessibilityIgnoresInvertColors = true
}
private let avatarImageView: UIImageView = configure(.init()) {
$0.translatesAutoresizingMaskIntoConstraints = false
$0.asAvatar(cornerRadius: 8)
$0.contentMode = .scaleAspectFill
$0.accessibilityIgnoresInvertColors = true
}
private let nameLabel: UILabel = configure(.init()) {
$0.translatesAutoresizingMaskIntoConstraints = false
$0.font = UIFont.designKit.title3
$0.textColor = .white
$0.numberOfLines = 1
}
我们分别调用configure()
函数来初始化三个 UI 控件的属性,backgroundImageView
用于显示背景图,avatarImageView
用于显示用户头像,而nameLabel
用于显示用户名字。
你可以根据下图,看到它们分别使用在哪里。
接着来看在第二步中如何配置 UI 控件的样式,我用setupUI()
方法来实现。
func setupUI() {
backgroundColor = UIColor.designKit.background
[backgroundImageView, avatarImageView, nameLabel].forEach {
addSubview($0)
}
}
在这里,我使用了 DesignKit 来设置了背景颜色,并把子控件添加到当前 View 里面。
然后看第三步如何设置自动布局的约束,其实现代码如下。
func setupConstraints() {
backgroundImageView.snp.makeConstraints {
$0.top.leading.trailing.equalToSuperview()
$0.bottom.equalToSuperview().offset(-Spacing.medium)
$0.height.equalTo(backgroundImageView.snp.width).multipliedBy(0.8).priority(999)
}
avatarImageView.snp.makeConstraints {
$0.right.equalToSuperview().offset(-Spacing.medium)
$0.bottom.equalToSuperview()
$0.height.equalTo(80)
$0.width.equalTo(80)
}
nameLabel.snp.makeConstraints {
$0.right.equalTo(self.avatarImageView.snp.left).offset(-Spacing.medium)
$0.centerY.equalTo(self.avatarImageView.snp.centerY)
}
}
其中backgroundImageView
的顶部和两边都延展到父控件,因为底部需要留白来显示用户头像,因此添加了medium
作为间距。背景图片的长宽比是 5:4。
avatarImageView
位于父控件的右下角,并设定长度和宽度都为 80pt。nameLabel
位于avatarImageView
的左边,并与之水平。这样我们就使用 SnapKit 完成用户资料 UI 的布局了。
最后一部分是调用update()
方法来更新 UI,其代码如下。
override func update(with viewModel: ListItemViewModel) {
guard let viewModel = viewModel as? UserProfileListItemViewModel else {
return
}
backgroundImageView.kf.setImage(with: viewModel.backgroundImageURL)
avatarImageView.kf.setImage(with: viewModel.avatarURL)
nameLabel.text = viewModel.name
}
因为UserProfileListItemViewModel
已经为UserProfileListItemView
准备好呈现所需的所有数据,因此,只要简单的赋值就可以更新 UI 了。
MomentListItemView
的代码结构和UserProfileListItemView
基本一样,你可以到拉勾教育的代码仓库进行查看。
总结
在这一讲中,我为你介绍了如何架构和实现一个通用的列表 UI 模块,有了这个模块,我们按照以下这几个步骤就可以完成 UI 的开发了。
- 初始化 UI 控件的属性,把 UI 分解成不同的子控件,然后通过
configure()
来初始化各个控件属性。 - 配置 UI 控件的样式,如配置背景颜色等,并把各个子控件添加到父控件里面。
- 设置自动布局的约束,推荐使用 SnapKit 来简化配置约束的工作。
- 重写
update(with viewModel: ListItemViewModel)
方法,根据 ViewModel 的数据来更新 UI。如果有数据绑定,那么使用 RxSwift 和 RxCocoa 把 ViewModel 的 Subject 属性绑定到 UI 控件上。如果不需要数据绑定,只需把 ViewModel 准备好的值赋给 UI 控件即可。
思考题
请问你们使用苹果提供的自动布局吗?如果是,是使用原生语法还是类似 SnapKit 那种库呢?或者说使用 Texture 等其他非苹果的框架进行布局?能分享你的使用经验吗?
可以把你的思考写到留言区哦,下一讲,我会介绍如何使用现有架构添加点赞功能。
源码地址
通用列表 UI 的源码地址:
https://github.com/lagoueduCol/iOS-linyongjian/tree/main/Moments/Moments/Foundations/Views 朋友圈时间轴页面实现的源码地址:
https://github.com/lagoueduCol/iOS-linyongjian/tree/main/Moments/Moments/Features/Moments/Views
22 功能实战:如何使用现有架构添加点赞功能?
你有没有遇到过接手一份新的代码却不知道如何下手的情况?其实,一套良好的开发框架就能有效地解决这种问题。规范的架构与框架不仅具有良好的可扩展性,例如,可以灵活地替换网络层、数据库甚至 UI 层的实现,而且还为开发者提供了统一的开发步骤与规范,方便新功能的快速迭代。
我们的 Moments App 使用了 MVVM 架构来支持快速开发,在这一讲中,我们再以添加点赞功能为例来看看如何一步一步去开发一个新功能。
如下面的动图所示,我们可以摇动手机来打开内部功能菜单页面,在该页面内点击开启点赞按钮来启动点赞功能。当重启 App 以后,我们就能在朋友圈页面里看到点赞按钮了。
![在这里插入图片描述](https://s2.51cto.com/images/blog/202406/22235103_6676f2e7ebbd751004.jpg?x-oss-process=image/watermark,size_16,text_QDUxQ1RP5Y2a5a6i,color_FFFFFF,t_30,g_se,x_10,y_10,shadow_20,type_ZmFuZ3poZW5naGVpdGk=#pic_center)
根据组件间的依赖关系,我们可以按照以下五个步骤来进行开发:
- 增加“添加点赞功能”的功能开关;
- 开发网络层来更新 BFF 的点赞信息;
- 开发 Repository 层来存储数据;
- 开发 ViewModel 层来准备 UI 所需的数据;
- 开发 UI/View 层呈现点赞按钮和点赞朋友列表。
下面我们就来详细说明这每一个步骤。
增加功能开关
当我们开发一个周期比较长的新功能时,通常会使用功能开关。
如果没有功能开关,当开发周期超过一周以上时,我们就不得不把开发中的功能放在一个“长命”功能分支下,直到整个功能完成后才合并到主分支,这往往会增加合并分支的难度。
另一种方法是延迟发布的时间,在功能完整开发出来后才进行发布。假如有多个团队一直在开发新功能,那么发布计划就可能一直在延迟。但如果我们使用了功能开关,就可以把未完成的功能一直隐藏着,直到通过完整的测试和产品验证后才把开关启动并进行发布。总之,有了功能开关,我们可以支持多个团队并行开发,并在此期间随时发布新版本的 App。
下面我们看看如何为添加点赞功能增加一个功能开关,具体代码如下:
enum InternalToggle: String, ToggleType {
case isLikeButtonForMomentEnabled
}
首先,我们为枚举类型InternalToggle
添加isLikeButtonForMomentEnabled
来表示启动点赞功能的功能开关。
接着在InternalTogglesDataStore
里把该值初始化为false
表示默认关闭该功能,这样就能保证 App Store 版本的 App 都看不到这个功能,代码如下:
struct InternalTogglesDataStore: TogglesDataStoreType {
private init(userDefaults: UserDefaults) {
self.userDefaults.register(defaults: [
InternalToggle.isLikeButtonForMomentEnabled.rawValue: false
])
}
}
最后一步是通过isLikeButtonForMomentEnabled
初始化InternalMenuFeatureToggleItemViewModel
,并添加到InternalMenuViewModel
的sections
属性里面 ,代码如下:
let featureTogglesSection = InternalMenuSection(
title: L10n.InternalMenu.featureToggles,
items: [
InternalMenuFeatureToggleItemViewModel(title: L10n.InternalMenu.likeButtonForMomentEnabled, toggle: InternalToggle.isLikeButtonForMomentEnabled)
])
sections = .just([
featureTogglesSection,
... // other sections
])
这样子就为内部隐藏菜单增加了启动点赞功能的功能开关。功能开关是其他模块的基础,你会看到我们在其他模块中也都会使用到该开关。
开发网络层
Moments App 使用了 BFF 来读取朋友圈信息,那我们也把点赞信息存储在 BFF 里面。因为 Moments App 的 BFF 使用了 GraphQL,要更新 BFF 上的数据,我们就需要使用 Mutation。和 Restful API 的 Post 操作不一样,在 GraphQL 的 Mutation 不仅能更新数据,还可以返回数据。
下面我们就来一起看看网络层的实现,首先定义一个名叫UpdateMomentLikeSessionType
的协议来提供更新点赞信息的接口,具体代码如下:
protocol UpdateMomentLikeSessionType {
func updateLike(_ isLiked: Bool, momentID: String, fromUserID userID: String) -> Observable<MomentsDetails>
}
该协议只定义了一个updateLike()
方法,该方法会接收以下的入口参数,并返回类型为MomentsDetails
的 Observable 序列。
isLiked
是一个布尔类型,用于表示是否点赞了。momentID
表示被点赞的那条朋友圈的 ID。userID
表示点赞的用户 ID。
接着我们定义了一个遵循UpdateMomentLikeSessionType
协议的结构体,它名叫UpdateMomentLikeSession
。UpdateMomentLikeSession
的实现方法和GetMomentsByUserIDSession
代码基本一致,我们已经在《18 | 网络层架构:如何设计网络访问与 JSON 数据解析?》那一讲中详细讲述了GetMomentsByUserIDSession
的实现,如有需要你可以回去复习一下。
不同的地方是在query
属性的定义里,UpdateMomentLikeSession
使用了mutation
而不是query
,具体定义如下:
private static let query = """
mutation updateMomentLike($momentID: ID!, $userID: ID!, $isLiked: Boolean!) {
// the response for updateMomentLike
}
"""
这样子,我们就能往 BFF 发送一个 Mutation 请求并接收更新后的MomentsDetails
信息了。
除了更新点赞信息以外,我们还要修改GetMomentsByUserIDSession
来读取点赞朋友的列表信息。
不过,点赞信息只有在功能开关开启的时候才能看到,因此在读取朋友圈信息的时候需要进行检查。要检查内部功能开关,需要使用到一个InternalTogglesDataStore
的实例,因此我们在初始化GetMomentsByUserIDSession
的时候可以将InternalTogglesDataStore.shared
传递进去,代码如下:
private let togglesDataStore: TogglesDataStoreType
init(togglesDataStore: TogglesDataStoreType = InternalTogglesDataStore.shared, sessionHandler: ...) {
self.togglesDataStore = togglesDataStore
}
当GetMomentsByUserIDSession
接收到InternalTogglesDataStore
的实例时,可以直接保存到togglesDataStore
属性里面,这样我们就能使用togglesDataStore
属性来检查点赞功能是否开启了。下面代码展示的是内嵌Session
结构体的init()
方法:
init(userID: String, togglesDataStore: TogglesDataStoreType) {
let variables: [AnyHashable: Encodable] = ["userID": userID,
"withLikes": togglesDataStore.isToggleOn(InternalToggle.isLikeButtonForMomentEnabled)]
}
我们通过调用togglesDataStore.isToggleOn(InternalToggle.isLikeButtonForMomentEnabled)
来判断是否开启了点赞功能,如果是,就把withLikes
属性赋值为true
,否则赋为false
。这样就可以在 Query 里面使用withLikes
属性了,代码如下:
private static let query = """
query getMomentsDetailsByUserID($userID: ID!, $withLikes: Boolean!) {
getMomentsDetailsByUserID(userID: $userID) {
// other fields
createdDate
isLiked @include(if: $withLikes)
likes @include(if: $withLikes) {
id
avatar
}
}
}
}
"""
在定义query
属性的地方,我们把withLikes
传递给getMomentsDetailsByUserID
Query,然后通过@include
来控制是否读取isLiked
和likes
属性,从而保证只有当isLikeButtonForMomentEnabled
开关开启时,才需要读取这两个属性。
到此为止,网络层的开发就完成了,下面我们再来看看 Repository 层的开发。
开发 Repository 层
在朋友圈功能里面,Respository 层的关键组件是MomentsRepo
。当它要更新点赞信息时,就会用UpdateMomentLikeSessionType
协议,因此我们在初始化的时候也注入对该协议的依赖,具体代码如下:
private let updateMomentLikeSession: UpdateMomentLikeSessionType
static let shared: MomentsRepo = {
return MomentsRepo(...,
updateMomentLikeSession: UpdateMomentLikeSession()
)
}()
init(..., updateMomentLikeSession: UpdateMomentLikeSessionType) {
self.updateMomentLikeSession = updateMomentLikeSession
}
我们把UpdateMomentLikeSession
结构体的实例赋值给updateMomentLikeSession
属性,当需要访问网络层时就可以使用该属性的方法,接着看一下updateLike()
方法的实现:
func updateLike(isLiked: Bool, momentID: String, fromUserID userID: String) -> Observable<Void> {
return updateMomentLikeSession
.updateLike(isLiked, momentID: momentID, fromUserID: userID)
.do(onNext: { persistentDataStore.save(momentsDetails: $0) })
.map { _ in () }
.catchErrorJustReturn(())
}
我们通过调用updateMomentLikeSession
属性的updateLike()
方法来更新点赞信息,然后把返回的结果通过persistentDataStore
的save()
方法保存到本地数据存储中。
到这里,Repository 层的开发也完成了,我们接着修改 ViewModel 层的代码来支持点赞功能。
开发 ViewModel 层
因为点赞功能只使用在 UI 层的MomentListItemView
里面,所以我们只需要更新该 View 所对应的 ViewModelMomentListItemViewModel
即可。为此,我们增加了两个属性:第一个是isLiked
属性,用于表示用户是否已经点赞了该朋友圈信息;第二个是likes
属性,用于显示点赞了朋友的头像列表。
有了这两个属性,我们就可以在init()
方法里面把MomentsDetails.Moment
数据映射到这两个属性中去,具体代码如下:
isLiked = moment.isLiked ?? false
likes = moment.likes?.compactMap { URL(string: $0.avatar) } ?? []
isLiked
属性的映射比较简单,只是简单的赋值即可。而likes
属性则需要我们把 BFF 返回的 URL 字符串转换为用于呈现图片的URL
类型。
当用户在页面中点击点赞按钮后,我们就需要调用MomentListItemViewModel
来完成具体的操作,因此我们在MomentListItemViewModel
也定义了两个方法,具体代码如下:
func like(from userID: String) -> Observable<Void> {
return momentsRepo.updateLike(isLiked: true, momentID: momentID, fromUserID: userID)
}
func unlike(from userID: String) -> Observable<Void> {
return momentsRepo.updateLike(isLiked: false, momentID: momentID, fromUserID: userID)
}
可以看到,like(from userID: String)
和unlike(from userID: String)
方法都调用了momentsRepo.updateLike()
方法来更新点赞信息。至此,ViewModel 层也开发完毕了。
开发 UI/View 层
其他模块开发完毕以后,最后就是更新 UI/View 层了。因为点赞按钮在每一条朋友圈信息里面,所以我们只需要修改MomentListItemView
就可以了。你可以从下面的这个示例图看到新加的组件:
从示例图可以看到,新加的组件主要有以下三个。
likesStakeView
用于存放点赞朋友的列表。likesContainerView
是一个用来存放likesStakeView
的容器视图,我们还可以使用它来设置背景颜色和配置圆角效果。favoriteButton
表示点赞按钮。
这些 UI 组件的属性定义如下:
private let likesContainerView: UIView = configure(.init()) {
$0.translatesAutoresizingMaskIntoConstraints = false
$0.backgroundColor = UIColor.designKit.secondaryBackground
$0.layer.cornerRadius = 4
}
private let likesStakeView: UIStackView = configure(.init()) {
$0.translatesAutoresizingMaskIntoConstraints = false
$0.spacing = Spacing.twoExtraSmall
}
private let favoriteButton: UIButton = configure(.init()) {
$0.translatesAutoresizingMaskIntoConstraints = false
$0.asHeartFavoriteButton()
}
有了这些属性以后,我们还需要把它们添加到 UI 里面,下面是setupUI()
方法的代码:
func setupUI() {
if togglesDataStore.isToggleOn(InternalToggle.isLikeButtonForMomentEnabled) {
likesContainerView.addSubview(likesStakeView)
verticalStackView.addArrangedSubview(likesContainerView)
addSubview(favoriteButton)
}
}
从上面的代码可以看到,只有当isLikeButtonForMomentEnabled
开关开启时,才需要添加新的组件。添加新组件的逻辑相对比较简单,我们把likesStakeView
添加到likesContainerView
里面,然后把likesContainerView
添加到verticalStackView
,这样就可以把点赞的朋友列表放在父视图的底部,最后再把favoriteButton
放到父视图里面。
接着我们为新的组件配置自动布局的约束条件,这就一起来看看setupConstraints()
方法的实现:
func setupConstraints() {
if togglesDataStore.isToggleOn(InternalToggle.isLikeButtonForMomentEnabled) {
likesStakeView.snp.makeConstraints {
$0.top.leading.equalToSuperview().offset(Spacing.twoExtraSmall)
$0.bottom.trailing.equalToSuperview().offset(-Spacing.twoExtraSmall)
}
favoriteButton.snp.makeConstraints {
$0.bottom.trailing.equalToSuperview().offset(-Spacing.medium)
}
}
}
Moments App 使用了 SnapKit 库来配置约束。在这个例子中,我们通过调用equalToSuperview().offset(Spacing.twoExtraSmall)
为likesStakeView
添加填充(padding),然后把favoriteButton
放置在父视图的右下角。
配置好布局以后,我们通过绑定的方式来处理点赞按钮的点击事件,具体代码如下:
func setupBindings() {
if togglesDataStore.isToggleOn(InternalToggle.isLikeButtonForMomentEnabled) {
favoriteButton.rx.tap
.bind(onNext: { [weak self] in
guard let self = self else { return }
if self.favoriteButton.isSelected {
self.viewModel?.like(from: self.userDataStore.userID).subscribe().disposed(by: self.disposeBag)
} else {
self.viewModel?.unlike(from: self.userDataStore.userID).subscribe().disposed(by: self.disposeBag)
}
})
.disposed(by: disposeBag)
}
}
同样地,在进行绑定前,我们先检查isLikeButtonForMomentEnabled
开关是否开启。当开关开启了,我们就使用 RxCocoa 中UIButton
的.rx.tap
扩展属性来绑定favoriteButton
的点击事件。当用户点击了点赞按钮时,就会调用viewModel
的like()
或者unlike()
方法来更新点赞状态。
到此为止,我们已经开发了一个完整的点赞功能。
总结
在这一讲中,我们以添加点赞功能为例讲解了如何快速开发一个新功能。因为 Moments App 使用了 MVVM 和 RxSwift 来进行架构,这就保证了每一层都有明确的责任与分工。
当你开发新功能时,就可以按照我今天讲解的这些步骤一层层来进行开发:添加功能开关,开发网络层、Repository 层、ViewModel 层和 View 层。这样能大大减低代码接手的难度,使得整个团队都遵循统一的步骤与规范,从而降低沟通成本,并同时保证代码的质量。
思考题
你可能已经注意到,当一个类型需要依赖其他类型时,例如当 GetMomentsByUserIDSession 使用 TogglesDataStoreType 时,我们都是通过 init() 方法进行注入的。那为什么我们不在 GetMomentsByUserIDSession 定义 togglesDataStore 属性时直接初始化呢?
你可以把自己的思考写到下面的留言区哦,这一讲就介绍到这里了,下一讲我将介绍如何使用 TDD 来保证功能模块的高质量。