iOS: какой самый быстрый и эффективный способ сделать снимок экрана программным способом? - PullRequest
77 голосов
/ 01 ноября 2011

в своем приложении для iPad я бы хотел сделать скриншот UIView, занимающего большую часть экрана.К сожалению, подпредставления довольно глубоко вложены, поэтому для создания скриншота и анимации скручивания страницы требуется много времени.

Есть ли более быстрый способ, чем "обычный"?

UIGraphicsBeginImageContext(self.bounds.size);
[self.layer renderInContext:UIGraphicsGetCurrentContext()];
UIImage *resultingImage = UIGraphicsGetImageFromCurrentImageContext();
UIGraphicsEndImageContext();

Если возможно, я бы хотел избежать кэширования или реструктуризации своего представления.

Ответы [ 6 ]

111 голосов
/ 05 ноября 2011

Я нашел лучший метод, который использует API снимков всякий раз, когда это возможно.

Надеюсь, это поможет.

class func screenshot() -> UIImage {
    var imageSize = CGSize.zero

    let orientation = UIApplication.shared.statusBarOrientation
    if UIInterfaceOrientationIsPortrait(orientation) {
        imageSize = UIScreen.main.bounds.size
    } else {
        imageSize = CGSize(width: UIScreen.main.bounds.size.height, height: UIScreen.main.bounds.size.width)
    }

    UIGraphicsBeginImageContextWithOptions(imageSize, false, 0)
    for window in UIApplication.shared.windows {
        window.drawHierarchy(in: window.bounds, afterScreenUpdates: true)
    }

    let image = UIGraphicsGetImageFromCurrentImageContext()
    UIGraphicsEndImageContext()
    return image!
}

Хотите узнать больше о снимках iOS 7?

Версия Objective-C:

+ (UIImage *)screenshot
{
    CGSize imageSize = CGSizeZero;

    UIInterfaceOrientation orientation = [UIApplication sharedApplication].statusBarOrientation;
    if (UIInterfaceOrientationIsPortrait(orientation)) {
        imageSize = [UIScreen mainScreen].bounds.size;
    } else {
        imageSize = CGSizeMake([UIScreen mainScreen].bounds.size.height, [UIScreen mainScreen].bounds.size.width);
    }

    UIGraphicsBeginImageContextWithOptions(imageSize, NO, 0);
    CGContextRef context = UIGraphicsGetCurrentContext();
    for (UIWindow *window in [[UIApplication sharedApplication] windows]) {
        CGContextSaveGState(context);
        CGContextTranslateCTM(context, window.center.x, window.center.y);
        CGContextConcatCTM(context, window.transform);
        CGContextTranslateCTM(context, -window.bounds.size.width * window.layer.anchorPoint.x, -window.bounds.size.height * window.layer.anchorPoint.y);
        if (orientation == UIInterfaceOrientationLandscapeLeft) {
            CGContextRotateCTM(context, M_PI_2);
            CGContextTranslateCTM(context, 0, -imageSize.width);
        } else if (orientation == UIInterfaceOrientationLandscapeRight) {
            CGContextRotateCTM(context, -M_PI_2);
            CGContextTranslateCTM(context, -imageSize.height, 0);
        } else if (orientation == UIInterfaceOrientationPortraitUpsideDown) {
            CGContextRotateCTM(context, M_PI);
            CGContextTranslateCTM(context, -imageSize.width, -imageSize.height);
        }
        if ([window respondsToSelector:@selector(drawViewHierarchyInRect:afterScreenUpdates:)]) {
            [window drawViewHierarchyInRect:window.bounds afterScreenUpdates:YES];
        } else {
            [window.layer renderInContext:context];
        }
        CGContextRestoreGState(context);
    }

    UIImage *image = UIGraphicsGetImageFromCurrentImageContext();
    UIGraphicsEndImageContext();
    return image;
}
18 голосов
/ 01 ноября 2011

РЕДАКТИРОВАТЬ 3 октября. 2013 Обновлено для поддержки нового сверхбыстрого drawViewHierarchyInRect: afterScreenUpdates: метод в iOS 7.


Нет.RenderInContext от CALayer: насколько я знаю, единственный способ сделать это.Вы можете создать категорию UIView следующим образом, чтобы вам было легче двигаться вперед:

UIView + Screenshot.h

#import <UIKit/UIKit.h>

@interface UIView (Screenshot)

- (UIImage*)imageRepresentation;

@end

UIView + Screenshot.m

#import <QuartzCore/QuartzCore.h>
#import "UIView+Screenshot.h"

@implementation UIView (Screenshot)

- (UIImage*)imageRepresentation {

    UIGraphicsBeginImageContextWithOptions(self.bounds.size, YES, self.window.screen.scale);

    /* iOS 7 */
    if ([self respondsToSelector:@selector(drawViewHierarchyInRect:afterScreenUpdates:)])            
        [self drawViewHierarchyInRect:self.bounds afterScreenUpdates:NO];
    else /* iOS 6 */
        [self.layer renderInContext:UIGraphicsGetCurrentContext()];

    UIImage* ret = UIGraphicsGetImageFromCurrentImageContext();

    UIGraphicsEndImageContext();

    return ret;

}

@end

Таким образом, вы можете сказать [self.view.window imageRepresentation] в контроллере представления и получить полный скриншот вашего приложения.Это может исключить строку состояния, хотя.

РЕДАКТИРОВАТЬ:

И я могу добавить.Если у вас есть UIView с прозрачным содержимым, и вам также необходимо представление изображения С подложным содержимым, вы можете получить представление изображения представления контейнера и обрезать это изображение, просто взяв прямоугольник подпредставления и преобразовав его в контейнерсистема координат представления.

[view convertRect:self.bounds toView:containerView]

Чтобы обрезать, см. ответ на этот вопрос: Обрезка UIImage

10 голосов
/ 21 сентября 2013

В iOS 7 появился новый метод, который позволяет рисовать иерархию представлений в текущем графическом контексте. Это может быть использовано, чтобы получить UIImage очень быстро.

Реализовано как метод категории в UIView:

- (UIImage *)pb_takeSnapshot {
    UIGraphicsBeginImageContextWithOptions(self.bounds.size, NO, [UIScreen mainScreen].scale);

    [self drawViewHierarchyInRect:self.bounds afterScreenUpdates:YES];

    UIImage *image = UIGraphicsGetImageFromCurrentImageContext();
    UIGraphicsEndImageContext();
    return image;
}

Это значительно быстрее, чем существующий метод renderInContext:.

ОБНОВЛЕНИЕ ДЛЯ SWIFT : расширение, которое делает то же самое:

extension UIView {

    func pb_takeSnapshot() -> UIImage {
        UIGraphicsBeginImageContextWithOptions(self.bounds.size, false, UIScreen.mainScreen().scale);

        self.drawViewHierarchyInRect(self.bounds, afterScreenUpdates: true)

        // old style: self.layer.renderInContext(UIGraphicsGetCurrentContext())

        let image = UIGraphicsGetImageFromCurrentImageContext();
        UIGraphicsEndImageContext();
        return image;
    }
}
3 голосов
/ 09 января 2014

Я объединил ответы на одну функцию, которая будет работать для любых версий iOS, даже для устройств с сетчаткой или без удержания.

- (UIImage *)screenShot {
    if ([[UIScreen mainScreen] respondsToSelector:@selector(scale)])
        UIGraphicsBeginImageContextWithOptions(self.view.bounds.size, NO, [UIScreen mainScreen].scale);
    else
        UIGraphicsBeginImageContext(self.view.bounds.size);

    #ifdef __IPHONE_7_0
        #if __IPHONE_OS_VERSION_MAX_ALLOWED >= 70000
            [self.view drawViewHierarchyInRect:self.view.bounds afterScreenUpdates:YES];
        #endif
    #else
            [self.view.layer renderInContext:UIGraphicsGetCurrentContext()];
    #endif

    UIImage *image = UIGraphicsGetImageFromCurrentImageContext();
    UIGraphicsEndImageContext();
    return image;
}
2 голосов
/ 12 октября 2013

Для меня настройка качества интерполяции прошла долгий путь.

CGContextSetInterpolationQuality(ctx, kCGInterpolationNone);

Если вы снимаете очень подробные изображения, это решение может быть неприемлемым. Если вы снимаете текст, вы вряд ли заметите разницу.

Это значительно сокращает время, необходимое для создания снимка, а также делает изображение, которое потребляет гораздо меньше памяти.

Это все еще выгодно с drawViewHierarchyInRect: afterScreenUpdates: метод.

1 голос
/ 08 ноября 2011

В качестве альтернативы вы просите прочитать графический процессор (поскольку экран составлен из любого количества полупрозрачных изображений), что также является по своей сути медленной операцией.

...