国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

10個Vue開發技巧

開發 前端
在組件中使用 $route 會使之與其對應路由形成高度耦合,從而使組件只能在某些特定的 URL 上使用,限制了其靈活性。

路由參數解耦

一般在組件內使用路由參數,大多數人會這樣做:

  1. export default { 
  2.     methods: { 
  3.         getParamsId() { 
  4.             return this.$route.params.id 
  5.         } 
  6.     } 

在組件中使用 $route 會使之與其對應路由形成高度耦合,從而使組件只能在某些特定的 URL 上使用,限制了其靈活性。

正確的做法是通過 props 解耦

  1. const router = new VueRouter({ 
  2.     routes: [{ 
  3.         path: '/user/:id', 
  4.         component: User, 
  5.         props: true 
  6.     }] 
  7. }) 

將路由的 props 屬性設置為 true 后,組件內可通過 props 接收到 params 參數

  1. export default { 
  2.     props: ['id'], 
  3.     methods: { 
  4.         getParamsId() { 
  5.             return this.id 
  6.         } 
  7.     } 

另外你還可以通過函數模式來返回 props

  1. const router = new VueRouter({ 
  2.     routes: [{ 
  3.         path: '/user/:id', 
  4.         component: User, 
  5.         props: (route) => ({ 
  6.             id: route.query.id 
  7.         }) 
  8.     }] 
  9. }) 

文檔:router.vuejs.org/zh/guide/es…

函數式組件

函數式組件是無狀態,它無法實例化,沒有任何的生命周期和方法。創建函數式組件也很簡單,只需要在模板添加 functional 聲明即可。一般適合只依賴于外部數據的變化而變化的組件,因其輕量,渲染性能也會有所提高。

組件需要的一切都是通過 context 參數傳遞。它是一個上下文對象,具體屬性查看文檔。這里 props 是一個包含所有綁定屬性的對象。

函數式組件:

  1. <template functional> 
  2.     <div class="list"> 
  3.         <div class="item" v-for="item in props.list" :key="item.id" @click="props.itemClick(item)"> 
  4.             <p>{{item.title}}</p> 
  5.             <p>{{item.content}}</p> 
  6.         </div> 
  7.     </div> 
  8. </template> 

父組件使用:

  1. <template> 
  2.     <div> 
  3.         <List :list="list" :itemClick="item => (currentItem = item)" /> 
  4.     </div> 
  5. </template> 
  1. import List from '@/components/List.vue' 
  2. export default { 
  3.     components: { 
  4.         List 
  5.     }, 
  6.     data() { 
  7.         return { 
  8.             list: [{ 
  9.                 title: 'title', 
  10.                 content: 'content' 
  11.             }], 
  12.             currentItem: '' 
  13.         } 
  14.     } 

文檔:cn.vuejs.org/v2/guide/re…

樣式穿透

在開發中修改第三方組件樣式是很常見,但由于 scoped 屬性的樣式隔離,可能需要去除 scoped 或是另起一個 style 。這些做法都會帶來副作用(組件樣式污染、不夠優雅),樣式穿透在css預處理器中使用才生效。

我們可以使用 >>> 或 /deep/ 解決這一問題:

  1. <style scoped> 
  2. 外層 >>> .el-checkbox { 
  3.   display: block; 
  4.   font-size: 26px; 
  5.  
  6.   .el-checkbox__label { 
  7.     font-size: 16px; 
  8.   } 
  9. </style> 
  1. <style scoped> 
  2. /deep/ .el-checkbox { 
  3.   display: block; 
  4.   font-size: 26px; 
  5.  
  6.   .el-checkbox__label { 
  7.     font-size: 16px; 
  8.   } 
  9. </style> 

watch高階使用

1. 立即執行

watch 是在監聽屬性改變時才會觸發,有些時候,我們希望在組件創建后 watch 能夠立即執行。

可能想到的的方法就是在 create 生命周期中調用一次,但這樣的寫法不優雅,或許我們可以使用這樣的方法:

  1. export default { 
  2.     data() { 
  3.         return { 
  4.             name: 'Joe' 
  5.         } 
  6.     }, 
  7.     watch: { 
  8.         name: { 
  9.             handler: 'sayName', 
  10.             immediate: true 
  11.         } 
  12.     }, 
  13.     methods: { 
  14.         sayName() { 
  15.             console.log(this.name) 
  16.         } 
  17.     } 

深度監聽在監聽對象時,對象內部的屬性被改變時無法觸發 watch ,我們可以為其設置深度監聽:

  1. export default { 
  2.     data: { 
  3.         studen: { 
  4.             name: 'Joe', 
  5.             skill: { 
  6.                 run: { 
  7.                     speed: 'fast' 
  8.                 } 
  9.             } 
  10.         } 
  11.     }, 
  12.     watch: { 
  13.         studen: { 
  14.             handler: 'sayName', 
  15.             deep: true 
  16.         } 
  17.     }, 
  18.     methods: { 
  19.         sayName() { 
  20.             console.log(this.studen) 
  21.         } 
  22.     } 

2. 觸發監聽

執行多個方法使用數組可以設置多項,形式包括字符串、函數、對象:

  1. export default { 
  2.     data: { 
  3.         name: 'Joe' 
  4.     }, 
  5.     watch: { 
  6.         name: [ 
  7.             'sayName1', 
  8.             function(newVal, oldVal) { 
  9.                 this.sayName2() 
  10.             }, 
  11.             { 
  12.                 handler: 'sayName3', 
  13.                 immaediate: true 
  14.             } 
  15.         ] 
  16.     }, 
  17.     methods: { 
  18.         sayName1() { 
  19.             console.log('sayName1==>', this.name) 
  20.         }, 
  21.         sayName2() { 
  22.             console.log('sayName2==>', this.name) 
  23.         }, 
  24.         sayName3() { 
  25.             console.log('sayName3==>', this.name) 
  26.         } 
  27.     } 

文檔:cn.vuejs.org/v2/api/#wat…

watch監聽多個變量

watch本身無法監聽多個變量。但我們可以將需要監聽的多個變量通過計算屬性返回對象,再監聽這個對象來實現“監聽多個變量”

  1. export default { 
  2.     data() { 
  3.         return { 
  4.             msg1: 'apple', 
  5.             msg2: 'banana' 
  6.         } 
  7.     }, 
  8.     compouted: { 
  9.         msgObj() { 
  10.             const { msg1, msg2 } = this 
  11.             return { 
  12.                 msg1, 
  13.                 msg2 
  14.             } 
  15.         } 
  16.     }, 
  17.     watch: { 
  18.         msgObj: { 
  19.             handler(newVal, oldVal) { 
  20.                 if (newVal.msg1 != oldVal.msg1) { 
  21.                     console.log('msg1 is change') 
  22.                 } 
  23.                 if (newVal.msg2 != oldVal.msg2) { 
  24.                     console.log('msg2 is change') 
  25.                 } 
  26.             }, 
  27.             deep: true 
  28.         } 
  29.     } 

事件參數$event

$event 是事件對象的特殊變量,在一些場景能給我們實現復雜功能提供更多可用的參數

1. 原生事件

在原生事件中表現和默認的事件對象相同:

  1. <template> 
  2.     <div> 
  3.         <input type="text" @input="inputHandler('hello', $event)" /> 
  4.     </div> 
  5. </template> 
  1. export default { 
  2.     methods: { 
  3.         inputHandler(msg, e) { 
  4.             console.log(e.target.value) 
  5.         } 
  6.     } 

2. 自定義事件

在自定義事件中表現為捕獲從子組件拋出的值

my-item.vue:

  1. export default { 
  2.     methods: { 
  3.         customEvent() { 
  4.             this.$emit('custom-event', 'some value') 
  5.         } 
  6.     } 
  7. 復制代碼 

App.vue:

  1. <template> 
  2.     <div> 
  3.         <my-item v-for="(item, index) in list" @custom-event="customEvent(index, $event)"> 
  4.             </my-list> 
  5.     </div> 
  6. </template> 
  1. export default { 
  2.     methods: { 
  3.         customEvent(index, e) { 
  4.             console.log(e) // 'some value' 
  5.         } 
  6.     } 

文檔:

  • cn.vuejs.org/v2/guide/ev…
  • cn.vuejs.org/v2/guide/co…

自定義組件雙向綁定

組件 model 選項:

允許一個自定義組件在使用 v-model 時定制 prop 和 event。默認情況下,一個組件上的 v-model 會把 value 用作 prop 且把 input 用作 event,但是一些輸入類型比如單選框和復選框按鈕可能想使用 value prop 來達到不同的目的。使用 model 選項可以回避這些情況產生的沖突。

input 默認作為雙向綁定的更新事件,通過 $emit 可以更新綁定的值

  1. <my-switch v-model="val"></my-switch> 
  1. export default { 
  2.     props: { 
  3.         value: { 
  4.             type: Boolean, 
  5.             default: false 
  6.         } 
  7.     }, 
  8.     methods: { 
  9.         switchChange(val) { 
  10.             this.$emit('input', val) 
  11.         } 
  12.     } 

修改組件的 model 選項,自定義綁定的變量和事件

  1. <my-switch v-model="num" value="some value"></my-switch> 
  1. export default { 
  2.     model: { 
  3.         prop: 'num', 
  4.         event: 'update' 
  5.     }, 
  6.     props: { 
  7.         value: { 
  8.             type: String, 
  9.             default: '' 
  10.         }, 
  11.         num: { 
  12.             type: Number, 
  13.             default: 0 
  14.         } 
  15.     }, 
  16.     methods: { 
  17.         numChange() { 
  18.             this.$emit('update', this.num++) 
  19.         } 
  20.     } 

文檔:cn.vuejs.org/v2/api/#mod…

監聽組件生命周期

通常我們監聽組件生命周期會使用 $emit ,父組件接收事件來進行通知

子組件:

  1. export default { 
  2.     mounted() { 
  3.         this.$emit('listenMounted') 
  4.     } 

父組件:

  1. <template> 
  2.     <div> 
  3.         <List @listenMounted="listenMounted" /> 
  4.     </div> 
  5. </template> 

其實還有一種簡潔的方法,使用 @hook 即可監聽組件生命周期,組件內無需做任何改變。同樣的, created 、 updated 等也可以使用此方法。

  1. <template> 
  2.     <List @hook:mounted="listenMounted" /> 
  3. </template> 

程序化的事件偵聽器

比如,在頁面掛載時定義計時器,需要在頁面銷毀時清除定時器。這看起來沒什么問題。但仔細一看 this.timer 唯一的作用只是為了能夠在 beforeDestroy 內取到計時器序號,除此之外沒有任何用處。

  1. export default { 
  2.     mounted() { 
  3.         this.timer = setInterval(() => { 
  4.             console.log(Date.now()) 
  5.         }, 1000) 
  6.     }, 
  7.     beforeDestroy() { 
  8.         clearInterval(this.timer) 
  9.     } 

如果可以的話最好只有生命周期鉤子可以訪問到它。這并不算嚴重的問題,但是它可以被視為雜物。

我們可以通過 $on 或 $once 監聽頁面生命周期銷毀來解決這個問題:

  1. export default { 
  2.     mounted() { 
  3.         this.creatInterval('hello') 
  4.         this.creatInterval('world') 
  5.     }, 
  6.     creatInterval(msg) { 
  7.         let timer = setInterval(() => { 
  8.             console.log(msg) 
  9.         }, 1000) 
  10.         this.$once('hook:beforeDestroy', function() { 
  11.             clearInterval(timer) 
  12.         }) 
  13.     } 

使用這個方法后,即使我們同時創建多個計時器,也不影響效果。因為它們會在頁面銷毀后程序化的自主清除。

文檔:cn.vuejs.org/v2/guide/co…

手動掛載組件

在一些需求中,手動掛載組件能夠讓我們實現起來更加優雅。比如一個彈窗組件,最理想的用法是通過命令式調用,就像 elementUI 的 this.$message 。而不是在模板中通過狀態切換,這種實現真的很糟糕。

先來個最簡單的例子:

  1. import Vue from 'vue' 
  2. import Message from './Message.vue' 
  3.  
  4. // 構造子類 
  5. let MessageConstructor = Vue.extend(Message) 
  6. // 實例化組件 
  7. let messageInstance = new MessageConstructor() 
  8. // $mount可以傳入選擇器字符串,表示掛載到該選擇器 
  9. // 如果不傳入選擇器,將渲染為文檔之外的的元素,你可以想象成 document.createElement()在內存中生成dom 
  10. messageInstance.$mount() 
  11. // messageInstance.$el獲取的是dom元素 
  12. document.body.appendChild(messageInstance.$el) 

下面實現一個簡易的 message 彈窗組件

Message/index.vue:

  1. <template> 
  2.     <div class="wrap"> 
  3.         <div class="message" :class="item.type" v-for="item in notices" :key="item._name"> 
  4.             <div class="content">{{item.content}}</div> 
  5.         </div> 
  6.     </div> 
  7. </template> 
  1. // 默認選項 
  2. const DefaultOptions = { 
  3.     duration: 1500, 
  4.     type: 'info', 
  5.     content: '這是一條提示信息!', 
  6. let mid = 0 
  7. export default { 
  8.     data() { 
  9.         return { 
  10.             notices: [] 
  11.         } 
  12.     }, 
  13.     methods: { 
  14.         add(notice = {}) { 
  15.             // name標識 用于移除彈窗 
  16.             let _name = this.getName() 
  17.             // 合并選項 
  18.             notice = Object.assign({ 
  19.                 _name 
  20.             }, DefaultOptions, notice) 
  21.  
  22.             this.notices.push(notice) 
  23.  
  24.             setTimeout(() => { 
  25.                 this.removeNotice(_name) 
  26.             }, notice.duration) 
  27.         }, 
  28.         getName() { 
  29.             return 'msg_' + (mid++) 
  30.         }, 
  31.         removeNotice(_name) { 
  32.             let index = this.notices.findIndex(item => item._name === _name) 
  33.             this.notices.splice(index, 1) 
  34.         } 
  35.     } 
  1. .wrap { 
  2.     position: fixed; 
  3.     top: 50px; 
  4.     left: 50%; 
  5.     display: flex; 
  6.     flex-direction: column; 
  7.     align-items: center; 
  8.     transform: translateX(-50%); 
  9.  
  10. .message { 
  11.     --borderWidth: 3px; 
  12.     min-width: 240px; 
  13.     max-width: 500px; 
  14.     margin-bottom: 10px; 
  15.     border-radius: 3px; 
  16.     box-shadow: 0 0 8px #ddd; 
  17.     overflow: hidden; 
  18.  
  19. .content { 
  20.     padding: 8px; 
  21.     line-height: 1.3; 
  22.  
  23. .message.info { 
  24.     border-left: var(--borderWidth) solid #909399; 
  25.     background: #F4F4F5; 
  26.  
  27. .message.success { 
  28.     border-left: var(--borderWidth) solid #67C23A; 
  29.     background: #F0F9EB; 
  30.  
  31. .message.error { 
  32.     border-left: var(--borderWidth) solid #F56C6C; 
  33.     background: #FEF0F0; 
  34.  
  35. .message.warning { 
  36.     border-left: var(--borderWidth) solid #E6A23C; 
  37.     background: #FDF6EC; 

Message/index.js:

  1. import Vue from 'vue' 
  2. import Index from './index.vue' 
  3.  
  4. let messageInstance = null 
  5. let MessageConstructor = Vue.extend(Index) 
  6.  
  7. let init = () => { 
  8.     messageInstance = new MessageConstructor() 
  9.     messageInstance.$mount() 
  10.     document.body.appendChild(messageInstance.$el) 
  11.  
  12. let caller = (options) => { 
  13.     if (!messageInstance) { 
  14.         init(options) 
  15.     } 
  16.     messageInstance.add(options) 
  17.  
  18. export default { 
  19.     // 返回 install 函數 用于 Vue.use 注冊 
  20.     install(vue) { 
  21.         vue.prototype.$message = caller 
  22.     } 

main.js:

  1. import Message from '@/components/Message/index.js' 
  2.  
  3. Vue.use(Message) 

使用:

  1. this.$message({ 
  2.     type: 'success', 
  3.     content: '成功信息提示', 
  4.     duration: 3000 
  5. }) 

文檔:cn.vuejs.org/v2/api/#vm-…

 

責任編輯:趙寧寧 來源: 前端工匠
相關推薦

2022-12-15 16:38:17

2015-07-27 09:36:09

storyboard

2020-03-31 09:47:04

Vue開發代碼

2020-06-07 16:16:01

Python開發工具

2017-11-06 14:33:54

Web開發服務器網絡

2020-06-10 10:30:48

Python 開發編程語言

2019-07-23 09:00:00

vuejavascript前端

2025-02-03 00:00:00

Vue.js組件緩存

2020-06-15 10:29:10

JavaScript開發 技巧

2014-08-20 10:02:54

GitGit能力

2020-04-14 09:26:43

開發工具Chrome

2011-07-07 10:21:56

2019-02-25 15:15:44

Windows 10Windows技巧

2018-11-28 08:15:09

2025-01-06 08:57:19

Vue技巧

2023-02-07 08:00:00

MySQL數據庫技巧

2011-07-07 18:39:22

SEO

2021-03-15 08:13:19

JavaScript開發代碼

2014-07-29 13:55:10

程序員代碼

2020-06-08 07:52:31

Python開發工具
點贊
收藏

51CTO技術棧公眾號

久久激情久久| 美臀av在线| 国模一区二区三区| 免费看欧美黑人毛片| 91免费观看在线| 免费av在线电影| 久久最新资源网| 好看的日韩av电影| 91色琪琪电影亚洲精品久久| 少妇高潮一区二区三区99| 国内精品小视频| 日本伦理一区二区| 久久91精品国产91久久小草| 麻豆传媒网站在线观看| 成人国产在线观看| 欧美狂欢多p性派对| 在线亚洲人成电影网站色www| jizz性欧美| 久久午夜a级毛片| 91精品成人| 在线观看亚洲视频啊啊啊啊| 亚洲国产精品成人综合| 在线观看h片| 九九热r在线视频精品| 久久久久免费av| 欧美美女黄色网| 亚洲成人一区二区| 国产拍在线视频| 日本成人精品在线| 久久国产福利国产秒拍| 97碰碰碰免费公开在线视频| 欧美日本在线播放| 久久久久久久久成人| 91精品国产一区二区三区动漫| 国产传媒久久文化传媒| 中文字幕在线影视资源| 精品视频在线播放| 无码一区二区三区视频| 人体内射精一区二区三区| 日韩欧美黄色动漫| 涩涩在线视频| 91欧美日韩一区| 99精品视频免费在线观看| 91高清在线| 欧美孕妇性xx| 国产精品亚洲综合一区在线观看| 日韩电影在线观看完整版| 欧美成人精品h版在线观看| 男人的天堂成人在线| 久久久久久久久久久久久久久久久久久| 亚洲第一男人av| 中文字幕一区二区三区欧美日韩 | 精品一区毛片| 亚洲精品日韩在线观看| 图片区小说区国产精品视频| 成人国产精品入口免费视频| 国产精品乱码| 亚洲精品一二三| 国产一区高清| 日韩精品一区二区三区色偷偷| 婷婷开心激情综合| 国产精品中文字幕制服诱惑| 国产 日韩 亚洲 欧美| 精品国产区一区| 一区精品久久| 青青青免费视频在线2| 97在线观看免费| 懂色av中文字幕一区二区三区| av毛片在线看| 不卡视频一区二区| 精品久久久久久久久久久久久久 | 亚洲欧洲日韩综合一区二区| 国产精品蜜月aⅴ在线| 亚洲午夜精品久久久久久浪潮| 91福利视频久久久久| 精品国产a一区二区三区v免费| 国产乱子夫妻xx黑人xyx真爽| 日韩大陆欧美高清视频区| 国产视频欧美| 麻豆传媒在线观看| 精品久久sese| 欧美高清hd18日本| 一区二区三区四区五区精品视频| 男生女生差差差的视频在线观看| 国产日产亚洲精品| 亚洲国产一二三| 精品国产乱码久久久久久蜜坠欲下| 手机在线看福利| 欧美老女人性生活| 久久精品夜夜夜夜久久| 日本成人精品| 亚洲不卡视频在线| 97精品免费视频| 亚洲美女一区二区三区| 国产永久精品大片wwwapp| 色老板在线视频| 国产噜噜噜噜噜久久久久久久久| 亚洲国产精品久久人人爱| 俺要去色综合狠狠| 日韩av免费观影| 激情视频一区二区| 欧美一区二区成人6969| 日韩电影免费在线| 高清视频在线观看三级| 米仓穗香在线观看| www日韩欧美| 中文字幕+乱码+中文字幕一区| 91精品综合久久久久久久久久久| 亚洲不卡视频| 韩国黄色一级大片| 精品亚洲va在线va天堂资源站| 日韩精品一级中文字幕精品视频免费观看 | 麻豆网站免费在线观看| 熟女视频一区二区三区| 亚洲人成欧美中文字幕| jizz一区二区| 日本一道高清一区二区三区| 国产香蕉视频在线观看| 超碰97网站| 亚洲成人三级在线| 成人99免费视频| 麻豆一区二区| 污污软件在线观看| 欧美一区二区三区电影在线观看| 91麻豆精品国产自产在线观看一区| 免费不卡在线观看| 亚洲综合资源| 意大利激情丛林无删减版dvd| 亚洲一区国产精品| 日韩视频在线你懂得| 成人午夜精品在线| 日韩在线麻豆| 亚洲色图美腿丝袜| 91精品国产91久久久久福利| 欧美精品免费在线观看| 欧美日韩成人在线视频| 午夜精品久久久久| 久草成人资源| 午夜在线视频| 日本男女交配视频| 日本高清不卡在线| 欧美另类变人与禽xxxxx| 国产成人av电影在线播放| 日韩三级av| 黄av在线播放| 日本黄色三级大片| 7777精品伊久久久大香线蕉语言| 亚洲精品一线二线三线无人区| 97久久人人超碰| 中文在线日韩| 日韩成人综合网| 麻豆av电影在线观看| 久久香蕉视频网站| 国产精品亚洲一区二区三区| 精品久久久三级丝袜| 国产精品久久久一区麻豆最新章节| 欧美午夜久久| 6080成人| av在线导航| 妞干网免费视频| 青娱乐一区二区| 欧美重口另类videos人妖| 欧美一区二区在线看| 国产精品少妇自拍| 日本中文一区二区三区| 亚洲69av| 久久sese| 牛牛澡牛牛爽一区二区| 人妻av中文系列| 久久久com| 琪琪第一精品导航| 亚洲精品国产精品自产a区红杏吧| 一区二区三区成人在线视频| 久草这里只有精品视频| 久久一区91| 日韩三级一区| 麻豆网站视频在线观看| caoporn97免费视频公开| 337p亚洲精品色噜噜狠狠p| 国产传媒一区二区三区| 97高清免费视频| 中文字幕av一区二区| 日韩一区二区三免费高清| 亚洲高清不卡在线观看| 久久久99精品免费观看不卡| 美国一区二区三区在线播放| 欧美精品日韩| 成人国产精品一级毛片视频| 不卡的国产精品| 中文在线а√天堂| 羞羞网站在线看| 人成在线免费视频| 97看剧电视剧大全| 国产精品无码一本二本三本色| 五月天色婷婷综合| 日韩精品无码一区二区三区| 久久国产精品久久| 国产三级精品在线不卡| 99在线影院| 成人综合av网|