mirror of
https://github.com/Swiftgram/Telegram-iOS.git
synced 2025-06-15 21:45:19 +00:00
131 lines
6.5 KiB
Swift
131 lines
6.5 KiB
Swift
import Foundation
|
|
import UIKit
|
|
import AsyncDisplayKit
|
|
import SwiftSignalKit
|
|
|
|
enum NavigationSplitContainerScrollToTop {
|
|
case master
|
|
case detail
|
|
}
|
|
|
|
final class NavigationSplitContainer: ASDisplayNode {
|
|
private var theme: NavigationControllerTheme
|
|
|
|
private let masterScrollToTopView: ScrollToTopView
|
|
private let detailScrollToTopView: ScrollToTopView
|
|
private let masterContainer: NavigationContainer
|
|
private let detailContainer: NavigationContainer
|
|
private let separator: ASDisplayNode
|
|
|
|
private(set) var masterControllers: [ViewController] = []
|
|
private(set) var detailControllers: [ViewController] = []
|
|
|
|
var canHaveKeyboardFocus: Bool = false {
|
|
didSet {
|
|
self.masterContainer.canHaveKeyboardFocus = self.canHaveKeyboardFocus
|
|
self.detailContainer.canHaveKeyboardFocus = self.canHaveKeyboardFocus
|
|
}
|
|
}
|
|
|
|
var isInFocus: Bool = false {
|
|
didSet {
|
|
if self.isInFocus != oldValue {
|
|
self.inFocusUpdated(isInFocus: self.isInFocus)
|
|
}
|
|
}
|
|
}
|
|
func inFocusUpdated(isInFocus: Bool) {
|
|
self.masterContainer.isInFocus = isInFocus
|
|
self.detailContainer.isInFocus = isInFocus
|
|
}
|
|
|
|
init(theme: NavigationControllerTheme, controllerRemoved: @escaping (ViewController) -> Void, scrollToTop: @escaping (NavigationSplitContainerScrollToTop) -> Void) {
|
|
self.theme = theme
|
|
|
|
self.masterScrollToTopView = ScrollToTopView(frame: CGRect())
|
|
self.masterScrollToTopView.action = {
|
|
scrollToTop(.master)
|
|
}
|
|
self.detailScrollToTopView = ScrollToTopView(frame: CGRect())
|
|
self.detailScrollToTopView.action = {
|
|
scrollToTop(.detail)
|
|
}
|
|
|
|
self.masterContainer = NavigationContainer(isFlat: false, controllerRemoved: controllerRemoved)
|
|
self.masterContainer.clipsToBounds = true
|
|
|
|
self.detailContainer = NavigationContainer(isFlat: false, controllerRemoved: controllerRemoved)
|
|
self.detailContainer.clipsToBounds = true
|
|
|
|
self.separator = ASDisplayNode()
|
|
self.separator.backgroundColor = theme.navigationBar.separatorColor
|
|
|
|
super.init()
|
|
|
|
self.addSubnode(self.masterContainer)
|
|
self.addSubnode(self.detailContainer)
|
|
self.addSubnode(self.separator)
|
|
self.view.addSubview(self.masterScrollToTopView)
|
|
self.view.addSubview(self.detailScrollToTopView)
|
|
}
|
|
|
|
func hasNonReadyControllers() -> Bool {
|
|
if self.masterContainer.hasNonReadyControllers() {
|
|
return true
|
|
}
|
|
if self.detailContainer.hasNonReadyControllers() {
|
|
return true
|
|
}
|
|
return false
|
|
}
|
|
|
|
func updateTheme(theme: NavigationControllerTheme) {
|
|
self.separator.backgroundColor = theme.navigationBar.separatorColor
|
|
}
|
|
|
|
func update(layout: ContainerViewLayout, masterControllers: [ViewController], detailControllers: [ViewController], detailsPlaceholderNode: NavigationDetailsPlaceholderNode?, transition: ContainedViewLayoutTransition) {
|
|
let masterWidth: CGFloat = min(max(320.0, floor(layout.size.width / 3.0)), floor(layout.size.width / 2.0))
|
|
let detailWidth = layout.size.width - masterWidth
|
|
|
|
self.masterScrollToTopView.frame = CGRect(origin: CGPoint(x: 0.0, y: -1.0), size: CGSize(width: masterWidth, height: 1.0))
|
|
self.detailScrollToTopView.frame = CGRect(origin: CGPoint(x: masterWidth, y: -1.0), size: CGSize(width: detailWidth, height: 1.0))
|
|
|
|
transition.updateFrame(node: self.masterContainer, frame: CGRect(origin: CGPoint(), size: CGSize(width: masterWidth, height: layout.size.height)))
|
|
transition.updateFrame(node: self.detailContainer, frame: CGRect(origin: CGPoint(x: masterWidth, y: 0.0), size: CGSize(width: detailWidth, height: layout.size.height)))
|
|
transition.updateFrame(node: self.separator, frame: CGRect(origin: CGPoint(x: masterWidth, y: 0.0), size: CGSize(width: UIScreenPixel, height: layout.size.height)))
|
|
|
|
if let detailsPlaceholderNode {
|
|
let needsTiling = layout.size.width > layout.size.height
|
|
detailsPlaceholderNode.updateLayout(size: CGSize(width: detailWidth, height: layout.size.height), needsTiling: needsTiling, transition: transition)
|
|
transition.updateFrame(node: detailsPlaceholderNode, frame: CGRect(origin: CGPoint(x: masterWidth, y: 0.0), size: CGSize(width: detailWidth, height: layout.size.height)))
|
|
}
|
|
|
|
self.masterContainer.update(layout: ContainerViewLayout(size: CGSize(width: masterWidth, height: layout.size.height), metrics: layout.metrics, deviceMetrics: layout.deviceMetrics, intrinsicInsets: layout.intrinsicInsets, safeInsets: layout.safeInsets, additionalInsets: UIEdgeInsets(), statusBarHeight: layout.statusBarHeight, inputHeight: layout.inputHeight, inputHeightIsInteractivellyChanging: layout.inputHeightIsInteractivellyChanging, inVoiceOver: layout.inVoiceOver), canBeClosed: false, controllers: masterControllers, transition: transition)
|
|
self.detailContainer.update(layout: ContainerViewLayout(size: CGSize(width: detailWidth, height: layout.size.height), metrics: layout.metrics, deviceMetrics: layout.deviceMetrics, intrinsicInsets: layout.intrinsicInsets, safeInsets: layout.safeInsets, additionalInsets: layout.additionalInsets, statusBarHeight: layout.statusBarHeight, inputHeight: layout.inputHeight, inputHeightIsInteractivellyChanging: layout.inputHeightIsInteractivellyChanging, inVoiceOver: layout.inVoiceOver), canBeClosed: true, controllers: detailControllers, transition: transition)
|
|
|
|
var controllersUpdated = false
|
|
if self.detailControllers.last !== detailControllers.last {
|
|
controllersUpdated = true
|
|
} else if self.masterControllers.count != masterControllers.count {
|
|
controllersUpdated = true
|
|
} else {
|
|
for i in 0 ..< masterControllers.count {
|
|
if masterControllers[i] !== self.masterControllers[i] {
|
|
controllersUpdated = true
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
self.masterControllers = masterControllers
|
|
self.detailControllers = detailControllers
|
|
|
|
if controllersUpdated {
|
|
let data = self.detailControllers.last?.customData
|
|
for controller in self.masterControllers {
|
|
controller.updateNavigationCustomData(data, progress: 1.0, transition: transition)
|
|
}
|
|
}
|
|
}
|
|
}
|