import Foundation import Display import QuickLook import Postbox import SwiftSignalKit import AsyncDisplayKit import TelegramCore enum AvatarGalleryEntry: Equatable { case topImage([TelegramMediaImageRepresentation]) case image(TelegramMediaImage) var representations: [TelegramMediaImageRepresentation] { switch self { case let .topImage(representations): return representations case let.image(image): return image.representations } } static func ==(lhs: AvatarGalleryEntry, rhs: AvatarGalleryEntry) -> Bool { switch lhs { case let .topImage(lhsRepresentations): if case let .topImage(rhsRepresentations) = rhs, lhsRepresentations == rhsRepresentations { return true } else { return false } case let .image(lhsImage): if case let .image(rhsImage) = rhs, lhsImage.isEqual(rhsImage) { return true } else { return false } } } } final class AvatarGalleryControllerPresentationArguments { let transitionArguments: (AvatarGalleryEntry) -> GalleryTransitionArguments? init(transitionArguments: @escaping (AvatarGalleryEntry) -> GalleryTransitionArguments?) { self.transitionArguments = transitionArguments } } class AvatarGalleryController: ViewController { private var galleryNode: GalleryControllerNode { return self.displayNode as! GalleryControllerNode } private let account: Account private let _ready = Promise() override var ready: Promise { return self._ready } private var didSetReady = false private let disposable = MetaDisposable() private var entries: [AvatarGalleryEntry] = [] private var centralEntryIndex: Int? private let centralItemTitle = Promise() private let centralItemTitleView = Promise() private let centralItemNavigationStyle = Promise() private let centralItemFooterContentNode = Promise() private let centralItemAttributesDisposable = DisposableSet(); private let _hiddenMedia = Promise(nil) var hiddenMedia: Signal { return self._hiddenMedia.get() } private let replaceRootController: (ViewController, ValuePromise?) -> Void init(account: Account, peer: Peer, replaceRootController: @escaping (ViewController, ValuePromise?) -> Void) { self.account = account self.replaceRootController = replaceRootController super.init(navigationBarTheme: GalleryController.darkNavigationTheme) self.navigationItem.leftBarButtonItem = UIBarButtonItem(title: "Done", style: .done, target: self, action: #selector(self.donePressed)) self.statusBar.statusBarStyle = .White var initialEntries: [AvatarGalleryEntry] = [] if let user = peer as? TelegramUser, !user.photo.isEmpty { initialEntries.append(.topImage(user.photo)) } else if let group = peer as? TelegramGroup { initialEntries.append(.topImage(group.photo)) } else if let channel = peer as? TelegramChannel { initialEntries.append(.topImage(channel.photo)) } let remoteEntriesSignal: Signal<[AvatarGalleryEntry], NoError> = requestPeerPhotos(account: account, peerId: peer.id) |> map { photos -> [AvatarGalleryEntry] in var result: [AvatarGalleryEntry] = [] if photos.isEmpty { result = initialEntries } else { for photo in photos { if result.isEmpty, let first = initialEntries.first { let image = TelegramMediaImage(imageId: photo.image.imageId, representations: first.representations) result.append(.image(image)) } else { result.append(.image(photo.image)) } } } return result } let entriesSignal: Signal<[AvatarGalleryEntry], NoError> = .single(initialEntries) |> then(remoteEntriesSignal) self.disposable.set((entriesSignal |> deliverOnMainQueue).start(next: { [weak self] entries in if let strongSelf = self { strongSelf.entries = entries strongSelf.centralEntryIndex = 0 if strongSelf.isViewLoaded { strongSelf.galleryNode.pager.replaceItems(strongSelf.entries.map({ PeerAvatarImageGalleryItem(account: account, entry: $0) }), centralItemIndex: 0, keepFirst: true) let ready = strongSelf.galleryNode.pager.ready() |> timeout(2.0, queue: Queue.mainQueue(), alternate: .single(Void())) |> afterNext { [weak strongSelf] _ in strongSelf?.didSetReady = true } strongSelf._ready.set(ready |> map { true }) } } })) self.centralItemAttributesDisposable.add(self.centralItemTitle.get().start(next: { [weak self] title in self?.navigationItem.title = title })) self.centralItemAttributesDisposable.add(self.centralItemTitleView.get().start(next: { [weak self] titleView in self?.navigationItem.titleView = titleView })) self.centralItemAttributesDisposable.add(self.centralItemFooterContentNode.get().start(next: { [weak self] footerContentNode in self?.galleryNode.updatePresentationState({ $0.withUpdatedFooterContentNode(footerContentNode) }, transition: .immediate) })) } required init(coder aDecoder: NSCoder) { fatalError("init(coder:) has not been implemented") } deinit { self.disposable.dispose() self.centralItemAttributesDisposable.dispose() } @objc func donePressed() { self.dismiss(forceAway: false) } private func dismiss(forceAway: Bool) { var animatedOutNode = true var animatedOutInterface = false let completion = { [weak self] in if animatedOutNode && animatedOutInterface { self?._hiddenMedia.set(.single(nil)) self?.presentingViewController?.dismiss(animated: false, completion: nil) } } if let centralItemNode = self.galleryNode.pager.centralItemNode(), let presentationArguments = self.presentationArguments as? AvatarGalleryControllerPresentationArguments { if !self.entries.isEmpty { if centralItemNode.index == 0, let transitionArguments = presentationArguments.transitionArguments(self.entries[centralItemNode.index]), !forceAway { animatedOutNode = false centralItemNode.animateOut(to: transitionArguments.transitionNode, completion: { animatedOutNode = true completion() }) } } } self.galleryNode.animateOut(animateContent: animatedOutNode, completion: { animatedOutInterface = true completion() }) } override func loadDisplayNode() { let controllerInteraction = GalleryControllerInteraction(presentController: { [weak self] controller, arguments in if let strongSelf = self { strongSelf.present(controller, in: .window, with: arguments) } }, dismissController: { [weak self] in self?.dismiss(forceAway: true) }, replaceRootController: { [weak self] controller, ready in if let strongSelf = self { strongSelf.replaceRootController(controller, ready) } }) self.displayNode = GalleryControllerNode(controllerInteraction: controllerInteraction) self.displayNodeDidLoad() self.galleryNode.statusBar = self.statusBar self.galleryNode.navigationBar = self.navigationBar self.galleryNode.transitionNodeForCentralItem = { [weak self] in if let strongSelf = self { if let centralItemNode = strongSelf.galleryNode.pager.centralItemNode(), let presentationArguments = strongSelf.presentationArguments as? AvatarGalleryControllerPresentationArguments { if let transitionArguments = presentationArguments.transitionArguments(strongSelf.entries[centralItemNode.index]) { return transitionArguments.transitionNode } } } return nil } self.galleryNode.dismiss = { [weak self] in self?._hiddenMedia.set(.single(nil)) self?.presentingViewController?.dismiss(animated: false, completion: nil) } self.galleryNode.pager.replaceItems(self.entries.map({ PeerAvatarImageGalleryItem(account: self.account, entry: $0) }), centralItemIndex: self.centralEntryIndex) self.galleryNode.pager.centralItemIndexUpdated = { [weak self] index in if let strongSelf = self { var hiddenItem: AvatarGalleryEntry? if let index = index { hiddenItem = strongSelf.entries[index] if let node = strongSelf.galleryNode.pager.centralItemNode() { strongSelf.centralItemTitle.set(node.title()) strongSelf.centralItemTitleView.set(node.titleView()) strongSelf.centralItemNavigationStyle.set(node.navigationStyle()) strongSelf.centralItemFooterContentNode.set(node.footerContent()) } } if strongSelf.didSetReady { strongSelf._hiddenMedia.set(.single(hiddenItem)) } } } let ready = self.galleryNode.pager.ready() |> timeout(2.0, queue: Queue.mainQueue(), alternate: .single(Void())) |> afterNext { [weak self] _ in self?.didSetReady = true } self._ready.set(ready |> map { true }) } override func viewDidAppear(_ animated: Bool) { super.viewDidAppear(animated) var nodeAnimatesItself = false if let centralItemNode = self.galleryNode.pager.centralItemNode(), let presentationArguments = self.presentationArguments as? AvatarGalleryControllerPresentationArguments { self.centralItemTitle.set(centralItemNode.title()) self.centralItemTitleView.set(centralItemNode.titleView()) self.centralItemNavigationStyle.set(centralItemNode.navigationStyle()) self.centralItemFooterContentNode.set(centralItemNode.footerContent()) if let transitionArguments = presentationArguments.transitionArguments(self.entries[centralItemNode.index]) { nodeAnimatesItself = true centralItemNode.animateIn(from: transitionArguments.transitionNode) self._hiddenMedia.set(.single(self.entries[centralItemNode.index])) } } self.galleryNode.animateIn(animateContent: !nodeAnimatesItself) } override func containerLayoutUpdated(_ layout: ContainerViewLayout, transition: ContainedViewLayoutTransition) { super.containerLayoutUpdated(layout, transition: transition) self.galleryNode.frame = CGRect(origin: CGPoint(), size: layout.size) self.galleryNode.containerLayoutUpdated(layout, navigationBarHeight: self.navigationHeight, transition: transition) } }