【組件封裝】改造 Element-UI 多選框組件 (el-checkbox)

code.jpg

el-checkbox多選框組件

element-ui官網(wǎng)的多選框組件盒揉,是采用簡(jiǎn)單數(shù)組進(jìn)行配置的,顯示的lable值與勾選后的值是同一個(gè)

el-checkbox.jpg

實(shí)際業(yè)務(wù)場(chǎng)景

在實(shí)際業(yè)務(wù)開(kāi)發(fā)中兑徘,我們更多遇到是如下數(shù)組對(duì)象的格式刚盈。

我們需要額外將數(shù)組進(jìn)行l(wèi)abel、value的拆分才能使用 el-checkbox 挂脑。

如果要實(shí)現(xiàn)勾選藕漱、全選、默認(rèn)賦值的場(chǎng)景崭闲,將需要更加復(fù)雜的代碼處理肋联。


[

  { label: '小紅', value: '1' },

  { label: '小明', value: '2' },

  { label: '小芳', value: '3' }

]

組件設(shè)計(jì)(簡(jiǎn)易版本)

調(diào)用效果及代碼

show1.png

<!--

 * @Date: 2020-12-09 17:52:54

 * @Author: surewinT 840325271@qq.com

 * @LastEditTime: 2022-05-09 14:22:01

 * @LastEditors: surewinT 840325271@qq.com

 * @Description: 調(diào)用頁(yè)面

-->

<template>

    <div class="">

        <Test :checkboxList="checkboxList" v-model="result" />

    </div>

</template>

<script>

import Test from "./test.vue";

export default {

    components: {

        Test,

    },

    props: [],

    data() {

        return {

            checkboxList: [

                { label: "小紅", value: "1" },

                { label: "小明", value: "2" },

                { label: "小芳", value: "3" },

            ],

            result: [],

        };

    },

    mounted() {},

    watch: {

        result() {

            console.log("變化了:", this.result);

        },

    },

    methods: {},

};

</script>

<style lang='scss' scoped>

</style>

改造后的組件(test.vue)


<!--

 * @Date: 2022-05-09 11:52:02

 * @Author: surewinT 840325271@qq.com

 * @LastEditTime: 2022-05-09 14:22:10

 * @LastEditors: surewinT 840325271@qq.com

 * @Description: 改造后的組件

-->

<template>

    <div class="">

        <el-checkbox-group v-model="labelResult">

            <el-checkbox v-for="(item, index) in labelList" :key="index" :label="item"> </el-checkbox>

        </el-checkbox-group>

    </div>

</template>

<script>

export default {

    components: {},

    props: {

        checkboxList: {

            type: Array,

            default: () => {

                return [];

            },

        },

    },

    data() {

        return {

            labelResult: [],

            labelList: [],

            labelMap: {},

        };

    },

    mounted() {

        this.loadData();

    },

    watch: {},

    methods: {

        // 初始化數(shù)據(jù)

        loadData() {

            this.checkboxList.forEach((res) => {

                this.labelList.push(res["label"]);

                this.labelMap[res["label"]] = res["value"];

            });

        },

    },

};

</script>

<style lang='scss' scoped>

</style>

組件設(shè)計(jì)(完整版)

組件簡(jiǎn)介

p-el-checkbox.png

調(diào)用效果及代碼

show2.png

<!--

 * @Date: 2020-12-09 17:52:54

 * @Author: surewinT 840325271@qq.com

 * @LastEditTime: 2022-05-09 14:49:13

 * @LastEditors: surewinT 840325271@qq.com

 * @Description: 調(diào)用頁(yè)面

-->

<template>

    <div class="">

        <p-el-checkbox

            v-model="result"

            resultType="Array"

            label="name"

            prop="id"

            :defaultCheck="defaultCheck"

            :checkboxList="checkboxList"

        />

    </div>

</template>

<script>

import Pelcheckbox from "./p-el-checkbox.vue";

export default {

    components: {

        "p-el-checkbox": Pelcheckbox,

    },

    props: [],

    data() {

        return {

            defaultCheck: ["1", "3"],

            checkboxList: [

                { name: "小紅", id: "1" },

                { name: "小明", id: "2" },

                { name: "小芳", id: "3" },

            ],

            result: {},

        };

    },

    mounted() {},

    watch: {

        result: function () {

            console.log("變化了:", this.result);

        },

    },

    methods: {},

};

</script>

<style lang='scss' scoped>

</style>

組件源碼(p-el-checkbox.vue)


<!--

 * @Author: surewinT 840325271@qq.com

 * @LastEditTime: 2022-05-09 14:57:09

 * @LastEditors: surewinT 840325271@qq.com

 * @Description: 改造 Element-UI 多選框組件 (el-checkbox)

-->

<template>

    <div :class="customClass">

        <el-checkbox

            v-if="showCheckAll"

            :indeterminate="isIndeterminate"

            v-model="checkAll"

            @change="handleCheckAllChange"

        >

            全選

        </el-checkbox>

        <el-checkbox-group v-model="labelResult" :class="inline ? '' : 'al_checkbox'">

            <span v-for="(item, index) in labelList" :key="index">

                <el-checkbox :label="item" :class="labelClass" @change="handleCheckChange" v-if="judgeHidden(item)">

                    <span>

                        <slot

                            name="label"

                            :index="index"

                            :label="item"

                            :checkboxList="checkboxList"

                            :disabled="checkboxMap[item] && checkboxMap[item].disabled"

                            :data="checkboxList[index] ? JSON.parse(JSON.stringify(checkboxList[index])) : {}"

                        >

                            <span> {{ item }} </span>

                        </slot>

                    </span>

                </el-checkbox>

            </span>

        </el-checkbox-group>

    </div>

</template>

<script>

/**

 * @description:props說(shuō)明

 * @param {Boolean} showCheckAll -是否顯示全選按鈕

 * @param {Boolean} inline -是否是行內(nèi)表單

 * @param {Array<Object>} checkboxList -多選框組件的數(shù)據(jù)配置

 * @param {String} label -label對(duì)應(yīng)的字段

 * @param {String} prop -value對(duì)應(yīng)的字段

 * @param {Array} defaultCheck -默認(rèn)勾選的項(xiàng)

 * @param {String} resultType -綁定值的類型(支持對(duì)象、數(shù)組兩種格式)

 * @param {String} labelClass -單個(gè)選項(xiàng)綁定類名刁俭,可用于自定義樣式

 * @param {String} customClass -多選框的綁定類型橄仍,可用于自定義樣式

 * @param {Array} checkboxHidden -隱藏的表單項(xiàng),元素對(duì)應(yīng)checkboxList中prop的值

 */

export default {

    name: "p-el-checkbox",

    props: {

        showCheckAll: {

            type: Boolean,

            default: true,

        },

        inline: {

            type: Boolean,

            default: true,

        },

        checkboxList: {

            type: Array,

            default: () => {

                return [];

            },

        },

        label: {

            type: String,

            default: "label",

        },

        prop: {

            type: String,

            default: "value",

        },

        defaultCheck: {

            type: Array,

            default: () => {

                return [];

            },

        },

        resultType: {

            type: String,

            default: "Array", // 可選:Array 牍戚、Object

        },

        labelClass: {

            type: String,

            default: "",

        },

        customClass: {

            type: String,

            default: "",

        },

        checkboxHidden: {

            type: Array,

            default: function () {

                return [];

            },

        },

    },

    data() {

        return {

            labelList: [],

            labelResult: [],

            labelMap: {}, // label與res的映射(res指checkboxList每一項(xiàng)元素)

            isIndeterminate: false,

            checkAll: false,

            currentLabel: null,

            currentValue: null,

        };

    },

    mounted() {

        this.loadData();

    },

    computed: {

        arrayRusult: function () {

            let result = [];

            if (this.resultType == "Array") {

                this.labelResult.forEach((res) => {

                    result.push(this.labelMap[res]);

                });

            }

            return result;

        },

        objectRusult: function () {

            let obj = {};

            if (this.resultType == "Object") {

                for (let item in this.labelMap) {

                    if (this.checkboxHidden.indexOf(this.labelMap[item]) != -1) {

                        continue;

                    }

                    if (this.labelResult.indexOf(item) == -1) {

                        obj[this.labelMap[item]] = false;

                    } else {

                        obj[this.labelMap[item]] = true;

                    }

                }

            }

            return obj;

        },

        // 名字映射

        checkboxMap() {

            let obj = {};

            for (let res of this.checkboxList) {

                obj[res[this.label]] = res;

            }

            return obj;

        },

    },

    watch: {

        checkboxList() {

            this.loadData();

        },

        labelResult() {

            this.isIndeterminate = true;

            this.checkAll = false;

            if (this.labelResult.length == 0) {

                this.isIndeterminate = false;

                this.checkAll = false;

            } else if (this.labelResult.length == this.labelList.length - this.checkboxHidden.length) {

                this.isIndeterminate = false;

                this.checkAll = true;

            }

            // 返回?cái)?shù)組結(jié)果

            if (this.resultType == "Array") {

                this.$emit("input", this.arrayRusult);

            }

            // 返回對(duì)象結(jié)果

            if (this.resultType == "Object") {

                this.$emit("input", this.objectRusult);

            }

        },

        defaultCheck() {

            this.labelResult = [];

            this.checkboxList.forEach((res) => {

                if (this.defaultCheck.indexOf(res[this.prop]) != -1) {

                    this.labelResult.push(res[this.label]);

                }

            });

        },

    },

    methods: {

        // 初始化數(shù)據(jù)

        loadData() {

            this.checkboxList.forEach((res) => {

                this.labelList.push(res[this.label]);

                this.labelMap[res[this.label]] = res[this.prop];

                if (this.defaultCheck.indexOf(res[this.prop]) != -1) {

                    this.labelResult.push(res[this.label]);

                }

            });

        },

        // 全選按鈕

        handleCheckAllChange(val) {

            this.isIndeterminate = false;

            if (val) {

                this.isIndeterminate = false;

                this.checkAll = true;

                this.labelResult = JSON.parse(JSON.stringify(this.labelList));

                // 過(guò)濾隱藏選項(xiàng)

                this.labelResult = this.labelResult.filter((res) => {

                    let exist = this.checkboxHidden.indexOf(this.labelMap[res]);

                    return exist == -1;

                });

            } else {

                this.checkAll = false;

                this.labelResult = [];

            }

        },

        // 單擊多選框

        handleCheckChange(val, e) {

            this.currentLabel = this.labelMap[e.target.value];

            this.currentValue = val;

            // 返回?cái)?shù)組結(jié)果:(勾選結(jié)果侮繁,點(diǎn)擊的prop,點(diǎn)擊的prop是否勾選)

            if (this.resultType == "Array") {

                this.$emit("change", this.arrayRusult, this.currentLabel, this.currentValue);

            }

            // 返回對(duì)象結(jié)果:(勾選結(jié)果如孝,點(diǎn)擊的prop宪哩,點(diǎn)擊的prop是否勾選)

            if (this.resultType == "Object") {

                this.$emit("change", this.objectRusult, this.currentLabel, this.currentValue);

            }

        },

        // 隱藏判斷

        judgeHidden(name) {

            let show = this.checkboxHidden.indexOf(this.labelMap[name]) == -1 ? true : false;

            return show;

        },

    },

};

</script>

<style lang="scss" scoped>

.el-checkbox {

    display: inline-block;

    padding-right: 10px;

}

.al_checkbox {

    .el-checkbox {

        display: block;

        margin-right: 0;

    }

}

// 滾動(dòng)條樣式修改

::-webkit-scrollbar {

    width: 5px;

}

::-webkit-scrollbar-track {

    -webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.3);

    box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.3);

    -webkit-border-radius: 10px;

    border-radius: 10px;

}

::-webkit-scrollbar-thumb {

    -webkit-border-radius: 10px;

    border-radius: 10px;

    background: rgba(155, 155, 155, 0.8);

    -webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.2);

    box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.2);

}

::-webkit-scrollbar-thumb:window-inactive {

    background: rgba(155, 155, 155, 0.1);

}

</style>

倉(cāng)庫(kù)源碼

改造el-checkbox

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末第晰,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子茁瘦,更是在濱河造成了極大的恐慌,老刑警劉巖甜熔,帶你破解...
    沈念sama閱讀 206,126評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異纺非,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)烧颖,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,254評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)炕淮,“玉大人,你說(shuō)我怎么就攤上這事涂圆。” “怎么了润歉?”我有些...
    開(kāi)封第一講書(shū)人閱讀 152,445評(píng)論 0 341
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)踩衩。 經(jīng)常有香客問(wèn)我,道長(zhǎng)驱富,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 55,185評(píng)論 1 278
  • 正文 為了忘掉前任褐鸥,我火速辦了婚禮,結(jié)果婚禮上叫榕,老公的妹妹穿的比我還像新娘浑侥。我一直安慰自己,他們只是感情好晰绎,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,178評(píng)論 5 371
  • 文/花漫 我一把揭開(kāi)白布锭吨。 她就那樣靜靜地躺著,像睡著了一般寒匙。 火紅的嫁衣襯著肌膚如雪零如。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 48,970評(píng)論 1 284
  • 那天锄弱,我揣著相機(jī)與錄音考蕾,去河邊找鬼。 笑死会宪,一個(gè)胖子當(dāng)著我的面吹牛肖卧,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播掸鹅,決...
    沈念sama閱讀 38,276評(píng)論 3 399
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼塞帐,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼拦赠!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起葵姥,我...
    開(kāi)封第一講書(shū)人閱讀 36,927評(píng)論 0 259
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤荷鼠,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后榔幸,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體允乐,經(jīng)...
    沈念sama閱讀 43,400評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,883評(píng)論 2 323
  • 正文 我和宋清朗相戀三年削咆,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了牍疏。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 37,997評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡拨齐,死狀恐怖鳞陨,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情瞻惋,我是刑警寧澤炊邦,帶...
    沈念sama閱讀 33,646評(píng)論 4 322
  • 正文 年R本政府宣布,位于F島的核電站熟史,受9級(jí)特大地震影響馁害,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜蹂匹,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,213評(píng)論 3 307
  • 文/蒙蒙 一碘菜、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧限寞,春花似錦、人聲如沸计雌。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,204評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至反番,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間罢缸,已是汗流浹背枫疆。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,423評(píng)論 1 260
  • 我被黑心中介騙來(lái)泰國(guó)打工养铸, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留轧膘,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 45,423評(píng)論 2 352
  • 正文 我出身青樓鳞滨,卻偏偏與公主長(zhǎng)得像蟆淀,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子褒链,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,722評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容