如何在 Swift 中制作引脚注释标注?
How do I make a pin annotation callout in Swift?
我试图让标注工作,但那没有发生,因为我在准备 segue 时做错了什么。我想知道如何能够对另一个视图进行引脚注释标注?
点击callout中的按钮跳转到另一个场景的过程是这样的:
将地图视图的delegate
设置为视图控制器。您可以在 Interface Builder 的 "Connections Inspector" 中或以编程方式执行此操作。您想指定视图控制器也符合 MKMapViewDelegate
。
创建注释时,请确保也设置标题:
let annotation = MKPointAnnotation()
annotation.coordinate = coordinate
annotation.title = ...
mapView.addAnnotation(annotation)
定义一个带按钮标注的注释视图子类:
class CustomAnnotationView: MKPinAnnotationView { // or nowadays, you might use MKMarkerAnnotationView
override init(annotation: MKAnnotation?, reuseIdentifier: String?) {
super.init(annotation: annotation, reuseIdentifier: reuseIdentifier)
canShowCallout = true
rightCalloutAccessoryView = UIButton(type: .infoLight)
}
required init?(coder aDecoder: NSCoder) {
super.init(coder: aDecoder)
}
}
指示您的 MKMapView
使用此注释视图。 iOS 11 简化了该过程,但我将描述如何以两种方式进行操作:
如果您的最低 iOS 版本是 11(或更高版本),您只需将自定义注释视图注册为默认值即可。在 iOS 11 及更高版本中,您通常根本不会实现 mapView(_:viewFor:)
。 (您唯一可以实现该方法的情况是,如果您需要注册多个重用标识符,因为您有多种类型的自定义注释类型。)
override func viewDidLoad() {
super.viewDidLoad()
mapView.register(CustomAnnotationView.self, forAnnotationViewWithReuseIdentifier: MKMapViewDefaultAnnotationViewReuseIdentifier)
}
如果您需要支持 11 之前的 iOS 版本,请确保将您的视图控制器指定为 MKMapView
的委托,然后实现 mapView(_:viewFor:)
:
extension ViewController: MKMapViewDelegate {
func mapView(_ mapView: MKMapView, viewFor annotation: MKAnnotation) -> MKAnnotationView? {
if annotation is MKUserLocation { return nil }
let reuseIdentifier = "..."
var annotationView = mapView.dequeueReusableAnnotationView(withIdentifier: reuseIdentifier)
if annotationView == nil {
annotationView = CustomAnnotationView(annotation: annotation, reuseIdentifier: reuseIdentifier)
} else {
annotationView?.annotation = annotation
}
return annotationView
}
}
例如,这会产生如下所示的标注,右侧有 .infoLight
按钮:
实现 calloutAccessoryControlTapped
以编程方式执行 segue:
func mapView(_ mapView: MKMapView, annotationView view: MKAnnotationView, calloutAccessoryControlTapped control: UIControl) {
performSegue(withIdentifier: "SegueToSecondViewController", sender: view)
}
显然,这假设您已经在两个视图控制器之间定义了一个 segue。
当你segue时,将必要的信息传递给目标场景。例如,您可以传递对注释的引用:
override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
if let destination = segue.destination as? SecondViewController,
let annotationView = sender as? MKPinAnnotationView {
destination.annotation = annotationView.annotation as? MKPointAnnotation
}
}
有关详细信息,请参阅 位置和地图编程指南中的 Creating Callouts。
关于Swift2上面的实现,见previous revision of this answer。
我试图让标注工作,但那没有发生,因为我在准备 segue 时做错了什么。我想知道如何能够对另一个视图进行引脚注释标注?
点击callout中的按钮跳转到另一个场景的过程是这样的:
将地图视图的
delegate
设置为视图控制器。您可以在 Interface Builder 的 "Connections Inspector" 中或以编程方式执行此操作。您想指定视图控制器也符合MKMapViewDelegate
。创建注释时,请确保也设置标题:
let annotation = MKPointAnnotation() annotation.coordinate = coordinate annotation.title = ... mapView.addAnnotation(annotation)
定义一个带按钮标注的注释视图子类:
class CustomAnnotationView: MKPinAnnotationView { // or nowadays, you might use MKMarkerAnnotationView override init(annotation: MKAnnotation?, reuseIdentifier: String?) { super.init(annotation: annotation, reuseIdentifier: reuseIdentifier) canShowCallout = true rightCalloutAccessoryView = UIButton(type: .infoLight) } required init?(coder aDecoder: NSCoder) { super.init(coder: aDecoder) } }
指示您的
MKMapView
使用此注释视图。 iOS 11 简化了该过程,但我将描述如何以两种方式进行操作:如果您的最低 iOS 版本是 11(或更高版本),您只需将自定义注释视图注册为默认值即可。在 iOS 11 及更高版本中,您通常根本不会实现
mapView(_:viewFor:)
。 (您唯一可以实现该方法的情况是,如果您需要注册多个重用标识符,因为您有多种类型的自定义注释类型。)override func viewDidLoad() { super.viewDidLoad() mapView.register(CustomAnnotationView.self, forAnnotationViewWithReuseIdentifier: MKMapViewDefaultAnnotationViewReuseIdentifier) }
如果您需要支持 11 之前的 iOS 版本,请确保将您的视图控制器指定为
MKMapView
的委托,然后实现mapView(_:viewFor:)
:extension ViewController: MKMapViewDelegate { func mapView(_ mapView: MKMapView, viewFor annotation: MKAnnotation) -> MKAnnotationView? { if annotation is MKUserLocation { return nil } let reuseIdentifier = "..." var annotationView = mapView.dequeueReusableAnnotationView(withIdentifier: reuseIdentifier) if annotationView == nil { annotationView = CustomAnnotationView(annotation: annotation, reuseIdentifier: reuseIdentifier) } else { annotationView?.annotation = annotation } return annotationView } }
例如,这会产生如下所示的标注,右侧有
.infoLight
按钮:实现
calloutAccessoryControlTapped
以编程方式执行 segue:func mapView(_ mapView: MKMapView, annotationView view: MKAnnotationView, calloutAccessoryControlTapped control: UIControl) { performSegue(withIdentifier: "SegueToSecondViewController", sender: view) }
显然,这假设您已经在两个视图控制器之间定义了一个 segue。
当你segue时,将必要的信息传递给目标场景。例如,您可以传递对注释的引用:
override func prepare(for segue: UIStoryboardSegue, sender: Any?) { if let destination = segue.destination as? SecondViewController, let annotationView = sender as? MKPinAnnotationView { destination.annotation = annotationView.annotation as? MKPointAnnotation } }
有关详细信息,请参阅 位置和地图编程指南中的 Creating Callouts。
关于Swift2上面的实现,见previous revision of this answer。