Vue學(xué)習(xí)筆記進(jìn)階篇——函數(shù)化組件

本文為轉(zhuǎn)載灿渴,原文:Vue學(xué)習(xí)筆記進(jìn)階篇——函數(shù)化組件

介紹

之前創(chuàng)建的錨點(diǎn)標(biāo)題組件是比較簡單无拗,沒有管理或者監(jiān)聽任何傳遞給他的狀態(tài)茎用,也沒有生命周期方法绝淡。它只是一個(gè)接收參數(shù)的函數(shù)缠捌。
在這個(gè)例子中乳蛾,我們標(biāo)記組件為 functional, 這意味它是無狀態(tài)(沒有 data)鄙币,無實(shí)例(沒有 this 上下文)肃叶。
一個(gè) 函數(shù)化組件 就像這樣:

Vue.component('my-component', {
  functional: true,
  // 為了彌補(bǔ)缺少的實(shí)例
  // 提供第二個(gè)參數(shù)作為上下文
  render: function (createElement, context) {
    // ...
  },
  // Props 可選
  props: {
    // ...
  }
})

組件需要的一切都是通過上下文傳遞,包括:

  1. props: 提供props 的對(duì)象
  1. children: VNode 子節(jié)點(diǎn)的數(shù)組
    slots: slots 對(duì)象
    data: 傳遞給組件的 data 對(duì)象
    parent: 對(duì)父組件的引用
    listeners: (2.3.0+) 一個(gè)包含了組件上所注冊(cè)的 v-on 偵聽器的對(duì)象十嘿。這只是一個(gè)指向 data.on 的別名因惭。
    injections: (2.3.0+) 如果使用了 inject 選項(xiàng), 則該對(duì)象包含了應(yīng)當(dāng)被注入的屬性。

在添加 functional: true 之后绩衷,錨點(diǎn)標(biāo)題組件的 render 函數(shù)之間簡單更新增加context參數(shù)蹦魔,this.$slots.default 更新為 context.children,之后this.level 更新為 context.props.level咳燕。
因?yàn)楹瘮?shù)化組件只是一個(gè)函數(shù)勿决,所以渲染開銷也低很多。另外招盲,這也意味著函數(shù)化組件不會(huì)出現(xiàn)在 VueJS Chrome 開發(fā)者工具的組件樹里低缩。
在作為包裝組件時(shí)它們也同樣非常有用,比如曹货,當(dāng)你需要做這些時(shí):
程序化地在多個(gè)組件中選擇一個(gè)
在將 children, props, data 傳遞給子組件之前操作它們咆繁。
下面是一個(gè)依賴傳入 props 的值的smart-list組件例子,它能代表更多具體的組件:

var EmptyList = { /* ... */ }
var TableList = { /* ... */ }
var OrderedList = { /* ... */ }
var UnorderedList = { /* ... */ }
Vue.component('smart-list', {
  functional: true,
  render: function (createElement, context) {
    function appropriateListComponent () {
      var items = context.props.items
      if (items.length === 0)           return EmptyList
      if (typeof items[0] === 'object') return TableList
      if (context.props.isOrdered)      return OrderedList
      return UnorderedList
    }
    return createElement(
      appropriateListComponent(),
      context.data,
      context.children
    )
  },
  props: {
    items: {
      type: Array,
      required: true
    },
    isOrdered: Boolean
  }
})

slots()和children對(duì)比

你可能想知道為什么同時(shí)需要 slots()children顶籽。slots().default 不是和 children 類似的嗎玩般?在一些場景中,是這樣礼饱,但是如果是函數(shù)式組件和下面這樣的 children 呢坏为?

<my-functional-component>
  <p slot="foo">
    first
  </p>
  <p>second</p>
</my-functional-component>

對(duì)于這個(gè)組件,children 會(huì)給你兩個(gè)段落標(biāo)簽镊绪,而 slots().default 只會(huì)傳遞第二個(gè)匿名段落標(biāo)簽匀伏,slots().foo 會(huì)傳遞第一個(gè)具名段落標(biāo)簽。同時(shí)擁有 childrenslots() 镰吆,因此你可以選擇讓組件通過 slot() 系統(tǒng)分發(fā)或者簡單的通過 children 接收帘撰,讓其他組件去處理。

示例

漸進(jìn)過渡

之前的Vue學(xué)習(xí)筆記進(jìn)階篇——列表過渡及其他中可復(fù)用的過渡提到用函數(shù)組件實(shí)現(xiàn)合適万皿,下面就用函數(shù)化組件來實(shí)現(xiàn)那個(gè)漸進(jìn)過渡

<div id="app5">
    <input v-model="query">
    <my-transition :query="query" :list="list">
        <li v-for="(item, index) in computedList"
            :key="item.msg"
            :data-index="index">
            {{item.msg}}
        </li>
    </my-transition>
</div>
    Vue.component('my-transition', {
        functional:true,
        render:function (h, ctx) {
            var data = {
                props:{
                    tag:'ul',
                    css:false
                },
                on:{
                    beforeEnter:function (el) {
                        el.style.opacity = 0
                        el.style.height = 0
                    },
                    enter:function (el, done) {
                        var delay = el.dataset.index * 150
                        setTimeout(function () {
                            Velocity(el, {opacity:1, height:'1.6em'},{complete:done})
                        }, delay)
                    },
                    leave:function (el, done) {
                        var delay = el.dataset.index * 150
                        setTimeout(function () {
                            Velocity(el, {opacity:0, height:0}, {complete:done})
                        }, delay)
                    }
                }
            }
            return h('transition-group', data, ctx.children)
        },
        props:['query', 'list']
    })

    var app5 = new Vue({
        el:'#app5',
        data:{
            query:'',
            list:[
                {msg:'Bruce Lee'},
                {msg:'Jackie Chan'},
                {msg:'Chuck Norris'},
                {msg:'Jet Li'},
                {msg:'Kung Furry'},
                {msg:'Chain Zhang'},
                {msg:'Iris Zhao'},
            ]
        },
        computed:{
            computedList:function () {
                var vm = this
                return this.list.filter(function (item) {
                    return item.msg.toLowerCase().indexOf(vm.query.toLowerCase()) !== -1
                })
            }
        },
    })

運(yùn)行結(jié)果:


本文為原創(chuàng)摧找,轉(zhuǎn)載請(qǐng)注明出處核行。
上一節(jié):Vue學(xué)習(xí)筆記進(jìn)階篇——Render函數(shù)
返回目錄
下一節(jié):Vue學(xué)習(xí)筆記進(jìn)階篇——vue-cli安裝及介紹

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市蹬耘,隨后出現(xiàn)的幾起案子芝雪,更是在濱河造成了極大的恐慌,老刑警劉巖综苔,帶你破解...
    沈念sama閱讀 206,214評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件惩系,死亡現(xiàn)場離奇詭異,居然都是意外死亡如筛,警方通過查閱死者的電腦和手機(jī)堡牡,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,307評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來杨刨,“玉大人晤柄,你說我怎么就攤上這事⊙停” “怎么了芥颈?”我有些...
    開封第一講書人閱讀 152,543評(píng)論 0 341
  • 文/不壞的土叔 我叫張陵,是天一觀的道長赚抡。 經(jīng)常有香客問我爬坑,道長,這世上最難降的妖魔是什么涂臣? 我笑而不...
    開封第一講書人閱讀 55,221評(píng)論 1 279
  • 正文 為了忘掉前任盾计,我火速辦了婚禮泞坦,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘圆到。我一直安慰自己勿侯,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,224評(píng)論 5 371
  • 文/花漫 我一把揭開白布等孵。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪炫乓。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,007評(píng)論 1 284
  • 那天献丑,我揣著相機(jī)與錄音末捣,去河邊找鬼。 笑死创橄,一個(gè)胖子當(dāng)著我的面吹牛箩做,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播妥畏,決...
    沈念sama閱讀 38,313評(píng)論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼邦邦,長吁一口氣:“原來是場噩夢(mèng)啊……” “哼安吁!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起燃辖,我...
    開封第一講書人閱讀 36,956評(píng)論 0 259
  • 序言:老撾萬榮一對(duì)情侶失蹤鬼店,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后黔龟,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體妇智,經(jīng)...
    沈念sama閱讀 43,441評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,925評(píng)論 2 323
  • 正文 我和宋清朗相戀三年氏身,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了巍棱。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,018評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡蛋欣,死狀恐怖拉盾,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情豁状,我是刑警寧澤捉偏,帶...
    沈念sama閱讀 33,685評(píng)論 4 322
  • 正文 年R本政府宣布,位于F島的核電站泻红,受9級(jí)特大地震影響夭禽,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜谊路,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,234評(píng)論 3 307
  • 文/蒙蒙 一讹躯、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧缠劝,春花似錦潮梯、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,240評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至脱羡,卻和暖如春萝究,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背锉罐。 一陣腳步聲響...
    開封第一講書人閱讀 31,464評(píng)論 1 261
  • 我被黑心中介騙來泰國打工帆竹, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人脓规。 一個(gè)月前我還...
    沈念sama閱讀 45,467評(píng)論 2 352
  • 正文 我出身青樓栽连,卻偏偏與公主長得像,于是被迫代替她去往敵國和親侨舆。 傳聞我的和親對(duì)象是個(gè)殘疾皇子秒紧,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,762評(píng)論 2 345

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