自定义标注视图按钮映射

时间:2017-10-28 12:20:21

标签: ios swift android-mapview

我正在使用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
}
}

如果有人有想法会有所帮助,我会坚持这个问题。

提前谢谢。

3 个答案:

答案 0 :(得分:0)

选项1:在传递给CustomCalloutView

的闭包中捕获MKMapView实例

添加将在按钮操作上调用的闭包。闭包将捕获MKMapView实例,你可以将我们放在里面。

class CustomCalloutView: MKAnnotationView {
    var didTapGoButton: (() -> Void)?

    @IBAction func goButton(_ sender: AnyObject) {
        didTapGoButton?()
    }
}

选项2:添加对MKMapView的弱引用作为标注

的属性

这不是一个干净的解决方案,但在某些情况下可能是一种选择。您只需要在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非常有用。