亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > JavaScript > 正文

詳解vue 2.6 中 slot 的新用法

2019-11-19 11:12:46
字體:
來源:轉載
供稿:網友

最近發布不久的Vue 2.6,使用插槽的語法變得更加簡潔。 對插槽的這種改變讓我對發現插槽的潛在功能感興趣,以便為我們基于Vue的項目提供可重用性,新功能和更清晰的可讀性。 真正有能力的插槽是什么?

如果你是Vue的新手,或者還沒有看到2.6版的變化,請繼續閱讀。也許學習插槽的最佳資源是Vue自己的文檔,但是我將在這里給出一個綱要。

想閱讀更多優質文章請猛戳GitHub博客 ,一年百來篇優質文章等著你!

插槽是什么?

插槽是Vue組件的一種機制,它允許你以一種不同于嚴格的父子關系的方式組合組件。插槽為你提供了一個將內容放置到新位置或使組件更通用的出口。從一個簡單的例子開始:

// frame.vue<template> <div class="frame"> <slot></slot> </div></template>

這個組件最外層是一個 div 。假設 div 的存在是為了圍繞其內容創建一個樣式框架。這個組件可以通用地用于將框架包圍在wq你想要的任何內容上,來看看它是怎么用的。這里的 frame 組件指的是我們剛才做的組件。

// app.vue<template> <frame><img src="an-image.jpg"></frame></template>

在開始和結束 frame 標記之間的內容將插入到插槽所在的 frame 組件中,替換 slot 標記。這是最基本的方法。還可以簡單地通過填充指定要放入槽中的默認內容

// frame.vue<template> <div class="frame"> <slot>如果這里沒有指定任何內容,這就是默認內容</slot> </div></template>

所以現在如果我們這樣使用它:

// app.vue<template> <frame /></template>

“ 如果這里沒有指定任何內容,這就是默認內容 ”是默認內容,但是如果像以前那樣使用它,默認文本將被 img 標記覆蓋。

多個/命名的插槽

可以向組件添加多個插槽,但是如果這樣做了,那么除了其中一個之外,其他所有插槽都需要有名稱。如果有一個沒有名稱的槽,它就是默認槽。下面是如何創建多個插槽:

// titled-frame.vue<template> <div class="frame"> <header><h2> <slot name="header">Title</slot> </h2></header> <slot>如果這里沒有指定任何內容,這就是默認內容</slot> </div></template>

我們保留了相同的默認槽,但這次我們添加了一個名為 header 的槽,可以在其中輸入標題,用法如下:

// app.vue<template> <titled-frame> <template v-slot:header> <!-- The code below goes into the header slot --> My Image's Title </template> <!-- The code below goes into the default slot --> <img src="an-image.jpg"> </titled-frame></template>

就像之前一樣,如果我們想將內容添加到默認槽中,只需將其直接放在 titled-frame 組件中。但是,要將內容添加到命名槽中,我們需要用 v-slot 指令將代碼包裹在在 template 標記中。在 v-slot 之后添加冒號 (:) ,然后寫出要傳遞內容的 slot 的名稱。

注意, v-slot 是 Vue 2.6 的新版本,所以如果你使用的是舊版本,則需要閱讀 關于不推薦的slot語法的文檔。

作用域插槽

還需要知道的另一件事是插槽可以將數據/函數傳遞給他們的孩子。 為了證明這一點,我們需要一個完全不同的帶有插槽的示例組件:創建一個組件,該組件將當前用戶的數據提供給其插槽:

// current-user.vue<template> <span> <slot v-bind:user="user"> {{ user.lastName }} </slot> </span></template><script>export default { data () { return { user: ... } }}</script>

該組件有一個名為 user 的屬性,其中包含關于用戶的詳細信息。默認情況下,組件顯示用戶的姓,但請注意,它使用 v-bind 將用戶數據綁定到 slot 。這樣,我們就可以使用這個組件向它的后代提供用戶數據

// app.vue<template> <current-user> <template v-slot:default="slotProps">{{ slotProps.user.firstName }}</template>  </current-user></template>

為了訪問傳遞給 slot 的數據,我們使用v-slot指令的值指定作用域變量的名稱。

這里有幾點需要注意:

  • 我們指定了 default 的名稱,但是不需要為默認槽指定名稱。相反,我們可以使用v -slot="slotProps"
  • 不需要使用 slotProps 作為名稱,可以隨便叫它什么。
  • 如果只使用默認槽,可以跳過內部 template 標記,直接將 v-slot 指令放到當前 current-user 上。
  • 可以使用對象解構來創建對作用域插槽數據的直接引用,而不是使用單個變量名。換句話說,可以使用 v-slot="{user}" 代替 v-slot="slotProps" ,然后可以直接使用 user 而不是 slotProps.user 。

所以,上面的例子可以這樣重寫

// app.vue<template> <current-user v-slot="{user}"> {{ user.firstName }} </current-user></template>

還有幾點要記?。?/strong>

  • 可以使用 v-bind 指令綁定多個值。
  • 也可以將函數傳遞到作用域槽。許多庫使用它來提供可重用的函數組件。
  • v-slot 的別名是 # 。因此,可以用 #header="data" 來代替 v-slot:header="data" 。還可以使用 #header 來代替 v-slot:header (前提:不是作用域插槽時)。對于默認插槽,在使用別名時需要 指定默認名稱 。換句話說,需要這樣寫 #default="data" 而不是 #="data" 。

可以從 文檔 中了解更多的細節,但這足以幫助你理解在本文剩下部分中討論的內容。

你能用插槽做什么?

插槽不是為了一個目的而構建的,或者至少如果它們是,它們已經超越了最初的意圖,成為做許多不同事物的強大工具。

可重用的模式

組件總是被設計為可重用的,但是某些模式對于使用單個“普通”組件來實施是不切實際的,因為為了自定義它,需要的 props 數量可能過多或者需要通過 props 傳遞大部分內容或其它組件。

插槽可用包裹外部的HTML標簽或者組件,并允許其他HTML或組件放在具名插槽對應名稱的插槽上。

對于的第一個例子,從簡單的東西開始:一個按鈕。假設咱們的團隊正在使用 Bootstrap。使用Bootstrap,按鈕通常與基本的 “btn” 類和指定顏色的類綁定在一起,比如 “btn-primary” 。你還可以添加 size 類,比如 'btn-lg' 。

為了簡單起見,現在讓我們假設你的應用使用 btn 、 btn-primary 和 btn-lg 。你不希望總是必須在按鈕上寫下這三個類,或者你不相信新手會記得寫下這三個類。

在這種情況下,可以創建一個自動包含所有這三個類的組件,但是如何允許自定義內容? prop 不實用,因為允許按鈕包含各種HTML,因此我們應該使用一個插槽。

<!-- my-button.vue --><template> <button class="btn btn-primary btn-lg"> <slot>Click Me!</slot> </button></template>

現在我們可以在任何地方使用它,無論你想要什么內容

<!-- 使用 my-button.vue --><template> <my-button> <img src="/img/awesome-icon.jpg"> 我是小智! </my-button></template>

當然,你可以選擇比按鈕更大的東西。 堅持使用Bootstrap,讓我們看一個模態:

<!-- my-modal.vue --><template><div class="modal" tabindex="-1" role="dialog"> <div class="modal-dialog" role="document"> <div class="modal-content"> <div class="modal-header"> <slot name="header"></slot> <button type="button" class="close" data-dismiss="modal" aria-label="Close">  <span aria-hidden="true">×</span> </button> </div> <div class="modal-body"> <slot name="body"></slot> </div> <div class="modal-footer"> <slot name="footer"></slot> </div> </div> </div></div></template>

現在,使用它:

<!-- 使用 my-modal.vue --><template> <my-modal> <template #header> <h5>大家最棒!</h5> </template> <template #body> <p>大家加油</p> </template> <template #footer> <em>大家好樣的!</em> </template> </my-modal></template>

上述類型的插槽用例顯然非常有用,但它可以做得更多。

復用函數

Vue組件并不完全是關于HTML和CSS的。它們是用JavaScript構建的,所以也是關于函數的。插槽對于一次性創建函數并在多個地方使用功能非常有用。讓我們回到模態示例并添加一個關閉模態的函數

<!-- my-modal.vue --><template><div class="modal" tabindex="-1" role="dialog"> <div class="modal-dialog" role="document"> <div class="modal-content"> <div class="modal-header"> <slot name="header"></slot> <button type="button" class="close" data-dismiss="modal" aria-label="Close">  <span aria-hidden="true">×</span> </button> </div> <div class="modal-body"> <slot name="body"></slot> </div> <div class="modal-footer">  <slot name="footer" :closeModal="closeModal"></slot> </div> </div> </div></div></template><script>export default { //... methods: { closeModal () { // 關閉對話框時,需要做的事情 } }}</script>

當使用此組件時,可以向 footer 添加一個可以關閉模​​態的按鈕。 通常,在Bootstrap模式的情況下,可以將 data-dismiss =“modal” 添加到按鈕來進行關閉。

但我們希望隱藏Bootstrap 特定的東西。 所以我們傳遞給他們一個他們可以調用的函數,這樣使用者就不會知道我們有使用 Bootstrap 的東西。

<!-- 使用 my-modal.vue --><template> <my-modal> <template #header> <h5>Awesome Interruption!</h5> </template> <template #body> <p>大家加油!</p> </template> <template #footer="{closeModal}"> <button @click="closeModal"> 點我可以關閉煩人的對話框 </button> </template> </my-modal></template>

無渲染組件

最后,可以利用你所知道的關于使用插槽來傳遞可重用函數的知識,并剝離所有HTML,只使用插槽。這就是無渲染組件的本質:一個只提供函數而不包含任何HTML的組件。

使組件真正無渲染可能有點棘手,因為需要編寫 render 函數而不是使用模板來消除對根元素的依賴,但它可能并不總是必要的。 來看看一個先使用模板的簡單示例:

<template> <transition name="fade" v-bind="$attrs" v-on="$listeners"> <slot></slot> </transition></template><style>.fade-enter-active,.fade-leave-active { transition: opacity 0.3s;}.fade-enter, .fade-leave-to { opacity: 0;}</style>

這是一個無渲染組件的奇怪例子,因為它甚至沒有任何JavaScript。這主要是因為我們正在創建一個內置無渲染函數的預配置可重用版本: transition 。

是的,Vue有內置的無渲染組件。這個特殊的例子取自Cristi Jora的一篇關于 可重用transition 的文章,展示了一種創建無渲染組件的簡單方法,該組件可以標準化整個應用程序中使用的 transition 。

對于我們的另一個示例,我們將創建一個組件來處理切換 Promise 的不同狀態中顯示的內容: pending、resolved 和 failed。這是一種常見的模式,雖然它不需要很多代碼,但是如果沒有為了可重用性而提取邏輯,它會使很多組件變得混亂。

<!-- promised.vue --><template> <span> <slot name="rejected" v-if="error" :error="error"></slot> <slot name="resolved" v-else-if="resolved" :data="data"></slot> <slot name="pending" v-else></slot> </span></template><script>export default { props: { promise: Promise }, data: () => ({ resolved: false, data: null, error: null }),  watch: { promise: { handler (promise) { this.resolved = false this.error = null if (!promise) {  this.data = null  return } promise.then(data => {  this.data = data  this.resolved = true }) .catch(err => {  this.error = err  this.resolved = true }) }, immediate: true } }}</script>

這是怎么回事,小老弟?首先,請注意,該組件接收一個Promise 類型參數。在 watch 部分中,監聽 promise 的變化,當 promise 發生變化時,清除狀態,然后調用 then 并 catch promise,當 promise 成功完成或失敗時更新狀態。

然后,在模板中,我們根據狀態顯示一個不同的槽。請注意,我們沒有保持它真正的無渲染,因為我們需要一個根元素來使用模板。我們還將 data 和 error 傳遞到相關的插槽范圍。

<template> <div> <promised :promise="somePromise"> <template #resolved="{ data }"> Resolved: {{ data }} </template> <template #rejected="{ error }"> Rejected: {{ error }} </template> <template #pending>  請求中... </template> </promised> </div></template>...

我們將 somePromise 傳遞給無渲染組件。 然后等待它完成,對于 pending 的插槽,顯示“請求中...”。 如果成功,顯示“Resolved:對應的值”。 如果失敗,顯示“已Rejected:失敗的原因”。 現在我們不再需要跟蹤此組件中的 promise 的狀態,因為該部分被拉出到它自己的可重用組件中。

那么,我們可以做些什么來繞過 promised.vue 中的插槽? 要刪除它,我們需要刪除 template 部分并向我們的組件添加 render 函數:

render () { if (this.error) { return this.$scopedSlots['rejected']({error: this.error}) } if (this.resolved) { return this.$scopedSlots['resolved']({data: this.data}) } return this.$scopedSlots['pending']()}

這里沒有什么太復雜的。我們只是使用一些 if 塊來查找狀態,然后返回正確的作用域 slot (通過 this.$ scopedslot ['SLOTNAME'](…) ),并將相關數據傳遞到 slot 作用域。當你不使用模板時,可以跳過使用 .vue 文件擴展名,方法是將JavaScript從 script 標記中提取出來,然后將其放入 .js 文件中。在編譯這些Vue文件時,這應該會給你帶來非常小的性能提升。

總結

Vue的插槽將基于組件的開發提升到了一個全新的水平,雖然本文已經展示了許多可以使用插槽的好方法,但還有更多的插槽。

以上所述是小編給大家介紹的vue 2.6 中 slot 的新用法,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對武林網網站的支持!
如果你覺得本文對你有幫助,歡迎轉載,煩請注明出處,謝謝!

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
欧美日韩亚洲精品内裤| 国产精品日韩在线观看| 色与欲影视天天看综合网| 精品magnet| 91精品综合视频| 久久精品视频中文字幕| 欧美孕妇孕交黑巨大网站| 亚洲国产精久久久久久| 欧美成aaa人片在线观看蜜臀| 九九九久久国产免费| 国产精品毛片a∨一区二区三区|国| 欧美视频在线观看免费网址| 精品国产一区久久久| 欧美一级在线播放| 夜夜嗨av一区二区三区四区| 日韩高清电影免费观看完整| 日韩欧美在线视频免费观看| 亚洲免费视频观看| 亚洲一二在线观看| 久久精品久久久久电影| 日韩**中文字幕毛片| 日本人成精品视频在线| 国产精品福利在线观看网址| 色偷偷噜噜噜亚洲男人的天堂| 亚洲精品午夜精品| 久久这里有精品视频| 久久夜色精品国产| 欧美成人在线影院| 久久精品国产成人| 成人精品久久久| 久久久国产影院| 91网站在线看| 亚洲人成毛片在线播放| 欧美性极品xxxx做受| 日韩欧美中文第一页| www.99久久热国产日韩欧美.com| 欧美国产亚洲精品久久久8v| 91精品国产免费久久久久久| 中文日韩在线视频| 成人福利在线观看| 91精品视频免费看| www.欧美三级电影.com| 亚洲国内精品在线| 日韩国产欧美精品一区二区三区| 国产成人综合av| 久久电影一区二区| 欧美日本中文字幕| 96sao精品视频在线观看| 欧美国产视频一区二区| 亚洲精品网址在线观看| 久久久久久香蕉网| 国产精品久久久久久婷婷天堂| 亚洲欧美精品中文字幕在线| 欧美高清理论片| 国产精品成av人在线视午夜片| 精品国产鲁一鲁一区二区张丽| 国产一区欧美二区三区| 亚洲激情久久久| 91久久久国产精品| 日韩色av导航| 日韩精品在线观看一区二区| 久久色在线播放| 91精品视频一区| 国产亚洲日本欧美韩国| 亚洲色图35p| 亚洲天天在线日亚洲洲精| 成人激情视频在线| 欧美亚洲视频一区二区| 亚洲性线免费观看视频成熟| 国产精品一区av| 日韩最新av在线| 国产精品福利久久久| 亚洲午夜精品视频| 欧美日韩亚洲精品一区二区三区| 亚洲图片欧洲图片av| 国产精品视频最多的网站| 91精品久久久久久久久久| 久久影视电视剧免费网站| 日韩国产欧美精品一区二区三区| 久久综合电影一区| 欧美中文在线观看国产| 色综合视频一区中文字幕| 色综合老司机第九色激情| 91亚洲精品久久久久久久久久久久| 亚洲一区二区三区成人在线视频精品| 日韩久久午夜影院| 国语自产精品视频在线看| 中文字幕亚洲无线码在线一区| 亚洲福利小视频| 国产成人精品在线播放| 欧美三级xxx| 最近的2019中文字幕免费一页| 国产亚洲一区二区在线| 97色伦亚洲国产| 精品久久中文字幕久久av| 久久精品国产免费观看| 91网在线免费观看| 欧美激情亚洲国产| 亚洲va久久久噜噜噜| 久久久亚洲国产天美传媒修理工| 欧美成人合集magnet| 97在线视频国产| 亚洲少妇激情视频| 国产欧美精品一区二区| 97久久精品在线| 成人黄色生活片| 在线观看日韩av| 亚洲iv一区二区三区| 欧美视频一二三| 欧美激情综合色综合啪啪五月| 成人午夜激情网| 国产欧美精品一区二区三区-老狼| 另类专区欧美制服同性| 国产精品免费久久久久久| 国产精品久久久久av| 18一19gay欧美视频网站| www国产精品com| 国产美女久久久| 亚洲免费福利视频| 在线丨暗呦小u女国产精品| 欧美大片大片在线播放| 欧美成人一区在线| 日韩av在线资源| 欧美电影免费观看网站| 51ⅴ精品国产91久久久久久| 成人黄色免费网站在线观看| 久久久久久12| 日韩免费黄色av| 亚洲日本aⅴ片在线观看香蕉| 亚洲香蕉成视频在线观看| 45www国产精品网站| 欧美贵妇videos办公室| 亚洲国产精品热久久| 欧美午夜精品久久久久久久| 国产精品69精品一区二区三区| 91精品国产色综合久久不卡98| 91免费欧美精品| 欧美日韩综合视频| 日韩视频免费看| 欧美视频在线视频| 中文字幕日韩精品在线| 欧美疯狂xxxx大交乱88av| 亚洲va久久久噜噜噜久久天堂| 国外视频精品毛片| 欧美成人一区二区三区电影| 欧美性xxxx极品高清hd直播| 欧美亚洲成人网| 欧美日韩亚洲网| 久久影视免费观看| 91精品免费久久久久久久久| www.国产精品一二区| 91精品久久久久久久久久久| 亚洲国产精品999| 日韩欧美视频一区二区三区| 欧美性xxxxx极品娇小| 亚洲视频在线观看视频| 夜色77av精品影院| 欧美一区二区色| 国产精品福利在线观看| 精品久久久视频| 欧美大肥婆大肥bbbbb| 亚洲97在线观看| 日韩不卡在线观看| 中文字幕亚洲激情|