- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
在代码中构建布局,一切都很好,直到我需要设置 map 区域:
if let coordinate = item.location?.coordinate {
let region = MKCoordinateRegionMakeWithDistance(coordinate, 100, 100)
let span = MKCoordinateSpanMake(coordinate.latitude, coordinate.longitude)
let regionM = MKCoordinateRegionMake(coordinate, span)
print(mapView.centerCoordinate)
print(coordinate)
print(region)
mapView.region = region
mapView.region.center = coordinate
mapView.regionThatFits(region)
mapView.setCenter(coordinate, animated: false)
mapView.setRegion(region, animated: false)
mapView.setRegion(regionM, animated: false)
print(mapView.region)
print(region)
}
这是失败的测试:
func test_SettingItemInfo_SetsMapLocation() {
let coordinate = CLLocationCoordinate2DMake(51.2277, 6.7735)
let location = Location(name: "Cool", coordinate: coordinate)
let item = Item(title: "Here", location: location)
manager.add(item)
sut.info = (manager, 0)
sut.beginAppearanceTransition(true, animated: true)
sut.endAppearanceTransition()
// possibly asynchronous
XCTAssertEqual(sut.mapView.centerCoordinate.latitude, coordinate.latitude, accuracy: 0.001)
XCTAssertEqual(sut.mapView.centerCoordinate.longitude, coordinate.longitude, accuracy: 0.001)
}
这是打印日志的结果:
CLLocationCoordinate2D(latitude: 37.787685747888986, longitude: -122.40971999999998)
CLLocationCoordinate2D(latitude: 51.227699999999999, longitude: 6.7735000000000003)
MKCoordinateRegion(center: __C.CLLocationCoordinate2D(latitude: 51.227699999999999, longitude: 6.7735000000000003), span: __C.MKCoordinateSpan(latitudeDelta: 0.00089885544064104096, longitudeDelta: 0.0014315673851581932))
MKCoordinateRegion(center: __C.CLLocationCoordinate2D(latitude: 37.787685747888986, longitude: -122.40971999999998), span: __C.MKCoordinateSpan(latitudeDelta: 90.0, longitudeDelta: 180.0))
MKCoordinateRegion(center: __C.CLLocationCoordinate2D(latitude: 51.227699999999999, longitude: 6.7735000000000003), span: __C.MKCoordinateSpan(latitudeDelta: 0.00089885544064104096, longitudeDelta: 0.0014315673851581932))
这是失败的测试日志:
error: -[TDDS4ToDoTests.DetailViewControllerTests test_SettingItemInfo_SetsMapLocation] : XCTAssertEqualWithAccuracy failed: ("37.787685747889") is not equal to ("51.2277") +/- ("0.001") -
error: -[TDDS4ToDoTests.DetailViewControllerTests test_SettingItemInfo_SetsMapLocation] : XCTAssertEqualWithAccuracy failed: ("-122.40972") is not equal to ("6.7735") +/- ("0.001") -
Test Case '-[TDDS4ToDoTests.DetailViewControllerTests test_SettingItemInfo_SetsMapLocation]' failed (213.944 seconds).
理想情况下,第一行应该修复该区域,但我尝试了几种不同的方法(如上所示)。对 mapview.region
的调用是否不会立即设置,因为它是异步的?
最佳答案
MKMapView 中的几乎所有内容都是异步更新的。您需要使用 XCTestExpectation (或类似的)来监听 MKMapViewDelegate 回调。
这是一个令人难以置信的乏味和困难的测试 - 充满了微妙的陷阱。将这个独立的测试放在这里供后代使用:
import MapKit
import XCTest
/// a callback to pipe region changes into our test
protocol MapViewControllerDelegate: class {
func regionChanged(to mapRect: MKMapRect)
}
/// the view controller we want to test
class MapViewController: UIViewController, MKMapViewDelegate {
weak var delegate: MapViewControllerDelegate?
var mapView: MKMapView {
return view as! MKMapView
}
override func loadView() {
let mapView = MKMapView()
mapView.delegate = self
view = mapView
}
func mapView(_ mapView: MKMapView, regionDidChangeAnimated animated: Bool) {
delegate?.regionChanged(to: mapView.visibleMapRect)
}
}
/// a "Spy pattern"-like object for capturing the inputs we want to validate with our test
class Delegate: MapViewControllerDelegate {
var mapRect: MKMapRect?
var regionChangedExpectation: XCTestExpectation?
func regionChanged(to mapRect: MKMapRect) {
guard let regionChangedExpectation = regionChangedExpectation else {
XCTFail("Delegate was not setup correctly. Missing expectation reference")
return
}
self.mapRect = mapRect
regionChangedExpectation.fulfill()
}
}
/// our tests
class TestCase: XCTestCase {
var viewController: MapViewController!
override func setUp() {
super.setUp()
viewController = MapViewController()
// force the view to load
_ = viewController.view
// VERY IMPORTANT: If you don't put the map view in a window none of the delegate callbacks get fired.
// I assume this is some kind of MapKit optimization.
UIApplication.shared.keyWindow?.rootViewController = viewController
}
func testExample() {
// set up an expectation to test the async call
let regionChangedExpectation = expectation(description: "Should update region")
regionChangedExpectation.expectedFulfillmentCount = 1
// create our Spy object to stand in for our usual delegate
let delegate = Delegate()
delegate.regionChangedExpectation = regionChangedExpectation
viewController.delegate = delegate
// the test region
var firstRect = MKMapRect(origin: MKMapPoint(x: 61089821.0, y: 110138019.04306149),
size: MKMapSize(width: 361250.00249999744, height: 691590.91387700045))
// adjust the mapRect so it fits the mapView
firstRect = viewController.mapView.mapRectThatFits(firstRect, edgePadding: .zero)
// trigger a map viewport change
viewController.mapView.visibleMapRect = firstRect
// block and wait for the async call to finish
wait(for: [regionChangedExpectation], timeout: 10)
// make sure we received the region change
guard let mapRect = delegate.mapRect else {
XCTFail("no visible map rect set")
return
}
// make sure we received the RIGHT region change
XCTAssertEqual(firstRect.origin.y, mapRect.origin.y, accuracy: 0.002)
XCTAssertEqual(firstRect.origin.x, mapRect.origin.x, accuracy: 0.002)
XCTAssertEqual(firstRect.size.width, mapRect.size.width, accuracy: 0.002)
XCTAssertEqual(firstRect.size.height, mapRect.size.height, accuracy: 0.002)
}
}
关于swift - 设置 MKMapKit 区域在测试中失败,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48993072/
有没有办法在 .swift 文件(编译成 .swift 模块)中声明函数,如下所示: 你好.swift func hello_world() { println("hello world")
我正在尝试使用 xmpp_messenger_ios 和 XMPPFramework 在 iOS 上执行 MUC 这是加入房间的代码。 func createOrJoinRoomOnXMPP()
我想在我的应用程序上创建一个 3D Touch 快捷方式,我已经完成了有关快捷方式本身的所有操作,它显示正确,带有文本和图标。 当我运行这个快捷方式时,我的应用程序崩溃了,因为 AppDelegate
我的代码如下: let assetTag = Expression("asset_tag") let query2 = mdm.select(mdm[assetTag],os, mac, lastRe
我的 swift 代码如下所示 Family.arrayTuple:[(String,String)]? = [] Family.arrayTupleStorage:String? Family.ar
这是我的 JSON,当我读取 ord 和 uniq 数据时出现错误 let response2 : [String: Any] = ["Response":["status":"SUCCESS","
我想将 swift 扩展文件移动到 swift 包中。但是,将文件移动到 swift 包后,我遇到了这种错误: "Type 'NSAttributedString' has no member 'ma
使用CocoaPods,我们可以设置以下配置: pod 'SourceModel', :configurations => ['Debug'] 有什么方法可以用 Swift Package Manag
我正在 Xcode 中开发一个 swift 项目。我将其称为主要项目。我大部分都在工作。我在日期选择器、日期范围和日期数学方面遇到了麻烦,因此我开始了另一个名为 StarEndDate 的项目,其中只
这是 ObjectiveC 代码: CCSprite *progress = [CCSprite spriteWithImageNamed:@"progress.png"]; mProgressBar
我正在创建一个命令行工具,在 Xcode 中使用 Swift。我想使用一个类似于 grunt 的配置文件确实如此,但我希望它是像 Swift 包管理器的 package.swift 文件那样的快速代码
我假设这意味着使用系统上安装的任何 swift 运行脚本:#!/usr/bin/swift 如何指定脚本适用的解释器版本? 最佳答案 Cato可用于此: #!/usr/bin/env cato 1.2
代码说完全没问题,没有错误,但是当我去运行模拟器的时候,会出现这样的字样: (Swift.LazyMapCollection (_base:[ ] 我正在尝试创建一个显示报价的报价应用。 这是导入
是否可以在运行 Swift(例如 Perfect、Vapor、Kitura 等)的服务器上使用 RealmSwift 并使用它来存储数据? (我正在考虑尝试将其作为另一种解决方案的替代方案,例如 no
我刚开始学习编程,正在尝试完成 Swift 编程书中的实验。 它要求““编写一个函数,通过比较两个 Rank 值的原始值来比较它们。” enum Rank: Int { case Ace = 1 ca
在您将此问题标记为重复之前,我检查了 this question 它对我不起作用。 如何修复这个错误: error: SWIFT_VERSION '5.0' is unsupported, suppo
从 Xcode 9.3 开始,我在我的模型中使用“Swift.ImplicitlyUnwrappedOptional.some”包裹了我的字符串变量 我不知道这是怎么发生的,但它毁了我的应用程序! 我
这个问题在这里已经有了答案: How to include .swift file from other .swift file in an immediate mode? (2 个答案) 关闭 6
我正在使用 Swift Package Manager 创建一个应用程序,我需要知道构建项目的配置,即 Debug 或 Release。我试图避免使用 .xcodeproj 文件。请有人让我知道这是否
有一个带有函数定义的文件bar.swift: func bar() { println("bar") } 以及一个以立即模式运行的脚本foo.swift: #!/usr/bin/xcrun s
我是一名优秀的程序员,十分优秀!