Как сделать скриншот видимой области UIScrollView?

Как сделать снимок экрана 1: 1 видимой области UIScrollView? Контент может быть больше или меньше, чем границы UIScrollView, а также наполовину скрыт (я реализовал пользовательскую прокрутку для контента меньшего размера, поэтому он не находится в верхнем левом углу). Я достиг желаемого результата на симуляторе, но не на самом устройстве:

-(UIImage *)imageFromCombinedContext:(UIView *)background {
      UIImage *image;
      CGRect vis = background.bounds;
      CGSize size = vis.size;
      UIGraphicsBeginImageContext(size);
      [background.layer affineTransform];
      [background.layer renderInontext:UIGraphicsGetCurrentContext()];
      image = UIGraphicsGetImageFromCurrentImageContext();
      UIGraphicsEndImageContext();
      CGImageRef imref = CGImageCreateWithImageInRect([image CGImage], vis);
      image = [UIImage imageWithCGImage:imref];
      CGImageRelease(imref);
      return image;
}

Ответы на вопрос(8)

UIScrollView extension с переводом, без медленной обрезки

extension UIScrollView {

    var snapshotVisibleArea: UIImage? {
        UIGraphicsBeginImageContext(bounds.size)
        UIGraphicsGetCurrentContext()?.translateBy(x: -contentOffset.x, y: -contentOffset.y)
        layer.render(in: UIGraphicsGetCurrentContext()!)
        let image = UIGraphicsGetImageFromCurrentImageContext()
        UIGraphicsEndImageContext()
        return image
    }
}

так как еслиUIScrollView содержит большую область содержимого, мы нарисуем всю эту область содержимого (даже за пределами видимых границ). Ответ @ Concuror имеет проблему, заключающуюся в том, что он также нарисует все, что находится поверхUIScrollView.

Моим решением было поставитьUIScrollView внутриUIView называетсяcontainerView с теми же границами, а затем визуализироватьcontainerView:

containerView.renderInContext(context)
Решение Вопроса

а затем обрезал его до размера и положенияUIScrollView Рамка.

-(UIImage *)imageFromCombinedContext:(UIView *)background 
{
      UIImage *image;
      CGSize size = self.view.frame.size;
      UIGraphicsBeginImageContext(size);
      [background.layer affineTransform];
      [self.view.layer.layer renderInContext:UIGraphicsGetCurrentContext()];
      image = UIGraphicsGetImageFromCurrentImageContext();
      UIGraphicsEndImageContext();
      CGImageRef imgRef = CGImageCreateWithImageInRect([image CGImage],background.frame);
      image = [UIImage imageWithCGImage:imref];
      CGImageRelease(imref);
      return image;
}
 Concuror12 июн. 2012 г., 15:21
Мне еще 20 часов подождать)
 08 февр. 2014 г., 17:34
Разве это не ужасно неэффективно для больших представлений прокрутки? Что не так с ответом от ответа @AbduliamRehmanius?
 26 февр. 2015 г., 10:17
Любить это!! Благодаря тонну!!
 14 дек. 2016 г., 10:26
@AbuliamRegmanius Ответ лучше! Это более эффективно.
 11 июн. 2012 г., 13:06
примите свой ответ тогда :)

contentOffset отрегулировать видимую область и захват слояonly в настоящее время видимая областьUIScrollView.

UIScrollView *contentScrollView;....//scrollview instance

UIGraphicsBeginImageContextWithOptions(contentScrollView.bounds.size, 
                                       YES, 
                                       [UIScreen mainScreen].scale);

//this is the key
CGPoint offset=contentScrollView.contentOffset;
CGContextTranslateCTM(UIGraphicsGetCurrentContext(), -offset.x, -offset.y); 

[contentScrollView.layer renderInContext:UIGraphicsGetCurrentContext()];
UIImage *visibleScrollViewImage = UIGraphicsGetImageFromCurrentImageContext();
UIGraphicsEndImageContext();

Ура :)

 10 февр. 2016 г., 06:42
Работал для меня в масштабируемой прокрутки в Swift. Благодарю.
 21 окт. 2013 г., 14:40
это намного проще и оптимизирует память, так как нет необходимости обрезать увеличенное изображение.
 06 нояб. 2015 г., 02:07
Кому-нибудь повезло с этим на Swift? У меня есть "масштабируемый" scrollview, вызывающий эту функцию расширения и получающий пустое изображение: func screenshot () - & gt; UIImage {let offset = self.contentOffset UIGraphicsBeginImageContextWithOptions (self.bounds.size, false, UIScreen.mainScreen (). Scale) CGContextTranslateCTM (UIGraphicsGetCurrentContext (), -offset.x, -offset.y) )!) let image = UIGraphicsGetImageFromCurrentImageContext () UIGraphicsEndImageContext () возвращаемое изображение}
 23 мая 2014 г., 10:19
Этот ответ единственный, который работал на меня. Благодарю.

func screenshot() -> UIImage {

        UIGraphicsBeginImageContextWithOptions(self.scrollCrop.bounds.size, true, UIScreen.mainScreen().scale);
        //this is the key
        let offset:CGPoint = self.scrollCrop.contentOffset;
        CGContextTranslateCTM(UIGraphicsGetCurrentContext(), -offset.x, -offset.y);
        self.scrollCrop.layer.renderInContext(UIGraphicsGetCurrentContext()!);
        let visibleScrollViewImage: UIImage = UIGraphicsGetImageFromCurrentImageContext();
        UIGraphicsEndImageContext();
        return visibleScrollViewImage;
    }

Swift 4 version:

func screenshot() -> UIImage {
UIGraphicsBeginImageContextWithOptions(self.scrollCrop.bounds.size, false, UIScreen.main.scale)
        let offset = self.scrollCrop.contentOffset
        let thisContext = UIGraphicsGetCurrentContext()
        thisContext?.translateBy(x: -offset.x, y: -offset.y)
        self.scrollCrop.layer.render(in: thisContext!)
        let visibleScrollViewImage = UIGraphicsGetImageFromCurrentImageContext()
        UIGraphicsEndImageContext()
return visibleScrollViewImage
}

func getImage(fromCombinedContext background: UIView) -> UIImage {
        var image: UIImage?
        let size: CGSize = view.frame.size
        UIGraphicsBeginImageContext(size)
        background.layer.affineTransform()
        view.layer.render(in: UIGraphicsGetCurrentContext()!)
        image = UIGraphicsGetImageFromCurrentImageContext()
        UIGraphicsEndImageContext()
        let imgRef = image?.cgImage?.cropping(to: background.frame)
        image = UIImage(cgImage: imgRef!)
        // CGImageRelease(imgRef!) // Removing on Swift - 'CGImageRelease' is unavailable: Core Foundation objects are automatically memory managed
        return image ?? UIImage()
    }

 func captureScreen() -> UIImage? {
    UIGraphicsBeginImageContextWithOptions(self.yourScrollViewName.bounds.size, true, UIScreen.main.scale)
    let offset:CGPoint = self.yourScrollViewName.contentOffset;
    UIGraphicsGetCurrentContext()!.translateBy(x: -offset.x, y: -offset.y);
    self.yourScrollViewName.layer.render(in: UIGraphicsGetCurrentContext()!)
    let image = UIGraphicsGetImageFromCurrentImageContext()
    UIGraphicsEndImageContext()
    return image
}

и использовать его как: пусть Image = captureScreen ()

ругой вид. Получить представление контейнера для визуализации в контексте. сделанный.

В приведенном ниже коде cropView содержит представление прокрутки для захвата. Решение действительно так просто.

Как я понимаю вопрос и почему я нашел эту страницу, весь контент представления прокрутки не нужен - только видимая часть.

func captureCrop() -> UIImage {
    UIGraphicsBeginImageContextWithOptions(self.cropView.frame.size, true, 0.0)
    self.cropView.layer.renderInContext(UIGraphicsGetCurrentContext())
    let image = UIGraphicsGetImageFromCurrentImageContext()
    UIGraphicsEndImageContext()
    return image;
}

Ваш ответ на вопрос