CGPDFPage Drawing вызывает проблемы с памятью (Swift 3)
В настоящее время я занимаюсь разработкой приложения и хочу внедрить программу для чтения PDF поверх UICollectionView.
Я использую пользовательский UIView в каждой ячейке, которая отображает соответствующую страницу PDF:
weak var page: CGPDFPage! {
didSet { setNeedsDisplay() }
}
override func draw(_ rect: CGRect) {
if page == nil {
print("page is nil")
return }
let context = UIGraphicsGetCurrentContext()
context?.clear(bounds)
context?.setFillColor(red: 1, green: 1, blue: 1, alpha: 1)
context?.fill(bounds)
context?.translateBy(x: 0.0, y: bounds.size.height);
context?.scaleBy(x: 1.0, y: -1.0);
var cropBox = page.getBoxRect(.cropBox)
cropBox = CGRect(x: cropBox.origin.x, y: cropBox.origin.y, width: ceil(cropBox.width), height: ceil(cropBox.height))
let scaleFactor = min(bounds.size.width/cropBox.size.width, bounds.size.height/cropBox.size.height)
let scale = CGAffineTransform(scaleX: scaleFactor, y: scaleFactor)
let scaledInnerRect = cropBox.applying(scale)
let translate = CGAffineTransform(translationX: ((bounds.size.width - scaledInnerRect.size.width) / 2) - scaledInnerRect.origin.x, y: ((bounds.size.height - scaledInnerRect.size.height) / 2) - scaledInnerRect.origin.y)
let concat = translate.scaledBy(x: scaleFactor, y: scaleFactor)
context?.concatenate(concat)
let clipRect = cropBox
context?.addRect(clipRect)
context?.clip()
context?.drawPDFPage(page)
UIGraphicsEndPDFContext()
}
Все идет нормально. Он отображает страницу в ячейке, но вызывает проблемы с памятью.
Каким-то образом контекст сохраняет ссылку на все страницы, отображаемые в ячейках представления коллекции, и они не освобождаются. CGPDFPageRelease больше не доступен.:(
С другой стороны, если этот подкласс uiview увеличен внутри прокрутки, как я могу перерисовать pdf, чтобы не потерять качество?
Любая помощь будет высоко оценена. Спасибо!
3 ответа
У меня был тот же код и та же проблема, но как только я сменил его на это новое решение, оно мне помогло. Теперь у меня нет проблем с памятью при рендеринге PDF.
autoreleasepool
Это очень важно, так как это поможет вам выпустить визуализированные страницы. Вот мой код
СВИФТ
class PDFPageRenderView: UIView {
func render(page: CGPDFPage) {
for subview in subviews { subview.removeFromSuperview() }
let imageView = UIImageView(frame: bounds)
imageView.contentMode = .scaleAspectFit
imageView.clipsToBounds = true
imageView.backgroundColor = UIColor.white
autoreleasepool {
let pageRect = page.getBoxRect(.mediaBox)
let renderer = UIGraphicsImageRenderer(size: pageRect.size)
let data = renderer.jpegData(withCompressionQuality: kImageCompression, actions: { cnv in
UIColor.white.set()
cnv.fill(pageRect)
cnv.cgContext.translateBy(x: 0.0, y: pageRect.size.height)
cnv.cgContext.scaleBy(x: 1.0, y: -1.0)
cnv.cgContext.drawPDFPage(page)
})
imageView.image = UIImage(data: data)
}
addSubview(imageView)
}
}
Используйте CATiledLayer для решения проблемы рендеринга при масштабировании страницы.
Создайте подкласс CATiledLayer и переопределите указанный ниже метод в своем подклассе представления.
override class var layerClass: AnyClass {
get {
return YourCATiledLayerSubclass.self
}
}
Мы наблюдали ту же утечку, и она, похоже, влияет только на iOS 10. Похоже, это общая проблема: https://forums.developer.apple.com/message/174710 Мы также протестировали ее в iOS 10.2 и не можем найти утечку больше. Вы пытались увидеть, работает ли это для вас?