forked from home-assistant/iOS
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathImageAttachmentViewController.swift
180 lines (153 loc) · 6.26 KB
/
ImageAttachmentViewController.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
import MobileCoreServices
import PromiseKit
import Shared
import UIKit
import UserNotifications
import UserNotificationsUI
import WebKit
class ImageAttachmentViewController: UIViewController, NotificationCategory {
let attachmentURL: URL
let needsEndSecurityScoped: Bool
let image: UIImage
let imageData: Data
let imageUTI: CFString
enum ImageViewType {
case imageView(UIImageView)
case webView(WKWebView)
var view: UIView {
switch self {
case let .imageView(imageView): return imageView
case let .webView(webView): return webView
}
}
}
let visibleView: ImageViewType
required init(api: HomeAssistantAPI, notification: UNNotification, attachmentURL: URL?) throws {
guard let attachmentURL = attachmentURL else {
throw ImageAttachmentError.noAttachment
}
self.needsEndSecurityScoped = attachmentURL.startAccessingSecurityScopedResource()
// rather than hard-coding an acceptable list of UTTypes it's probably easier to just try decoding
// https://developer.apple.com/documentation/usernotifications/unnotificationattachment
// has the full list of what is advertised - at time of writing (iOS 14.5) it's jpeg, gif and png
// but iOS 14 also supports webp, so who knows if it'll be added silently or not
do {
let data = try Data(contentsOf: attachmentURL, options: .alwaysMapped)
guard let image = UIImage(data: data) else {
throw ImageAttachmentError.imageDecodeFailure
}
self.image = image
self.imageData = data
if let imageSource = CGImageSourceCreateWithData(data as CFData, nil),
let uti = CGImageSourceGetType(imageSource) {
self.imageUTI = uti
} else {
// can't figure out, just assume JPEG
self.imageUTI = kUTTypeJPEG
}
if UTTypeConformsTo(imageUTI, kUTTypeGIF) {
// use a WebView for gif so we can animate without pulling in a third party library
let config = with(WKWebViewConfiguration()) {
$0.userContentController = with(WKUserContentController()) {
// we can't use `loadHTMLString` with `<img>` inside to do styling because the webview can't get
// the security scoped file if loaded by the service extension so we need to load data directly
$0.addUserScript(WKUserScript(source: """
var style = document.createElement('style');
style.innerHTML = `
img { width: 100%; height: 100%; }
`;
document.head.appendChild(style);
""", injectionTime: .atDocumentEnd, forMainFrameOnly: true))
}
}
self.visibleView = .webView(with(WKWebView(frame: .zero, configuration: config)) {
$0.scrollView.isScrollEnabled = false
$0.isOpaque = false
$0.backgroundColor = .clear
$0.scrollView.backgroundColor = .clear
})
} else {
self.visibleView = .imageView(with(UIImageView()) {
$0.contentMode = .scaleAspectFit
})
}
} catch {
attachmentURL.stopAccessingSecurityScopedResource()
throw error
}
self.attachmentURL = attachmentURL
super.init(nibName: nil, bundle: nil)
}
@available(*, unavailable)
required init?(coder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
deinit {
if needsEndSecurityScoped {
attachmentURL.stopAccessingSecurityScopedResource()
}
}
enum ImageAttachmentError: Error {
case noAttachment
case notImage
case imageDecodeFailure
}
private var aspectRatioConstraint: NSLayoutConstraint? {
willSet {
aspectRatioConstraint?.isActive = false
}
didSet {
aspectRatioConstraint?.isActive = true
}
}
private var lastAttachmentURL: URL? {
didSet {
oldValue?.stopAccessingSecurityScopedResource()
}
}
func start() -> Promise<Void> {
lastAttachmentURL = attachmentURL
switch visibleView {
case let .webView(webView):
let mime = UTTypeCopyPreferredTagWithClass(imageUTI, kUTTagClassMIMEType)?.takeRetainedValue() as String?
webView.load(
imageData,
mimeType: mime ?? "image/gif",
characterEncodingName: "UTF-8",
baseURL: attachmentURL
)
case let .imageView(imageView):
imageView.image = image
}
aspectRatioConstraint = NSLayoutConstraint.aspectRatioConstraint(on: visibleView.view, size: image.size)
return .value(())
}
override func loadView() {
class UnanimatingView: UIView {
override func layoutSubviews() {
// avoids the image view sizing up from nothing when initially displaying
// since we don't control our own view's expansion, we need to disable animation at our level
UIView.performWithoutAnimation {
super.layoutSubviews()
}
}
}
view = UnanimatingView()
}
override func viewDidLoad() {
super.viewDidLoad()
let subview = visibleView.view
view.addSubview(subview)
subview.translatesAutoresizingMaskIntoConstraints = false
NSLayoutConstraint.activate([
subview.topAnchor.constraint(equalTo: view.topAnchor),
subview.leadingAnchor.constraint(equalTo: view.leadingAnchor),
subview.trailingAnchor.constraint(equalTo: view.trailingAnchor),
subview.bottomAnchor.constraint(equalTo: view.bottomAnchor),
])
}
var mediaPlayPauseButtonType: UNNotificationContentExtensionMediaPlayPauseButtonType { .none }
var mediaPlayPauseButtonFrame: CGRect?
func mediaPlay() {}
func mediaPause() {}
}