HTML+JS+CSS 鼠标上下移动页面(非滚动条)

HTML+JS+CSS 鼠标上下移动页面(非滚动条)

html 复制代码
<!DOCTYPE html>
<html lang="en">

<head>
    <meta charset="UTF-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Mouse Drag Scroll - IE11 Compatible</title>
    <style>
        body {
            height: 2000px;
            /* 为了方便测试滚动效果,设置一个较长的页面 */
            /* 使用自定义的鼠标指针图片,这里假设图片名为 grab.cur */
            cursor: url('file:///C:/grab.cur'), default;
        }

        /* 定义按下鼠标左键时的鼠标指针样式 */
        body.dragging {
            /* 使用自定义的鼠标指针图片,这里假设图片名为 grabbing.cur */
            cursor: url('file:///C:/grabbing.cur'), default;
        }
    </style>
</head>

<body>
    <!-- 这里可以添加具体的页面内容 -->
    <h1>鼠标拖动滚动页面示例 - 兼容 IE11</h1>
    <script>
        // 定义变量来跟踪鼠标状态和初始位置
        var isDragging = false;
        var startY = 0;
        var initialScrollTop = 0;
        var body = document.body;

        // 获取文档对象
        var doc = document.documentElement || document.body;

        // 监听鼠标按下事件
        if (document.attachEvent) {
            document.attachEvent('onmousedown', mouseDownHandler);
        } else {
            document.addEventListener('mousedown', mouseDownHandler);
        }

        function mouseDownHandler(event) {
            event = event || window.event;
            // 只有按下鼠标左键时才进行操作
            if ((event.which && event.which === 1) || (event.button && event.button === 1)) {
                // 判断是否点击在滚动条上
                if (isClickOnScrollbar(event)) {
                    return; // 如果点击在滚动条上,不执行后续拖动逻辑
                }
                isDragging = true;
                // 记录鼠标按下时的垂直位置
                startY = event.clientY;
                // 记录当前页面的滚动位置
                initialScrollTop = doc.scrollTop;
                // 添加 dragging 类,改变鼠标指针样式
                addClass(body, 'dragging');
            }
        }

        // 判断鼠标是否点击在滚动条上
        function isClickOnScrollbar(event) {
            // 获取窗口宽度
            var windowWidth = window.innerWidth;
            // 获取文档元素宽度
            var docWidth = doc.clientWidth;
            // 计算滚动条宽度
            var scrollbarWidth = windowWidth - docWidth;
            // 判断鼠标点击的水平位置是否在滚动条范围内
            return event.clientX >= windowWidth - scrollbarWidth;
        }

        // 监听鼠标移动事件
        if (document.attachEvent) {
            document.attachEvent('onmousemove', mouseMoveHandler);
        } else {
            document.addEventListener('mousemove', mouseMoveHandler);
        }

        function mouseMoveHandler(event) {
            event = event || window.event;
            if (isDragging) {
                // 计算鼠标垂直移动的距离
                var deltaY = event.clientY - startY;
                // 根据鼠标移动距离来滚动页面
                doc.scrollTop = initialScrollTop - deltaY;
            }
        }

        // 监听鼠标松开事件
        if (document.attachEvent) {
            document.attachEvent('onmouseup', mouseUpHandler);
        } else {
            document.addEventListener('mouseup', mouseUpHandler);
        }

        function mouseUpHandler() {
            // 停止拖动状态
            isDragging = false;
            // 移除 dragging 类,恢复鼠标指针样式
            removeClass(body, 'dragging');
        }

        // 监听鼠标离开窗口事件,防止鼠标移出窗口后仍在拖动
        if (document.attachEvent) {
            document.attachEvent('onmouseleave', mouseLeaveHandler);
        } else {
            document.addEventListener('mouseleave', mouseLeaveHandler);
        }

        function mouseLeaveHandler() {
            isDragging = false;
            // 移除 dragging 类,恢复鼠标指针样式
            removeClass(body, 'dragging');
        }

        // 兼容 IE11 的添加类名函数
        function addClass(element, className) {
            if (element.className.indexOf(className) === -1) {
                if (element.className === '') {
                    element.className = className;
                } else {
                    element.className += ' ' + className;
                }
            }
        }

        // 兼容 IE11 的移除类名函数
        function removeClass(element, className) {
            var classNames = element.className.split(' ');
            var newClassNames = [];
            for (var i = 0; i < classNames.length; i++) {
                if (classNames[i] !== className) {
                    newClassNames.push(classNames[i]);
                }
            }
            element.className = newClassNames.join(' ');
        }
    </script>
</body>

</html>
相关推荐
海盐泡泡龟2 小时前
Javascript本地存储的方式有哪些?区别及应用场景?(含Deep Seek讲解)
开发语言·javascript·ecmascript
Elastic 中国社区官方博客4 小时前
JavaScript 中使用 Elasticsearch 的正确方式,第一部分
大数据·开发语言·javascript·数据库·elasticsearch·搜索引擎·全文检索
万物得其道者成4 小时前
从零开始创建一个 Next.js 项目并实现一个 TodoList 示例
开发语言·javascript·ecmascript
sunbyte5 小时前
50天50个小项目 (Vue3 + Tailwindcss V4) ✨ | Expanding Cards (展开式卡片)
javascript·vue.js·ecmascript
肠胃炎5 小时前
React Contxt详解
javascript·react.js·ecmascript
xx24065 小时前
React Native简介
javascript·react native·react.js
重生之后端学习6 小时前
02-前端Web开发(JS+Vue+Ajax)
java·开发语言·前端·javascript·vue.js
布鲁斯的快乐小屋6 小时前
axios的基本使用
javascript·ajax
来自星星的坤9 小时前
【Vue 3 + Vue Router 4】如何正确重置路由实例(resetRouter)——避免“VueRouter is not defined”错误
前端·javascript·vue.js
香蕉可乐荷包蛋13 小时前
浅入ES5、ES6(ES2015)、ES2023(ES14)版本对比,及使用建议---ES6就够用(个人觉得)
前端·javascript·es6