elementui源碼學(xué)習(xí)之仿寫一個(gè)el-divider組件
正文
本篇文章記錄仿寫一個(gè)el-divider組件細(xì)節(jié),從而有助于大家更好理解餓了么ui對(duì)應(yīng)組件具體工作細(xì)節(jié)。本文是elementui源碼學(xué)習(xí)仿寫系列的又一篇文章,后續(xù)空閑了會(huì)不斷更新并仿寫其他組件。源碼在github上,大家可以拉下來(lái),npm start運(yùn)行跑起來(lái),結(jié)合注釋有助于更好的理解。
github倉(cāng)庫(kù)地址如下:github.com/shuirongshu…
組件需求分析
關(guān)于分割線的組件,一般使用的場(chǎng)景有:
- 分割線之水平分割(如:段落與段落之間分隔)
- 水平分割線加上文字描述(文字描述位置一般是位于分割線的左側(cè)、居中、右側(cè))
- 分割類型之垂直分割(橫向的行內(nèi)文字與文字之間的分割)
- 分割線樣式類型(solid分割線或dashed分割線或dotted分割線)
因?yàn)榇怪狈指罹嚯x不夠,所以垂直分割沒(méi)有文字描述
組件中用到的知識(shí)點(diǎn)
函數(shù)式組件
我們知道,一個(gè)函數(shù),其實(shí)就是一個(gè)加工流水線,我們?nèi)雲(yún)ⅲ?jīng)過(guò)函數(shù)的加工以后,得到return加工后的結(jié)果。
對(duì)于組件而言,其實(shí)也可以當(dāng)做一個(gè)函數(shù)(函數(shù)式組件),我們傳遞參數(shù),組件props接收,然后組件再根據(jù)傳參的不同,渲染出對(duì)應(yīng)的樣式效果。
這種方式和原來(lái)的我們使用組件的方式其實(shí)差不多,只不過(guò)更加精簡(jiǎn),畢竟只考慮接收參數(shù),沒(méi)有生命周期、this實(shí)例之類的。所以函數(shù)式組件的渲染效率會(huì)比普通組件高。
所以有些情況下,使用函數(shù)式組件效率會(huì)更高
函數(shù)式組件的兩種定義方式
比如,通過(guò)傳參方式去渲染文字,可以有以下兩種方式
方式一:render函數(shù)定義
<script>
export default {
functional: true, // 一定要指明是函數(shù)式組件
props: {
bookName: {
type: String,
default: "斗破蒼穹",
},
},
// context上下文,有點(diǎn)像this的意思。但不一樣
render: function (createElement, context) {
// 創(chuàng)建一個(gè)h3標(biāo)簽,標(biāo)簽內(nèi)容是上下文對(duì)象中的props中的bookName字段
return createElement("h3", {}, context.props.bookName);
},
};
</script>
方式二:template定義
<!-- 模板標(biāo)簽上加上functional關(guān)鍵字表明是函數(shù)式組件 -->
<template functional>
<div>
<h3>{{ props.bookName }}</h3>
</div>
</template>
<script>
export default {
props: {
bookName: {
type: String,
default: "斗破蒼穹",
},
},
}
</script>
關(guān)于函數(shù)式組件的具體細(xì)節(jié),詳情見(jiàn)官方文檔:cn.vuejs.org/v2/guide/re…
這里為什么要提到函數(shù)式組件呢?因?yàn)?code>el-divider組件就是使用函數(shù)式組件進(jìn)行封裝的(因?yàn)榻M件只是做分割線條的展示,所以不需要有this實(shí)例,生命周期之類的)
故:為了提升渲染效率,這里使用函數(shù)式組件
使用函數(shù)式組件的情況大致有以下:
- 只做展示的組件,畢竟沒(méi)啥邏輯操作相關(guān)
- 高階組件的操作
- 循環(huán)中也可以考慮使用
- 整體來(lái)說(shuō),函數(shù)式組件還是比較靈活的
解決一像素太粗的問(wèn)題
先看一下問(wèn)題圖

只需要給對(duì)應(yīng)dom元素稍微縮放一下即可:
transform: scaleY(0.95); 或 transform: scaleX(0.95);
組件封裝
el-divider組件整體還是比較簡(jiǎn)單的,主要是函數(shù)式組件的參數(shù)使用細(xì)節(jié)
組件封裝的效果圖

組件封裝的代碼
使用的代碼
<template>
<div class="wrap">
<h3>實(shí)線solid靠左</h3>
<my-divider class="staticClass_must_hava" content-position="left"
>早上好</my-divider
>
<h3>實(shí)線solid居中</h3>
<my-divider content-position="center">中午好</my-divider>
<h3>實(shí)線solid靠右</h3>
<my-divider content-position="right">下午好</my-divider>
<h3>虛線dashed靠左</h3>
<my-divider lineType="dashed" content-position="left">早上好</my-divider>
<h3>虛線dashed居中</h3>
<my-divider lineType="dashed" content-position="center">中午好</my-divider>
<h3>虛線dashed靠右</h3>
<my-divider lineType="dashed" content-position="right">下午好</my-divider>
<h3>圓點(diǎn)dotted靠左</h3>
<my-divider lineType="dotted" content-position="left">早上好</my-divider>
<h3>圓點(diǎn)dotted居中</h3>
<my-divider lineType="dotted" content-position="center">中午好</my-divider>
<h3>圓點(diǎn)dotted靠右</h3>
<my-divider lineType="dotted" content-position="right">下午好</my-divider>
<h3>豎向分割線</h3>
<span>早上好</span>
<my-divider direction="vertical"></my-divider>
<span>中午好</span>
<my-divider direction="vertical"></my-divider>
<span>下午好</span>
</div>
</template>
<script>
import myDivider from "./myDivider.vue";
export default {
components: { myDivider },
};
</script>
<style>
.wrap { padding: 24px; }
.staticClass_must_hava { border-bottom: 1px solid #baf !important; }
</style>
封裝的代碼
<template functional>
<!-- functional關(guān)鍵字加上后,即代表為函數(shù)式組件,就沒(méi)有生命周期、this實(shí)例等了,故渲染更快,性能更高 -->
<!-- v-bind和v-on這里不加也行。畢竟函數(shù)式組件個(gè)人愚見(jiàn)簡(jiǎn)潔建議一些,attrs存放傳遞來(lái)的數(shù)據(jù),listeners傳遞任何事件監(jiān)聽(tīng)器 -->
<!-- data.staticClass需要加上,否則外層給組件加的class屬性在審查元素的時(shí)候不會(huì)出現(xiàn)的,相當(dāng)于沒(méi)加 -->
<div
v-bind="data.attrs"
v-on="listeners"
:class="[data.staticClass, `${props.direction}`, `${props.lineType}`]"
>
<!-- 當(dāng)分割線為水平的時(shí)候,且插槽有內(nèi)容的時(shí)候,才渲染對(duì)應(yīng)數(shù)據(jù) -->
<!-- slots()方法返回插槽數(shù)據(jù)對(duì)象,default為默認(rèn)插槽;相當(dāng)于this.$slots.default。本系列文章中仿寫一個(gè)el-tabs 也提到了此知識(shí)點(diǎn) -->
<div
v-if="props.direction === 'horizontal' && slots().default"
:class="['text', `${props.contentPosition}`]"
>
<slot />
</div>
</div>
</template>
<script>
export default {
name: "myDivider",
props: {
// 分割線的方向,默認(rèn)水平horizontal(垂直vertical可選)
direction: {
type: String,
default: "horizontal",
validator(val) {
return ["horizontal", "vertical"].includes(val);
},
},
// 文本內(nèi)容位置,默認(rèn)居中(僅當(dāng)分割線為水平方向時(shí),才渲染對(duì)應(yīng)內(nèi)容)
contentPosition: {
type: String,
default: "center",
validator(val) {
return ["left", "center", "right"].includes(val);
},
},
// 分割線線條類型,默認(rèn)實(shí)線,可選虛線,圓點(diǎn)線
lineType: {
type: String,
default: "solid",
validator(val) {
return ["solid", "dashed", "dotted"].includes(val);
},
},
},
};
</script>
<style lang="less" scoped>
// 水平樣式
.horizontal {
width: 100%;
border-bottom: 1px solid #333; // 默認(rèn)solid分割線樣式
display: block;
margin: 12px 0;
position: relative;
// 縮放0.95倍可以解決一像素問(wèn)題(一像素看著比實(shí)際粗)橫線Y縮放
transform: scaleY(0.95);
}
// dashed分割線
.dashed {
border-bottom: 1px dashed #333;
}
// dotted分割線
.dotted {
border-bottom: 1px dotted #333;
}
// 垂直樣式
.vertical {
display: inline-block;
width: 1px;
height: 1em;
margin: 0 8px;
background-color: #333;
vertical-align: middle;
position: relative;
top: -1px; // 微調(diào)一下位置
// 縮放0.95倍可以解決一像素問(wèn)題(一像素看著比實(shí)際粗)豎線X縮放
transform: scaleX(0.95);
}
// 文本樣式
.text {
position: absolute;
padding: 0 12px;
color: #333;
font-size: 14px;
transform: translateY(-50%); // y軸移動(dòng)自身的一半,使其垂直方向居中
background-color: #fff; // 通過(guò)背景色和padding去蓋住水平分割線條
}
// 文本靠左
.text.left {
left: 36px;
}
// 文本靠右
.text.right {
right: 36px;
}
// 文本居中
.text.center {
left: 50%;
// 居中的話,別忘了要回正。即XY軸方向都平移自身的一半
transform: translateX(-50%) translateY(-50%);
}
</style>以上就是elementui源碼學(xué)習(xí)之仿寫一個(gè)el-divider組件的詳細(xì)內(nèi)容,更多關(guān)于elementui仿寫el-divider的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
vue2中基于vue-simple-upload實(shí)現(xiàn)文件分片上傳組件功能
這篇文章主要介紹了vue2中基于vue-simple-upload的文件分片上傳組件,本文通過(guò)示例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-06-06
Vue數(shù)據(jù)驅(qū)動(dòng)模擬實(shí)現(xiàn)2
這篇文章主要介紹了Vue數(shù)據(jù)驅(qū)動(dòng)模擬實(shí)現(xiàn)的相關(guān)資料,實(shí)現(xiàn)Observer構(gòu)造函數(shù),監(jiān)聽(tīng)已有數(shù)據(jù)data中的所有屬性,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-01-01
vue插件vue-resource的使用筆記(小結(jié))
本篇文章主要介紹了vue插件vue-resource的使用筆記(小結(jié)),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-08-08
Vue封裝一個(gè)Tabbar組件?帶組件路由跳轉(zhuǎn)方式
這篇文章主要介紹了Vue封裝一個(gè)Tabbar組件?帶組件路由跳轉(zhuǎn)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-04-04
vue使用監(jiān)聽(tīng)實(shí)現(xiàn)全選反選功能
最近做的項(xiàng)目用到了全選全不選功能,于是我就自己動(dòng)手寫了一個(gè),基于vue使用監(jiān)聽(tīng)實(shí)現(xiàn)全選反選功能,具體實(shí)例代碼大家參考下本文2018-07-07
Vue-Access-Control 前端用戶權(quán)限控制解決方案
Vue-Access-Control是一套基于Vue/Vue-Router/axios 實(shí)現(xiàn)的前端用戶權(quán)限控制解決方案。這篇文章主要介紹了Vue-Access-Control:前端用戶權(quán)限控制解決方案,需要的朋友可以參考下2017-12-12
解決Vue調(diào)用springboot接口403跨域問(wèn)題
這篇文章主要介紹了解決Vue調(diào)用springboot接口403跨域問(wèn)題,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2019-09-09
vue使用動(dòng)畫實(shí)現(xiàn)滾動(dòng)表格效果
這篇文章主要為大家詳細(xì)介紹了vue使用動(dòng)畫實(shí)現(xiàn)滾動(dòng)表格效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-04-04
Vue.js組件使用props傳遞數(shù)據(jù)的方法
這篇文章主要為大家詳細(xì)介紹了Vue.js組件使用props傳遞數(shù)據(jù)的方法,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-10-10

