Tangram系列(一)Android Tangram入門


背景

Tangram是阿里出品茧痕、用于快速實(shí)現(xiàn)組合布局的框架模型突梦,在手機(jī)天貓 Android & iOS版 內(nèi)廣泛使用。
移動端原生App最大的弱點(diǎn)就是不能像Web端那樣即改即用姆吭,需要等到下一個版本發(fā)布之后才能看到效果已艰,而衍生出的如多動態(tài)方案(如RN)在動態(tài)化或性能方面都不太完美痊末。所以產(chǎn)生了用于快速實(shí)現(xiàn)組合布局的框架模型Tangram。

Tangram基本理念是粗顆粒度組件化+靈活布局容器哩掺,重點(diǎn)關(guān)注高性能(頁面渲染效率&組件回收復(fù)用凿叠,跨父節(jié)點(diǎn)組件的高效回收與復(fù)用);面向業(yè)務(wù)(組件業(yè)務(wù)化、動態(tài)化盒件,通過布局+組件的形式搭建整個頁面蹬碧,而不是從基本的UI元素搭建頁面);多終端一致性(一個json描述的布局可以同時在iOS端與Android端使用履恩,且表現(xiàn)一致)锰茉。

整體頁面由卡片組成呢蔫,卡片由組件組成切心。在Tangram中組件模型是抽象出的最小的可復(fù)用單元,承載業(yè)務(wù)邏輯和UI展示片吊,以盡可能小的業(yè)務(wù)單元為顆粒度绽昏,而卡片模型負(fù)責(zé)邏輯,粗顆燎渭梗化全谤,不提供布局細(xì)節(jié)描述,只聲明布局方式爷贫。

本篇會從Tangram布局框架VLayout(Android)认然、VirtualView初探兩方面來進(jìn)行初步的介紹。

VLayout(Android)

思路

針對需要在長列表下做各種形態(tài)來分配不同元素的電商頁首頁漫萄,傳統(tǒng)的復(fù)用容器(ListView或RecyclerView)會出現(xiàn)復(fù)用性降低以及需要處理嵌套滑動的情況卷员,所以VLayout提供了一個基于RecyclerView的自定義LayoutManger,可以實(shí)現(xiàn)不同布局格式混排腾务。

RecyclerView中正常只有一種布局毕骡,如果列表需要使用不同的布局,可以通過設(shè)置不同的ItemType岩瘦,提供多種ViewHolder實(shí)現(xiàn)未巫。而VLayout使用了不同的思路兜辞,其Adapter是串聯(lián)自多個繼承RecyclerView.Adapter的Adapter來管理視圖的適配和數(shù)據(jù)蕉毯,同時實(shí)現(xiàn)緩存提陶。

VLayout中具體布局邏輯在LayoutHelper中實(shí)現(xiàn)丧荐,每個Adapter和LayoutHelper負(fù)責(zé)一個區(qū)域范圍內(nèi)的組件氢妈,不同范圍內(nèi)的組件之間如果類型相同愈污,可以在滑動過程中回收復(fù)用拨拓。

系統(tǒng)默認(rèn)布局類型

實(shí)現(xiàn)原理

LayoutHelper包含了它負(fù)責(zé)的組件的位置起始區(qū)域狠轻,它們會被傳遞給自定義的LayoutManager苏遥。當(dāng)RecyclerView開始渲染頁面或者滑動時饼拍,它內(nèi)部維護(hù)了一個布局狀態(tài),獲取當(dāng)前屏幕范圍內(nèi)還有多少區(qū)域是空白的田炭,下一個要加載的View的位置是多少师抄,然后把這些信息告訴LayoutManager去加載View做布局。我們?nèi)缓蠼唤oLayoutHelper去布局教硫,不同的LayoutHelper會按照約定的協(xié)議進(jìn)行進(jìn)一的自定義LayoutManager拿到這個位置之后叨吮,就反向查找對應(yīng)的LayoutHelper去布局辆布。

public void appendGroup(@Nullable List<L> cards) {
        if (cards == null || cards.size() == 0)
            return;
        createSnapshot();
        final List<LayoutHelper> helpers = new LinkedList<>(getLayoutHelpers());
        //根據(jù)組件大小進(jìn)行擴(kuò)容
        mCards.ensureCapacity(mCards.size() + cards.size());
        //add helper負(fù)責(zé)布局內(nèi)的組件以及數(shù)據(jù)
        helpers.addAll(transformCards(cards, mData, mCards));
        setLayoutHelpers(helpers);
        diffWithSnapshot();
        //刷新展示
        notifyDataSetChanged();
    }

VirtualView初探

Tangram中已經(jīng)實(shí)現(xiàn)了頁面布局的動態(tài)化,我們可以通過配置json文件自由的布局茶鉴;但還有一個局限性锋玲,json中使用的卡片或者組件的type,必須是在客戶端已經(jīng)實(shí)現(xiàn)好的才能使用涵叮;如果想要在原來的版本中新增一個type類型的組件惭蹂,這是沒有辦法做到的,還是只能通過升級客戶端來實(shí)現(xiàn)割粮。于是2.0版本出現(xiàn)了虛擬化控件VirtualView盾碗。

虛擬化

VirtualView抽象&封裝了Canvas繪制視圖流程,通過使用Canvas來實(shí)現(xiàn)UI控件的繪制舀瓢,虛擬化就是實(shí)現(xiàn)依賴于宿主容器而沒有實(shí)際View(不同于常見的Button廷雅、TextView)。通過XML引用繪制好的UI組件京髓,從而創(chuàng)建出界面模板航缀,客戶端通過解析和加載XML界面模板最終渲染出界面。

虛擬組件

不論是虛擬化組件還是原生組件堰怨,都要經(jīng)過計算尺寸階段芥玉、布局階段、繪制階段來定義诚些,再加上相同的尺寸計算接口飞傀、布局接口、繪制接口诬烹,這樣對于宿主容器來說砸烦,包裝在內(nèi)部的組件就不分虛擬化還是原生,一視同仁绞吁,暴露給外面的接口也是一樣的幢痘,只要將宿主容器像普通的 View 一樣添加到的視圖界面上,就可以在后續(xù)的渲染過程中顯示出來家破。如果虛擬組件使用的越多颜说,View 的個數(shù)就越少,對于系統(tǒng)來說層級越扁平汰聋。以示例的組件來說门粪,最終呈現(xiàn)的 View 只有宿主容器和兩個圖片組件,如果將圖片也用虛擬化的方式實(shí)現(xiàn)烹困,最終 View 只有一個宿主容器玄妈,而界面仍然保持不變。

虛擬組件

優(yōu)化問題及特點(diǎn)

優(yōu)化問題
1.動態(tài)更新UI組件:實(shí)現(xiàn)了模板與數(shù)據(jù)分離,使用XML描述視圖然后在
端上綁定動態(tài)下發(fā)的界面模板&數(shù)據(jù)拟蜻,最終渲染绎签。
2.提高性能:通過使用Canvas繪制減少視圖的層級和個數(shù)來提高布局加載
效率。

特點(diǎn)
1.渲染性能高:渲染出來的視圖結(jié)構(gòu)呈現(xiàn)扁平化酝锅;
2.組件熱更新:通過配套XML模板更新sdk诡必;
3.跨平臺:一套XML模板,可以Android搔扁、iOS通用爸舒;
4.兼容性好:支持加載&渲染原生基礎(chǔ)組件;
5.使用方便:內(nèi)置一系列基礎(chǔ)組件可以直接使用阁谆。

VirtualView接入

在Tangram里使用VirtualView的時候碳抄,很多步驟已經(jīng)內(nèi)置到 Tangram 的初始化里了愉老,外部只需要注冊業(yè)務(wù)組件類型场绿、加載模板數(shù)據(jù)、提供事件處理器嫉入。

        //Step 1: init tangram
        TangramBuilder.init(this, new IInnerImageSetter() {
            @Override
            public <IMAGE extends ImageView> void doLoadImageUrl(@NonNull IMAGE view,
                @Nullable String url) {
                Picasso.with(BannerTestActivity.this).load(url).into(view);
            }
        }, ImageView.class);

        //Tangram.switchLog(true);
        mMainHandler = new Handler(getMainLooper());

        //Step 2: register build=in cells and cards
        builder = TangramBuilder.newInnerBuilder(this);
        //Step 3: register business cells and cards 注冊組件和卡片
        builder.registerCell(1, TestView.class);
        builder.registerCell(10, SimpleImgView.class);
        builder.registerCell(2, SimpleImgView.class);
        builder.registerCell(110,
            TestViewHolderCell.class,
            new ViewHolderCreator<>(R.layout.item_holder, TestViewHolder.class, TextView.class));
        builder.registerCell(199,SingleImageView.class);
        builder.registerVirtualView("vvtest");//注冊組件焰盗,只需要提供組件類型名稱即可
        //Step 4: new engine
        engine = builder.build();
        engine.setVirtualViewTemplate(VVTEST.BIN);//加載模板數(shù)據(jù)
        //注冊事件處理器
        engine.getService(VafContext.class).setImageLoaderAdapter(new IImageLoaderAdapter() {.../}

基礎(chǔ)布局描述實(shí)例

基本布局描術(shù)文件是一個XML文件,并附帶一個json數(shù)據(jù)文件咒林,其中的相關(guān)數(shù)據(jù)來源都可以從數(shù)據(jù)json文件中使用表達(dá)式獲取熬拒。一個XML就是一個組件,Tangram通過加載這個XML文件即可使用該XML文件所描述的組件垫竞,從而實(shí)現(xiàn)了動態(tài)新增組件類型的功能澎粟。


描述布局文件示例
json文件示例

參考

https://github.com/alibaba/tangram-android
https://www.sohu.com/a/122226581_505818
http://www.reibang.com/p/48764ff8449f
http://www.reibang.com/p/cd634106f533
http://pingguohe.net/2017/02/28/vlayout-design.html

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市欢瞪,隨后出現(xiàn)的幾起案子活烙,更是在濱河造成了極大的恐慌,老刑警劉巖遣鼓,帶你破解...
    沈念sama閱讀 216,496評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件啸盏,死亡現(xiàn)場離奇詭異,居然都是意外死亡骑祟,警方通過查閱死者的電腦和手機(jī)回懦,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,407評論 3 392
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來次企,“玉大人怯晕,你說我怎么就攤上這事「卓茫” “怎么了舟茶?”我有些...
    開封第一講書人閱讀 162,632評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我稚晚,道長崇堵,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,180評論 1 292
  • 正文 為了忘掉前任客燕,我火速辦了婚禮鸳劳,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘也搓。我一直安慰自己赏廓,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,198評論 6 388
  • 文/花漫 我一把揭開白布傍妒。 她就那樣靜靜地躺著幔摸,像睡著了一般。 火紅的嫁衣襯著肌膚如雪颤练。 梳的紋絲不亂的頭發(fā)上既忆,一...
    開封第一講書人閱讀 51,165評論 1 299
  • 那天,我揣著相機(jī)與錄音嗦玖,去河邊找鬼患雇。 笑死,一個胖子當(dāng)著我的面吹牛宇挫,可吹牛的內(nèi)容都是我干的苛吱。 我是一名探鬼主播,決...
    沈念sama閱讀 40,052評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼器瘪,長吁一口氣:“原來是場噩夢啊……” “哼翠储!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起橡疼,我...
    開封第一講書人閱讀 38,910評論 0 274
  • 序言:老撾萬榮一對情侶失蹤援所,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后衰齐,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體任斋,經(jīng)...
    沈念sama閱讀 45,324評論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,542評論 2 332
  • 正文 我和宋清朗相戀三年耻涛,在試婚紗的時候發(fā)現(xiàn)自己被綠了废酷。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,711評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡抹缕,死狀恐怖澈蟆,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情卓研,我是刑警寧澤趴俘,帶...
    沈念sama閱讀 35,424評論 5 343
  • 正文 年R本政府宣布睹簇,位于F島的核電站,受9級特大地震影響寥闪,放射性物質(zhì)發(fā)生泄漏太惠。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,017評論 3 326
  • 文/蒙蒙 一疲憋、第九天 我趴在偏房一處隱蔽的房頂上張望凿渊。 院中可真熱鬧,春花似錦缚柳、人聲如沸埃脏。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,668評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽彩掐。三九已至,卻和暖如春灰追,著一層夾襖步出監(jiān)牢的瞬間堵幽,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,823評論 1 269
  • 我被黑心中介騙來泰國打工监嗜, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留谐檀,地道東北人抡谐。 一個月前我還...
    沈念sama閱讀 47,722評論 2 368
  • 正文 我出身青樓裁奇,卻偏偏與公主長得像,于是被迫代替她去往敵國和親麦撵。 傳聞我的和親對象是個殘疾皇子刽肠,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,611評論 2 353

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