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) } } } }