Table 滚动条始终停靠在可视区域的底部

news2024/11/27 11:35:12

1. 话题引入

存在这样一个场景:当页面尺寸发生变化时,希望滚动条能够随之动态调整,始终展示在 table 的可视区域的最下方,而不是整个 table 本身的最底部。

这种行为可以提升用户的使用体验,尤其是在处理大数据表格时,用户不需要滚动到整个页面的最底部才能看到滚动条,而是始终在当前可视区域操作。

效果如下,在 Table 未完全展示下的滚动条:

完全展示后,滚动条在 右侧 和 底部。

2. 如何实现?

这个实现方法比较困难,不能通过控制 Table 自带的滚动条或简单的 CSS 来实现效果。我尝试了很久,都不OK,因此决定手动实现一个与其本身风格一致的自定义滚动条。

需要解决的问题包括:

1、监听横向滚动条,实时捕获水平滚动状态并动态更新滚动条的位置和宽度。样式部分可以简单处理,不是重点。

2、监听视口可视区域高度变化,动态展示滚动条,使用 position 固定到可视区域底部。

3、实现拖拽滚动功能,通过鼠标按下滚动条并拖动,能够实现同步滚动的效果。

3. 解决实现

感兴趣的小伙伴可以自行实现下哦O。

HTML 结构

在 Table 下使用一个 div 实现滚动条效果,📢 外层的 div 是必须的,负责实时的定位效果。

<div class="custom-container" style={{ position: "relative" }}>
  <ElTable ref={tableRef} {...props} {...attrs} v-slots={slots}>
    {slots.default && slots.default()}
  </ElTable>
  <div
    class="custom-scrollbar"
    style={{
    zIndex: 999,
    width: `${scrollBarWidthRef.value}px`,
    height: `${SCROLLBAR_HEIGHT}px`,
    position: "absolute",
    top: `${scrollBarTopRef.value}px`,
    left: `${scrollBarLeftRef.value}px`
    }}
    onMousedown={onMouseDown}
    ></div>
</div>

第一个问题:滚动条宽度和位置处理

难点:监听横向滚动条变化

const getScrollBarWidth = () => {
  // 获取横向滚动条元素
  const scrollWrapper = tableRef.value?.$el.querySelector(".el-scrollbar__wrap");
  // 获取 table 元素大小及其相对位置
  const tableRect = tableRef.value?.$el.getBoundingClientRect();
  
  // ---- 获取滚动条的宽度(源码) ----
  const GAP = 4;
  const offsetWidth = scrollWrapper.offsetWidth - GAP;
  const originalWidth = offsetWidth ** 2 / scrollWrapper.scrollWidth;
  const scrollbarWidth = Math.max(originalWidth, SCROLLBAR_MIN_WIDTH);
  scrollBarWidthRef.value = scrollbarWidth;
  return {
    tableRect, // 表格大小
    scrollbarWidth // 滚动条宽度
  };
};

接下来设置滚动条的位置

// 设置滚动条宽度
const setScrollBarWidth = () => {
  const { scrollbarWidth } = getScrollBarWidth();
  scrollBarWidthRef.value = scrollbarWidth;
};

// 设置滚动条高度
const setScrollBarTop = () => {
  const { tableRect } = getScrollBarWidth();
  scrollBarTopRef.value =
    window.innerHeight - tableRect.top - SCROLLBAR_HEIGHT > tableRect.height
    ? tableRect.height - SCROLLBAR_HEIGHT
    : window.innerHeight - tableRect.top - SCROLLBAR_HEIGHT;
};

// 监听滚动的同时设置 Left
const handleScroll = e => {
  const { tableRect } = getScrollBarWidth();
  setScrollBarTop();
  setScrollBarWidth();
  if (e.target) {
    // 比例计算
    scrollBarLeftRef.value = (e.target.scrollLeft / e.target.scrollWidth) * tableRect.width;
  }
};

const handleWindowScroll = () => {
  setScrollBarTop();
};

既然方法都写好啦,那就开始调用。

onMounted(() => {
  // 初始化时执行
  // 为什么在 setTimeout 呢?避免过早地执行这些方法,它们依赖于表格 DOM 完全渲染完成后的尺寸信息。
  setTimeout(() => {
    setScrollBarWidth();
    setScrollBarTop();
  }, 0);

  // 确保 DOM 已完全更新
  nextTick(() => {
    if (tableRef.value) {
      scrollWrapperRef.value = tableRef.value?.$el.querySelector(".el-scrollbar__wrap");
      bodyWrapperRef.value = tableRef.value?.$el.querySelector(".el-table__body-wrapper");
      if (scrollWrapperRef.value) {
        scrollWrapperRef.value.addEventListener("scroll", handleScroll);
      }
      if (bodyWrapperRef.value) {
        window.addEventListener("scroll", handleWindowScroll);
      }
    }
  });
});

注意:当窗口尺寸变化时,需要重新计算滚动条的高度和宽度。

const handleResize = () => {
  setScrollBarWidth();
  setScrollBarTop();
};
window.addEventListener("resize", handleResize);

OK,到此滚动条的位置完成,接下来处理滚动条的拖拽。

这个就比较简单了,监听鼠标抬起,移动的事件,设置滚动距离。

let isDragging = false;
let startX = 0;
let startLeft = 0;
let rafId: number | null = null;

// 鼠标抬起
const onMouseUp = () => {
  isDragging = false;
  document.removeEventListener("mousemove", onMouseMove);
  document.removeEventListener("mouseup", onMouseUp);
};
// 鼠标移动
const onMouseMove = (e: MouseEvent) => {
  if (!isDragging) return;
  if (rafId !== null) {
    cancelAnimationFrame(rafId);
  }
  rafId = requestAnimationFrame(() => {
    const moveX = e.clientX - startX; // 计算拖动距离
    const newLeft = startLeft + moveX; // 计算滚动条的新位置
    scrollBarLeftRef.value = Math.max(
      0,
      Math.min(newLeft, tableRef.value?.$el.scrollWidth - scrollBarWidthRef.value)
    );
    // 更新表格滚动位置
    if (scrollWrapperRef.value) {
      scrollWrapperRef.value.scrollLeft = (scrollBarLeftRef.value / tableRef.value?.$el.scrollWidth) * scrollWrapperRef.value.scrollWidth;
    }
  });
};
// 鼠标按下
const onMouseDown = (e: MouseEvent) => {
  isDragging = true;
  startX = e.clientX;
  startLeft = scrollBarLeftRef.value; // 当前滚动条的位置
  document.addEventListener("mousemove", onMouseMove);
  document.addEventListener("mouseup", onMouseUp);
};

按下滚动条可滑动 Table。

最后清空一下监听的事件。

onBeforeUnmount(() => {
  if (scrollWrapperRef.value) {
    scrollWrapperRef.value.removeEventListener("scroll", handleScroll);
  }
  window.removeEventListener("scroll", handleWindowScroll);
  window.removeEventListener("resize", handleResize);
});

自此就成功实现一个符合要求的滚动条 👏🏻 👏🏻 👏🏻。 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2248380.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【漏洞复现】CVE-2020-13925

漏洞信息 NVD - CVE-2020-13925 Similar to CVE-2020-1956, Kylin has one more restful API which concatenates the API inputs into OS commands and then executes them on the server; while the reported API misses necessary input validation, which causes the hac…

基于Springboot的心灵治愈交流平台系统的设计与实现

基于Springboot的心灵治愈交流平台系统 介绍 基于Springboot的心灵治愈交流平台系统&#xff0c;后端框架使用Springboot和mybatis&#xff0c;前端框架使用Vuehrml&#xff0c;数据库使用mysql&#xff0c;使用B/S架构实现前台用户系统和后台管理员系统&#xff0c;和不同级别…

快速理解微服务中Gateway的概念

一.基本概念 定义&#xff1a; 在微服务架构中&#xff0c;Spring Cloud Gateway 是一个用于API网关的框架&#xff0c;它是一个基于 Spring Framework 的高效、可扩展的路由器和反向代理&#xff0c;它能够将外部请求转发到适当的微服务&#xff0c;并提供一些与请求处理相关…

Java【多线程】(1)进程与线程

目录 1.前言 2.正文 2.1什么是进程 2.2PCB&#xff08;进程控制块&#xff09; 2.2.1进程id 2.2.2内存指针 2.2.3文件描述符表 2.2.4进程状态 2.2.4.1就绪状态 2.2.4.2阻塞状态 2.2.5进程优先级 2.2.6进程上下文 2.2.7进程的记账信息 2.3CPU操作进程的方法 2.4什…

计算机毕业设计Python+大模型美食推荐系统 美食可视化 美食数据分析大屏 美食爬虫 美团爬虫 机器学习 大数据毕业设计 Django Vue.js

温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 作者简介&#xff1a;Java领…

华为鸿蒙内核成为HarmonyOS NEXT流畅安全新基座

HDC2024华为重磅发布全自研操作系统内核—鸿蒙内核&#xff0c;鸿蒙内核替换Linux内核成为HarmonyOS NEXT稳定流畅新基座。鸿蒙内核具备更弹性、更流畅、更安全三大特征&#xff0c;性能超越Linux内核10.7%。 鸿蒙内核更弹性&#xff1a;元OS架构&#xff0c;性能安全双收益 万…

kafka生产者和消费者命令的使用

kafka-console-producer.sh 生产数据 # 发送信息 指定topic即可 kafka-console-producer.sh \ --bootstrap-server bigdata01:9092 \ --topic topicA # 主题# 进程 29124 ConsoleProducer kafka-console-consumer.sh 消费数据 # 消费数据 kafka-console-consumer.sh \ --boo…

构造函数的相关

文章目录 一、构造函数 今天我们要来讲解类的默认成员函数之一的构造函数。 一、构造函数 构造函数是特殊的成员函数&#xff0c;需要注意的是&#xff0c;构造函数虽然名称叫构造&#xff0c;但是构造函数的主要任务并不是开空间创建对象(我们常使用的局部对象是栈帧创建时&…

云服务器部署WebSocket项目

WebSocket是一种在单个TCP连接上进行全双工通信的协议&#xff0c;其设计的目的是在Web浏览器和Web服务器之间进行实时通信&#xff08;实时Web&#xff09; WebSocket协议的优点包括&#xff1a; 1. 更高效的网络利用率&#xff1a;与HTTP相比&#xff0c;WebSocket的握手只…

计算机网络八股整理(一)

计算机网络八股文整理 一&#xff1a;网络模型 1&#xff1a;网络osi模型和tcp/ip模型分别介绍一下 osi模型是国际标准的网络模型&#xff0c;它由七层组成&#xff0c;从上到下分别是&#xff1a;应用层&#xff0c;表示层&#xff0c;会话层&#xff0c;传输层&#xff0c;…

【Qt】控件7

1.QTextEdit的简单使用 使用简单的QTextEdit,获取到的内容显示到标签上 使用textChanged信号 在槽函数中需要获取QTextEdit的内容&#xff0c;对应操作是&#xff1a; QString curorui->textEdit->toPlainText();然后显示到标签上&#xff0c;对应操作是&#xff1a; …

【博主推荐】C#的winfrom应用中datagridview常见问题及解决方案汇总

文章目录 1.datagridview绘制出现鼠标悬浮数据变空白2.datagridview在每列前动态添加序号2.1 加载数据集完成后绘制序号2.2 RowPostPaint事件绘制 3.datagridview改变行样式4.datagridview后台修改指定列数据5.datagridview固定某个列宽6.datagridview某个列的显示隐藏7.datagr…

AI智能体崛起:从“工具”到“助手”的进化之路

目录 AI智能体的崛起 AI智能体的定义与决策模型 AI智能体的特点与优势 AI智能体的应用与类型 面临的挑战 未来展望 近年来&#xff0c;人工智能领域的焦点正从传统的聊天机器人&#xff08;Chat Bot&#xff09;快速转向更具潜力的AI智能体&#xff08;AI Agent&#xff…

【计网】自定义协议与序列化(一) —— Socket封装于服务器端改写

&#x1f30e; 应用层自定义协议与序列化 文章目录&#xff1a; Tcp协议Socket编程 应用层简介 序列化和反序列化       重新理解read/write/recv/send及tcp的全双工       Socket封装       服务器端改写 &#x1f680;应用层简介 我们程序员写的一个个解决…

鸿蒙动画开发07——粒子动画

1、概 述 粒子动画是在一定范围内随机生成的大量粒子产生运动而组成的动画。 动画元素是一个个粒子&#xff0c;这些粒子可以是圆点、图片。我们可以通过对粒子在颜色、透明度、大小、速度、加速度、自旋角度等维度变化做动画&#xff0c;来营造一种氛围感&#xff0c;比如下…

C语言学习 12(指针学习1)

一.内存和地址 1.内存 在讲内存和地址之前&#xff0c;我们想有个⽣活中的案例&#xff1a; 假设有⼀栋宿舍楼&#xff0c;把你放在楼⾥&#xff0c;楼上有100个房间&#xff0c;但是房间没有编号&#xff0c;你的⼀个朋友来找你玩&#xff0c;如果想找到你&#xff0c;就得挨…

【pyspark学习从入门到精通19】机器学习库_2

目录 估计器 分类 回归 聚类 管道 估计器 估计器可以被看作是需要估算的统计模型&#xff0c;以便对您的观测值进行预测或分类。 如果从抽象的 Estimator 类派生&#xff0c;新模型必须实现 .fit(...) 方法&#xff0c;该方法根据在 DataFrame 中找到的数据以及一些默认或…

结构方程模型(SEM)入门到精通:lavaan VS piecewiseSEM、全局估计/局域估计;潜变量分析、复合变量分析、贝叶斯SEM在生态学领域应用

目录 第一章 夯实基础 R/Rstudio简介及入门 第二章 结构方程模型&#xff08;SEM&#xff09;介绍 第三章 R语言SEM分析入门&#xff1a;lavaan VS piecewiseSEM 第四章 SEM全局估计&#xff08;lavaan&#xff09;在生态学领域高阶应用 第五章 SEM潜变量分析在生态学领域…

JQuery -- 第九课

文章目录 前言一、JQuery是什么&#xff1f;二、JQuery的使用步骤1.引入2.书写位置3. 表示方法 三、JQuery选择器1.层级选择器2. 筛选选择器3. 排他思想4. 精品展示 四、jQuery样式操作1. 修改样式2.类操作1. 添加2. 移除3. 切换 五、jQuery动画1. 显示和隐藏2. 滑动1. slide2.…

无人机探测:光电侦测核心技术算法详解!

核心技术 双光谱探测跟踪&#xff1a; 可见光成像技术&#xff1a;利用无人机表面反射的自然光或主动光源照射下的反射光&#xff0c;通过高灵敏度相机捕捉图像。该技术适用于日间晴朗天气下的无人机探测&#xff0c;具有直观、易于识别目标的特点。 红外成像技术&#xff1…