forked from kean/Nuke
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathImageProcessing.swift
193 lines (161 loc) · 7.06 KB
/
ImageProcessing.swift
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
// The MIT License (MIT)
//
// Copyright (c) 2015-2018 Alexander Grebenyuk (github.com/kean).
import Foundation
/// Performs image processing.
public protocol ImageProcessing: Equatable {
/// Returns processed image.
func process(image: Image, context: ImageProcessingContext) -> Image?
}
/// Image processing context used when selecting which processor to use.
public struct ImageProcessingContext {
public let request: ImageRequest
public let isFinal: Bool
public let scanNumber: Int? // need a more general purpose way to implement this
}
/// Composes multiple processors.
internal struct ImageProcessorComposition: ImageProcessing {
private let processors: [AnyImageProcessor]
/// Composes multiple processors.
public init(_ processors: [AnyImageProcessor]) {
self.processors = processors
}
/// Processes the given image by applying each processor in an order in
/// which they were added. If one of the processors fails to produce
/// an image the processing stops and `nil` is returned.
func process(image: Image, context: ImageProcessingContext) -> Image? {
return processors.reduce(image) { image, processor in
return autoreleasepool {
image.flatMap { processor.process(image: $0, context: context) }
}
}
}
/// Returns true if the underlying processors are pairwise-equivalent.
public static func == (lhs: ImageProcessorComposition, rhs: ImageProcessorComposition) -> Bool {
return lhs.processors == rhs.processors
}
}
/// Type-erased image processor.
public struct AnyImageProcessor: ImageProcessing {
private let _process: (Image, ImageProcessingContext) -> Image?
private let _processor: Any
private let _equals: (AnyImageProcessor) -> Bool
public init<P: ImageProcessing>(_ processor: P) {
self._process = { processor.process(image: $0, context: $1) }
self._processor = processor
self._equals = { ($0._processor as? P) == processor }
}
public func process(image: Image, context: ImageProcessingContext) -> Image? {
return self._process(image, context)
}
public static func == (lhs: AnyImageProcessor, rhs: AnyImageProcessor) -> Bool {
return lhs._equals(rhs)
}
}
internal struct AnonymousImageProcessor<Key: Hashable>: ImageProcessing {
private let _key: Key
private let _closure: (Image) -> Image?
init(_ key: Key, _ closure: @escaping (Image) -> Image?) {
self._key = key; self._closure = closure
}
func process(image: Image, context: ImageProcessingContext) -> Image? {
return self._closure(image)
}
static func == (lhs: AnonymousImageProcessor, rhs: AnonymousImageProcessor) -> Bool {
return lhs._key == rhs._key
}
}
extension ImageProcessing {
func process(image: ImageContainer, request: ImageRequest) -> Image? {
let context = ImageProcessingContext(request: request, isFinal: image.isFinal, scanNumber: image.scanNumber)
return process(image: image.image, context: context)
}
}
#if !os(macOS)
import UIKit
/// Decompresses and (optionally) scales down input images. Maintains
/// original aspect ratio.
///
/// Decompressing compressed image formats (such as JPEG) can significantly
/// improve drawing performance as it allows a bitmap representation to be
/// created in a background rather than on the main thread.
public struct ImageDecompressor: ImageProcessing {
/// An option for how to resize the image.
public enum ContentMode {
/// Scales the image so that it completely fills the target size.
/// Doesn't clip images.
case aspectFill
/// Scales the image so that it fits the target size.
case aspectFit
}
/// Size to pass to disable resizing.
public static let MaximumSize = CGSize(
width: CGFloat.greatestFiniteMagnitude,
height: CGFloat.greatestFiniteMagnitude
)
private let targetSize: CGSize
private let contentMode: ContentMode
private let upscale: Bool
/// Initializes `Decompressor` with the given parameters.
/// - parameter targetSize: Size in pixels. `MaximumSize` by default.
/// - parameter contentMode: An option for how to resize the image
/// to the target size. `.aspectFill` by default.
public init(targetSize: CGSize = MaximumSize, contentMode: ContentMode = .aspectFill, upscale: Bool = false) {
self.targetSize = targetSize
self.contentMode = contentMode
self.upscale = upscale
}
/// Decompresses and scales the image.
public func process(image: Image, context: ImageProcessingContext) -> Image? {
return decompress(image, targetSize: targetSize, contentMode: contentMode, upscale: upscale)
}
/// Returns true if both have the same `targetSize` and `contentMode`.
public static func == (lhs: ImageDecompressor, rhs: ImageDecompressor) -> Bool {
return lhs.targetSize == rhs.targetSize && lhs.contentMode == rhs.contentMode
}
#if !os(watchOS)
/// Returns target size in pixels for the given view. Takes main screen
/// scale into the account.
public static func targetSize(for view: UIView) -> CGSize { // in pixels
let scale = UIScreen.main.scale
let size = view.bounds.size
return CGSize(width: size.width * scale, height: size.height * scale)
}
#endif
}
internal func decompress(_ image: UIImage, targetSize: CGSize, contentMode: ImageDecompressor.ContentMode, upscale: Bool) -> UIImage {
guard let cgImage = image.cgImage else { return image }
let bitmapSize = CGSize(width: cgImage.width, height: cgImage.height)
let scaleHor = targetSize.width / bitmapSize.width
let scaleVert = targetSize.height / bitmapSize.height
let scale = contentMode == .aspectFill ? max(scaleHor, scaleVert) : min(scaleHor, scaleVert)
return decompress(image, scale: CGFloat(upscale ? scale : min(scale, 1)))
}
internal func decompress(_ image: UIImage, scale: CGFloat) -> UIImage {
guard let cgImage = image.cgImage else { return image }
let size = CGSize(
width: round(scale * CGFloat(cgImage.width)),
height: round(scale * CGFloat(cgImage.height))
)
// For more info see:
// - Quartz 2D Programming Guide
// - https://github.com/kean/Nuke/issues/35
// - https://github.com/kean/Nuke/issues/57
let alphaInfo: CGImageAlphaInfo = isOpaque(cgImage) ? .noneSkipLast : .premultipliedLast
guard let ctx = CGContext(
data: nil,
width: Int(size.width), height: Int(size.height),
bitsPerComponent: 8, bytesPerRow: 0,
space: CGColorSpaceCreateDeviceRGB(),
bitmapInfo: alphaInfo.rawValue) else {
return image
}
ctx.draw(cgImage, in: CGRect(origin: CGPoint.zero, size: size))
guard let decompressed = ctx.makeImage() else { return image }
return UIImage(cgImage: decompressed, scale: image.scale, orientation: image.imageOrientation)
}
private func isOpaque(_ image: CGImage) -> Bool {
let alpha = image.alphaInfo
return alpha == .none || alpha == .noneSkipFirst || alpha == .noneSkipLast
}
#endif