稀土掘金技术社区 01月04日
前端:“这需求是认真的吗?” —— el-select 的动态宽度解决方案
index_new5.html
../../../zaker_core/zaker_tpl_static/wap/tpl_guoji1.html

 

本文介绍了如何解决Element UI的el-select组件在内容过长时显示不全的问题。通常可以通过加tooltip或切割label值解决,但客户要求选择框宽度动态增加。作者通过分析el-select的结构,发现其宽度由内部input决定。解决方案是在input同级增加一个元素,利用el-select的prefix插槽,将用户选中的内容放入prefix,并调整样式,使prefix撑开选择框宽度。同时,处理了未选择内容时的最小宽度和图标对齐问题,最终实现了el-select宽度随内容动态变化的效果。

💡 **问题背景:** 客户要求el-select选择框宽度根据内容动态调整,以解决内容过长显示不全的问题,传统tooltip和切割label方案不被接受。

🔍 **结构分析:** 通过控制台分析,el-select的宽度由内部input元素决定,而input宽度受同级元素内容影响。

🛠️ **解决方案:** 利用el-select的prefix插槽,将选中的内容放入prefix,并通过调整prefix和input的定位,实现选择框宽度随内容自适应增加。

🎨 **细节调整:** 通过CSS样式调整prefix的位置和隐藏,使其与选择框内容重叠,并处理了未选择内容时的最小宽度以及图标对齐问题。

✅ **最终效果:** 最终实现了el-select选择框宽度随选项内容长度动态变化,解决了内容显示不全的问题。

原创 秋天的一阵风 2025-01-04 09:01 重庆

点击关注公众号,“技术干货”及时达!

点击关注公众号,“技术干货” 及时达!

Hello~大家好。我是秋天的一阵风 ~

前言

最近我遇到了一个神奇的需求,客户要求对 「el-select」「宽度」 进行动态设置。

简单来说,就是我们公司有一些选择框,展示的内容像“中华人民共和国/广西壮族自治区/南宁市/西乡塘区”这么长,一不小心就会内容超长,显示不全。详情请看下面动图:

一般来说,想解决「内容展示不全」的问题,有几种方法。

第一种:给选择框加个tooltip效果,在鼠标悬浮时展示完整内容。

第二种:对用户选择label值进行切割,只展示最后一层内容。

但是我们的客户对这两种方案都不接受,要求选择的时候让select选择框的宽度动态增加。

有什么办法呢?客户就是上帝,必须满足,他们说什么就是什么,所以我们只能开动脑筋,动手解决。

思路

我们打开控制台,来侦察一下el-select的结构,发现它是一个el-input--suffixdiv包裹着一个input,如下图所示。

内层input的宽度是100%,外层div的宽度是由这个内层input决定的。也就是说,内层input的宽度如果动态增加,外层div的宽度也会随之增加。那么问题来了,如何将内层input的宽度动态增加呢?

tips:

如果你对width的100%和auto有什么区别感兴趣,可以点击查看我之前的文章

探究 width:100%与width:auto区别

解决方案

为了让我们的el-select宽度能够跟着内容走,我们可以在内层input同级别增加一个元素,内容就是用户选中的内容。内容越多,它就像一个胃口很大的小朋友,把外层div的宽度撑开。下面来看图示例 :

借助prefix

幸运的是,el-select本身有一个prefix的插槽选项,我们可以借助这个选项实现:

我们添加一个prefix的插槽,再把prefix的定位改成relative,并且把input的定位改成绝对定位absolute。最后将prefix的内容改成我们的选项内容。看看现在的效果:


<template>
  <div>
    <el-select class="autoWidth" v-model="value" placeholder="请选择">
      <template slot="prefix">
        {{optionLabel}}
      </template>
      <el-option
        v-for="item in options"
        :key="item.value"
        :label="item.label"
        :value="item.value"
      >

      </el-option>
    </el-select>
  </div>

</template>
<script>
export default {
  data() {
    return {
      options: [
        {
          value: "选项1",
          label: "中华人民共和国/广东省/深圳市/福田区",
        },
        {
          value: "选项2",
          label: "中华人民共和国/广西壮族自治区/南宁市/西乡塘区",
        },
        {
          value: "选项3",
          label: "中华人民共和国/北京市",
        },
        {
          value: "选项4",
          label: "中华人民共和国/台湾省",
        },
        {
          value: "选项5",
          label: "中华人民共和国/香港特别行政区",
        },
      ],
      value: "",
    };
  },
  computed: {
    optionLabel() {
      return (this.options.find((item) => item.value === this.value) || {})
        .label;
    },
  },
};
</
script>
<style lang="scss" scoped>
::v-deep .autoWidth .el-input__prefix {
  position: relative;
}
::v-deep .autoWidth input {
  position: absolute;
}
</style>

细节调整

现在el-select已经可以根据选项label的内容长短动态增加宽度了,但是我们还需要继续处理一下细节部分,将prefix的内容调整到和select框中的内容位置重叠,并且将它隐藏。看看现在的效果

::v-deep .autoWidth .el-input__prefix {
  position: relative;
  box-sizing: border-box;
  border: 1px solid #fff;
  padding: 0 30px;
  height: 40px;
  line-height: 40px;
  left: 0px;
  visibility: hidden;
}

调整初始化效果(用户未选择内容)

目前已经基本实现了效果了,还有最后一个问题,当用户没有选择内容的时候,select的宽度是“没有”的,如下图所示。

所以我们还得给他加上一个最小宽度

我们加上最小宽度以后,发现这个select的图标又没对齐,这是因为我们在重写.el-input__prefix样式的时候设置了padding: 0 30px,当用户没有选择内容的时候,select的图标应该是默认位置,我们需要继续调整代码,最后效果如下图所示:

完整代码

最后附上完整代码:


<template>
  <div>
    <el-select
      class="autoWidth"
      :class="{ 'has-content': optionLabel }"
      v-model="value"
      placeholder="请选择"
      clearable
    >

      <template slot="prefix">
        {{ optionLabel }}
      </template>
      <el-option
        v-for="item in options"
        :key="item.value"
        :label="item.label"
        :value="item.value"
      >

      </el-option>
    </el-select>
  </div>

</template>
<script>
export default {
  data() {
    return {
      options: [
        {
          value: "选项1",
          label: "中华人民共和国/广东省/深圳市/福田区",
        },
        {
          value: "选项2",
          label: "中华人民共和国/广西壮族自治区/南宁市/西乡塘区",
        },
        {
          value: "选项3",
          label: "中华人民共和国/北京市",
        },
        {
          value: "选项4",
          label: "中华人民共和国/台湾省",
        },
        {
          value: "选项5",
          label: "中华人民共和国/香港特别行政区",
        },
      ],
      value: "",
    };
  },
  computed: {
    optionLabel() {
      return (this.options.find((item) => item.value === this.value) || {})
        .label;
    },
  },
};
</
script>
<style lang="scss" scoped>
.autoWidth {
  min-width: 180px;
}
::v-deep .autoWidth .el-input__prefix {
  position: relative;
  box-sizing: border-box;
  border: 1px solid #fff;
  padding: 0 30px;
  height: 40px;
  line-height: 40px;
  left: 0px;
  visibility: hidden;
}
::v-deep .autoWidth input {
  position: absolute;
}
.autoWidth {
  // 当.has-content存在时设置样式
  &.has-content {
    ::v-deep .el-input__suffix {
      right: 5px;
    }
  }
  // 当.has-content不存在时的默认或备选样式
  &:not(.has-content) {
    ::v-deep .el-input__suffix {
      right: -55px;
    }
  }
}
</style>

点击关注公众号,“技术干货” 及时达!

阅读原文

跳转微信打开

Fish AI Reader

Fish AI Reader

AI辅助创作,多种专业模板,深度分析,高质量内容生成。从观点提取到深度思考,FishAI为您提供全方位的创作支持。新版本引入自定义参数,让您的创作更加个性化和精准。

FishAI

FishAI

鱼阅,AI 时代的下一个智能信息助手,助你摆脱信息焦虑

联系邮箱 441953276@qq.com

相关标签

el-select 动态宽度 Element UI 前端开发 CSS
相关文章