树组件宽度自适应横向滚动 | 解决方案

不知不觉已经年底了,个人的 KPI 还远远没达成.... 一个有趣的解决方案,分享一下,也记录一下解决过程。本次代号: 滚动的树

背景

项目上一个有意思的问题,我们使用树组件时,一般都会有一个固定的父容器宽度,那么在层级过多时,树组件并不会自动增加宽度。这样,更深层级的目录就没法看到,也不支持横向滚动查看。

解决思路

方案一:调整布局

我们实现一个弹性容器组件,正如下面的 ·绿色·区域,拖动中间的基线,可改变左右内容的宽度,这样就能保证想要看全树组件更多层级时,把基线拖动到右边,留出更多的内容给到数组件区域就好了。实现上我就不重点讲述了,本文主要探讨第二种方案。

方案二: 横向滚动

其实第一种方案是非常合适了,不过面对复杂的应用场景,右边的区域可能是不允许被压缩的。那么我们只能在左边固定的区域内做文章。那么最简单的方式就是当层级超过 x 层时横向滚动。

实现这个方案的原理就是:层级超出容器最大层级后,不断增加树组件宽度

计算公式

<math xmlns="http://www.w3.org/1998/Math/MathML" display="block"> 树组件宽度 = 容器宽度 + 超出宽度 树组件宽度 = 容器宽度 + 超出宽度 </math>树组件宽度=容器宽度+超出宽度
<math xmlns="http://www.w3.org/1998/Math/MathML" display="block"> 树组件宽度 = 容器宽度 + (超出层级 ∗ 层级扩展宽度基数) 树组件宽度 = 容器宽度 + (超出层级 * 层级扩展宽度基数) </math>树组件宽度=容器宽度+(超出层级∗层级扩展宽度基数)
<math xmlns="http://www.w3.org/1998/Math/MathML" display="block"> 超出层级 = 最高层级 − 父容器最大层级 超出层级 = 最高层级 - 父容器最大层级 </math>超出层级=最高层级−父容器最大层级
<math xmlns="http://www.w3.org/1998/Math/MathML" display="block"> 父容器最大层级 = 容器宽度 / 层级缩进 父容器最大层级 = 容器宽度/层级缩进 </math>父容器最大层级=容器宽度/层级缩进

层级扩展宽度基数:超出一层级,要增加的宽度值(px)
最高层级:树组件展开的最深层级
父容器基础层级:父容器能容纳的最大层级
层级缩进: 树组件每一层的缩减宽度(px)

代码实现

上面的公式没看懂也没事,主要逻辑如下:

树组件宽度 = 容器宽度 + (超出层级 * 层级扩展宽度基数)

示例代码:vue + el-tree

页面布局

jsx 复制代码
<template>
    <div>
        <div class="left overflow-x-auto">
            <tree :style="treeStyles" />
        </div>
        <div class="right">
        </div>
    </div>
</template>

容器宽度
this.$el?.getBoundingClientRect()?.width

超出宽度 treeStyles

js 复制代码
const treeStyles = computed(() => {
    const containerWidth = this.$el?.getBoundingClientRect()?.width
    
    // el-tree 每一级的宽度(估算的)
    const step = 28
    
    // 如果所有宽度大于父容器宽度,则增加子容器宽度并触发横向滚动
    if (containerWidth && step * this.deepExpandLevel >= containerWidth) {
      const base = containerWidth / step
      const over = this.deepExpandLevel - Math.floor(base)
      
      return {
        width: `calc(100% + ${(over + 1) * 20}px)`,
        overflowX: 'auto',
      }
    }
    return {}
})

获取展开的最深层级 deepExpandLevel

js 复制代码
async getDeepExpandNodeLevel(node) {
  await this.$nextTick()
  let topNode = node
  let deepLevel = 0
  while (topNode.parent) {
    topNode = topNode.parent
  }

  walkTree(
    [topNode],
    node => {
      // 如果父元素都没有展开,就要没必要继续遍历其子元素了
      if (!node.expanded) return 'continue'
      if (node.expanded && node.level > deepLevel) {
        deepLevel = node.level
      }
    },
    { children: 'childNodes' },
  )

  this.deepExpandLevel = deepLevel
}

getDeepExpandNodeLevel 触发时机

  • 折叠节点
  • 展开节点
  • 点击节点(如果会展开)

实现效果

结尾

往后也会督促自己,往后也会多记录一些解决方案。如果大家有更好的方案欢迎也可以在评论区写出。

相关推荐
蟾宫曲3 小时前
在 Vue3 项目中实现计时器组件的使用(Vite+Vue3+Node+npm+Element-plus,附测试代码)
前端·npm·vue3·vite·element-plus·计时器
秋雨凉人心3 小时前
简单发布一个npm包
前端·javascript·webpack·npm·node.js
liuxin334455663 小时前
学籍管理系统:实现教育管理现代化
java·开发语言·前端·数据库·安全
qq13267029403 小时前
运行Zr.Admin项目(前端)
前端·vue2·zradmin前端·zradmin vue·运行zradmin·vue2版本zradmin
LCG元4 小时前
Vue.js组件开发-使用vue-pdf显示PDF
vue.js
魏时烟4 小时前
css文字折行以及双端对齐实现方式
前端·css
哥谭居民00015 小时前
将一个组件的propName属性与父组件中的variable变量进行双向绑定的vue3(组件传值)
javascript·vue.js·typescript·npm·node.js·css3
烟波人长安吖~5 小时前
【目标跟踪+人流计数+人流热图(Web界面)】基于YOLOV11+Vue+SpringBoot+Flask+MySQL
vue.js·pytorch·spring boot·深度学习·yolo·目标跟踪
踢足球的,程序猿5 小时前
Android native+html5的混合开发
javascript
2401_882726485 小时前
低代码配置式组态软件-BY组态
前端·物联网·低代码·前端框架·编辑器·web