成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

一個(gè)案例搞懂Vue.js的作用域插槽

開(kāi)發(fā) 前端
當(dāng)你無(wú)法理解一個(gè)東西的時(shí)候,最好的辦法就是在解決問(wèn)題的過(guò)程中體會(huì)它的應(yīng)用。本文將向你展示如何使用作用域插槽構(gòu)建一個(gè)可復(fù)用的列表組件。

 [[279199]]

作用域插槽是 Vue.js 中一個(gè)很有用的特性,可以顯著提高組件的通用性和可復(fù)用性。問(wèn)題在于,它實(shí)在不太好理解。嘗試搞清楚父子作用域之間錯(cuò)綜復(fù)雜的關(guān)系,其痛苦程度不亞于求解一個(gè)棘手的數(shù)學(xué)方程。

當(dāng)你無(wú)法理解一個(gè)東西的時(shí)候,最好的辦法就是在解決問(wèn)題的過(guò)程中體會(huì)它的應(yīng)用。本文將向你展示如何使用作用域插槽構(gòu)建一個(gè)可復(fù)用的列表組件。

注意: 完整代碼可以去 Codepen    查看

最基礎(chǔ)的組件

我們即將構(gòu)建的組件叫做 my-list ,用來(lái)展示一系列的項(xiàng)目。它的特別之處就在于,你可以在每次使用組件的時(shí)候自定義列表項(xiàng)目的渲染方式。

我們先從最簡(jiǎn)單的單個(gè)列表開(kāi)始:一個(gè)包含幾何圖形名字和邊數(shù)的數(shù)組。

app.js 

  1. Vue.component('my-list', {  
  2.   template: '#my-list',  
  3.   data() {  
  4.     return {  
  5.       title: 'Shapes',  
  6.       shapes: [   
  7.         { name: 'Square', sides: 4 },   
  8.         { name: 'Hexagon', sides: 6 },   
  9.         { name: 'Triangle', sides: 3 }  
  10.       ]  
  11.     };  
  12.   }  
  13. });  
  14. new Vue({  
  15.   el: '#app'  
  16. }); 

index.html 

  1. <div id="app">  
  2.   <my-list></my-list>  
  3. </div>  
  4. <script type="text/x-template" id="my-list">  
  5.   <div class="my-list">  
  6.     <div class="title">{{ title }}</div>  
  7.     <div class="list">  
  8.       <div class="list-item" v-for="shape in shapes">  
  9.         <div>{{ shape.name }} <small>({{ shape.sides }} sides)</small></div>  
  10.       </div>  
  11.     </div>  
  12.   </div>  
  13. </script> 

在加上一點(diǎn)樣式,大概就會(huì)是下圖這個(gè)樣子:

更通用的 my-list

現(xiàn)在我們想要讓 my-list 更加通用,可以渲染任何類型的列表。這次我們展示的是一堆顏色的名字以及對(duì)應(yīng)的顏色方塊。

為此,我們需要將上例列表獨(dú)有的數(shù)據(jù)進(jìn)行抽象化。由于列表中的項(xiàng)目可能有不同的結(jié)構(gòu),我們將會(huì)給 my-list 一個(gè)插槽,讓父組件來(lái)定義列表的展示方式。

app.js 

  1. Vue.component('my-list', {  
  2.   template: '#my-list',  
  3.   props: [ 'title' ]  
  4. }); 

index.html 

  1. <script type="text/x-template" id="my-list">  
  2.   <div class="my-list">  
  3.     <div class="title">{{ title }}</div>  
  4.     <div class="list">  
  5.       <slot></slot>  
  6.     </div>  
  7.   </div>  
  8. </script> 

現(xiàn)在,我們?cè)诟鶎?shí)例中創(chuàng)建 my-list 組件的兩個(gè)實(shí)例,分別展示兩個(gè)測(cè)試用例列表:lists:

app.js 

  1. new Vue({  
  2.   el: '#app',  
  3.   data: {  
  4.     shapes: [   
  5.       { name: 'Square', sides: 4 },   
  6.       { name: 'Hexagon', sides: 6 },   
  7.       { name: 'Triangle', sides: 3 }  
  8.     ],  
  9.     colors: [  
  10.       { name: 'Yellow', hex: '#F4D03F', },  
  11.       { name: 'Green', hex: '#229954' },  
  12.       { name: 'Purple', hex: '#9B59B6' }  
  13.     ]  
  14.   }  
  15. });  
  1. <div id="app">  
  2.   <my-list :title="Shapes">  
  3.     <div class="list-item" v-for="item in shapes">  
  4.       <div>{{ shape.name }} <small>({{ shape.sides }} sides)</small></div>  
  5.     </div>  
  6.   </my-list>  
  7.   <my-list :title="Colors">  
  8.     <div class="list-item" v-for="color in colors">  
  9.       <div>  
  10.         <div class="swatch" :style="{ background: color.hex }"></div>  
  11.         {{ color.name }}  
  12.       </div>  
  13.     </div>  
  14.   </my-list>  
  15. </div> 

效果如下圖:

大材小用的組件

我們剛才創(chuàng)建的組件確實(shí)符合要求,但那段代碼算不上很好。my-list 本來(lái)應(yīng)該是一個(gè)展示列表的組件,但我們卻把渲染列表需要的邏輯部分抽象到了父組件中,這樣一來(lái),子組件在這里只不過(guò)是用來(lái)包裹列表而已,未免顯得大材小用了。

更糟糕的是,在兩個(gè)組件的聲明中存在著大量重復(fù)代碼(例如,<div class="list-item" v-for="item in ...">)。如果我們能夠在子組件中編寫(xiě)這些代碼,那么子組件就不再是“打醬油的角色”了。

作用域插槽

普通插槽無(wú)法滿足我們的需求,這時(shí)候,作用域插槽就派上用場(chǎng)了。作用域插槽允許你傳遞一個(gè)模板而不是已經(jīng)渲染好的元素給插槽。之所以叫做”作用域“插槽,是因?yàn)槟0咫m然是在父級(jí)作用域中渲染的,卻能拿到子組件的數(shù)據(jù)。

例如,帶有作用域插槽的組件 child 大概是下面這個(gè)樣子: 

  1. <div>  
  2.   <slot my-prop="Hello from child"></slot>  
  3. </div> 

使用這個(gè)組件的父組件將會(huì)在插槽中聲明一個(gè) template 元素。這個(gè)模板元素會(huì)有一個(gè) scope (譯者注:Vue 2.6 后改為 v-slot 屬性)屬性指向一個(gè)對(duì)象,任何添加到插槽(位于子組件模板)中的屬性都會(huì)作為這個(gè)對(duì)象的屬性。 

  1. <child>  
  2.   <template scope="props">  
  3.     <span>Hello from parent</span>  
  4.     <span>{{ props.my-prop }}</span>  
  5.   </template>  
  6. </child> 

將會(huì)渲染成: 

  1. <div>  
  2.   <span>Hello from parent</span>  
  3.   <span>Hello from child</span>  
  4. </div> 

在 my-list 中使用作用域插槽

我們將兩個(gè)列表數(shù)組通過(guò) props 傳遞給 my-list。之后將普通插槽替換為作用域插槽,這樣,my-list 就能夠負(fù)責(zé)迭代列表項(xiàng)目,同時(shí)父組件依然能夠定義每個(gè)項(xiàng)目具體的展示方式。

index.html 

  1. <div id="app">  
  2.   <my-list title="Shapes" :items="shapes">  
  3.     <!--在這里書(shū)寫(xiě) template-->  
  4.   </my-list>  
  5.   <my-list title="Colors" :items="colors">  
  6.     <!--在這里書(shū)寫(xiě) template-->  
  7.   </my-list>     
  8. </div> 

接著我們讓 my-list 迭代項(xiàng)目。在 v-for 循環(huán)中,item 是當(dāng)前迭代項(xiàng)目的別名。我們可以創(chuàng)建一個(gè)插槽并通過(guò) v-bind="item" 將那個(gè)項(xiàng)目綁定到插槽中。

app.js 

  1. Vue.component('my-list', {  
  2.   template: '#my-list',  
  3.   props: [ 'title', 'items' ]  
  4. }); 

index.html 

  1. <script type="text/x-template" id="my-list">  
  2.   <div class="my-list">  
  3.     <div class="title">{{ title }}</div>  
  4.     <div class="list">  
  5.       <div v-for="item in items">  
  6.         <slot v-bind="item"></slot>  
  7.       </div>  
  8.     </div>  
  9.   </div>  
  10. </script> 

注意:也許你之前沒(méi)見(jiàn)過(guò)不帶參數(shù)的 v-bind 用法。這種用法將會(huì)把整個(gè)對(duì)象的所以屬性都綁定到當(dāng)前元素上。在涉及作用域插槽時(shí),這種用法很常見(jiàn),因?yàn)榻壎ǖ膶?duì)象可能有很多屬性,而一一將它們列舉出來(lái)并手動(dòng)綁定顯然太麻煩了。

現(xiàn)在,回到根實(shí)例這里來(lái),在 my-list 的插槽中聲明一個(gè)模板。首先看一下幾何圖形列表(第一個(gè)例子中的列表),我們聲明的模板必須帶有一個(gè) scope 屬性,這里將其賦值為 shape。shape 這個(gè)別名可以讓我們?cè)L問(wèn)作用域插槽。在模板中,我們可以繼續(xù)沿用最初例子中的標(biāo)記來(lái)展示項(xiàng)目。 

  1. <my-list title="Shapes" :items="shapes">  
  2.   <template scope="shape">  
  3.     <div>{{ shape.name }} <small>({{ shape.sides }} sides)</small></div>  
  4.   </template>  
  5. </my-list> 

整個(gè)模板大概是下面這樣: 

  1. <div id="app">  
  2.   <my-list title="Shapes" :items="shapes">  
  3.     <template scope="shape">  
  4.       <div>{{ shape.name }} <small>({{ shape.sides }} sides)</small></div>  
  5.     </template>  
  6.   </my-list>  
  7.   <my-list title="Colors" :items="colors">  
  8.     <template scope="color">  
  9.       <div>  
  10.         <div class="swatch" :style="{ background: color.hex }"></div>  
  11.         {{ color.name }}  
  12.       </div>  
  13.     </template>  
  14.   </my-list>     
  15. </div> 

結(jié)論

雖然用上作用域插槽之后,代碼量并未減少,但是我們將通用的功能都交由子組件負(fù)責(zé),這顯著提高了代碼的健壯性。

完整代碼的 Codepen 在這里:

https://codepen.io/anthonygor...

譯者注: Vue.js 2.6.0 之后將 slot-scope 改為 v-slot

 

 

責(zé)任編輯:龐桂玉 來(lái)源: segmentfault
相關(guān)推薦

2020-05-25 17:03:47

Vue嵌套插槽開(kāi)發(fā)

2024-01-16 12:19:08

MySQL重要機(jī)制高并發(fā)

2021-04-14 07:52:00

Vue 作用域插槽

2018-01-31 15:45:07

前端Vue.js組件

2011-03-31 11:15:52

網(wǎng)頁(yè)設(shè)計(jì)Web

2022-02-10 10:48:23

JavaScriptVue.js數(shù)據(jù)

2023-07-13 12:21:18

2022-04-05 16:44:59

系統(tǒng)Vue.js響應(yīng)式

2019-04-29 14:51:05

前后端JavaVue.js

2022-04-09 17:53:56

Vue.js分支切換嵌套的effect

2011-03-04 10:07:34

Win7SQL Server連接

2023-03-07 16:09:08

2019-10-11 09:59:55

開(kāi)發(fā)者技能工具

2020-04-30 09:17:28

數(shù)據(jù)分析電商分析思維

2020-12-07 06:26:32

模式交付工作

2022-01-13 22:37:26

VSCode代碼 編輯器

2023-04-26 01:25:05

案例故障模型

2020-03-24 08:32:24

vue作用域前端

2021-03-12 08:21:54

JavaScript 前端原生js

2023-10-12 12:43:16

組件Vue
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

主站蜘蛛池模板: 日日夜夜精品免费视频 | 国产黄色在线 | 欧美a在线| 中文字幕在线免费观看 | 爱爱综合网 | 亚洲a在线视频 | 国产精品一区一区三区 | 午夜在线视频 | 色999日韩 | 国产免费一区 | 免费一区二区三区 | av黄色在线 | 欧美精品乱码久久久久久按摩 | 中文字幕精品一区二区三区精品 | 久久精品色欧美aⅴ一区二区 | 日韩成人免费视频 | 99热在线播放 | 精品一区二区三区四区 | 国产一区亚洲二区三区 | 一区二区三区成人 | 97久久国产| 一区二区三区四区在线视频 | 久久久久亚洲国产| 97精品视频在线 | 日本黄色免费大片 | 国产91丝袜在线播放 | 久久免费精品视频 | av一区二区三区四区 | 久久中文字幕电影 | 欧美日韩一区二区视频在线观看 | 99re国产视频 | www.久久精品| 成人av大全 | 在线观看成人av | 亚洲成人精品免费 | 亚洲欧美v | 国产视频中文字幕 | 99久久婷婷国产综合精品电影 | 欧美一级片在线看 | 国产高清在线观看 | 涩涩视频在线观看 |