288 lines
8.6 KiB
Swift
288 lines
8.6 KiB
Swift
//
|
|
// ModalPreviewHandler.swift
|
|
// Telegram
|
|
//
|
|
// Created by keepcoder on 02/02/2017.
|
|
// Copyright © 2017 Telegram. All rights reserved.
|
|
//
|
|
|
|
import Cocoa
|
|
import TelegramCore
|
|
|
|
import TGUIKit
|
|
import SwiftSignalKit
|
|
|
|
|
|
enum QuickPreviewMedia : Equatable {
|
|
case file(FileMediaReference, ModalPreviewControllerView.Type)
|
|
case image(ImageMediaReference, ModalPreviewControllerView.Type)
|
|
|
|
static func ==(lhs: QuickPreviewMedia, rhs: QuickPreviewMedia) -> Bool {
|
|
switch lhs {
|
|
case let .file(lhsReference, _):
|
|
if case let .file(rhsReference, _) = rhs {
|
|
return lhsReference.media.isEqual(to: rhsReference.media)
|
|
} else {
|
|
return false
|
|
}
|
|
case let .image(lhsReference, _):
|
|
if case let .image(rhsReference, _) = rhs {
|
|
return lhsReference.media.isEqual(to: rhsReference.media)
|
|
} else {
|
|
return false
|
|
}
|
|
}
|
|
}
|
|
|
|
var fileReference: FileMediaReference? {
|
|
switch self {
|
|
case let .file(reference, _):
|
|
return reference
|
|
default:
|
|
return nil
|
|
}
|
|
}
|
|
var imageReference: ImageMediaReference? {
|
|
switch self {
|
|
case let .image(reference, _):
|
|
return reference
|
|
default:
|
|
return nil
|
|
}
|
|
}
|
|
|
|
var viewType: ModalPreviewControllerView.Type {
|
|
switch self {
|
|
case let .file(_, type), let .image(_, type):
|
|
return type
|
|
}
|
|
}
|
|
}
|
|
|
|
extension GridNode : ModalPreviewProtocol {
|
|
func fileAtLocationInWindow(_ point: NSPoint) -> (QuickPreviewMedia, NSView?)? {
|
|
let point = self.documentView!.convert(point, from: nil)
|
|
var reference: (QuickPreviewMedia, NSView?)? = nil
|
|
self.forEachItemNode { node in
|
|
if NSPointInRect(point, node.frame) {
|
|
if let c = node as? ModalPreviewRowViewProtocol {
|
|
reference = c.fileAtPoint(node.convert(point, from: nil))
|
|
}
|
|
}
|
|
}
|
|
return reference
|
|
}
|
|
}
|
|
|
|
extension TableView : ModalPreviewProtocol {
|
|
func fileAtLocationInWindow(_ point: NSPoint) ->(QuickPreviewMedia, NSView?)? {
|
|
let index = self.row(at: documentView!.convert(point, from: nil))
|
|
if index != -1 {
|
|
let item = self.item(at: index)
|
|
if let view = self.viewNecessary(at: item.index), let c = view as? ModalPreviewRowViewProtocol {
|
|
return c.fileAtPoint(view.convert(point, from: nil))
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
}
|
|
|
|
protocol ModalPreviewRowViewProtocol {
|
|
func fileAtPoint(_ point:NSPoint) -> (QuickPreviewMedia, NSView?)?
|
|
}
|
|
|
|
protocol ModalPreviewProtocol {
|
|
func fileAtLocationInWindow(_ point:NSPoint) -> (QuickPreviewMedia, NSView?)?
|
|
|
|
}
|
|
|
|
protocol ModalPreviewControllerView : NSView {
|
|
func update(with reference: QuickPreviewMedia, context: AccountContext, animated: Bool)
|
|
func getContentView() -> NSView
|
|
}
|
|
|
|
fileprivate var handler:ModalPreviewHandler?
|
|
|
|
|
|
|
|
func startModalPreviewHandle(_ global:ModalPreviewProtocol, window:Window, context: AccountContext) {
|
|
handler = ModalPreviewHandler(global, window: window, context: context)
|
|
handler?.startHandler()
|
|
}
|
|
|
|
class ModalPreviewHandler : NSObject {
|
|
private let global:ModalPreviewProtocol
|
|
private let context:AccountContext
|
|
private let window:Window
|
|
private let modal:PreviewModalController
|
|
init(_ global:ModalPreviewProtocol, window:Window, context: AccountContext) {
|
|
self.global = global
|
|
self.window = window
|
|
self.context = context
|
|
|
|
self.modal = PreviewModalController(context)
|
|
}
|
|
|
|
func startHandler() {
|
|
guard window.isKeyWindow else {
|
|
return
|
|
}
|
|
let initial = global.fileAtLocationInWindow(window.mouseLocationOutsideOfEventStream)
|
|
if let initial = initial {
|
|
modal.update(with: initial.0)
|
|
let animation:ModalAnimationType
|
|
if let view = initial.1 {
|
|
var rect = view.convert(view.bounds, to: nil)
|
|
rect.origin.y = window.contentView!.frame.maxY - rect.maxY
|
|
animation = .scaleFrom(rect)
|
|
} else {
|
|
animation = .bottomToCenter
|
|
}
|
|
showModal(with: modal, for: window, animationType: animation)
|
|
|
|
window.set(mouseHandler: { [weak self] (_) -> KeyHandlerResult in
|
|
if let strongSelf = self, let reference = strongSelf.global.fileAtLocationInWindow(strongSelf.window.mouseLocationOutsideOfEventStream) {
|
|
strongSelf.modal.update(with: reference.0)
|
|
}
|
|
return .invoked
|
|
}, with: self, for: .leftMouseDragged, priority: .supreme)
|
|
|
|
window.set(mouseHandler: { [weak self] (_) -> KeyHandlerResult in
|
|
self?.stopHandler()
|
|
return .invoked
|
|
}, with: self, for: .leftMouseUp, priority: .supreme)
|
|
}
|
|
|
|
}
|
|
|
|
func stopHandler() {
|
|
window.removeAllHandlers(for: self)
|
|
if let view = self.global.fileAtLocationInWindow(self.window.mouseLocationOutsideOfEventStream)?.1 {
|
|
let content = modal.genericView.contentView?.getContentView() ?? modal.genericView
|
|
let rect = view.convert(view.bounds, to: modal.genericView.superview?.superview?.superview)
|
|
modal.close(animationType: .scaleToRect(rect, content))
|
|
} else {
|
|
modal.close()
|
|
}
|
|
|
|
handler = nil
|
|
}
|
|
|
|
deinit {
|
|
|
|
}
|
|
}
|
|
|
|
|
|
private final class PreviewModalView: View {
|
|
fileprivate var contentView: ModalPreviewControllerView?
|
|
required init(frame frameRect: NSRect) {
|
|
super.init(frame: frameRect)
|
|
}
|
|
|
|
deinit {
|
|
var bp:Int = 0
|
|
bp += 1
|
|
}
|
|
|
|
override func layout() {
|
|
super.layout()
|
|
}
|
|
|
|
func update(with preview: QuickPreviewMedia, context: AccountContext, animated: Bool) {
|
|
|
|
let viewType = preview.viewType
|
|
var changed = false
|
|
if contentView == nil || !contentView!.isKind(of: viewType) {
|
|
if animated {
|
|
let current = self.contentView
|
|
self.contentView = nil
|
|
current?.layer?.animateScaleSpring(from: 1, to: 0, duration: 0.2, removeOnCompletion: false, completion: { [weak current] completed in
|
|
if completed {
|
|
current?.removeFromSuperview()
|
|
}
|
|
})
|
|
} else {
|
|
self.contentView?.removeFromSuperview()
|
|
}
|
|
|
|
self.contentView = viewType.init(frame:NSZeroRect)
|
|
self.addSubview(self.contentView!)
|
|
changed = true
|
|
}
|
|
contentView?.frame = bounds
|
|
contentView?.update(with: preview, context: context, animated: animated && !changed)
|
|
|
|
if animated {
|
|
contentView?.layer?.animateScaleSpring(from: 0.5, to: 1.0, duration: 0.2)
|
|
}
|
|
}
|
|
|
|
required init?(coder: NSCoder) {
|
|
fatalError("init(coder:) has not been implemented")
|
|
}
|
|
}
|
|
|
|
class PreviewModalController: ModalViewController {
|
|
fileprivate let context:AccountContext
|
|
fileprivate var reference:QuickPreviewMedia?
|
|
init(_ context: AccountContext) {
|
|
self.context = context
|
|
|
|
super.init(frame: NSMakeRect(0, 0, min(context.window.frame.width - 50, 500), min(500, context.window.frame.height - 50)))
|
|
bar = .init(height: 0)
|
|
}
|
|
|
|
override var hasOwnTouchbar: Bool {
|
|
return false
|
|
}
|
|
|
|
override var containerBackground: NSColor {
|
|
return .clear
|
|
}
|
|
|
|
override func becomeFirstResponder() -> Bool? {
|
|
return nil
|
|
}
|
|
|
|
override var handleEvents:Bool {
|
|
return false
|
|
}
|
|
|
|
override func viewDidLoad() {
|
|
super.viewDidLoad()
|
|
if let reference = reference {
|
|
genericView.update(with: reference, context: context, animated: false)
|
|
}
|
|
readyOnce()
|
|
}
|
|
|
|
func update(with reference:QuickPreviewMedia?) {
|
|
if self.reference != reference {
|
|
self.reference = reference
|
|
if isLoaded(), let reference = reference {
|
|
genericView.update(with: reference, context: context, animated: true)
|
|
}
|
|
}
|
|
}
|
|
|
|
fileprivate var genericView:PreviewModalView {
|
|
return view as! PreviewModalView
|
|
}
|
|
|
|
override func viewClass() -> AnyClass {
|
|
return PreviewModalView.self
|
|
}
|
|
|
|
deinit {
|
|
var bp:Int = 0
|
|
bp += 1
|
|
}
|
|
|
|
// override var isFullScreen: Bool {
|
|
// return true
|
|
// }
|
|
|
|
}
|