32

这是我的代码。我想添加我自己的自定义标注视图而不是 iOS 默认值。我知道只有左标注和右标注视图,但我需要添加一个带有我自己的背景和标签的视图工具提示类型。

    - (MKAnnotationView *)mapView:(MKMapView *)mapView viewForAnnotation:(id <MKAnnotation>)annotation
    {
        MKAnnotationView *userAnnotationView = nil;
        if ([annotation isKindOfClass:MKUserLocation.class])
        {
            userAnnotationView = (MKAnnotationView *)[mapView dequeueReusableAnnotationViewWithIdentifier:@"UserLocation"];
            if (userAnnotationView == nil)  {
            userAnnotationView = [[MKAnnotationView alloc] initWithAnnotation:annotation reuseIdentifier:@"UserLocation"];
            }
            else
                userAnnotationView.annotation = annotation;

            userAnnotationView.enabled = YES;


            userAnnotationView.canShowCallout = YES;
            userAnnotationView.image = [UIImage imageNamed:@"map_pin.png"];
            UIView *view = [[UIView alloc]initWithFrame:CGRectMake(0,0,141,108)];
            view.backgroundColor = [UIColor clearColor];
            UIImageView *imgView = [[UIImageView alloc]initWithImage:[UIImage imageNamed:@"toltip.png"]];
            [view addSubview:imgView];
            userAnnotationView.leftCalloutAccessoryView = view;



            return userAnnotationView;
        }

}

图片供参考

4

6 回答 6

133

我遇到了同样的问题,最终做了以下事情:

当我收到 mapView 委托回调时

-(void)mapView:(MKMapView *)mapView didSelectAnnotationView:(MKAnnotationView *)view

(是时候展示我的自定义 CalloutView 了)

我使用作为参数接收的视图*(MKAnnotationView )view (这是一个引脚视图),并使用简单地将我的自定义视图添加到该引脚视图

 [view addSubview:customView];

它会将您的自定义视图添加到该引脚视图的顶部,因此如果我们希望它位于引脚上方,我们必须更改自定义视图中心属性,如下所示:

CGRect customViewRect = customView.frame;
        CGRect rect = CGRectMake(-customViewRect.size.width/2, -customViewRect.size.height-7, customViewRect.size.width, customViewRect.size.height);
        customView.frame = rect;
[view addSubview:customView];

就我而言,它看起来像这样

在此处输入图像描述

!注意

有一个警告可以很容易地解决,您的自定义视图将忽略触摸事件,因为mapView以不同的方式处理触摸。这是一个快速修复

1)子类MKAnnotationView(或MKPinAnnotationView取决于你需要什么)

2)在您的 mapView 委托中

- (MKAnnotationView *)mapView:(MKMapView *)mapView viewForAnnotation:(id <MKAnnotation>)annotation

使用您的子类而不是 MKAnnotationView/MKPinAnnotationView

3)在您的子类 .m 文件中覆盖以下两种方法:

- (UIView*)hitTest:(CGPoint)point withEvent:(UIEvent*)event
{
    UIView* hitView = [super hitTest:point withEvent:event];
    if (hitView != nil)
    {
        [self.superview bringSubviewToFront:self];
    }
    return hitView;
}

- (BOOL)pointInside:(CGPoint)point withEvent:(UIEvent*)event
{
    CGRect rect = self.bounds;
    BOOL isInside = CGRectContainsPoint(rect, point);
    if(!isInside)
    {
        for (UIView *view in self.subviews)
        {
            isInside = CGRectContainsPoint(view.frame, point);
            if(isInside)
                break;
        }
    }
    return isInside;
}

全部完成!

于 2013-10-16T13:37:19.410 回答
10

我创建了一个库来显示自定义标注。

DXCustomCallout-ObjC

您可以为标注传递任何自定义视图!它也支持 UIControls 的事件。

自定义标注

于 2015-04-12T20:42:19.120 回答
2

到 Swift 上面的最佳答案

这可以节省一些时间来重写自己。

!笔记:

有一个警告可以很容易地修复,您的自定义视图将忽略触摸事件,因为 mapView 以不同的方式处理触摸。这是一个快速修复:

1)子类MKAnnotationView(或MKPinAnnotationView取决于你需要什么)

2) 在您的 mapView 委托中

func mapView(mapView: MKMapView, viewForAnnotation annotation: MKAnnotation) -> MKAnnotationView?

使用您的子类而不是 MKAnnotationView/MKPinAnnotationView

3)在您的子类 .m 文件中覆盖以下两种方法:

override func hitTest(point: CGPoint, withEvent event: UIEvent?) -> UIView? {

    let hitView = super.hitTest(point, withEvent: event)

    if (hitView != nil)
    {
        self.superview?.bringSubviewToFront(self)
    }

    return hitView;

}

override func pointInside(point: CGPoint, withEvent event: UIEvent?) -> Bool {

    let rect = self.bounds
    var isInside = CGRectContainsPoint(rect, point)

    if(!isInside) {
        for view in self.subviews {

            isInside = CGRectContainsPoint(view.frame, point)
            break;
        }
    }

    return isInside
}
于 2016-07-31T10:12:45.707 回答
1

我刚刚为我的地图视图创建了一个自定义标注,以避免在点击时标注消失的问题。是我采取的步骤的要点。

于 2013-11-16T02:51:11.030 回答
0

你可以使用这个库。它还为标注视图提供了一个现成的模板。为您的自定义视图添加很酷的动画和锚视图。

https://github.com/okhanokbay/MapViewPlus

在此处输入图像描述 在此处输入图像描述

于 2018-03-10T12:56:07.297 回答