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>

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

相关推荐
kyriewen8 小时前
我手写了一个 EventEmitter,面试官追问了 6 个问题——第 4 个我没答上来
前端·javascript·面试
IT_陈寒9 小时前
Java的Date类又坑了我一次,改用时间戳真香
前端·人工智能·后端
小林攻城狮9 小时前
使用 Transport 节流解决 Vercel AI SDK 流式渲染卡死问题
前端·react.js
前端缘梦9 小时前
告别 TS 运行时类型漏洞!Zod 完整入门实战教程(前端 / 全栈必备)
前端·react.js·全栈
the_answer10 小时前
Webpack vs Vite 深度对比分析
前端·webpack
转转技术团队10 小时前
验证码识别实战:前端不写页面,改训模型了?
前端
MomentYY10 小时前
Temperature:AI 的“脑洞旋钮”
前端·llm·ai编程
远航_10 小时前
OpenSpec 完整详细介绍
前端·后端
召钱熏10 小时前
状态枚举正确≠渲染正确:一个语音按钮的状态机边界修复实录
android·前端
SkyWalking中文站10 小时前
认识 Horizon UI · 1/17:SkyWalking 新一代可观测性控制台
运维·前端·监控