46

我有一个注释列表(MKPointAnnotation)。我有一个用于整个视图的 UIViewController,MKMapView 实现控制器,我认为这对于检测用户与地图的交互很有用,我自己的 MKPointAnnotation 实现(子类)控制器告诉如何显示注释。

但是,我对用户检测到点击事件感到震惊。

谷歌搜索告诉我,我必须通过实现以下功能来做一些事情。

- (void)mapView:(MKMapView *)mapView annotationView:(MKAnnotationView *)view calloutAccessoryControlTapped:(UIControl *)control

而且我必须在一些实现 MapViewDelegate (协议)的类中实现这一点。

但是我很迷茫,无法前进。谁能告诉我在哪里做什么?

对不起所有的大惊小怪!

4

3 回答 3

106

有两种方法可以检测用户与注释视图的交互。常见的技术是为您的MKAnnotationView. viewForAnnotation并且您在标准方法中为您的注释创建注释视图:

- (MKAnnotationView *)mapView:(MKMapView *)mapView viewForAnnotation:(id <MKAnnotation>)annotation
{
    if ([annotation isKindOfClass:[MKUserLocation class]])
        return nil;

    MKAnnotationView *annotationView = [[MKPinAnnotationView alloc] initWithAnnotation:annotation reuseIdentifier:@"loc"];
    annotationView.canShowCallout = YES;
    annotationView.rightCalloutAccessoryView = [UIButton buttonWithType:UIButtonTypeDetailDisclosure];

    return annotationView;
}

通过这样做,你会得到一个标注,但你正在添加一个正确的附件,在我上面的示例中,它是一个披露指示器。这样,他们点击您的注释视图(在我上面的示例中,地图上的一个图钉),他们会看到标注,并且当他们点击该标注的正确附件(本示例中的小披露指示器)时,您calloutAccessoryControlTapped被称为 (在下面的示例中,对某些详细视图控制器执行 segue):

- (void)mapView:(MKMapView *)mapView annotationView:(MKAnnotationView *)view calloutAccessoryControlTapped:(UIControl *)control
{
    [self performSegueWithIdentifier:@"DetailsIphone" sender:view];
}

这是 iPhone 小屏幕上非常典型的用户体验。

但是,如果您不喜欢那个 UX 并且您不想要标准标注,而是希望发生其他事情,您可以定义您的MKAnnotationView以便不显示标注,而是拦截它并执行其他操作(例如,在 iPad 地图应用程序上,您可能会显示一些更复杂的弹出框而不是标准标注)。例如,您可以MKAnnotationView不显示标注:

- (MKAnnotationView *)mapView:(MKMapView *)mapView viewForAnnotation:(id <MKAnnotation>)annotation
{
    if ([annotation isKindOfClass:[MKUserLocation class]])
        return nil;

    MKAnnotationView *annotationView = [[MKPinAnnotationView alloc] initWithAnnotation:annotation reuseIdentifier:@"loc"];
    annotationView.canShowCallout = NO;

    return annotationView;
}

但是您可以手动处理didSelectAnnotationView以检测用户何时点击了您的MKAnnotationView,在此示例中显示了一个弹出框:

- (void)mapView:(MKMapView *)mapView didSelectAnnotationView:(MKAnnotationView *)view
{
    [mapView deselectAnnotation:view.annotation animated:YES];

    DetailsViewController *controller = [self.storyboard instantiateViewControllerWithIdentifier:@"DetailsPopover"];
    controller.annotation = view.annotation;
    self.popover = [[UIPopoverController alloc] initWithContentViewController:controller];
    self.popover.delegate = self;
    [self.popover presentPopoverFromRect:view.frame
                                  inView:view.superview
                permittedArrowDirections:UIPopoverArrowDirectionAny
                                animated:YES];
}

我在此处的回答中包含了由上述代码生成的用户界面的一些屏幕快照。

于 2013-03-08T15:50:12.647 回答
9

Swift 3 的抢夺示例:

class MapViewController: UIViewController
{
    override func viewDidLoad()
    {
        super.viewDidLoad()

        // Create map view in storyboard
        view.delegate = self
    }
}

extension MapViewController: MKMapViewDelegate
{
    func mapView(_ mapView: MKMapView, viewFor annotation: MKAnnotation) -> MKAnnotationView?
    {
        let annotationView = MKPinAnnotationView(annotation: annotation, reuseIdentifier: "annotationView")
        annotationView.canShowCallout = true
        annotationView.rightCalloutAccessoryView = UIButton.init(type: UIButtonType.detailDisclosure)

        return annotationView
    }

    func mapView(_ mapView: MKMapView, annotationView view: MKAnnotationView, calloutAccessoryControlTapped control: UIControl)
    {
        guard let annotation = view.annotation else
        {
            return
        }

        let urlString = "http://maps.apple.com/?sll=\(annotation.coordinate.latitude),\(annotation.coordinate.longitude)"
        guard let url = URL(string: urlString) else
        {
            return
        }

        UIApplication.shared.openURL(url)
    }
}
于 2016-11-30T10:53:38.823 回答
6

在这个方法中添加按钮

-(MKAnnotationView *)mapView:(MKMapView *)mapView viewForAnnotation:(id <MKAnnotation>)annotation 
{
  pinView.rightCalloutAccessoryView = [UIButton buttonWithType:UIButtonTypeDetailDisclosure];
  pinView.canShowCallout = YES;
}

Then callout method
- (void)mapView:(MKMapView *)mapView annotationView:(MKAnnotationView *)view calloutAccessoryControlTapped:(UIControl *)control
{
      InfoView *infoView = [[InfoView alloc]initWithNibName:@"InfoView" bundle:nil];
        [self.navigationController pushViewController:infoView animated:YES];

} 
于 2013-03-08T11:21:18.123 回答