实现滚动穿透,保留原本元素的事件,仅实现滚动穿透

网络上都是一堆解决滚动穿透的,却没有一条是实现滚动穿透的方案。

小程序或者某些组件会有滚动穿透,但是如果是自己要造轮子,在某些场景下就需要去实现滚动穿透。比如我打开一个可滑动的弹框,在弹框上打开一个可滑动的组件,但是这个可滑动的组件在弹框的底部,高度不确定且又要让这个可滑动组件定位到弹框子item下面,这样子可能会导致这个可滑动组件的位置被底部遮挡或位置无法调整,无法滚动父级实现组件位置的调整,造成体验低下。

这里滚动穿透的核心就是:

1、position: fixed;保证元素覆盖父级

2、pointer-events: none;取消这个元素的事件实现事件穿透,但因为pointer-events没有单独指定某个事件,所以要js支持

3、监听鼠标滚轮事件,滚动时才添加pointer-events: none;,不滚动就取消添加。

简单代码示例:

html 复制代码
<!--
 * @轮子的作者: 轮子哥
 * @Date: 2023-12-15 10:14:35
 * @LastEditTime: 2023-12-15 14:43:16
-->
<div class="container">
    <div class="item">我是父级虚拟滑动列表</div>
    <div class="item">我是父级虚拟滑动列表</div>
    <div class="item">
        <div class="mask" onclick="writess()">
            实现滚动穿透(遮罩层),尝试在此遮罩区域滚动试试。
        </div>
        <div class="maskOv">
            <div class="maskItem">我是根据父级item定位的模拟滑动下拉框</div>
            <div class="maskItem">我是根据父级item定位的模拟滑动下拉框</div>
            <div class="maskItem">我是根据父级item定位的模拟滑动下拉框</div>
            <div class="maskItem">我是根据父级item定位的模拟滑动下拉框</div>
            <div class="maskItem">我是根据父级item定位的模拟滑动下拉框</div>
            <div class="maskItem">我是根据父级item定位的模拟滑动下拉框</div>
            <div class="maskItem">我是根据父级item定位的模拟滑动下拉框</div>
            <div class="maskItem">我是根据父级item定位的模拟滑动下拉框</div>
            <div class="maskItem">我是根据父级item定位的模拟滑动下拉框</div>
        </div>
    </div>
    <div class="item">我是父级虚拟滑动列表</div>
    <div class="item">我是父级虚拟滑动列表</div>
    <div class="item">我是父级虚拟滑动列表</div>
    <div class="item">我是父级虚拟滑动列表</div>
    <div class="item">我是父级虚拟滑动列表</div>
    <div class="item">我是父级虚拟滑动列表</div>
    <div class="item">我是父级虚拟滑动列表</div>
    <div class="item">我是父级虚拟滑动列表</div>
</div>
<div class="writeDom"></div>

<style>
    .container {
        position: relative;
        overflow: scroll;
        height: 500px;
        background: red;
    }

    .mask {
        position: fixed;
        top: 0;
        left: 0;
        width: 50%;
        height: 100%;
        background-color: #32323229;
        z-index: 1;
        text-align: center;
        line-height: 500px;
        color: blue;
    }

    .scrolled {
        pointer-events: none;
    }

    .item {
        position: relative;
        height: 150px;
        background-color: aqua;
        margin: 10px;
    }

    .maskOv {
        overflow: auto;
        height: 200px;
        position: absolute;
        z-index: 2;
        width: 400px;
    }

    .maskItem {
        position: relative;
        height: 50px;
        background-color: rgb(207, 233, 62);
        margin: 10px;
    }
</style>
<script>
    var mask = document.querySelector(".mask");
    let timer = null;

    //监听滚动
    mask.addEventListener("wheel", function (event) {
        mask.classList.add("scrolled");
        if (timer) {
            clearTimeout(timer);
            timer = setTimeout(() => {
                timer = null;
                mask.classList.remove("scrolled");
            }, 300)
        } else {
            timer = setTimeout(() => {
                timer = null;
                mask.classList.remove("scrolled");
            }, 300)
        }
    });


    let writeDom = document.querySelector(".writeDom");
    function writess() {
        console.log("点击触发,保留其它事件,仅实现滚动穿透")
        writeDom.innerHTML = "点击触发,保留其它事件,仅实现滚动穿透" + new Date()
    }

</script>
相关推荐
lichenyang45313 小时前
Docker 学习笔记(一):为什么需要镜像、容器和仓库?
前端
kyriewen13 小时前
别再对着 TypeScript 报错发呆了:我把 10 个最常见的红色波浪线翻译成了人话
前端·javascript·typescript
IT_陈寒13 小时前
SpringBoot自动配置的坑,我的API突然就404了
前端·人工智能·后端
free3514 小时前
从 0 实现一个 Tiny JavaScript VM:项目架构拆解
javascript
奇奇怪怪的14 小时前
Embedding 模型 10+ 横向评测
前端
陈广亮14 小时前
Monorepo 从 0 到 1 实操指南 2026 版:pnpm catalogs + Turborepo 2.x + changesets 全链路
前端
子兮曰14 小时前
OpenMontage 深度解剖:你的 AI 编程助手,其实是个视频工作室
前端·后端·ai编程
敲代码的鱼14 小时前
PDF 预览与签名批注写回 支持安卓 iOS 鸿蒙 UTS插件
android·前端·ios
子兮曰14 小时前
前端工具链的「Rust 化」:一场没有赢家的军备竞赛?
前端·后端·rust