基于Telegram二次開(kāi)發(fā) --- Controllers 篇

一允趟、Telegram 控制器

image.png

ViewController 使 UIViewController 的工作變成了 nodes 層次結(jié)構(gòu)的容器洞慎;與官方 node 控制器 ASViewController 不同,它沒(méi)有 可見(jiàn)深度智能預(yù)加載 等功能酝润。

@objc open class ViewController: UIViewController, ContainableController {
    // the root content node
    private var _displayNode: ASDisplayNode?
    public final var displayNode: ASDisplayNode {
        get {
            if let value = self._displayNode {
                return value
            }
            else {
                self.loadDisplayNode()
                ...
                return self._displayNode!
            }
        }
        ...
    }
    open func loadDisplayNode()
    open func displayNodeDidLoad()
    
    // shared components
    public let statusBar: StatusBar
    public let navigationBar: NavigationBar?
    private(set) var toolbar: Toolbar?
    private var scrollToTopView: ScrollToTopView?
    // customizations of navigationBar
    public var navigationOffset: CGFloat
    open var navigationHeight: CGFloat
    open var navigationInsetHeight: CGFloat
    open var cleanNavigationHeight: CGFloat
    open var visualNavigationInsetHeight: CGFloat
    public var additionalNavigationBarHeight: CGFloat
}
  • 每個(gè)ViewController 通過(guò)一個(gè) root node 來(lái)管理 node 層栏饮,該 root node 存儲(chǔ)在 displayNode 該類的屬性中;loadDisplayNodedisplayNodeDidLoad 函數(shù)里實(shí)現(xiàn)懶加載。
  • 作為基類绪撵,它為子類準(zhǔn)備了幾個(gè)共享的 node 組件:狀態(tài)欄導(dǎo)航欄祝蝠、工具欄返回到頂部功能 音诈;還提供簡(jiǎn)便屬性來(lái)支持自定義導(dǎo)航欄。
  • ViewController 很少單獨(dú)使用绎狭,項(xiàng)目中有上百個(gè)它的子類用于實(shí)現(xiàn)不同的用戶界面细溅; UIKit 中兩個(gè)最常用的容器控制器:UINavigationControllerUITabBarController ,被重新實(shí)現(xiàn)為 NavigationControllerTabBarController儡嘶。
open class NavigationController: UINavigationController, ContainableController, UIGestureRecognizerDelegate {
    private var _viewControllers: [ViewController] = []
    // NavigationControllerNode
    private var _displayNode: ASDisplayNode?
    private var theme: NavigationControllerTheme
    
    // manage layout and transition animation
    private func updateContainers(layout rawLayout: ContainerViewLayout, transition: ContainedViewLayoutTransition)
    
    // push with a completion handler
    public func pushViewController(_ controller: ViewController, animated: Bool = true, completion: @escaping () -> Void)
}

// NavigationLayout.swift
enum RootNavigationLayout {
    case split([ViewController], [ViewController])
    case flat([ViewController])
}

// NavigationContainer.swift
final class NavigationContainer: ASDisplayNode, UIGestureRecognizerDelegate
    override func didLoad() {
        // the interactive pop gesture
        let panRecognizer = InteractiveTransitionGestureRecognizer(target: self, action: #selector(self.panGesture(_:)), allowedDirections: ...)
    }
}

NavigationController 擴(kuò)展 UINavigationController 以借用其公共API喇聊,使得它可以像普通的 UIViewController 一樣被使用,它在內(nèi)部重寫了以下內(nèi)容:

  • 直接管理子控制器蹦狂;由于它只是一個(gè)簡(jiǎn)單的數(shù)組誓篱,因此可以對(duì)堆棧操作進(jìn)行自由調(diào)整。
  • 過(guò)渡動(dòng)畫(huà)凯楔;你可以在 ContainedViewLayoutTransition 中找到所有動(dòng)畫(huà)詳細(xì)信息窜骄。
  • 交互式pop手勢(shì);InteractiveTransitionGestureRecognizer 可以在整個(gè)屏幕范圍響應(yīng) pop 手勢(shì)摆屯。
  • 像 iPad 這樣的大屏設(shè)備分屏布局邻遏,它支持兩種類型的布局:flatsplit
  • 主題;通過(guò) theme 屬性可以很輕松的自定義外觀准验。

二赎线、Telegram 布局

AsyncDisplayKit 中的FlexBox 布局系統(tǒng)被混合布局機(jī)制所取代:

// NavigationBar.swift
//   layout in the main thread
open class NavigationBar: ASDisplayNode {
    override open func layout() {
        super.layout()
        
        if let validLayout = self.validLayout, self.requestedLayout {
            self.requestedLayout = false
            self.updateLayout(size: validLayout.0, defaultHeight: validLayout.1, additionalHeight: validLayout.2, leftInset: validLayout.3, rightInset: validLayout.4, appearsHidden: validLayout.5, transition: .immediate)
        }
    }
    func updateLayout(size: CGSize, defaultHeight: CGFloat, additionalHeight: CGFloat, leftInset: CGFloat, rightInset: CGFloat, appearsHidden: Bool, transition: ContainedViewLayoutTransition)
}

// TabBarController.swift
//   layout in the main thread
open class TabBarController: ViewController {
    override open func containerLayoutUpdated(_ layout: ContainerViewLayout, transition: ContainedViewLayoutTransition) {
        super.containerLayoutUpdated(layout, transition: transition)
        self.tabBarControllerNode.containerLayoutUpdated(layout, toolbar: self.currentController?.toolbar, transition: transition)
        ...
    }
}


// ListView.swift
//     asynchronously load visible items by the scrolling event
open class ListView: ASDisplayNode, ... {
    public func scrollViewDidScroll(_ scrollView: UIScrollView) {
        self.updateScrollViewDidScroll(scrollView, synchronous: false)
    }
    private func updateScrollViewDidScroll(_ scrollView: UIScrollView, synchronous: Bool) {
        ...
        self.enqueueUpdateVisibleItems(synchronous: synchronous)
    }
    private func enqueueUpdateVisibleItems(synchronous: Bool) {
        ...
        strongSelf.updateVisibleItemsTransaction(synchronous: synchronous, completion:...)
    }
    private func updateVisibleItemsTransaction(synchronous: Bool, completion: @escaping () -> Void)
}
  • 所有布局都是手動(dòng)完成的。
  • 簡(jiǎn)單 UI 的布局計(jì)算在主線程中進(jìn)行糊饱;布局代碼可以放在 nodelayout 方法中垂寥,也可以放在視圖控制器的 containerLayoutUpdated 方法中。
  • ListView 為其 item nodes 構(gòu)建靈活的布局機(jī)制济似,該機(jī)制支持同步和異步計(jì)算矫废。

后記

Telegram 集成 AsyncDisplayKit 的方式讓人驚嘆,它基于 ASDK 的 node 重構(gòu)了整個(gè) UIKit 用以提高效率和自由度砰蠢,雖然它的聊天 UI 界面非常復(fù)雜蓖扑,但在老機(jī)型上卻也有流暢體驗(yàn)。

參考資料:

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末台舱,一起剝皮案震驚了整個(gè)濱河市律杠,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌竞惋,老刑警劉巖柜去,帶你破解...
    沈念sama閱讀 218,755評(píng)論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異拆宛,居然都是意外死亡嗓奢,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,305評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門浑厚,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)股耽,“玉大人,你說(shuō)我怎么就攤上這事钳幅∥矧” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 165,138評(píng)論 0 355
  • 文/不壞的土叔 我叫張陵敢艰,是天一觀的道長(zhǎng)诬乞。 經(jīng)常有香客問(wèn)我,道長(zhǎng)钠导,這世上最難降的妖魔是什么震嫉? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,791評(píng)論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮牡属,結(jié)果婚禮上责掏,老公的妹妹穿的比我還像新娘。我一直安慰自己湃望,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,794評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著证芭,像睡著了一般瞳浦。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上废士,一...
    開(kāi)封第一講書(shū)人閱讀 51,631評(píng)論 1 305
  • 那天叫潦,我揣著相機(jī)與錄音,去河邊找鬼官硝。 笑死矗蕊,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的氢架。 我是一名探鬼主播傻咖,決...
    沈念sama閱讀 40,362評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼岖研!你這毒婦竟也來(lái)了卿操?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 39,264評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤孙援,失蹤者是張志新(化名)和其女友劉穎害淤,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體拓售,經(jīng)...
    沈念sama閱讀 45,724評(píng)論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡窥摄,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,900評(píng)論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了础淤。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片崭放。...
    茶點(diǎn)故事閱讀 40,040評(píng)論 1 350
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖值骇,靈堂內(nèi)的尸體忽然破棺而出莹菱,到底是詐尸還是另有隱情,我是刑警寧澤吱瘩,帶...
    沈念sama閱讀 35,742評(píng)論 5 346
  • 正文 年R本政府宣布道伟,位于F島的核電站,受9級(jí)特大地震影響使碾,放射性物質(zhì)發(fā)生泄漏蜜徽。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,364評(píng)論 3 330
  • 文/蒙蒙 一票摇、第九天 我趴在偏房一處隱蔽的房頂上張望拘鞋。 院中可真熱鬧,春花似錦矢门、人聲如沸盆色。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,944評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)隔躲。三九已至摩梧,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間宣旱,已是汗流浹背仅父。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,060評(píng)論 1 270
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留浑吟,地道東北人笙纤。 一個(gè)月前我還...
    沈念sama閱讀 48,247評(píng)論 3 371
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像组力,于是被迫代替她去往敵國(guó)和親省容。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,979評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容