Vue自定義指令原來這么簡單

本篇學習目標

  1. 能夠了解組件進階知識
  2. 能夠掌握自定義指令創建和使用
  3. 能夠完成tabbar案例的開發

1. 組件進階

1.0 組件進階 - 動態組件

目標: 多個組件使用同一個掛載點,并動態切換,這就是動態組件

需求: 完成一個注冊功能頁面, 2個按鈕切換, 一個填寫注冊信息, 一個填寫用戶簡介信息

效果如下:
在這里插入圖片描述

  1. 準備被切換的 - UserName.vue / UserInfo.vue 2個組件

  2. 引入到UseDynamic.vue注冊

  3. 準備變量來承載要顯示的"組件名"

  4. 設置掛載點, 使用is屬性來設置要顯示哪個組件

  5. 點擊按鈕 – 修改comName變量里的"組件名"

<template><div><button @click="comName = 'UserName'">賬號密碼填寫</button><button @click="comName = 'UserInfo'">個人信息填寫</button><p>下面顯示注冊組件-動態切換:</p><div style="border: 1px solid red;"><component :is="comName"></component></div></div>
</template><script>
// 目標: 動態組件 - 切換組件顯示
// 場景: 同一個掛載點要切換 不同組件 顯示
// 1. 創建要被切換的組件 - 標簽+樣式
// 2. 引入到要展示的vue文件內, 注冊
// 3. 變量-承載要顯示的組件名
// 4. 設置掛載點<component :is="變量"></component>
// 5. 點擊按鈕-切換comName的值為要顯示的組件名import UserName from '../components/01/UserName'
import UserInfo from '../components/01/UserInfo'
export default {data(){return {comName: "UserName"}},components: {UserName,UserInfo}
}
</script>

在App.vue - 引入01_UseDynamic.vue并使用顯示

總結: vue內置component組件, 配合is屬性, 設置要顯示的組件名字

1.1 組件進階 - 組件緩存

目標: 組件切換會導致組件被頻繁銷毀和重新創建, 性能不高

使用Vue內置的keep-alive組件, 可以讓包裹的組件保存在內存中不被銷毀

演示1: 可以先給UserName.vue和UserInfo.vue 注冊created和destroyed生命周期事件, 觀察創建和銷毀過程

演示2: 使用keep-alive內置的vue組件, 讓動態組件緩存而不是銷毀

語法:

? Vue內置的keep-alive組件 包起來要頻繁切換的組件

02_UseDynamic.vue

<div style="border: 1px solid red;"><!-- Vue內置keep-alive組件, 把包起來的組件緩存起來 --><keep-alive><component :is="comName"></component></keep-alive>
</div>

補充生命周期:

  • activated - 激活
  • deactivated - 失去激活狀態

總結: keep-alive可以提高組件的性能, 內部包裹的標簽不會被銷毀和重新創建, 觸發激活和非激活的生命周期方法

1.2 組件進階 - 激活和非激活

目標: 被緩存的組件不再創建和銷毀, 而是激活和非激活

補充2個鉤子方法名:

? activated – 激活時觸發

? deactivated – 失去激活狀態觸發

1.3 組件進階 - 組件插槽

目標: 用于實現組件的內容分發, 通過 slot 標簽, 可以接收到寫在組件標簽內的內容

vue提供組件插槽能力, 允許開發者在封裝組件時,把不確定的部分定義為插槽

插槽例子:
在這里插入圖片描述

需求: 以前折疊面板案例, 想要實現不同內容顯示, 我們把折疊面板里的Pannel組件, 添加組件插槽方式
在這里插入圖片描述

語法口訣:

  1. 組件內用占位
  2. 使用組件時夾著的地方, 傳入標簽替換slot

03/Pannel.vue - 組件(直接復制)

<template><div><!-- 按鈕標題 --><div class="title"><h4>芙蓉樓送辛漸</h4><span class="btn" @click="isShow = !isShow">{{ isShow ? "收起" : "展開" }}</span></div><!-- 下拉內容 --><div class="container" v-show="isShow"><p>寒雨連江夜入吳,</p><p>平明送客楚山孤。</p><p>洛陽親友如相問,</p><p>一片冰心在玉壺。</p></div></div>
</template><script>
export default {data() {return {isShow: false,};},
};
</script><style scoped>
h3 {text-align: center;
}.title {display: flex;justify-content: space-between;align-items: center;border: 1px solid #ccc;padding: 0 1em;
}.title h4 {line-height: 2;margin: 0;
}.container {border: 1px solid #ccc;padding: 0 1em;
}.btn {/* 鼠標改成手的形狀 */cursor: pointer;
}img {width: 50%;
}
</style>

views/03_UserSlot.vue - 使用組件(直接復制)

框: 在這個基礎重復使用組件

<template><div id="container"><div id="app"><h3>案例:折疊面板</h3></div></div>
</template><script>
export default {
};
</script><style>
#app {width: 400px;margin: 20px auto;background-color: #fff;border: 4px solid blueviolet;border-radius: 1em;box-shadow: 3px 3px 3px rgba(0, 0, 0, 0.5);padding: 1em 2em 2em;
}
</style>

views/03_UseSlot.vue - 組件插槽使用

<template><div id="container"><div id="app"><h3>案例:折疊面板</h3><Pannel><img src="../assets/mm.gif" alt=""><span>我是內容</span></Pannel><Pannel><p>寒雨連江夜入吳,</p><p>平明送客楚山孤。</p><p>洛陽親友如相問,</p><p>一片冰心在玉壺。</p></Pannel><Pannel></Pannel></div></div>
</template><script>
import Pannel from "../components/03/Pannel";
export default {components: {Pannel,},
};
</script>

總結: 組件內容分發技術, slot占位, 使用組件時傳入替換slot位置的標簽

1.4 組件進階 - 插槽默認內容

目標: 如果外面不給傳, 想給個默認顯示內容

口訣: 夾著內容默認顯示內容, 如果不給插槽slot傳東西, 則使用夾著的內容在原地顯示

<slot>默認內容</slot>

1.5 組件進階 - 具名插槽

目標: 當一個組件內有2處以上需要外部傳入標簽的地方

傳入的標簽可以分別派發給不同的slot位置

要求: v-slot一般用跟template標簽使用 (template是html5新出標簽內容模板元素, 不會渲染到頁面上, 一般被vue解析內部標簽)

components/04/Pannel.vue - 留下具名slot

<template><div><!-- 按鈕標題 --><div class="title"><slot name="title"></slot><span class="btn" @click="isShow = !isShow">{{ isShow ? "收起" : "展開" }}</span></div><!-- 下拉內容 --><div class="container" v-show="isShow"><slot name="content"></slot></div></div>
</template>

views/04_UseSlot.vue使用

<template><div id="container"><div id="app"><h3>案例:折疊面板</h3><Pannel><template v-slot:title><h4>芙蓉樓送辛漸</h4></template><template v-slot:content><img src="../assets/mm.gif" alt=""><span>我是內容</span></template></Pannel><Pannel><template #title><span style="color: red;">我是標題</span></template><template #content><p>寒雨連江夜入吳,</p><p>平明送客楚山孤。</p><p>洛陽親友如相問,</p><p>一片冰心在玉壺。</p></template></Pannel></div></div>
</template><script>
import Pannel from "../components/04/Pannel";
export default {components: {Pannel,},
};
</script>

v-slot可以簡化成#使用

v-bind可以省略成: v-on: 可以省略成@ 那么v-slot: 可以簡化成#

總結: slot的name屬性起插槽名, 使用組件時, template配合#插槽名傳入具體標簽

1.6 組件進階 - 作用域插槽

目標: 子組件里值, 在給插槽賦值時在父組件環境下使用

復習: 插槽內slot中顯示默認內容

例子: 默認內容在子組件中, 但是父親在給插槽傳值, 想要改變插槽顯示的默認內容

口訣:

  1. 子組件, 在slot上綁定屬性和子組件內的值
  2. 使用組件, 傳入自定義標簽, 用template和v-slot=“自定義變量名”
  3. scope變量名自動綁定slot上所有屬性和值

components/05/Pannel.vue - 定義組件, 和具名插槽, 給slot綁定屬性和值

<template><div><!-- 按鈕標題 --><div class="title"><h4>芙蓉樓送辛漸</h4><span class="btn" @click="isShow = !isShow">{{ isShow ? "收起" : "展開" }}</span></div><!-- 下拉內容 --><div class="container" v-show="isShow"><slot :row="defaultObj">{{ defaultObj.defaultOne }}</slot></div></div>
</template><script>
// 目標: 作用域插槽
// 場景: 使用插槽, 使用組件內的變量
// 1. slot標簽, 自定義屬性和內變量關聯
// 2. 使用組件, template配合v-slot="變量名"
// 變量名會收集slot身上屬性和值形成對象
export default {data() {return {isShow: false,defaultObj: {defaultOne: "無名氏",defaultTwo: "小傳同學"}};},
};
</script>

views/05_UseSlot.vue

<template><div id="container"><div id="app"><h3>案例:折疊面板</h3><Pannel><!-- 需求: 插槽時, 使用組件內變量 --><!-- scope變量: {row: defaultObj} --><template v-slot="scope"><p>{{ scope.row.defaultTwo }}</p></template></Pannel></div></div>
</template><script>
import Pannel from "../components/05/Pannel";
export default {components: {Pannel,},
};
</script>

總結: 組件內變量綁定在slot上, 然后使用組件v-slot=“變量” 變量上就會綁定slot身上屬性和值

1.7 組件進階 - 作用域插槽使用場景

目標: 了解作用域插槽使用場景, 自定義組件內標簽+內容

案例: 封裝一個表格組件, 在表格組件內循環產生單元格

準備MyTable.vue組件 – 內置表格, 傳入數組循環鋪設頁面, 把對象每個內容顯示在單元格里

準備UseTable.vue – 準備數據傳入給MyTable.vue使用

components/06/MyTable.vue - 模板(直接復制)

<template><div><table border="1"><thead><tr><th>序號</th><th>姓名</th><th>年齡</th><th>頭像</th></tr></thead><thead><tr><td></td><td></td><td></td><td></td></tr></thead></table></div>
</template><script>
export default {}
</script>

views/06_UseTable.vue - 準備數據, 傳入給MyTable.vue組件里循環使用

list: [{name: "小傳同學",age: 18,headImgUrl:"http://yun.itheima.com/Upload/./Images/20210303/603f2d2153241.jpg",},{name: "小黑同學",age: 25,headImgUrl:"http://yun.itheima.com/Upload/./Images/20210304/6040b101a18ef.jpg",},{name: "智慧同學",age: 21,headImgUrl:"http://yun.itheima.com/Upload/./Images/20210302/603e0142e535f.jpg",},
],

例子: 我想要給td內顯示圖片, 需要傳入自定義的img標簽
在這里插入圖片描述

正確做法:

? 在MyTable.vue的td中準備占位, 但是外面需要把圖片地址賦予給src屬性,所以在slot上把obj數據綁定

components/06/MyTable.vue - 正確代碼

<template><div><table border="1"><thead><tr><th>序號</th><th>姓名</th><th>年齡</th><th>頭像</th></tr></thead><tbody><tr v-for="(obj, index) in arr" :key="index"><td>{{ index + 1 }}</td><td>{{ obj.name }}</td><td>{{ obj.age }}</td><td><slot :row="obj"><!-- 默認值給上,如果使用組件不自定義標簽顯示默認文字 -->{{ obj.headImgUrl}}</slot></td></tr></tbody></table></div>
</template><script>
export default {props: {arr: Array}
}
</script>

? 在UseTable使用MyTable的時候, template上v-slot綁定變量, 傳入img組件設置圖片地址

<template><div><MyTable :arr="list"></MyTable><MyTable :arr="list"><!-- scope: {row: obj} --><template v-slot="scope"><a :href="scope.row.headImgUrl">{{ scope.row.headImgUrl }}</a></template></MyTable><MyTable :arr="list"><template v-slot="scope"><img style="width: 100px;" :src="scope.row.headImgUrl" alt=""></template></MyTable></div>
</template><script>
import MyTable from "../components/06/MyTable";
export default {components: {MyTable,},data() {return {list: [{name: "小傳同學",age: 18,headImgUrl:"http://yun.itheima.com/Upload/./Images/20210303/603f2d2153241.jpg",},{name: "小黑同學",age: 25,headImgUrl:"http://yun.itheima.com/Upload/./Images/20210304/6040b101a18ef.jpg",},{name: "智慧同學",age: 21,headImgUrl:"http://yun.itheima.com/Upload/./Images/20210302/603e0142e535f.jpg",},],};},
};
</script><style>
</style>

總結: 插槽可以自定義標簽, 作用域插槽可以把組件內的值取出來自定義內容

2. 自定義指令

自定義指令文檔

除了核心功能默認內置的指令 (v-modelv-show),Vue 也允許注冊自定義指令。 v-xxx

html+css的復用的主要形式是組件

你需要對普通 DOM 元素進行底層操作,這時候就會用到自定義指令

2.0 自定義指令-注冊

目標: 獲取標簽, 擴展額外的功能

局部注冊和使用

07_UseDirective.vue - 只能在當前組件.vue文件中使用

<template><div><!-- <input type="text" v-gfocus> --><input type="text" v-focus></div>
</template><script>
// 目標: 創建 "自定義指令", 讓輸入框自動聚焦
// 1. 創建自定義指令
// 全局 / 局部
// 2. 在標簽上使用自定義指令  v-指令名
// 注意:
// inserted方法 - 指令所在標簽, 被插入到網頁上觸發(一次)
// update方法 - 指令對應數據/標簽更新時, 此方法執行
export default {data(){return {colorStr: 'red'}},directives: {focus: {inserted(el){el.focus()}}}
}
</script><style></style>

全局注冊

在main.js用 Vue.directive()方法來進行注冊, 以后隨便哪個.vue文件里都可以直接用v-fofo指令

// 全局指令 - 到處"直接"使用
Vue.directive("gfocus", {inserted(el) {el.focus() // 觸發標簽的事件方法}
})

總結: 全局注冊自定義指令, 哪里都能用, 局部注冊, 只能在當前vue文件里用

2.1 自定義指令-傳值

目標: 使用自定義指令, 傳入一個值

需求: 定義color指令-傳入一個顏色, 給標簽設置文字顏色

main.js定義處修改一下

// 目標: 自定義指令傳值
Vue.directive('color', {inserted(el, binding) {el.style.color = binding.value},update(el, binding) {el.style.color = binding.value}
})

Direct.vue處更改一下

<p v-color="colorStr" @click="changeColor">修改文字顏色</p><script>data() {return {theColor: "red",};},methods: {changeColor() {this.theColor = 'blue';},},
</script>

總結: v-xxx, 自定義指令, 獲取原生DOM, 自定義操作

3. 案例-tabbar

完成如下案例和各步功能
在這里插入圖片描述

知識點:

  • 組件封裝
  • 動態組件
  • keep-alive
  • 作用域插槽
  • 自定義指令

3.0 案例-tabbar-初始化項目

目標: 創建項目文件夾, 引入字體圖標, 下載bootstrap, less, less-loader@5.0.0 axios, 在App.vue注冊組件
在這里插入圖片描述

  • 需求: 從0新建項目, 拆分組件, 創建使用

組件分析:

  • 組件拆分:

    • MyHeader.vue – 復用之前的
    • MyTabBar.vue – 底部導航
    • MyTable.vue – 封裝表格
  • 三個頁面

    • -MyGoodsList.vue – 商品頁
    • MyGoodsSearch.vue – 搜索頁
    • -MyUserInfo.vue – 用戶信息頁

思路分析:

? ①: vue create tabbar-demo

? ②: yarn add less less-loader@5.0.0 -D

? ③: yarn add bootstrap axios 并在main.js 引入和全局屬性

? ④: 根據需求-創建需要的頁面組件

? ⑤: 把昨天購物車案例-封裝的MyHeader.vue文件復制過來復用

? ⑥: 從App.vue – 引入組織相關標簽

新建工程:

vue create tabbar-demo
yarn add less less-loader@5.0.0 -D
yarn add bootstrap axios

在main.js中引入bootStrap.css和字體圖標樣式

import "bootstrap/dist/css/bootstrap.css"
import "./assets/fonts/iconfont.css"

創建/復制如下文件

從昨天案例中-直接復制過來components/MyHeader.vue

components/MyTabBar.vue

views/MyGoodsList.vue

views/MyGoodsSearch.vue

views/MyUserInfo.vue

components/MyTable.vue

3.1 案例-tabbar-底部封裝

目標: 實現MyTabBar.vue組件

在這里插入圖片描述

  • 需求: 把底部導航也靈活封裝起來

分析:

? ①: 基本標簽+樣式(md里復制)

? ②: 為tabbar組件指定數據源

? ③: 數據源最少2個, 最多5個(validator)

? ④: 從App.vue給MyTabBar.vue傳入底部導航的數據

? ⑤: MyTabBar.vue中循環展示

App.vue-數組準備

tabList: [{iconText: "icon-shangpinliebiao",text: "商品列表",componentName: "MyGoodsList"},{iconText: "icon-sousuo",text: "商品搜索",componentName: "MyGoodsSearch"},{iconText: "icon-user",text: "我的信息",componentName: "MyUserInfo"}
]

MyTabBar.vue - 標簽模板

<template><div class="my-tab-bar"><div class="tab-item"><!-- 圖標 --><span class="iconfont"></span><!-- 文字 --><span></span></div></div>
</template><script>
export default {}
</script><style lang="less" scoped>
.my-tab-bar {position: fixed;left: 0;bottom: 0;width: 100%;height: 50px;border-top: 1px solid #ccc;display: flex;justify-content: space-around;align-items: center;background-color: white;.tab-item {display: flex;flex-direction: column;align-items: center;}
}.current {color: #1d7bff;
}
</style>

MyTabBar.vue正確代碼(不可復制)

<template><div class="my-tab-bar"><divclass="tab-item"v-for="(obj, index) in arr":key="index"><!-- 圖標 --><span class="iconfont" :class="obj.iconText"></span><!-- 文字 --><span>{{ obj.text }}</span></div></div>
</template><script>
export default {props: {arr: {type: Array,required: true,// 自定義校驗規則validator(value) {// value就是接到數組if (value.length >= 2 && value.length <= 5) {return true; // 符合條件就return true} else {console.error("數據源必須在2-5項");return false;}},},}
};
</script>

不要忘了把tabList數組從App.vue -> MyTabBar.vue

3.2 案例-tabbar-底部高亮

目標: 點擊底部導航實現高亮效果

  • 需求: 點擊底部實現高亮效果

分析:

? ①: 綁定點擊事件, 獲取點擊的索引

? ②: 循環的標簽設置動態class, 遍歷的索引, 和點擊保存的索引比較, 相同則高亮

效果演示:
在這里插入圖片描述

MyTabBar.vue(正確代碼)

<template><div class="my-tab-bar"><div class="tab-item" v-for="(obj, index) in arr" :key="index":class="{current: activeIndex === index}"@click="activeIndex = index"><!-- 圖標 --><span class="iconfont" :class="obj.iconText"></span><!-- 文字 --><span>{{ obj.text }}</span></div></div>
</template><script>
export default {data(){return {activeIndex: 0 // 高亮元素下標}},// ....其他代碼
};
</script>

3.3 案例-tabbar-組件切換

目的: 點擊底部導航, 切換頁面組件顯示

需求: 點擊底部切換組件

分析:

? ①: 底部導航傳出動態組件名字符串到App.vue

? ②: 切換動態組件is屬性的值為要顯示的組件名

效果演示:

在這里插入圖片描述

補充: 給內容div.app- 設置上下內邊距

App.vue - 引入并注冊

<template><div><MyHeader:background="'blue'":fontColor="'white'"title="TabBar案例"></MyHeader><div class="main"><component :is="comName"></component></div><MyTabBar :arr="tabList"@changeCom="changeComFn"></MyTabBar></div>
</template><script>
import MyHeader from "./components/MyHeader";
// 目標: 完成底部封裝
// 1. MyTabBar.vue 組件標簽+樣式 準備
// 2. 字體圖標引入
// 3. 準備底部數據
// 4. 使用MyTabBar組件, 傳入數據(父->子), 循環產生底部導航
// 5. 子組件內props自定義檢驗規則(2-5項)
// 6. 子組件內循環產生底部導航
import MyTabBar from './components/MyTabBar'// 目標: 切換組件顯示
// 1. 創建組件 - 編寫內容
// 2. 引入App.vue注冊
// 3. 掛載點設置is
// 4. 切換comName的值(重要)
// 5. 底部導航點擊- MyTabBar.vue里
// 子 -> 父技術 (傳要切換的組件名出來)import MyGoodsList from './views/MyGoodsList'
import MyGoodsSearch from './views/MyGoodsSearch'
import MyUserInfo from './views/MyUserInfo'
export default {data() {return {comName: "MyGoodsList", // 默認顯示的組件tabList: [ // 底部導航的數據{iconText: "icon-shangpinliebiao",text: "商品列表",componentName: "MyGoodsList",},{iconText: "icon-sousuo",text: "商品搜索",componentName: "MyGoodsSearch",},{iconText: "icon-user",text: "我的信息",componentName: "MyUserInfo",},],};},components: {MyHeader,MyTabBar,MyGoodsList,MyGoodsSearch,MyUserInfo},methods: {changeComFn(cName){this.comName = cName; // MyTabBar里選出來的組件名賦予給is屬性的comName// 導致組件的切換}}
};
</script><style scoped>
.main{padding-top: 45px;padding-bottom: 51px;
}
</style>

MyTabBar.vue - 點擊傳遞過來組件名

methods: {btn(index, theObj) {this.selIndex = index; // 點誰, 就把誰的索引值保存起來this.$emit("changeCom", theObj.componentName); // 要切換的組件名傳App.vue},},

3.4 案例-tabbar-商品列表

目標: 為MyGoodsList頁面, 準備表格組件MyTable.vue-鋪設展示數據

  • 需求: 商品列表鋪設頁面

分析:

? ①: 封裝MyTable.vue – 準備標簽和樣式

? ②: axios在MyGoodsList.vue請求數據回來

? ③: 請求地址: https://www.escook.cn/api/goods

? ④: 傳入MyTable.vue中循環數據顯示

? ⑤: 給刪除按鈕添加bootstrap的樣式: btn btn-danger btn-sm

效果演示:
在這里插入圖片描述

MyTable.vue - 準備table整個表格標簽和樣式(可復制)

<template><table class="table table-bordered table-stripped"><!-- 表格標題區域 --><thead><tr><th>#</th><th>商品名稱</th><th>價格</th><th>標簽</th><th>操作</th></tr></thead><!-- 表格主體區域 --><tbody><tr ><td>1</td><td>商品</td><td>998</td><td>xxx</td><td>xxx</td></tr></tbody></table>
</template><script>
export default {name: 'MyTable'
}
</script><style scoped lang="less">
.my-goods-list {.badge {margin-right: 5px;}
}
</style>

使用axios請求數據, 把表格頁面鋪設出來

main.js - 注冊axios配置默認地址

import axios from "axios";
axios.defaults.baseURL = "https://www.escook.cn";

MyGoodsList.vue - 使用axios請求數據, 把數據傳入給MyTable.vue里循環鋪設

<template><div><MyTable :arr="list"></MyTable></div>
</template><script>
import MyTable from "../components/MyTable";
import axios from "axios";
axios.defaults.baseURL = "https://www.escook.cn";
// 目標: 循環商品列表表格
// 1. 封裝MyTable.vue 整體表格組件-一套標簽和樣式
// 2. axios請求數據
// 3. 傳入MyTable組件里循環tr顯示數據// 目標: 展示tags標簽
// 1. tags數組 - 某個td循環span使用文字
// 2. span設置bs的樣式// 目標: 刪除數據
// 1. 刪除按鈕 - 點擊事件
// 2. 作用域插槽把索引值關聯出來了
// scope身上就有row和index
// 3. 刪除中使用scope.index的索引值
// 4. 刪除事件里刪除數組里對應索引值的數據
export default {components: {MyTable,},data() {return {list: [] // 所有數據};},created() {axios({url: "/api/goods",}).then((res) => {console.log(res);this.list = res.data.data;});}
};
</script>

MyTable.vue里正確代碼(不可復制)

<template><table class="table table-bordered table-stripped"><!-- 表格標題區域 --><thead><tr><th>#</th><th>商品名稱</th><th>價格</th><th>標簽</th><th>操作</th></tr></thead><!-- 表格主體區域 --><tbody><tr v-for="(obj, index) in arr":key="obj.id"><td>{{ obj.id }}</td><td>{{ obj.goods_name }}</td><td>{{ obj.goods_price }}</td><td>{{ obj.tags }}</td><td><button class="btn btn-danger btn-sm">刪除</button></td></tr></tbody></table>
</template><script>
export default {name: 'MyTable',props: {arr: Array}
}
</script><style scoped lang="less">
.my-goods-list {.badge {margin-right: 5px;}
}
</style>

3.5_案例-tabbar-商品表格-插槽

目標: 使用插槽技術, 和作用域插槽技術, 給MyTable.vue組件, 自定義列標題, 自定義表格內容

  • 需求: 允許用戶自定義表格頭和表格單元格內容

分析:

? ①: 把MyTable.vue里準備slot

? ②: 使用MyTable組件時傳入具體標簽

步驟:

  1. 提高組件復用性和靈活性, 把表格列標題thead部分預留標簽, 設置name屬性
  2. 使用MyTable.vue時, 傳入列標題標簽
  3. 表格內容td部分也可以讓組件使用者自定義, 也給tbody下tr內留好標簽和name屬性名
  4. 使用插槽需要用到插槽內的obj對象上的數據, 使用作用域插槽技術

在這里插入圖片描述

MyTable.vue - 留好具名插槽

<template><table class="table table-bordered table-stripped"><!-- 表格標題區域 --><thead><tr><!-- <th>#</th><th>商品名稱</th><th>價格</th><th>標簽</th><th>操作</th> --><slot name="header"></slot></tr></thead><!-- 表格主體區域 --><tbody><tr v-for="(obj, index) in arr":key="obj.id"><!-- <td>{{ obj.id }}</td><td>{{ obj.goods_name }}</td><td>{{ obj.goods_price }}</td><td>{{ obj.tags }}</td><td><button class="btn btn-danger btn-sm">刪除</button></td> --><slot name="body" :row="obj" :index="index"></slot></tr></tbody></table>
</template><script>
export default {name: 'MyTable',props: {arr: Array}
}
</script>

MyGoodsList.vue 使用

<template><div><MyTable :arr="list"><template #header><th>#</th><th>商品名稱</th><th>價格</th><th>標簽</th><th>操作</th></template><!-- scope的值: {row: obj, index: 索引值} --><template #body="scope"><td>{{ scope.row.id }}</td><td>{{ scope.row.goods_name }}</td><td>{{ scope.row.goods_price }}</td><td>{{ scope.row.tags }}</td><td><button class="btn btn-danger btn-sm">刪除</button></td></template></MyTable></div>
</template><script>
import MyTable from "../components/MyTable";
import axios from "axios";
axios.defaults.baseURL = "https://www.escook.cn";
// 目標: 循環商品列表表格
// 1. 封裝MyTable.vue 整體表格組件-一套標簽和樣式
// 2. axios請求數據
// 3. 傳入MyTable組件里循環tr顯示數據// 目標: 展示tags標簽
// 1. tags數組 - 某個td循環span使用文字
// 2. span設置bs的樣式// 目標: 刪除數據
// 1. 刪除按鈕 - 點擊事件
// 2. 作用域插槽把索引值關聯出來了
// scope身上就有row和index
// 3. 刪除中使用scope.index的索引值
// 4. 刪除事件里刪除數組里對應索引值的數據
export default {components: {MyTable,},data() {return {list: [] // 所有數據};},created() {axios({url: "/api/goods",}).then((res) => {console.log(res);this.list = res.data.data;});}
};
</script><style>
</style>

3.6 案例-tabbar-商品表格-tags微標

目標: 把單元格里的標簽, tags徽章鋪設下

  • 需求: 標簽列自定義顯示

分析:

? ①: 插槽里傳入的td單元格

? ②: 自定義span標簽的循環展示-給予樣式

效果演示:
在這里插入圖片描述

bootstrap徽章: https://v4.bootcss.com/docs/components/badge/

MyGoodsList.vue - 插槽

<span v-for="(str, ind) in scope.row.tags" :key="ind"class="badge badge-warning">{{ str }}
</span>

下面額外添加樣式

<style lang="less" scoped>
.my-goods-list {.badge {margin-right: 5px;}
}
</style>

3.7 案例-tabbar-商品表格-刪除功能

目標: 點擊刪除對應這條數據

  • 需求: 點擊刪除按鈕刪除數據

分析:

? ①: 刪除按鈕綁定點擊事件

? ②: 作用域插槽綁定id值出來

? ③: 傳給刪除方法, 刪除MyGoodsList.vue里數組里數據

效果演示
在這里插入圖片描述

提示: id在MyTable.vue里, 但是MyGoodsList.vue里要使用, 而且在插槽位置, 使用作用域插槽已經把整個obj對象(包含id)帶出來了

MyTable.vue

<slot name="body" :row="obj"></slot>
  1. MyGoodsList.vue - 注冊點擊事件
<button class="btn btn-danger btn-sm"@click="removeBtn(scope.row.id)">刪除</button>

? 2. my-goods-list.vue 根據 id 刪除

removeBtn(id){let index = this.list.findIndex(obj => obj.id === id)this.list.splice(index, 1)
},

3.8 案例-tabbar-添加tab

目標: 實現點擊tab按鈕, 出現輸入框自動獲取焦點, 失去焦點關閉input, 回車新增tag, esc清空內容

  • 需求1: 點擊Tab, 按鈕消失, 輸入框出現
  • 需求2: 輸入框自動聚焦
  • 需求3: 失去焦點, 輸入框消失, 按鈕出
  • 需求4: 監測input回車, 無數據攔截
  • 需求5: 監測input取消, 清空數據
  • 需求6: 監測input回車, 有數據添加

效果目標:
在這里插入圖片描述

3.8.0 點擊按鈕消失, 輸入框出現

MyGoodsList.vue - 標簽位置添加

注意: 每個tab按鈕和input都是獨立變量控制, 那么直接在row身上的屬性控制即可

<inputclass="tag-input form-control"style="width: 100px;"type="text"v-if="scope.row.inputVisible"/><button v-else style="display: block;" class="btn btn-primary btn-sm add-tag"@click="scope.row.inputVisible = true">+Tag</button>

3.8.1 input自動獲取焦點

main.js - 定義全局自定義指令

// 全局指令
Vue.directive("focus", {inserted(el){el.focus()}
})

MyGoodsList.vue - 使用 v-focus指令

3.8.2 input失去焦點關閉input

監聽input失去焦點事件, 讓input消失

@blur="scope.row.inputVisible = false"

3.8.3 input回車新增tag

監聽input的回車事件, 如果無數據攔截代碼

@keydown.enter="enterFn(scope.row)"

事件處理函數

enterFn(obj){ // 回車// console.log(obj.inputValue);if (obj.inputValue.trim().length === 0) {alert('請輸入數據')return}obj.tags.push(obj.inputValue) // 表單里的字符串狀態tags數組obj.inputValue = ""
}

3.8.4 input框esc清空內容

@keydown.esc="scope.row.inputValue = ''"

今日總結

  1. 動態組件的使用步驟
  2. 組件緩存使用步驟和作用
  3. 組件插槽默認使用
  4. 插槽默認顯示的內容
  5. 多個插槽時, 具名插槽如何使用
  6. 作用域插槽如何使用以及目的
  7. 自定義命令如何使用
  8. 跟隨視頻完成tabbar案例

面試題

1. vue中solt的使用方式,以及solt作用域插槽的用法

使用方式:當組件當做標簽進行使用的時候,用slot可以用來接受組件標簽包裹的內容,當給solt標簽添加name屬性的 時候,可以調換響應的位置
(高級用法) 插槽作用域: 當傳遞的不是單一的標簽, 例如需要循環時, 把要循環的標簽傳入, 組件內使用v-for在slot標簽上, 內部可以v-bind:把值傳出來, 再外面把值賦予進去, 看示例

<current-user><template v-slot:default="slotProps">{{ slotProps.user.firstName }}</template>
</current-user>// current-user組件, user屬性和值, 綁定給slotProps上
<span><slot v-bind:user="user">{{ user.lastName }}</slot>
</span>

擴展閱讀: https://cn.vuejs.org/v2/guide/components-slots.html (了解即可, 一般用不上)

2. 跟keep-alive有關的生命周期是哪些?(必會)

? 1**)前言:**在開發Vue項目的時候,大部分組件是沒必要多次渲染的,所以Vue提供了一個內置組件keep-alive來緩存組件內部狀態,避免重新渲染,在開發Vue項目的時候,大部分組件是沒必要多次渲染的,所以Vue提供了一個內置組件keep-alive來緩存組件內部狀態,避免重新渲染

? 2**)生命周期函數:**在被keep-alive包含的組件/路由中,會多出兩個生命周期的鉤子:activated 與 deactivated。

? 1**、activated鉤子:**在在組件第一次渲染時會被調用,之后在每次緩存組件被激活時調用。

? 2**、Activated鉤子調用時機:** 第一次進入緩存路由/組件,在mounted后面,beforeRouteEnter守衛傳給 next 的回調函數之前調用,并且給因為組件被緩存了,再次進入緩存路由、組件時,不會觸發這些鉤子函數,beforeCreate created beforeMount mounted 都不會觸發

? 1**、deactivated鉤子:**組件被停用(離開路由)時調用。

? 2**、deactivated鉤子調用時機**:使用keep-alive就不會調用beforeDestroy(組件銷毀前鉤子)和destroyed(組件銷毀),因為組件沒被銷毀,被緩存起來了,這個鉤子可以看作beforeDestroy的替代,如果你緩存了組件,要在組件銷毀的的時候做一些事情,可以放在這個鉤子里,組件內的離開當前路由鉤子beforeRouteLeave => 路由前置守衛 beforeEach =>全局后置鉤子afterEach => deactivated 離開緩存組件 => activated 進入緩存組件(如果你進入的也是緩存路由)

3. 自定義指令(v-check、v-focus)的方法有哪些?它有哪些鉤子函數?還有哪些鉤子函數參數?(必會)

? 全局定義指令:在vue對象的directive方法里面有兩個參數,一個是指令名稱,另外一個是函數。組件內定義指令:directives

? 鉤子函數:bind(綁定事件觸發)、inserted(節點插入的時候觸發)、update(組件內相關更新)

? 鉤子函數參數:el、binding

4. is這個特性你有用過嗎?主要用在哪些方面?(高薪常問)

? 1**)動態組件**

? , componentName可以是在本頁面已經注冊的局部組件名和全局組件名,也可以是一個組件的選項對象。 當控制componentName改變時就可以動態切換選擇組件。

? 2**)is的用法**

?    有些HTML元素,諸如 <ul><ol><table><select>,對于哪些元素可以出現在其內部是有嚴格限制的。?    而有些HTML元素,諸如 <li><tr><option>,只能出現在其它某些特定的元素內部。
?    <ul>?      <card-list></card-list>?    </ul>

? 所以上面會被作為無效的內容提升到外部,并導致最終渲染結果出錯。應該這么寫:

?    <ul>?      <li is="cardList"></li>?     </ul>

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/248528.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/248528.shtml
英文地址,請注明出處:http://en.pswp.cn/news/248528.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

重載(overload)與重寫(override)的區別

overload&#xff08;重載&#xff09;:在同一個類中&#xff0c;方法名相同&#xff0c;參數列表不相同。與返回值類型無關。 override&#xff08;重寫&#xff09;:存在同一個類中&#xff0c;或者父子接口中&#xff0c;方法名相同個&#xff0c;參數列表相同。遵循“兩同兩…

python學習,day3:函數式編程,*arge,**kwargs

對于不固定長度的參數&#xff0c;需要使用*arge&#xff0c;**kwargs來調用&#xff0c;區別是*arge是轉換為元組&#xff0c;而kwargs轉化為字典 # codingutf-8 # Author: RyAn Bi def test(*args): #參數組print(args)test(1,2,4,6,7,8) #方式1 test(*[1,2,4,5,6]) #方式2 #…

那些被人忽略的Vue導航知識

本篇學習目標 能夠了解單頁面應用概念和優缺點能夠掌握vue-router路由系統使用能夠掌握鏈接導航和編程式導航用法能夠掌握路由嵌套和路由守衛能夠掌握vant組件庫基礎使用 1. vue路由簡介和基礎使用 1.0 什么是路由 目標: 設備和ip的映射關系 目標: 接口和服務的映射關系 目…

passwd命令

-n 在這幾天你不能更改密碼&#xff01; -x 在n<時間<x在這段時間內你必須修改密碼&#xff01; -w 當距離x日期還有w天的時候開始提醒你改密碼&#xff01; -i 密碼過期i天之后&#xff0c;此密碼停用&#xff0c;你也就無法用此密碼登陸這個用戶了。注意是密碼過期之后…

一文帶你吃透Vue生命周期(結合案例通俗易懂)

文章目錄本篇學習目標1. vue生命周期1.0_人的-生命周期1.1_鉤子函數1.2_初始化階段1.3_掛載階段1.4_更新階段1.5_銷毀階段2. axios2.0_axios基本使用2.1_axios基本使用-獲取數據2.2_axios基本使用-傳參2.3_axios基本使用-發布書籍2.4_axios基本使用-全局配置3. nextTick和nextT…

[SCOI2012]滑雪 (最小生成樹 Kruskal)

題目描述 a180285非常喜歡滑雪。他來到一座雪山&#xff0c;這里分布著M條供滑行的軌道和N個軌道之間的交點&#xff08;同時也是景點&#xff09;&#xff0c;而且每個景點都有一編號i(1≤i≤N)和一高度Hi?。a180285能從景點ii滑到景點j當且僅當存在一條i和j之間的邊&#xf…

來學習ansibie(1)

# ansible 批量在遠程主機上執行命令 python2.7編寫 ## 安裝 第一步:下載epel源 shellwget -O /etc/yum.repos.d/epel.repo http://mirrors.aliyun.com/repo/epel-7.repo 第二步:安裝 shellyum install -y ansible ## ansible 命令格式 shellUsage: ansible <host-pattern&g…

CQYZOJ P1392 拔河問題

題目\(1\) Description 一個學校舉行拔河比賽&#xff0c;所有的人被分成了兩組&#xff0c;每個人必須&#xff08;且只能夠&#xff09;在其中的一組&#xff0c;且兩個組內的所有人體重加起來盡可能地接近. Input 第\(1\)行是一個\(n\)&#xff0c;表示參加拔河比賽的總人數…

靈活的Vue組件——原來這么簡單

本篇學習目標 能夠理解vue組件概念和作用能夠掌握封裝組件能力能夠使用組件之間通信能夠完成todo案例 1. vue組件 1.0_為什么用組件 以前做過一個折疊面板 需求: 現在想要多個收起展開的部分 方案1: 復制代碼 代碼重復 冗余不利于維護 案例用less寫的樣式, 所以下載 ya…

FOI冬令營 Day 3

目錄 T1、簽到題&#xff08;sort&#xff09;傳送門 Code T2、送分題&#xff08;queue&#xff09;傳送門 Code T3、簡單題&#xff08;game&#xff09;傳送門 Code 咕咕咕T1、簽到題&#xff08;sort&#xff09; 傳送門 原題&#xff1a;LOJ 2767 Code //2019/2/14 //50…

委托事件觀察者模式

委托的默認返回類型&#xff1a;void 聲明委托的關鍵字&#xff1a;delegate 多播委托&#xff1a;將多個方法綁定到一個委托變量 在調用方法時 可以執行綁定的方法 委托的描述&#xff1a; 委托是一個類 定義了方法的類型 可以將方法當做另一個方法進行傳遞 委托并不等同于方法…

贏在CSDN——名利兼收

文章目錄&#x1f30a; 相識CSDN&#x1f30a; 益于CSDN流量將成為你我的亮點我的專欄收益到賬啦學習會員助你拿捏專欄更多曝光自己的機會CSDN問答為你準備的零花錢&#x1f30a; 忠于CSDN&#x1f30a; 相識CSDN 小編自注冊CSDN至今兩年有余&#xff0c;記得初衷也僅僅是為了…

124angular1實現無限表單(僅供自己看)

//將本行的內容對象作為參數&#xff0c;傳給點擊函數&#xff0c;點擊函數向后臺發送請求&#xff0c;把獲取的返回值作為內容對象的一個屬性。 (function (angular) {angular.module(myModule, []).directive(treeModel, [$compile, function ($compile) {return {restrict: …

了解 Vue SSR 這一篇足以

文章目錄1 - 什么是服務器端渲染&#xff1f;1.1 新建server文件夾1.2 生成一個node項目1.3 安裝express1.4 服務端渲染小案例1.5 運行查看效果1.6 打開瀏覽器1.7 右鍵查看源代碼2 - 什么是客戶端渲染&#xff1f;2.1 新建client文件夾2.2 生成一個vue項目2.3 安裝依賴并啟動2.…

3 數組中的重復數字

題目描述 在一個長度為 n 的數組里的所有數字都在 0 到 n-1 的范圍內。數組中某些數字是重復的&#xff0c;但不知道有幾個數字是重復的&#xff0c;也不知道每個數字重復幾次。請找出數組中任意一個重復的數字。 Input: {2, 3, 1, 0, 2, 5}Output: 2 思路 給出了長度為n且數組…

小型軟件項目開發流程探討

一&#xff0e;導言國內很多項目都是小型項目, 參與人員少(兩到五個人), 要快速交付(一兩個月) . 要成功完成這種項目, 除了使用成熟且被團隊成員熟練使用的技術之外, 有一個良好的開發流程, 也是很必要的. 二&#xff0e;小型軟件項目開發流程下圖是我對小型軟件項目開發流程…

Vue2的核心原理剖析

? 用了這么久的Vue2了你真的 知其然&#xff0c;知其所以然么&#xff1f; ?今天博主就為大家帶來一篇對Vue核心功能的部分剖析\textcolor{pink}{今天博主就為大家帶來一篇對Vue核心功能的部分剖析}今天博主就為大家帶來一篇對Vue核心功能的部分剖析 ?后續文章會用更多小案…

Scrum之成敗——從自身案例說起,僅供參考

從07年中初次接觸Scrum的概念到其中幾年項目中逐漸實踐CI、TDD&#xff0c;到親自掌握項目實踐Scrum近一年&#xff0c;最終我們放棄了Scrum這個框架和所謂的“自組織”。原因為何&#xff1f; 1.成員放棄了Scrum所“賦予”的“權利” 比如領用任務、評估工作量、自組織協作、決…

sanic官方文檔解析之下載和Configuration

1,sanic框架是做什么的? sanic的官方網址:https://sanic.readthedocs.io/en/latest/sanic框架是一個類似于flask框架的在Python3.5以上版本的文本服務器,他能夠快速的編寫,它是通過驚人的開發效率完成開發,希望通過這篇文章得到激勵sanic框架的理念是:簡單,高效 sanic的應用如…

首秀 Express 框架

文章目錄框架特性express的使用初始化項目&#xff1a;下載框架模塊&#xff1a;測試代碼&#xff1a;總結以上代碼&#xff1a;請求處理的中間件概念&#xff1a;中間件——app.use基本用法&#xff1a;next的用法app.use中間件的應用路由的保護網站維護公告自定義404&#xf…