elementUI之下拉选项加多选框功能实现vue3+ts

根据
@牛先森家的牛奶的代码修改后实现

javascript 复制代码
<template>
    <div class="select-checked">
      <el-select
        v-model="selected"
        :class="{ all: optionsAll }"
        multiple
        placeholder="请选择"
        :popper-append-to-body="false"
      >
        <el-option :value="''" label="全部" class="multiple">
          <el-checkbox v-model="optionsAll" @change="handleOptionsAllChange" style="width: 100%;">
            全部
          </el-checkbox>
        </el-option>
        <el-option
          class="multiple"
          :value="item.value"
          :label="item.label"
          v-for="(item, key) in optionsData"
          :key="key"
        >
          <el-checkbox v-model="item.check" @change="handleTaskItemChange(item)" style="width: 100%;">
            {{ item.label }}
          </el-checkbox>
        </el-option>
      </el-select>
    </div>
  </template>
  
<script lang="ts">
  import { defineComponent } from 'vue';
  
  export default defineComponent({
    props: {
      options: {
        type: Array as () => Array<any>,
        required: true,
      }
    },
    data() {
      return {
        optionsData: [] as any[],
        optionsAll: false,
        selectedOptions: [] as any[],
      };
    },
    watch: {
      options: {
        handler(newVal) {
          this.optionsData = newVal;
          let checkedData = newVal.filter((item: any) => item.check);
          this.selectedOptions = checkedData.map((item: any) => item.value);
          this.optionsAll = checkedData.length === newVal.length;
        },
        immediate: true,
      }
    },
    mounted() {        
        // 在控件载入时触发父级的selected方法
        this.$emit('selected', this.selectedOptions);
    },
    computed: {
      selected: {
        get() {
          return this.selectedOptions.length > this.options.length ? [''] : this.selectedOptions;
        },
        set(value: any[]) {
          this.selectedOptions = value;
        },
      },
    },
    methods: {
      handleOptionsAllChange(isAll: boolean) {
        this.optionsData.forEach((elm: any) => {
          elm.check = isAll;
        });
        this.selectedOptions = isAll ? this.optionsData.map((item: any) => item.value) : [];
        this.$emit('selected', this.selectedOptions);
      },
      handleTaskItemChange(item: any) {
        if (!item.check) {
          this.selectedOptions = this.selectedOptions.filter((value: any) => value !== item.value);
        } else {
          this.selectedOptions.push(item.value);
        }        
        this.optionsAll = this.selectedOptions.length === this.optionsData.length;
        this.$emit('selected', this.selectedOptions);
      },
    },
  });
  </script>
  <style lang="scss">
  //.select-checked {
    .el-select-dropdown.is-multiple .el-select-dropdown__item.selected::after {
      content: '';
    }
    .el-checkbox {
      width: 100%;
      padding: 0 30px;
      .el-checkbox__label {
        margin-left: 20px;
      }
    }
    .el-select-dropdown__item {
      padding: 0;
    }
    .el-tag__close,
    .el-icon-close {
      display: none;
    }
    .el-tag.el-tag--info {
      background: transparent;
      border: 0;
    }
  
    .el-select {
      .el-select__tags {
        flex-wrap: nowrap;
        overflow: hidden;
        white-space: nowrap;
      }
      .el-tag {
        background-color: #fff;
        border: none;
        color: #606266;
        font-size: 13px;
        padding-right: 0;
        & ~ .el-tag {
          margin-left: 0;
        }
        &:not(:last-child)::after {
          content: ',';
        }
      }
    }
  //}
  </style>

具体参考原博主文章,这里只对部分细节调整,记录一下

相关推荐
浮华似水20 分钟前
简洁之道 - React Hook Form
前端
正小安2 小时前
如何在微信小程序中实现分包加载和预下载
前端·微信小程序·小程序
_.Switch4 小时前
Python Web 应用中的 API 网关集成与优化
开发语言·前端·后端·python·架构·log4j
一路向前的月光4 小时前
Vue2中的监听和计算属性的区别
前端·javascript·vue.js
长路 ㅤ   4 小时前
vite学习教程06、vite.config.js配置
前端·vite配置·端口设置·本地开发
长路 ㅤ   4 小时前
vue-live2d看板娘集成方案设计使用教程
前端·javascript·vue.js·live2d
Fan_web4 小时前
jQuery——事件委托
开发语言·前端·javascript·css·jquery
安冬的码畜日常4 小时前
【CSS in Depth 2 精译_044】第七章 响应式设计概述
前端·css·css3·html5·响应式设计·响应式
莹雨潇潇5 小时前
Docker 快速入门(Ubuntu版)
java·前端·docker·容器
Jiaberrr5 小时前
Element UI教程:如何将Radio单选框的圆框改为方框
前端·javascript·vue.js·ui·elementui