Tangram系列(一)Android Tangram入門


背景

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

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

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

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

VLayout(Android)

思路

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

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

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

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

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

LayoutHelper包含了它負(fù)責(zé)的組件的位置起始區(qū)域,它們會(huì)被傳遞給自定義的LayoutManager。當(dāng)RecyclerView開始渲染頁(yè)面或者滑動(dòng)時(shí),它內(nèi)部維護(hù)了一個(gè)布局狀態(tài),獲取當(dāng)前屏幕范圍內(nèi)還有多少區(qū)域是空白的,下一個(gè)要加載的View的位置是多少,然后把這些信息告訴LayoutManager去加載View做布局。我們?nèi)缓蠼唤oLayoutHelper去布局,不同的LayoutHelper會(huì)按照約定的協(xié)議進(jìn)行進(jìn)一的自定義LayoutManager拿到這個(gè)位置之后,就反向查找對(duì)應(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)了頁(yè)面布局的動(dòng)態(tài)化,我們可以通過配置json文件自由的布局;但還有一個(gè)局限性,json中使用的卡片或者組件的type,必須是在客戶端已經(jīng)實(shí)現(xiàn)好的才能使用;如果想要在原來的版本中新增一個(gè)type類型的組件,這是沒有辦法做到的,還是只能通過升級(jí)客戶端來實(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)過計(jì)算尺寸階段、布局階段、繪制階段來定義,再加上相同的尺寸計(jì)算接口、布局接口、繪制接口,這樣對(duì)于宿主容器來說,包裝在內(nèi)部的組件就不分虛擬化還是原生,一視同仁,暴露給外面的接口也是一樣的,只要將宿主容器像普通的 View 一樣添加到的視圖界面上,就可以在后續(xù)的渲染過程中顯示出來。如果虛擬組件使用的越多,View 的個(gè)數(shù)就越少,對(duì)于系統(tǒng)來說層級(jí)越扁平。以示例的組件來說,最終呈現(xiàn)的 View 只有宿主容器和兩個(gè)圖片組件,如果將圖片也用虛擬化的方式實(shí)現(xiàn),最終 View 只有一個(gè)宿主容器,而界面仍然保持不變。

虛擬組件

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

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

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

VirtualView接入

在Tangram里使用VirtualView的時(shí)候,很多步驟已經(jīng)內(nèi)置到 Tangram 的初始化里了,外部只需要注冊(cè)業(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 注冊(cè)組件和卡片
        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");//注冊(cè)組件,只需要提供組件類型名稱即可
        //Step 4: new engine
        engine = builder.build();
        engine.setVirtualViewTemplate(VVTEST.BIN);//加載模板數(shù)據(jù)
        //注冊(cè)事件處理器
        engine.getService(VafContext.class).setImageLoaderAdapter(new IImageLoaderAdapter() {.../}

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

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


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

參考

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

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,461評(píng)論 6 532
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,538評(píng)論 3 417
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,423評(píng)論 0 375
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,991評(píng)論 1 312
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,761評(píng)論 6 410
  • 開封第一講書人閱讀 55,207評(píng)論 1 324
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,268評(píng)論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,419評(píng)論 0 288
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,959評(píng)論 1 335
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,782評(píng)論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 42,983評(píng)論 1 369
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,528評(píng)論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,222評(píng)論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,653評(píng)論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,901評(píng)論 1 286
  • 我被黑心中介騙來泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,678評(píng)論 3 392
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 47,978評(píng)論 2 374

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