通过 JavaScript 获取页面上的鼠标位置

May 12th, 2010 Add Comment
image

用 JS 计算鼠标在页面上的位置并非难事, 只要把握好各浏览器的区别就可以轻易算出鼠标位置. (这是 DEMO)

视窗 (浏览器可视窗口) 就像是页面上的掩板开了一个洞. 滚动条可以改变页面和视窗之间的偏移量, 从而可以通过视窗看到页面的各个位置.
鼠标在页面上的位置 = 页面和视窗之间的偏移量 + 鼠标在视窗中的位置

右图中的 cursorX 和 cursorY 分别是鼠标在视窗中的横向和纵向位置, scrollY 是页面和视窗之间的纵向距离. 当然, 当视窗宽度小于页面宽度的时候, 还会存在 scrollX. 那么鼠标在页面上的位置就是: (scrollX+cursorX, scrollY+cursorY)

IE 以外的浏览器 (本人测试过 Firefox 3.6, Opera 10.10, Chrome 4.1 和 Safari 4.0.4) 均可以通过 pageXOffset 和 pageYOffset 来获取页面和视窗间的横纵距离. 但 IE (本人测试过 IE6, IE7, IE8) 只能通过滚动位移来获取页面和视窗间的距离, 并存在一到两个像素的偏离.

以下是获取鼠标在页面上位置的代码实现. (这是 DEMO)

/**
 * 获取鼠标在页面上的位置
 * @param ev		触发的事件
 * @return			x:鼠标在页面上的横向位置, y:鼠标在页面上的纵向位置
 */
function getMousePoint(ev) {
	// 定义鼠标在视窗中的位置
	var point = {
		x:0,
		y:0
	};
 
	// 如果浏览器支持 pageYOffset, 通过 pageXOffset 和 pageYOffset 获取页面和视窗之间的距离
	if(typeof window.pageYOffset != 'undefined') {
		point.x = window.pageXOffset;
		point.y = window.pageYOffset;
	}
	// 如果浏览器支持 compatMode, 并且指定了 DOCTYPE, 通过 documentElement 获取滚动距离作为页面和视窗间的距离
	// IE 中, 当页面指定 DOCTYPE, compatMode 的值是 CSS1Compat, 否则 compatMode 的值是 BackCompat
	else if(typeof document.compatMode != 'undefined' && document.compatMode != 'BackCompat') {
		point.x = document.documentElement.scrollLeft;
		point.y = document.documentElement.scrollTop;
	}
	// 如果浏览器支持 document.body, 可以通过 document.body 来获取滚动高度
	else if(typeof document.body != 'undefined') {
		point.x = document.body.scrollLeft;
		point.y = document.body.scrollTop;
	}
 
	// 加上鼠标在视窗中的位置
	point.x += ev.clientX;
	point.y += ev.clientY;
 
	// 返回鼠标在视窗中的位置
	return point;
}

记得以前我写过一个减速滚动置顶的 JavaScript 方法, 也是通过计算视窗和页面高度来实现的, 其计算方法与本文的方法不同小异, 也可以搬过来用.

这几个月来, 专注前端项目的开发, 也算小有收获, 最近我会将一些积累作成文章分享于众. 以此作为铺垫, 由浅入深, 希望可以带出一些有用的内容.

声明: 本文采用 BY-NC-SA 协议进行授权. 转载请注明转自: 通过 JavaScript 获取页面上的鼠标位置

  1. http://1.gravatar.com/avatar/b2986defdd28dcaa196e317a94d82e31?s=32&d=http%3A%2F%2F1.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D32&r=G

    IE 以外的浏览器 (本人测试过 Firefox 3.6, Opera 10.10, Chrome 4.1 和 Safari 4.0.4) 均可以通过 pageXOffset 和 pageYOffset 来获取页面和视窗间的横纵距离.

    pageXOffset是早期Netscape浏览器才有的,现在浏览器都不支持的

  2. http://0.gravatar.com/avatar/a56b2e1c61ae0f27f3cd916d1a807b86?s=32&d=http%3A%2F%2F0.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D32&r=G

    u.u no se japones o coreano pero kiero descargar minimal y tengo mucho lsd y ganjah desde colombia u.u

  3. http://0.gravatar.com/avatar/024a121d6bd8cc17ff2af05568a5a0c9?s=32&d=http%3A%2F%2F0.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D32&r=G

    ie6不好用,横纵坐标多2像素

  4. http://0.gravatar.com/avatar/ac555be5964e168cba0475c0469fd98f?s=32&d=http%3A%2F%2F0.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D32&r=G

    呵呵,真好,我写的鼠标动半天数字才变一点点~

  5. http://1.gravatar.com/avatar/bb6eeca6c19f5f11c8aaaa061ccde5e7?s=32&d=http%3A%2F%2F1.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D32&r=G
  6. http://0.gravatar.com/avatar/490cf262668eebb0f0f1a50d9d48d702?s=32&d=http%3A%2F%2F0.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D32&r=G

    k4y :

    Leeiio :
    去了阿里的一般博客都会鲜有更新了。。

    确实是,群里也N久不说话啦 :P

    哪个群? 我群太多, 谈技术的我一般都发言 (我上 Q 都不谈技术).

    @Jutoy
    高个毛, 记录一下, 以后自己好找.

    @Platero
    那已经是 4, 5 年前的事了. 感叹道: 年轻真好.

    @ZhouQi
    难道是那本很厚的犀牛封面的绿色的 JavaScript 书籍?

  7. http://1.gravatar.com/avatar/fe17613a53cc59eea0a1079474f14a3a?s=32&d=http%3A%2F%2F1.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D32&r=G

    好的!最近也在自己动手搞代码

  8. http://1.gravatar.com/avatar/f54f7e8880a77d9be41f85c973d4b7d0?s=32&d=http%3A%2F%2F1.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D32&r=G

    不懂技术。只能路过。。

  9. http://1.gravatar.com/avatar/bb6eeca6c19f5f11c8aaaa061ccde5e7?s=32&d=http%3A%2F%2F1.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D32&r=G

    查了下犀牛,document.documentElement就是对应html元素~

  10. http://1.gravatar.com/avatar/bb6eeca6c19f5f11c8aaaa061ccde5e7?s=32&d=http%3A%2F%2F1.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D32&r=G

    我试了下:
    function getMouseXY(ev) {
    var point = {
    x:0,
    y:0
    };

    point.x = Math.max(document.documentElement.scrollLeft,document.body.scrollLeft) + ev.clientX;
    point.y = Math.max(document.documentElement.scrollTop,document.body.scrollTop) + ev.clientY;

    return point;
    }
    各个浏览器都行,很奇怪chrome为什么一body元素来算,我之前这么写:
    point.y = document.getElementsByTagName("html")[0].scrollTop + ev.clientY;
    其它浏览器都行,但是chrome必须:
    point.y = document.getElementsByTagName("body")[0].scrollTop + ev.clientY;
    在css中chrome的顶级元素也是html,为啥在js里要用body?博主知道吗?

  11. http://1.gravatar.com/avatar/b2986defdd28dcaa196e317a94d82e31?s=32&d=http%3A%2F%2F1.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D32&r=G
  12. http://0.gravatar.com/avatar/eafff4918fffe6108e0f1ce887ee42e6?s=32&d=http%3A%2F%2F0.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D32&r=G

    哈哈,期待更多文章~
    图片的放大显示效果很爽~

  13. http://0.gravatar.com/avatar/eafff4918fffe6108e0f1ce887ee42e6?s=32&d=http%3A%2F%2F0.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D32&r=G

    "不同小异"哦~

  14. http://1.gravatar.com/avatar/bbb99c856872bdf1bd23790198834431?s=32&d=http%3A%2F%2F1.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D32&r=G

    学习了。每次来都有收获。 :cool:

  15. http://0.gravatar.com/avatar/64c0ecd205d00cd44717590f1d7822e9?s=32&d=http%3A%2F%2F0.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D32&r=G
  16. http://1.gravatar.com/avatar/390c74dedb216010128634bcc7a5bef2?s=32&d=http%3A%2F%2F1.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D32&r=G

    啊!
    突然发现my opera上的mg12就是你呀!!!
    果真大名鼎鼎!

  17. http://1.gravatar.com/avatar/78b0fd25b4aadfa50d9dde0ba2cc5d50?s=32&d=http%3A%2F%2F1.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D32&r=G

    高端应用,拜读

  18. http://1.gravatar.com/avatar/90d04faa49701e4e6e4225c4f29d595e?s=32&d=http%3A%2F%2F1.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D32&r=G

    难得这么靠前的位置 拜读了 :lol:

  19. http://0.gravatar.com/avatar/2f64db6c95d30046036a70048f682356?s=32&d=http%3A%2F%2F0.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D32&r=G

    Leeiio :
    去了阿里的一般博客都会鲜有更新了。。

    确实是,群里也N久不说话啦 :P

  20. http://0.gravatar.com/avatar/6ae83ef4bfaa6f98356a5187b6f171f6?s=32&d=http%3A%2F%2F0.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D32&r=G

    JS小菜路过。。。学习。。。 :grin: :grin:

  21. http://0.gravatar.com/avatar/8aa97fe8ca30db7b8b4c992f013736bf?s=32&d=http%3A%2F%2F0.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D32&r=G

    来学习JS~吼一声

  22. http://0.gravatar.com/avatar/2a9c4f525050372a6fe0b10ce4fe8957?s=32&d=http%3A%2F%2F0.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D32&r=G

    不错,期待更多JavaScript的内容,跟你学习了

  23. http://0.gravatar.com/avatar/490cf262668eebb0f0f1a50d9d48d702?s=32&d=http%3A%2F%2F0.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D32&r=G

    @underone
    我这栏文笔, 能出书吗?

    @xiao3
    是的, 恭喜.

    @行骏
    我想由浅入深. 嗯...

    @Leeiio
    项目不忙的时候我还是会坚持更新的. 可预见的未来一段时间我会写几个关于 JS 的文章.

  24. http://1.gravatar.com/avatar/ffa60157af668384f5325116efd02385?s=32&d=http%3A%2F%2F1.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D32&r=G

    去了阿里的一般博客都会鲜有更新了。。

  25. http://0.gravatar.com/avatar/eefde0b28ef0c51408b6d3039e99117a?s=32&d=http%3A%2F%2F0.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D32&r=G

    由浅入深... ...
    你想干嘛?

  26. http://0.gravatar.com/avatar/8985ffa1735009786d741e0b9c7a193c?s=32&d=http%3A%2F%2F0.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D32&r=G
  27. http://0.gravatar.com/avatar/e8f87528ed0a0eaba60009f8580df401?s=32&d=http%3A%2F%2F0.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D32&r=G

    等待一个系列
    再出本书为自己纪念一下:)

  1. Loading...