简体   繁体   English

iOS7中MKMapView的快照

[英]Snapshot of MKMapView in iOS7

I am trying to create a snapshot of a MKMapView in iOS7 application the same way it's recommended everywhere for previous iOS versions: 我试图在iOS7应用程序中创建MKMapView的快照,就像之前iOS版本推荐的那样:

- (UIImage*) renderMapViewToImage
{
   UIGraphicsBeginImageContextWithOptions(mapView.frame.size, NO, 0.0);
   [mapView.layer renderInContext:UIGraphicsGetCurrentContext()];
   UIImage *image = UIGraphicsGetImageFromCurrentImageContext();
   UIGraphicsEndImageContext(); 
   return image;
}

However, the image returned is a black rectangle with a blue current location dot on top of it. 但是,返回的图像是一个黑色矩形,顶部有一个蓝色的当前位置点。 I've tried using different sublayers of the mapView as well, but the result is always the same. 我也试过使用mapView的不同子层,但结果总是一样的。

Does anyone know how to take MKMapView snapshots in iOS7 ? 有谁知道如何在iOS7中拍摄MKMapView快照?

You can use MKMapSnapshotter and grab the image from the resulting MKMapSnapshot . 您可以使用MKMapSnapshotter并从生成的MKMapSnapshot获取image See the discussion of it WWDC 2013 session video, Putting Map Kit in Perspective . 请参阅WWDC 2013会话视频的讨论, 将Map Kit放入透视图中

For example: 例如:

MKMapSnapshotOptions *options = [[MKMapSnapshotOptions alloc] init];
options.region = self.mapView.region;
options.scale = [UIScreen mainScreen].scale;
options.size = self.mapView.frame.size;

MKMapSnapshotter *snapshotter = [[MKMapSnapshotter alloc] initWithOptions:options];
[snapshotter startWithCompletionHandler:^(MKMapSnapshot *snapshot, NSError *error) {
    UIImage *image = snapshot.image;
    NSData *data = UIImagePNGRepresentation(image);
    [data writeToFile:[self snapshotFilename] atomically:YES];
}];

Having said that, the renderInContext solution still works for me. 话虽如此, renderInContext解决方案仍适用于我。 There are notes about only doing that in the main queue in iOS7, but it still seems to work. 有关于仅在iOS7的主队列中执行此操作的说明,但它似乎仍然有效。 But MKMapSnapshotter seems like the more appropriate solution for iOS7. MKMapSnapshotter似乎是更适合iOS7的解决方案。


If you want to include some annotations in the snapshot, you have to draw them manually (!). 如果要在快照中包含一些注释,则必须手动绘制它们(!)。 This is discussed in some detail at the end of the Putting Map Kit in Perspective video. 在透视视频中的Putting Map Kit结尾处详细讨论。 I have to say that this is one of the least elegant implementations that I've ever seen Apple advise. 我不得不说这是我见过苹果建议的最不优雅的实现之一。 Anyway, in iOS, it might look like: 无论如何,在iOS中,它可能看起来像:

MKMapSnapshotOptions *options = [[MKMapSnapshotOptions alloc] init];
options.region = self.mapView.region;
options.scale = [UIScreen mainScreen].scale;
options.size = self.mapView.frame.size;

MKMapSnapshotter *snapshotter = [[MKMapSnapshotter alloc] initWithOptions:options];
[snapshotter startWithQueue:dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0) completionHandler:^(MKMapSnapshot *snapshot, NSError *error) {

    // get the image associated with the snapshot

    UIImage *image = snapshot.image;

    // Get the size of the final image

    CGRect finalImageRect = CGRectMake(0, 0, image.size.width, image.size.height);

    // Get a standard annotation view pin. Clearly, Apple assumes that we'll only want to draw standard annotation pins!

    MKAnnotationView *pin = [[MKPinAnnotationView alloc] initWithAnnotation:nil reuseIdentifier:@""];
    UIImage *pinImage = pin.image;

    // ok, let's start to create our final image

    UIGraphicsBeginImageContextWithOptions(image.size, YES, image.scale);

    // first, draw the image from the snapshotter

    [image drawAtPoint:CGPointMake(0, 0)];

    // now, let's iterate through the annotations and draw them, too

    for (id<MKAnnotation>annotation in self.mapView.annotations)
    {
        CGPoint point = [snapshot pointForCoordinate:annotation.coordinate];
        if (CGRectContainsPoint(finalImageRect, point)) // this is too conservative, but you get the idea
        {
            CGPoint pinCenterOffset = pin.centerOffset;
            point.x -= pin.bounds.size.width / 2.0;
            point.y -= pin.bounds.size.height / 2.0;
            point.x += pinCenterOffset.x;
            point.y += pinCenterOffset.y;

            [pinImage drawAtPoint:point];
        }
    }

    // grab the final image

    UIImage *finalImage = UIGraphicsGetImageFromCurrentImageContext();
    UIGraphicsEndImageContext();

    // and save it

    NSData *data = UIImagePNGRepresentation(finalImage);
    [data writeToFile:[self snapshotFilename] atomically:YES];
}];

For MacOS implementation, see that video for more information, but the technique is basically the same (the mechanism for creating the images is slightly different). 对于MacOS实现,请参阅该视频以获取更多信息,但技术基本相同(创建图像的机制略有不同)。

For Swift 3 对于Swift 3

Here is a swift 3 version I modified from this article: Render a Map as an Image using MapKit 这是我从本文修改的一个快速3版本: 使用MapKit将地图渲染为图像

The following code allows you to snapshot a region based on both Point(1 coordinate) and Polyline(several coordinates) 以下代码允许您基于Point(1坐标)和Polyline(几个坐标)对区域进行快照

func takeSnapShot() {
    let mapSnapshotOptions = MKMapSnapshotOptions()

    // Set the region of the map that is rendered. (by one specified coordinate)
    // let location = CLLocationCoordinate2DMake(24.78423, 121.01836) // Apple HQ
    // let region = MKCoordinateRegionMakeWithDistance(location, 1000, 1000)

    // Set the region of the map that is rendered. (by polyline)
    // var yourCoordinates = [CLLocationCoordinate2D]()  <- initinal this array with your polyline coordinates
    let polyLine = MKPolyline(coordinates: &yourCoordinates, count: yourCoordinates.count)
    let region = MKCoordinateRegionForMapRect(polyLine.boundingMapRect)

    mapSnapshotOptions.region = region

    // Set the scale of the image. We'll just use the scale of the current device, which is 2x scale on Retina screens.
    mapSnapshotOptions.scale = UIScreen.main.scale

    // Set the size of the image output.
    mapSnapshotOptions.size = CGSize(width: IMAGE_VIEW_WIDTH, height: IMAGE_VIEW_HEIGHT)

    // Show buildings and Points of Interest on the snapshot
    mapSnapshotOptions.showsBuildings = true
    mapSnapshotOptions.showsPointsOfInterest = true

    let snapShotter = MKMapSnapshotter(options: mapSnapshotOptions)

    snapShotter.start() { snapshot, error in
        guard let snapshot = snapshot else {
            return
        }
        self.imageView.image = snapshot.image
    }
}

For iOS 10 and above you can use UIGraphicsImageRenderer class for rendering any view to image (just in case if you don't want to use MKMapSnapshotter , since i am using MapBox ). 对于iOS 10及更高版本,您可以使用UIGraphicsImageRenderer类将任何视图呈现为图像(以防万一您不想使用MKMapSnapshotter ,因为我使用的是MapBox )。

let render = UIGraphicsImageRenderer(size: self.mapView.bounds.size)
let image = render.image { ctx in
  self.mapView.drawHierarchy(in: self.mapView.bounds, afterScreenUpdates: true)
}

Result : 结果

在此输入图像描述

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM