vue this.$refs.xxx获取dom注意事项 v-if v-for渲染的dom不能直接使用

 更新时间:2023年03月27日 08:58:47   作者:yehaocheng520  
这篇文章主要介绍了vue this.$refs.xxx获取dom注意事项 v-if v-for渲染的dom不能直接使用问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

最近被借调到其他部门写代码,嘿嘿,我是一块砖,哪里需要哪里搬……

今天遇到一个问题

vue项目:

v-for渲染的元素,如果内容超过一行,则右侧展示一个"更多"按钮,点击“更多”按钮后,再展示全部的数据。

在上家公司的时候,我也遇到过类似的要求,但是当时水平菜的底气硬,直接一个不会,愣是把需求改为了全部展示,今天遇到了这个问题,想着还是要解决一下,困难就是成长。

我的思路是

一行的内容高度是50px,我只需要判断渲染完成后,内容区的高度是否超过50px即可。

于是:html部分代码:

<div class="searchWrap">
  <div class="searchItem" v-for="(tag, tagIndex) in pinPaiList" :key="tagIndex">
    <div class="searchLeft">{{ tag.tagSortTitle }}</div>
    <div class="searchList" :ref="'tagParent' + tagIndex">
      <div v-for="(p, pIndex) in tag.items" :key="pIndex" class="listCls">
        <span @click="selectPP(tagIndex, pIndex)" :class="{ active: p.checked }">{{ p.value | filterStr }}</span></div></div>
    <div class="searchMore">
      <div>
       	<div class="el-icon-arrow-down" v-if="tag.moreFlag" @click="moreData(tagIndex)">更多</div>
        <div class="el-icon-circle-check" v-if="tag.selectMore" @click="sureSearch(tagIndex)"
          :class="{ active: tag.selectMore }" >确定</div>
        <div @click="changeType(tagIndex)" :class="{ active: tag.selectMore }" class="el-icon-circle-plus-outline">{{ !tag.selectMore ? "单选" : "多选" }}
        </div>
      </div>
    </div>
  </div>
</div>

上面代码的重点在于,我要在v-for渲染的dom元素上绑定ref,动态绑定属性的方式跟其他的动态绑定一致:

:ref="'tagParent' + tagIndex"

tag.moreFlag就是判断是否要展示更多按钮的关键,这个是根据dom元素的高度来判断的。

vue项目中获取dom元素的高度——this.$refs.xxx

vue项目中获取dom元素的高度可以通过this.$refs.xxx的方式来处理:

但是我在使用的过程中,this.$refs.xxx的方式获取dom,经常会出现undefined,也有少数情况是可以获取到的。

具体原因就是:如果是用v-if或者v-for渲染的dom元素,是不能直接使用this.$refs.xxx的方式来获取dom的,即便是放在this.$nextTick里面也是不行的。

具体原因可以查看下面的链接:

https://www.jb51.net/article/279018.htm

具体的解决方法就是

需要在获取到数据之后,再通过this.$nextTick的方式来处理:

我这边的处理方法如下:

...
//此处的this.pinPaiList是从上面的接口中获取到的数据,需要循环遍历,依次判断高度
this.pinPaiList && this.pinPaiList.forEach((p, pIndex) => {
   this.$nextTick(() => {
   //一定要注意通过this.$refs.xxx的方式获取到的dom要取index为0的一项,然后获取高度通过clientHight来获取,如果不是v-if或者v-for渲染的dom,则不需要加这个0。
     var height = this.$refs[`tagParent${pIndex}`][0].clientHeight;
     if (height > 50) {
       this.pinPaiList[pIndex].moreFlag = true;
       //如果判断出来某一项的高度超出一行,则延迟将此行的高度设置为一行,进而展示更多按钮,点击更多按钮时将height改为auto即可
       setTimeout(() => {
         this.$refs[`tagParent${pIndex}`][0].style.height = "35px";
       }, 0);
     } else {
       this.pinPaiList[pIndex].moreFlag = false;
     }
     this.$set(this.pinPaiList, pIndex, this.pinPaiList[pIndex]);
   });
 });
...
//点击更多按钮
moreData(index) {
  this.pinPaiList[index].moreFlag = false;
  this.$set(this.pinPaiList, index, this.pinPaiList[index]);//这个是vue2的缺陷,数据改变视图不渲染的解决方案
  setTimeout(() => {
    this.$refs[`tagParent${index}`][0].style.height = "auto";
  }, 0);
}

问题解决!!!

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • vite分目录打包及去掉默认的.gz 文件的操作方法

    vite分目录打包及去掉默认的.gz 文件的操作方法

    Vite在默认配置下会将资源打包至assets文件夹并添加哈希值,不同于Webpack的多文件夹存放方式,Vite只对public文件夹不进行打包处理,而Webpack不打包public和static文件夹,本文介绍vite分目录打包及去掉默认的.gz 文件的操作方法,感兴趣的朋友一起看看吧
    2024-09-09
  • Vue3项目中通过LuckySheet实现Excel在线编辑功能

    Vue3项目中通过LuckySheet实现Excel在线编辑功能

    在实现Excel文件导入时,领导要求实现在前端导入文件后,不调用后端的接口,而是直接显示excel文件的内容,等待用户修改完以后,再调用后端接口进行文件的提交,所以本文介绍了Vue3项目中通过LuckySheet实现Excel在线编辑,需要的朋友可以参考下
    2025-04-04
  • 基于vue封装一个带眼睛的密码子组件

    基于vue封装一个带眼睛的密码子组件

    这篇文章给大家介绍了基于vue封装一个带眼睛的密码子组件的方法,文章中有详细的代码讲解,对大家的学习或工作有一定的帮助,需要的朋友可以参考下
    2023-09-09
  • Vue 组件之间的通信方式详解

    Vue 组件之间的通信方式详解

    在 Vue.js 中,组件是构建应用程序的基本单位,然而,当你的应用程序变得复杂时,组件之间的通信变得至关重要,本文将介绍几种 Vue 组件之间通信的方式,帮助你更好地管理和组织代码,感兴趣的朋友一起看看吧
    2024-06-06
  • vue列表如何自动滚动到制定位置

    vue列表如何自动滚动到制定位置

    这篇文章主要介绍了vue列表如何自动滚动到制定位置问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-10-10
  • 使用Vue3和ApexCharts实现3D径向条形图的代码

    使用Vue3和ApexCharts实现3D径向条形图的代码

    径向条形图是一种用于可视化单一数据点及其与目标或理想值的关系的图表类型,它在显示进度、完成率或其他类似度量时非常有用,本文给大家介绍了使用Vue3和ApexCharts实现3D径向条形图,感兴趣的小伙伴可以参考阅读下
    2024-06-06
  • Vue关键字搜索功能实战小案例

    Vue关键字搜索功能实战小案例

    在vue项目中,搜索功能是我们经常需要使用的一个场景,下面这篇文章主要给大家介绍了关于Vue关键字搜索功能的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2023-06-06
  • 如何使用 vxe-table 实现左边是树右边是表格联动功能

    如何使用 vxe-table 实现左边是树右边是表格联动功能

    使用 vxe-table 来实现左边是树,右边是表格联动功能,当需要实现左右两侧联动时,表格 vxe-grid 配合分割模板 vxe-split 就很容易实现了,下面通过实例代码给大家介绍使用 vxe-table 来实现左边是树,右边是表格联动功能,感兴趣的朋友一起看看吧
    2025-03-03
  • vue移动端写的拖拽功能示例代码

    vue移动端写的拖拽功能示例代码

    这篇文章主要介绍了vue移动端写的拖拽功能,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-09-09
  • vue如何监听对象或者数组某个属性的变化详解

    vue如何监听对象或者数组某个属性的变化详解

    这篇文章主要给大家介绍了关于vue如何监听对象或者数组某个属性的变化,在Vue.js中可以通过watch监听属性变化并动态修改其他属性的值,watch通过监听器函数接收新旧值,实现属性间的数据联动,需要的朋友可以参考下
    2024-12-12

最新评论