我正在使用MapView,点击任何自定义注释引脚,我显示自定义标注视图(从xib文件加载)。
从这个自定义标注我有一个UIButton,我已经可以检测到点击这个按钮但我想在地图上访问,如:基本标注中的:view?.rightCalloutAccessoryView。
func mapView(_ mapView: MKMapView, didSelect view: MKAnnotationView, {
if view.annotation!.isKind(of: MKUserLocation.self){
return
}
let customView = (Bundle.main.loadNibNamed("CustomCalloutView", owner: self, options: nil))?[0] as! CustomCalloutView;
let calloutViewFrame = customView.frame;
customView.frame = CGRect(x: -calloutViewFrame.size.width/2.23, y: -calloutViewFrame.size.height+10, width: 315, height: 170)
view.addSubview(customView)
let region = MKCoordinateRegion(center: pinToZoomOn!.coordinate, span: span)
mapView.setRegion(region, animated: true)
}
路线是从经典标注中正确计算的,但我不知道如何从我的自定义标注按钮访问我的地图。
我的CustomCalloutViewClass:
import UIKit
import MapKit
class CustomCalloutView: MKAnnotationView {
@IBOutlet weak var goButton: UIButton!
@IBAction func goButton(_ sender: AnyObject) {
print("Button clicked sucessfully")
}
// MARK: - Detect taps on callout
override func hitTest(_ point: CGPoint, with event: UIEvent?) -> UIView? {
let hitView = super.hitTest(point, with: event)
if hitView != nil {
superview?.bringSubview(toFront: self)
}
return hitView
}
override func point(inside point: CGPoint, with event: UIEvent?) -> Bool {
let rect = self.bounds
var isInside = rect.contains(point)
if !isInside {
for view in subviews {
isInside = view.frame.contains(point)
if isInside {
break
}
}
}
return isInside
}
}
如果有人有想法会有所帮助,我会坚持这个问题。
提前谢谢。
答案 0 :(得分:0)
添加将在按钮操作上调用的闭包。闭包将捕获MKMapView
实例,你可以将我们放在里面。
class CustomCalloutView: MKAnnotationView {
var didTapGoButton: (() -> Void)?
@IBAction func goButton(_ sender: AnyObject) {
didTapGoButton?()
}
}
这不是一个干净的解决方案,但在某些情况下可能是一种选择。您只需要在MKMapView
CustomCalloutView
实例引用的弱属性
class CustomCalloutView: MKAnnotationView {
weak var mapView: MKMapView?
}
这就是为两种解决方案连接CustomCalloutView
的方法。请记住使用swift捕获列表来捕获对MKMapView
实例的弱引用。没有它,您可以创建一个强大的参考周期。
func mapView(_ mapView: MKMapView, didSelect view: MKAnnotationView) {
// ...
let customView = (Bundle.main.loadNibNamed("CustomCalloutView", owner: self, options: nil))?[0] as! CustomCalloutView;
// Option 1
customView.didTapGoButton = { [weak mapView ] in
print(mapView?.annotations.count)
}
// Option 2
customView.mapView = mapView
// ...
}
答案 1 :(得分:0)
非常感谢你的帮助!
我使用第一个选项来捕获MKMapView实例。
如果有人遇到同样的问题,你可以使用第一个选项并在你的didSelect MapView函数中添加它:
let selectedLoc = view.annotation
print("Annotation '\(String(describing: selectedLoc?.title!))' has been selected")
let location = view.annotation as! YourCustomClassType
let launchOptions = [MKLaunchOptionsDirectionsModeKey : MKLaunchOptionsDirectionsModeDriving]
customView.didTapGoButton = { location.mapItem().openInMaps(launchOptions: launchOptions) }
感谢的
答案 2 :(得分:0)
var didTapGoButton:(()-> Void)?这种类型的闭包对于通过mkmapview的调用layoutviews中的按钮动作从视图导航到viewControllers非常有用。