我正在尝试将UIView
上的扩展名进行单位测试,该扩展名调用animateWithDuration:animations:completion:
。
不幸的是,测试案例中的所有动画都会立即取消,因此完成块的isFinished
参数始终为false
。
有人知道如何使动画在测试案例中工作吗?
这是我的游乐场代码:
import UIKit
import XCTest
extension UIView {
func fadeOut(duration: TimeInterval, completion: ((Bool) -> Void)?) {
UIView.animate(withDuration: duration,
animations: {
self.alpha = 0.0
},
completion: { isFinished in
self.isHidden = isFinished
completion?(isFinished)
})
}
}
class UIViewTests: XCTestCase {
func testFadeView() {
// Given
let expectation = self.expectation(description: "Expect completion handler to be called.")
let view = UIView()
view.alpha = 1.0
// When
print("Date before", Date())
view.fadeOut(duration: 1.0) { (isFinished) in
print("Date completed", Date())
print("isFinished", isFinished)
expectation.fulfill()
}
// Then
wait(for: [expectation], timeout: 1.1)
XCTAssertTrue(view.isHidden) // <- this assertion always fails, cause `isFinished` inside the completion handler is always `false`.
XCTAssertEqual(view.alpha, 0.0, accuracy: CGFloat.ulpOfOne)
}
}
UIViewTests.defaultTestSuite.run()
断言XCTAssertTrue(view.isHidden)
总是失败。此外,日志语句输出:
Date before 2019-05-20 23:30:35 +0000
Date completed 2019-05-20 23:30:35 +0000
基本上动画立即被杀死。
使动画配有正确的标志,视图需要在可见的uiwindow中。
let window = UIWindow()
window.addSubview(view)
window.isHidden = false
这样,您的测试就成功了。但是Uikit不会在测试结束时给跑步循环额外踢,而不会清理窗户。所以添加
func tearDown() {
super.tearDown()
RunLoop.current.run(until: Date())
}
然后将窗口(以及其中的任何内容(被交易。
现在它正在工作,您可以通过减少持续时间来节省时间。我使用duration: 0.001