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

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

小程序或者某些组件会有滚动穿透,但是如果是自己要造轮子,在某些场景下就需要去实现滚动穿透。比如我打开一个可滑动的弹框,在弹框上打开一个可滑动的组件,但是这个可滑动的组件在弹框的底部,高度不确定且又要让这个可滑动组件定位到弹框子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>
相关推荐
Mintopia18 小时前
Gemini-Essay-Writer 技术解析:基于 Gemini 的长文写作生成与质量控制实践
前端
蜡台18 小时前
Node Vue 项目开发常见问题解决
前端·javascript·vue.js·git·node
嘉琪00118 小时前
Day1 完整学习包(var/let/const + 作用域)——2026 0310
前端·javascript·学习
Moment19 小时前
2026 年 Next.js 站点的 SEO 优化指南
前端·javascript·面试
We་ct19 小时前
LeetCode 46. 全排列:深度解析+代码拆解
前端·数据结构·算法·leetcode·typescript·深度优先·回溯
problc19 小时前
前端预览pdf有哪些方案
前端·pdf
小小仙。19 小时前
IT自学第三十二天
服务器·前端·javascript
@大迁世界19 小时前
01.什么是 ReactJS?
前端·javascript·react.js·前端框架·ecmascript
l1t19 小时前
利用DuckDB 1.5的json和struct功能分析llama web-ui导出的对话json文件
前端·ui·json
猫头虎-前端技术19 小时前
这个项目需要Node 16,那个项目需要Node 18:如何解决多项目Node.js版本管理问题
前端·javascript·chrome·typescript·node.js·json·firefox