-
Notifications
You must be signed in to change notification settings - Fork 895
/
Copy pathAvatarLayer.swift
313 lines (261 loc) · 9.97 KB
/
AvatarLayer.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
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
//
// AvatarControl.swift
// Telegram-Mac
//
// Created by keepcoder on 15/10/2016.
// Copyright © 2016 Telegram. All rights reserved.
//
import Cocoa
import TelegramCore
import Postbox
import TGUIKit
import SwiftSignalKit
private class AvatarNodeParameters: NSObject {
let account: Account
let peerId: Peer
let letters: [String]
let font: NSFont
init(account: Account, peerId: Peer, letters: [String], font: NSFont) {
self.account = account
self.peerId = peerId
self.letters = letters
self.font = font
super.init()
}
}
enum AvatarNodeState: Equatable {
case Empty
case PeerAvatar(Peer, [String], TelegramMediaImageRepresentation?, PeerNameColor?, Message?, NSSize?, Bool)
case ArchivedChats
}
func ==(lhs: AvatarNodeState, rhs: AvatarNodeState) -> Bool {
switch (lhs, rhs) {
case (.Empty, .Empty):
return true
case let (.PeerAvatar(lhsPeer, lhsLetters, lhsPhotoRepresentations, lhsPeerNameColor, _, lhsSize, lhsForum), .PeerAvatar(rhsPeer, rhsLetters, rhsPhotoRepresentations, rhsPeerNameColor, _, rhsSize, rhsForum)):
return lhsPeer.isEqual(rhsPeer) && lhsLetters == rhsLetters && lhsPhotoRepresentations == rhsPhotoRepresentations && lhsSize == rhsSize && lhsForum == rhsForum && lhsPeerNameColor == rhsPeerNameColor
case (.ArchivedChats, .ArchivedChats):
return true
default:
return false
}
}
class AvatarControl: NSView {
private var trackingArea:NSTrackingArea?
private var displaySuspended:Bool = false
var userInteractionEnabled:Bool = true
private let longOverHandleDisposable = MetaDisposable()
private var handlers:[(ControlEvent,(AvatarControl) -> Void)] = []
var font: NSFont {
didSet {
if oldValue !== font {
if !self.displaySuspended {
self.needsDisplay = true
}
}
}
}
private let disposable = MetaDisposable()
private var state: AvatarNodeState = .Empty
private var account:Account?
private var disableForum: Bool = false
private var contentScale: CGFloat = 0
var contentUpdated: ((Any?)->Void)?
func callContentUpdater() {
self.contentUpdated?(self.imageContents)
}
var imageContents: Any? {
didSet {
self.layer?.contents = imageContents
self.contentUpdated?(imageContents)
}
}
public var animated: Bool = false
private var _attemptLoadNextSynchronous: Bool = false
public var attemptLoadNextSynchronous: Bool {
get {
let result = _attemptLoadNextSynchronous
_attemptLoadNextSynchronous = false
return result
}
set {
_attemptLoadNextSynchronous = newValue
}
}
public init(font: NSFont) {
self.font = font
super.init(frame: NSZeroRect)
wantsLayer = true
}
required init?(coder aDecoder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
required override init(frame frameRect: NSRect) {
fatalError("init(frame:) has not been implemented")
}
override public var frame: CGRect {
get {
return super.frame
} set(value) {
let updateImage = !value.size.equalTo(super.frame.size)
super.frame = value
if updateImage && !self.displaySuspended {
self.needsDisplay = true
}
}
}
public func setState(account: Account, state: AvatarNodeState) {
self.account = account
if state != self.state {
contentScale = 0
self.state = state
self.viewDidChangeBackingProperties()
}
}
public func setPeer(account: Account, peer: Peer?, message: Message? = nil, size: NSSize? = nil, disableForum: Bool = false) {
self.account = account
self.disableForum = disableForum
let state: AvatarNodeState
if let peer = peer {
state = .PeerAvatar(peer, peer.displayLetters, peer.smallProfileImage, peer.nameColor, message, size, peer.isForum && !disableForum)
} else {
state = .Empty
}
if self.state != state || self.imageContents == nil {
self.state = state
contentScale = 0
self.viewDidChangeBackingProperties()
}
}
func set(handler:@escaping (AvatarControl) -> Void, for event:ControlEvent) -> Void {
handlers.append((event,handler))
}
override open func mouseEntered(with event: NSEvent) {
if userInteractionEnabled {
let disposable = (Signal<Void,Void>.single(Void()) |> delay(0.3, queue: Queue.mainQueue())).start(next: { [weak self] in
if let strongSelf = self, strongSelf._mouseInside() {
strongSelf.send(event: .LongOver)
}
})
longOverHandleDisposable.set(disposable)
} else {
super.mouseEntered(with: event)
}
}
override func mouseDown(with event: NSEvent) {
if userInteractionEnabled {
longOverHandleDisposable.set(nil)
} else {
super.mouseDown(with: event)
}
}
override open func mouseUp(with event: NSEvent) {
if userInteractionEnabled {
if _mouseInside() {
if event.clickCount == 1 {
send(event: .SingleClick)
}
send(event: .Click)
}
} else {
super.mouseUp(with: event)
}
}
func removeAllHandlers() ->Void {
handlers.removeAll()
}
func send(event:ControlEvent) -> Void {
for (e,handler) in handlers {
if e == event {
handler(self)
}
}
}
override func viewDidChangeBackingProperties() {
layer?.contentsScale = backingScaleFactor
layer?.contentsGravity = .resizeAspectFill
if let account = account, self.state != .Empty {
if contentScale != backingScaleFactor {
contentScale = backingScaleFactor
self.displaySuspended = true
self.imageContents = nil
let photo: PeerPhoto?
var updatedSize: NSSize = self.frame.size
switch state {
case let .PeerAvatar(peer, letters, representation, nameColor, message, size, _):
if let peer = peer as? TelegramUser, peer.firstName == nil && peer.lastName == nil {
photo = nil
self.setState(account: account, state: .Empty)
let icon = theme.icons.deletedAccount
self.setSignal(generateEmptyPhoto(updatedSize, type: .icon(colors: theme.colors.peerColors(Int(peer.id.id._internalGetInt64Value() % 7)), icon: icon, iconSize: icon.backingSize.aspectFitted(NSMakeSize(min(50, updatedSize.width - 20), min(updatedSize.height - 20, 50))), cornerRadius: nil)) |> map {($0, false)})
return
} else {
photo = .peer(peer, representation, nameColor, letters, message)
}
updatedSize = size ?? frame.size
case .Empty:
photo = nil
default:
photo = nil
}
if let photo = photo {
setSignal(peerAvatarImage(account: account, photo: photo, displayDimensions: updatedSize, scale:backingScaleFactor, font: self.font, synchronousLoad: attemptLoadNextSynchronous, disableForum: disableForum), force: false)
} else {
let content = self.imageContents
self.displaySuspended = false
self.imageContents = content
}
}
} else {
self.state = .Empty
}
}
public func setSignal(_ signal: Signal<(CGImage?, Bool), NoError>, force: Bool = true) {
if force {
self.state = .Empty
}
self.disposable.set((signal |> deliverOnMainQueue).start(next: { [weak self] image, animated in
if let strongSelf = self {
strongSelf.imageContents = image
if animated {
strongSelf.layer?.animateContents()
}
}
}))
}
open override func updateTrackingAreas() {
super.updateTrackingAreas();
if let trackingArea = trackingArea {
self.removeTrackingArea(trackingArea)
}
trackingArea = nil
if let _ = window {
let options:NSTrackingArea.Options = [.cursorUpdate, .mouseEnteredAndExited, .mouseMoved, .activeAlways, .inVisibleRect]
self.trackingArea = NSTrackingArea(rect: self.bounds, options: options, owner: self, userInfo: nil)
self.addTrackingArea(self.trackingArea!)
}
}
open override func viewDidMoveToSuperview() {
super.viewDidMoveToSuperview()
updateTrackingAreas()
}
deinit {
if let trackingArea = self.trackingArea {
self.removeTrackingArea(trackingArea)
}
disposable.dispose()
longOverHandleDisposable.dispose()
}
override func copy() -> Any {
let view = NSView()
view.wantsLayer = true
view.background = .clear
view.layer?.frame = NSMakeRect(0, visibleRect.minY == 0 ? 0 : visibleRect.height - frame.height, frame.width, frame.height)
view.layer?.contents = self.imageContents
view.layer?.masksToBounds = true
view.frame = self.visibleRect
view.layer?.shouldRasterize = true
view.layer?.rasterizationScale = backingScaleFactor
return view
}
}