JS | JS之元素偏移量 offset 系列属性详解

目录

一、offset 概述

定位父级

offsetParent 

偏移量

offsetWidth

offsetHeight

offsetLeft

offsetTop

计算页面偏移

注意事项

二、offset 与 style 区别

偏移offset

样式style

三、案例

★ 案例:获取鼠标在盒子内的坐标

★ 案例:模态框拖拽

★ 案例:仿京东放大镜

四、番外:元素视图方法


偏移量(offset dimension)是javascript中的一个重要的概念。涉及到偏移量的主要是offsetLeft、offsetTop、offsetHeight、offsetWidth这四个属性。当然,还有一个偏移参照——定位父级offsetParent。本文将详细介绍该部分内容。

一、offset 概述

offset 翻译过来就是偏移量, 我们使用 offset系列相关属性可以动态的得到该元素的位置(偏移)、大小等。

  • 获得元素距离带有定位父元素的位置

  • 获得元素自身的大小(宽度高度)

注意返回的数值都不带单位

♠ offset 系列常用属性 ♠

以上五个偏移量属性,它们都是只读属性。如果元素设置了display:none,则它的偏移量属性均为0。每次访问偏移量属性都需要重新计算,重复访问偏移量属性需耗费大量性能,如需重复访问,应将其值保存在变量中,以提高性能。 

参考分析图

示例代码

<!DOCTYPE html>
<html lang="en"><head><meta charset="UTF-8" /><meta name="viewport" content="width=device-width, initial-scale=1.0" /><title>Document</title><style>* {margin: 0;padding: 0;}.father {margin: 100px;width: 200px;height: 200px;background-color: pink;}.son {margin-left: 30px;width: 100px;height: 100px;background-color: purple;}.box {margin: 0 100px;width: 200px;height: 200px;padding: 20px;background-color: skyblue;border: 10px solid red;}</style></head><body><div class="father"><div class="son"></div></div><div class="box"></div><script>// offset 系列let father = document.querySelector(".father");let son = document.querySelector(".son");// 1. offsetParent返回带有定位的父亲,否则返回的是bodyconsole.log(son.offsetParent); // 返回<body>...</body>// 注意:offsetParent与parentNode的值不一定相等// 返回最近一级的父亲(亲爸爸) 不管父亲有没有定位console.log(son.parentNode); // 返回<div class="father">...</div>// 2. 可以得到元素的偏移量(位置),返回的是不带单位的数值console.log(father.offsetTop); // 100 ,因为css为其设置的margin为100console.log(father.offsetLeft); // 100 ,因为css为其设置的margin为100// 它以带有定位的父亲为准,如果没有父亲或者父亲没有定位 则以 body 为准console.log(son.offsetLeft); // 130 ,因为虽然其有父亲,但是其父亲没有定位,所以以body为准,即100+30=130// 3. 可以得到元素的大小(宽度和高度),是包含padding + border + width/height的,不包括marginlet box = document.querySelector(".box");console.log(box.offsetWidth); // 260console.log(box.offsetHeight); // 260</script></body>
</html>

下面我们把offset的5个偏移属性,分别一一介绍一下

定位父级

offsetParent 

offsetParent属性是一个偏移参照,翻译为定位父级,即当前元素最近的有定位的父级元素(position不等于static)。定位父级分为下列四种情况。

在理解偏移大小之前,首先要理解offsetParent。人们并没有把offsetParent翻译为偏移父级,而是翻译成定位父级,很大原因是offsetParent与定位有关。

(1)当前元素自身有fixed定位,offsetParent返回null

当元素自身有fixed固定定位时,我们知道固定定位的元素相对于视口进行定位,此时没有定位父级,offsetParent的结果为null。

注意:firefox浏览器有兼容性问题——firefox没有考虑固定定位问题,返回body元素

<!-- 元素自身有fixed定位,offsetParent的结果为null -->
<div id="box" style="position:fixed;"></div>
<script>let box = document.querySelector('#box');console.log(box.offsetParent); // null//兼容问题:firefox浏览器没有考虑固定定位的问题,返回body元素,其他浏览器都返回null
</script>

(2)当前元素自身无fixed定位,且不存在有定位的父级元素,offsetParent返回body元素

<!-- 元素自身无fixed定位,且父级元素都未经过定位,offsetParent的结果为<body> -->
<div><span id="box"></span>
</div>
<script>let box = document.querySelector('#box');console.log(box.offsetParent); // <body>...</boby>
</script>

(3)当前元素自身无fixed定位,且存在有定位的父级元素,offsetParent返回最近的有定位的父级元素

<!--元素自身无fixed定位,且父级元素存在经过定位的元素,offsetParent的结果为离自身元素最近的经过定位的父级元素-->
<div id="div0" style="position:absolute;"><div id="div1" style="position:absolute;"><span id="box"></span></div>
</div>
<script>let box = document.querySelector('#box');console.log(span.offsetParent); // <div id="div1" style="position:absolute;">...</div>
</script>

(4)body元素无父元素节点,offsetParent返回null 

<!-- <body>元素的parentNode是null -->
<script>console.log(document.body.offsetParent); // null
</script>

★ IE7 浏览器Bug

对于定位父级offsetParent来说,IE7-浏览器存在以下bug

【bug1】元素本身经过绝对定位或相对定位,且父级元素无定位时,IE7-浏览器下offsetParent返回的结果是<html>

<div id="test" style="position:absolute;"></div>    
<script>
//IE7-浏览器返回<html>,其他浏览器返回<body>
let box = document.querySelector('#test');
console.log(box.offsetParent);
</script>
<div id="test" style="position:relative;"></div>    
<script>
//IE7-浏览器返回<html>,其他浏览器返回<body>
let box = document.querySelector('#test');
console.log(box.offsetParent);
</script>
<div id="test" style="position:fixed;"></div>    
<script>
// firefox并没有考虑固定定位的问题,返回<body>,其他浏览器都返回null
let box = document.querySelector('#test');
console.log(box.offsetParent);
</script>

 【bug2】如果父级元素存在触发haslayout的元素或有定位的元素,则offsetParent的结果为离自身元素最近的有定位或触发haslayout的父级元素

注意:关于haslayout的详细信息haslayout详解 - 小火柴的蓝色理想 - 博客园

<div id="div0" style="display:inline-block;"><div id='test'></div>    
</div>
<script>
let box = document.querySelector('#test');
//IE7-浏览器返回<div id="div0">,其他浏览器返回<body>
console.log(box.offsetParent);
</script>
<div id="div0" style="position:absolute;"><div id="div1" style="display:inline-block;"><div id='test'></div>    </div>    
</div>
<script>
let box = document.querySelector('#test');
//IE7-浏览器返回<div id="div1">,其他浏览器返回<div id="div0">
console.log(box.offsetParent);
</script>
<div id="div0" style="display:inline-block;"><div id="div1" style="position:absolute;"><div id='test'></div>    </div>    
</div>
<script>
let box = document.querySelector('#test');
//所有浏览器都返回<div id="div1">
console.log(box.offsetParent);
</script>

偏移量

偏移量共包括offsetHeight、offsetWidth、offsetLeft、offsetTop这四个属性

参考:彻底搞懂clientHeight、offsetHeight、scrollHeight的区别

offsetWidth

获取dom元素视觉上的宽度,返回结果是一个数字,单位像素。(不包含外边距

offsetWidth表示元素在水平方向上占用的空间大小,无单位(以像素px计)

<script>// 边框/内边距/宽/垂直滚动条element.offsetWidth = border + padding + width + scroll 
<script>
offsetHeight

获取dom元素视觉上的高度,返回结果是一个数字,单位像素。(不包含外边距

offsetHeight表示元素在垂直方向上占用的空间大小,无单位(以像素px计)

<script>// 边框/内边距/高/水平滚动条element.offsetWidth = border + padding + height + scroll; 
<script>

示例:offsetWidth / offsetHeight

<!DOCTYPE html>
<html lang="en"><head><title></title><meta charset="UTF-8" /><meta name="viewport" content="width=device-width, initial-scale=1" /><style>div {width: 100px;height: 200px;padding: 10px;margin: 20px;border: 5px solid black;}</style></head><body><div id="box"></div><script>let box = document.querySelector("#box");// element.offsetWidth =  border-left-width + padding-left + width + padding-right + border-right-width; console.log(box.offsetWidth); // 130=5*2+10*2+100// element.offsetHeight =  border-top-width + padding-top + height + padding-bottom + border-bottom-widthconsole.log(box.offsetHeight); // 130=5*2+10*2+200</script></body>
</html>

注意:如果存在垂直滚动条,offsetWidth也包括垂直滚动条的宽度;如果存在水平滚动条,offsetHeight也包括水平滚动条的高度

有滚动条时,offsetWidth和offsetHeight包含滚动条的宽度。IE8浏览器将垂直和水平滚动条的宽度计算在width宽度和height高度中,所以出现滚动条时width和height不变,而其他浏览器则是把垂直滚动条的宽度从width宽度中移出,把水平滚动条的高度从height高度中移出。

<!DOCTYPE html>
<html lang="en"><head><title></title><meta charset="UTF-8" /><meta name="viewport" content="width=device-width, initial-scale=1" /><style>#box {width: 100px;height: 100px;padding: 10px;margin: 10px;border: 1px solid black;overflow: scroll;}</style></head><body><div id="box"></div><script>//IE8-浏览器将垂直滚动条的宽度计算在width宽度和height高度中,width和height的值仍然是100px;//而其他浏览器则把垂直滚动条的宽度从width宽度中移出,把水平滚动条的高度从height高度中移出,因为滚动条默认宽度为17px,所以width宽度和height高度为剩下的83pxlet box = document.querySelector("#box");if (window.getComputedStyle) {// 83px 83pxconsole.log(getComputedStyle(box).width, getComputedStyle(box).height);} else {//ie8及以下版本的浏览器console.log(box.currentStyle.width, box.currentStyle.height); //100px}// 122=border*2+padding*2+100console.log(box.offsetWidth, box.offsetHeight); </script></body>
</html>

执行结果,如下图:

有童鞋可能会疑惑,宽度和高度不应该是83px吗?怎么各自多出了0.2px?我们看一下浏览器中呈现的样式。

盒子模型中,元素宽高的确是多出了0.2px,而且边框少了0.2,这是因为我的电脑系统缩放与布局调的是125%,这就是症结所在。所以将系统缩放调整到100%,这样浏览器就可以正确显示‌了。

补:currentStyle是一个在旧版本的IE浏览器中引入的属性,用于获取元素计算后的样式。‌ 它现在已经被废弃,仅支持IE浏览器使用‌。

currentStyle属性用于获取指定元素的当前样式,返回的是一个样式属性对象。它可以查询元素的宽度、高度、文本对齐方式、位置等所有CSS属性值。然而,这个属性仅能在IE浏览器中使用,如果需要在其他浏览器(如Firefox)中实现相同的效果,应使用getComputedStyle属性‌。

补:浏览器滚动条默认的宽度和高度通常是17px。‌ 这是因为在IE7及以上版本的IE浏览器、Chrome和Firefox中,滚动栏所占用的宽度都是17px‌。

不同浏览器的默认滚动条宽度可能有所不同,例如在Chrome或Safari中,可以通过设置::-webkit-scrollbar的宽度来调整滚动条的宽度。例如:

::-webkit-scrollbar {width: 4px;
}

这段代码将Chrome或Safari浏览器的滚动条宽度设置为4px‌。

在Firefox中,可以通过设置scrollbar-width属性来隐藏或显示滚动条。例如:

scrollbar-width: none;

这段代码将隐藏Firefox浏览器的滚动条‌。

在IE浏览器中,可以通过设置-ms-overflow-style属性来隐藏滚动条。例如:

-ms-overflow-style: none;

这段代码将隐藏IE浏览器的滚动条‌

番外:网页尺寸offsetHeight、offsetWidth
offsetHeight和offsetWidth,获取网页内容高度和宽度(包括滚动条等边线,会随窗口的显示大小改变)。

① 值

offsetHeight = clientHeight + 水平滚动条 + 边框。
offsetWidth = clientWidth + 滚动条 + 边框。

② 浏览器兼容性

var w= document.documentElement.offsetWidth || document.body.offsetWidth;
var h= document.documentElement.offsetHeight || document.body.offsetHeight;

offsetLeft

获取当前元素与其最近的有定位父级元素之间的x轴距离(计算左上顶点)。若当前元素没有有定位父级元素,获取其距离文档的x轴距离。返回结果是一个数字,单位像素。

offsetLeft表示元素的左外边框至offsetParent元素的左内边框之间的像素距离

offsetTop

获取当前元素与其最近的有定位父级元素之间的y轴距离(计算左上顶点)。若当前元素没有有定位的父级元素,获取其距离文档的y轴距离。返回结果是一个数字,单位像素。

offsetTop表示元素的上外边框至offsetParent元素的上内边框之间的像素距离

<!DOCTYPE html>
<html lang="en"><head><meta charset="UTF-8" /><meta name="viewport" content="width=device-width, initial-scale=1.0" /><title>Document</title><style>div#out {padding: 5px;position: relative;background-color: red;margin: 6px;border: 1px solid black;}div#inline {width: 50px;height: 50px;margin: 10px;background-color: green;}</style></head><body><div id="out"><div id="inline"></div></div><script>//parseInt(getComputedStyle(inline,null)['marginTop']) +//parseInt(getComputedStyle(out,null)['paddingTop'])=10+5=15console.log(inline.offsetTop); //15//parseInt(getComputedStyle(inline,null)['marginLeft']) +//parseInt(getComputedStyle(out,null)['paddingLeft'])=10+5=15console.log(inline.offsetLeft); //15</script></body>
</html>

★ IE7 浏览器Bug

IE7-浏览器在offsetTop属性的处理上存在bug

【1】若父级设置position: relative,则在IE7-浏览器下,offsetTop值为offsetParent元素的paddingBottom值

<div id="out" style="padding: 5px;position: relative;"><div id="test" style="width:100px; height:100px; margin:10px;"></div>        
</div>
<script>
// 其他浏览器返回15(5+10),而IE7-浏览器返回5
console.log(test.offsetTop);
</script>

【2】若父级设置position: aboslute(或其他触发haslayout的条件),offsetTop值为offsetParent元素的paddingBottom值和当前元素的marginTop值的较大值

<div id="out" style="padding: 5px;position:absolute;"><div id="test" style="width:100px; height:100px; margin:10px;"></div>        
</div>
<script>
// 其他浏览器返回15(5+10),而IE7-浏览器返回10(10和5的较大值)
console.log(test.offsetTop);
</script>

—— 小结:偏移量 ——

偏移量:包括元素在屏幕上占用的所有可见空间,元素的可见大小有其高度,宽度决定,包括所有内边距,滚动条和边框大小(注意,不包括外边距)。

以下4个属性可以获取元素的偏移量

1. offsetHeight:元素在垂直方向上占用的空间大小,以像素计。包括元素的高度(可见的),水平滚动条的高度,上边框高度和下边框高度。

2. offsetWidth:元素在水平方向上占用的空间大小,以像素计。包括元素的宽度(可见的),垂直滚动条的宽度,左边框宽度和右边框宽度。

3: offsetLeft:元素的左外边框至包含元素的左内边框之间的像素距离。

4: offsetTop:元素的上外边框至包含元素的上内边框之间的像素距离。

其中offsetLeft,offsetTop属性与包含元素有关,包含元素的引用保存在offsetParent中,请注意offsetParent与parentNode的值不一定相等

计算页面偏移

计算元素在页面上的偏移量只需将当前元素的offsetLeft和offsetTop与其offsetParent的相同属性相加,再加上offsetParent相应方向的边框,如此一直循环到根元素,就可以得到元素到页面的偏移量。

注意:在默认情况下,IE8 浏览器下如果使用currentStyle()方法获取<html>和<body>(甚至普通div元素)的边框宽度都是medium,而如果使用clientLeft(或clientTop)获取边框宽度,则是实际的数值

function getElementLeft(element){var actualLeft = element.offsetLeft;var current = element.offsetParent;while(current != null){actualLeft += current.offsetLeft + current.clientLeft;current = current.offsetParent;}return actualLeft + 'px';
}
function getElementTop(element){var actualTop = element.offsetTop;var current = element.offsetParent;while(current != null){actualTop += current.offsetTop + current.clientTop;current = current.offsetParent;}return actualTop + 'px';
}

完整示例代码如下:

<!DOCTYPE html>
<html lang="en"><head><meta charset="UTF-8" /><meta name="viewport" content="width=device-width, initial-scale=1.0" /><title>Document</title><style>html,body {border: 0;}body {margin: 0;}</style></head><body><div style="padding: 20px; border: 1px solid black; position: absolute"><div id="test" style="width: 100px; height: 100px; margin: 10px"></div></div><script>function getElementLeft(element) {var actualLeft = element.offsetLeft;var current = element.offsetParent;while (current != null) {actualLeft += current.offsetLeft + current.clientLeft;current = current.offsetParent;}return actualLeft + "px";}function getElementTop(element) {var actualTop = element.offsetTop;var current = element.offsetParent;while (current != null) {actualTop += current.offsetTop + current.clientTop;current = current.offsetParent;}return actualTop + "px";}//其他浏览器返回31(10+20+1),而IE7-浏览器返回21((20和10的较大值)+1)console.log(getElementTop(test));//所有浏览器返回31(10+20+1)console.log(getElementLeft(test));</script></body>
</html>

注意事项

【1】所有偏移量属性都是只读的

<div id="test" style="width:100px; height:100px; margin:10px;"></div>        
<script>
console.log(test.offsetWidth);//100
//IE8-浏览器会报错,其他浏览器则静默失败
test.offsetWidth = 10;
console.log(test.offsetWidth);//100
</script>

【2】如果给元素设置了display:none,则它的偏移量属性都为0

<div id="test" style="width:100px; height:100px; margin:10px;display:none"></div>
<script>
console.log(test.offsetWidth);//0
console.log(test.offsetTop);//0
</script>

 【3】每次访问偏移量属性都需要重新计算

<div id="test" style="width:100px; height:100px; margin:10px;"></div>        
<script>
console.time("time");
for(var i = 0; i < 100000; i++){var a = test.offsetWidth;
}
console.timeEnd('time');//65.129ms
</script>
<div id="test" style="width:100px; height:100px; margin:10px;"></div>        
<script>
console.time("time");
var a = test.offsetWidth;
for(var i = 0; i < 100000; i++){var b = a;
}
console.timeEnd('time');//1.428ms
</script>

由上面代码对比可知,重复访问偏移量属性需要耗费大量的性能,所以要尽量避免重复访问这些属性。如果需要重复访问,则把它们的值保存在变量中,以提高性能。


—— 小结:‌offset偏移属性 ——

‌offset偏移主要包括以下几个属性:​ offsetLeft、offsetTop、offsetWidth、offsetHeight ​和offsetParent。‌

  • offsetLeft‌:表示元素的左外边框至包含元素的左内边框之间的像素距离。
  • offsetTop‌:表示元素的上外边框至包含元素的上内边框之间的像素距离。
  • offsetWidth‌:表示元素在水平方向上占用的空间大小,包括元素的宽度(可见的)、垂直滚动条的宽度、左边框宽度和右边框宽度。
  • offsetHeight‌:表示元素在垂直方向上占用的空间大小,包括元素的高度(可见的)、水平滚动条的高度、上边框高度和下边框高度。
  • offsetParent‌:表示当前元素最近的有定位的父级元素,即position不等于static的父元素。如果元素自身有fixed定位,offsetParent结果为null;如果元素自身无fixed定位且父级元素存在经过定位的元素,offsetParent的结果为离自身元素最近的经过定位的父级元素。

这些属性用于获取或设置元素相对于其最近祖先或文档边界框的偏移量,常用于动态获取元素的位置和大小信息‌。 

二、offset 与 style 区别

偏移offset

  • offset 可以得到任意样式表中的样式值

  • offset 系列获得的数值是没有单位的

  • offsetWidth 包含padding+border+width,不含外边距

  • offsetWidth 等属性是只读属性,只能获取不能赋值

所以,我们想要获取元素大小位置,用offset更合适

样式style

  • style 只能得到行内样式表中的样式值

  • style.width 获得的是带有单位的字符串

  • style.width 获得不包含padding和border 的值

  • style.width 是可读写属性,可以获取也可以赋值

所以,我们想要给元素更改值,则需要用style改变

<body><div class="box" style="width: 200px;"></div><script>// offset与style的区别var box = document.querySelector('.box');console.log(box.offsetWidth); // 不带单位的数字 paddiing+border+widthconsole.log(box.style.width); // 只能得到行内样式 带单位的字符串 width// box.offsetWidth = '300px'; // 只能获取不能赋值box.style.width = '300px'; // 可以获取可以赋值// offset更合适获取元素大小位置// style更适合给元素更改值</script>
</body>

三、案例

★ 案例:获取鼠标在盒子内的坐标

  1. 我们在盒子内点击,想要得到鼠标距离盒子左右的距离。
  2. 首先得到鼠标在页面中的坐标(e.pageX, e.pageY)
  3. 其次得到盒子在页面中的距离 ( box.offsetLeft, box.offsetTop)
  4. 用鼠标距离页面的坐标减去盒子在页面中的距离,得到 鼠标在盒子内的坐标
  5. 如果想要移动一下鼠标,就要获取最新的坐标,使用鼠标移动
var box = document.querySelector(".box");
box.addEventListener("mousemove", function (e) {var x = e.pageX - this.offsetLeft;var y = e.pageY - this.offsetTop;this.innerHTML = "x坐标是" + x + " y坐标是" + y;
});

:e.pageX、e.clientX、e.screenX、e.offsetX的区别

e.pageX,e.pageY:返回的值是相对于文档的定位,文档的左上角为(0,0),向右为正,向下为正,IE不支持(不包含上方工具栏);

e.screenX,e.screenY:返回的是相对于屏幕的坐标,浏览器上面的工具栏(包含上方工具栏);

e.clientX,e.clientY:返回的值是相对于屏幕可见区域的坐标,如果页面有滚动条,呗滚动条隐藏的那部分不进行计算,也可以说是相对于屏幕的坐标,但是不计算上方的工具栏(不包含上方工具栏);

e.offsetX,e.offsetY:返回的是元素距离带有定位的父元素的位置,如果没有定位的父元素就是对body,和e.pageX,e.pageY作用相同,但是只有IE支持(不包含上方工具栏)。 

clientWidth, clientHeight:包含width+padding,不包含border;
scrollWidth,scrollHeight: 包含width+padding,不包含border;
offsetWidth, offsetHeight: 包含border+padding+width;
style.width, style.height: 包含width,不包含border+padding


★ 案例:模态框拖拽

弹出框,我们也称为模态框。

案例需求:

  1. ​ 点击弹出层,会弹出模态框, 并且显示灰色半透明的遮挡层。
  2. ​ 点击关闭按钮,可以关闭模态框,并且同时关闭灰色半透明遮挡层。
  3. ​ 鼠标放到模态框最上面一行,可以按住鼠标拖拽模态框在页面中移动。
  4. ​ 鼠标松开,可以停止拖动模态框移动

案例分析:

  1. 点击弹出层, 模态框和遮挡层就会显示出来 display:block;
  2. 点击关闭按钮,模态框和遮挡层就会隐藏起来 display:none;
  3. 在页面中拖拽的原理:鼠标按下并且移动, 之后松开鼠标
  4. 触发事件是鼠标按下mousedown,鼠标移动mousemove 鼠标松开 mouseup
  5. 拖拽过程: 鼠标移动过程中,获得最新的值赋值给模态框的left和top值,这样模态框可以跟着鼠标走了
  6. 鼠标按下触发的事件源是最上面一行,就是 id 为 title
  7. 鼠标的坐标减去 鼠标在盒子内的坐标, 才是模态框真正的位置。
  8. 鼠标按下,我们要得到鼠标在盒子的坐标。
  9. 鼠标移动,就让模态框的坐标 设置为 :鼠标坐标 减去盒子坐标即可,注意移动事件写到按下事件里面。
  10. 鼠标松开,就停止拖拽,就是可以让鼠标移动事件解除

代码实现:

// 1. 获取元素
var login = document.querySelector(".login");
var mask = document.querySelector(".login-bg");
var link = document.querySelector("#link");
var closeBtn = document.querySelector("#closeBtn");
var title = document.querySelector("#title");
// 2. 点击弹出层这个链接 link  让mask 和login 显示出来
link.addEventListener("click", function () {mask.style.display = "block";login.style.display = "block";
});
// 3. 点击 closeBtn 就隐藏 mask 和 login
closeBtn.addEventListener("click", function () {mask.style.display = "none";login.style.display = "none";
});
// 4. 开始拖拽
// (1) 当我们鼠标按下, 就获得鼠标在盒子内的坐标
title.addEventListener("mousedown", function (e) {var x = e.pageX - login.offsetLeft;var y = e.pageY - login.offsetTop;// (2) 鼠标移动的时候,把鼠标在页面中的坐标,减去 鼠标在盒子内的坐标就是模态框的left和top值document.addEventListener("mousemove", move);function move(e) {login.style.left = e.pageX - x + "px";login.style.top = e.pageY - y + "px";}// (3) 鼠标弹起,就让鼠标移动事件移除document.addEventListener("mouseup", function () {document.removeEventListener("mousemove", move);});
});

完整代码整理:

<!DOCTYPE html>
<html lang="en"><head><meta charset="UTF-8" /><meta name="viewport" content="width=device-width, initial-scale=1.0" /><title>Document</title><!-- css样式 --><style>* {margin: 0;padding: 0;box-sizing: border-box;}.login-bg {display: none;width: 100%;height: 100%;position: fixed;top: 0;left: 0;background: rgba(0, 0, 0, 0.3);}.login-header {font-size: 20px;margin: 10px 0;text-align: center;}a {text-decoration: none;color: #333;}.login {position: fixed;left: 50%;top: 50%;transform: translate(-50%, -50%);display: none;width: 480px;padding-bottom: 40px;background-color: #fff;border: 1px solid #ebebeb;box-shadow: 0px 0px 20px #ddd;z-index: 3;}.login-title {position: relative;text-align: center;padding: 20px 0 30px;font-size: 18px;cursor: move;}.close-login {position: absolute;top: -20px;right: -20px;display: block;text-align: center;line-height: 40px;width: 40px;height: 40px;border-radius: 50%;font-size: 12px;border: 1px solid #ccc;background-color: #fff;}.login-input-content {margin-left: 20px;font-size: 14px;}.login-input:nth-child(1) label {margin-left: 14px;}.login-input-content input {width: 340px;height: 30px;outline: none;border: none;border: 1px solid #ccc;margin-bottom: 30px;padding-left: 5px;}.login-button {padding-bottom: 30px;width: 260px;height: 30px;border: 1px solid #ccc;margin: 0 auto;text-align: center;line-height: 30px;}</style></head><body><!-- html结构 --><div class="login-header"><a href="javascript:;">点击,弹出登录框</a></div><div class="login"><div class="login-title">登录会员<span><a href="javascript:;" class="close-login">关闭</a></span></div><div class="login-input-content"><div class="login-input"><label for="">用户名:</label><inputtype="text"placeholder="请输入用户名"name="info[username]"id="username"class="username"/></div><div class="login-input"><label for="">登录密码:</label><inputtype="text"placeholder="请输入登录密码"name="info[password]"id="password"class="password"/></div></div><div class="login-button" id="loginBtn"><a href="javascript:;" id="login-button-submit">登录会员</a></div></div><!-- 遮盖层 --><div id="bg" class="login-bg"></div><!-- js代码 --><script>// 1. 获取元素let loginHeader = document.querySelector(".login-header");let login = document.querySelector(".login");let bg = document.getElementById("bg");let close = document.querySelector(".close-login");// 2. 点击,弹出登录框和遮盖层loginHeader.onclick = function () {login.style.display = "block";bg.style.display = "block";};// 3. 点击关闭,隐藏登录框和遮盖层close.onclick = function () {login.style.display = "none";bg.style.display = "none";};// 4. 鼠标放到模态框最上面的登录会员一行,可以按住鼠标拖拽模态框在页面中移动// 由于鼠标拖拽模态框时,鼠标的位置是不变的,所以要先获取鼠标在登录会员框中的位置// (1) 当鼠标按下,就能获得鼠标在盒子内的坐标let title = document.querySelector(".login-title");title.onmousedown = function (e) {let x = e.pageX - login.offsetLeft;let y = e.pageY - login.offsetTop;// (2) 鼠标移动的时候,把鼠标在页面中的坐标,减去鼠标在盒子内的坐标就是模态框的left和top值document.addEventListener("mousemove", move);// 由于后面会移除事件,为了方便使用所以这里给回调函数单独设置并命名为movefunction move(e) {login.style.left = e.pageX - x + "px";login.style.top = e.pageY - y + "px";}// (3) 鼠标弹起mouseup,就让鼠标移动事件移除document.addEventListener("mouseup", function () {document.removeEventListener("mousemove", move);});};</script></body>
</html>

★ 案例:仿京东放大镜

案例需求

  1. 整个案例可以分为三个功能模块
  2. 鼠标经过小图片盒子, 黄色的遮挡层 和 大图片盒子显示,离开隐藏2个盒子功能
  3. 黄色的遮挡层跟随鼠标功能。
  4. 移动黄色遮挡层,大图片跟随移动功能。

案例分析:

  1. 黄色的遮挡层跟随鼠标功能。
  2. 把鼠标坐标给遮挡层不合适。因为遮挡层坐标以父盒子为准。
  3. 首先是获得鼠标在盒子的坐标。
  4. 之后把数值给遮挡层做为left 和top值。
  5. 此时用到鼠标移动事件,但是还是在小图片盒子内移动。
  6. 发现,遮挡层位置不对,需要再减去盒子自身高度和宽度的一半。
  7. 遮挡层不能超出小图片盒子范围。
  8. 如果小于零,就把坐标设置为0
  9. 如果大于遮挡层最大的移动距离,就把坐标设置为最大的移动距离
  10. 遮挡层的最大移动距离:小图片盒子宽度 减去 遮挡层盒子宽度

代码实现:

window.addEventListener('load', function() {var preview_img = document.querySelector('.preview_img');var mask = document.querySelector('.mask');var big = document.querySelector('.big');// 1. 鼠标经过 preview_img 显示和隐藏 mask 遮挡层 和 big 大盒子preview_img.addEventListener('mouseover', function() {mask.style.display = 'block';big.style.display = 'block';})preview_img.addEventListener('mouseout', function() {mask.style.display = 'none';big.style.display = 'none';})// 2. 鼠标移动的时候,让黄色的盒子跟着鼠标来走preview_img.addEventListener('mousemove', function(e) {// (1). 先计算出鼠标在盒子内的坐标var x = e.pageX - this.offsetLeft;var y = e.pageY - this.offsetTop;// console.log(x, y);// (2) (鼠标在盒子内的坐标)-(盒子高度的一半)=(mask的最终left和top值)var maskX = x - mask.offsetWidth / 2;var maskY = y - mask.offsetHeight / 2;// (3) 如果x坐标小于0 就让他停在0的位置// 遮挡层的最大移动距离var maskMax = preview_img.offsetWidth - mask.offsetWidth;if (maskX <= 0) {maskX = 0;} else if (maskX >= maskMax) {maskX = maskMax;}if (maskY <= 0) {maskY = 0;} else if (maskY >= maskMax) {maskY = maskMax;}mask.style.left = maskX + 'px';mask.style.top = maskY + 'px';// 3. 大图片的移动距离 = 遮挡层移动距离 * 大图片最大移动距离 / 遮挡层的最大移动距离// 大图var bigIMg = document.querySelector('.bigImg');// 大图片最大移动距离var bigMax = bigIMg.offsetWidth - big.offsetWidth;// 大图片的移动距离 X Yvar bigX = maskX * bigMax / maskMax;var bigY = maskY * bigMax / maskMax;bigIMg.style.left = -bigX + 'px';bigIMg.style.top = -bigY + 'px';})
})

四、番外:元素视图方法

元素视图方法概述

元素视图有三个方法,分别是getBoundingClientRect、getClientRects(忽略)和elementFromPoint(忽略)。

参考:深入理解元素视图的3个方法 - 小火柴的蓝色理想 - 博客园

getBoundingClientRect

获取dom元素相对于浏览器视口的位置集合。不同浏览器,集合中的元素略有不同,ie8及以下版本的浏览器中没有width和height属性,若出现width和height属性,则它们是偏移宽高offset。位置集合中主要包含四个属性,分别是top、right、bottom和left,其中top和left是dom元素左上顶点相对于文档的偏移坐标,bottom和right是dom元素右下顶点相对于文档的偏移坐标。该方法的返回结果不是实时的,相当于一个快照。该方法是es5添加的方法,兼容性于所有主流浏览器。

<div id="div" style="width:100px;height:200px;padding:100px;"></div>
<script>console.log(div.offsetWidth);  //"300"console.log(div.offsetHeight); //"400"console.log(div.getBoundingClientRect());
</script>

<div id="div" style="width:100px;height:100px;"></div>
<script>var box = div.getBoundingClientRect();console.log(box.width); // "100"div.style.width = "200px";console.log(box.width); // "100", getBoundingClientRect返回结果是静态的
</script>

参考:JS | JS中的getBoundingClientRect()方法详解,秒懂!- 烤地瓜的CSDN博客


● 参考资料 ●

scrollTop及offsetTop的对比_scrolltop和offsettop-CSDN博客

深入理解元素视图的3个方法-getBoundingClientRect/getClientRects/elementFromPoint - 博客园

—— JS | JavaScrip之深入理解滚动scroll大小系列属性- 烤地瓜的CSDN博客 ——

—— JS | JavaScrip之深入理解客户区尺寸client系列属性 - 烤地瓜的CSDN博客 ——

—— JS | 详解图片懒加载的6种实现方案之利用JS监听scroll滚动事件 - 烤地瓜的CSDN博客 ——

一文看懂JS中元素偏移量(offsetLeft,offsetTop,offsetWidth,offsetHeight) - 博客园

Javascript进阶篇之DOM节点 - 获取浏览器窗口可视区域大小+获取网页尺寸) - 博客园


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

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

相关文章

软件测试学习笔记丨Pytest的使用

本文转自测试人社区&#xff0c;原文链接&#xff1a;https://ceshiren.com/t/topic/22158 1. 简介 pytest是一个成熟的全功能python测试框架测试用例的skip和xfail&#xff0c;自动失败重试等处理能够支持简单的单元测试和复杂的功能测试&#xff0c;还可以用来做selenium/ap…

2024互联网大厂营收排名:京东/阿里/华为前三,超多技术岗都在热招!

2024年已经过去一大半&#xff0c;各大互联网大厂的竞争如火如荼&#xff0c;营收都取得了不俗的成绩&#xff0c;京东、阿里、华为分别占领前三&#xff01; 第四第五名则为华为和拼多多。 根据排行榜里的公司名单&#xff0c;小码特意为大家整理了一批各大厂的招聘岗位。 阿…

【论文#快速算法】Fast Intermode Decision in H.264/AVC Video Coding

目录 摘要1.前言2.帧间模式决策概览2.1 H.264/AVC中的帧间模式决策2.2 发现和动机 3.同质性和平稳性的确定3.1 同质性区域的确定3.2 稳定性区域的决定3.3 整体算法 4.实验结果4.1 IPPP序列的测试4.2 IBBP序列测试 5.结论 《Fast Intermode Decision in H.264/AVC Video Coding》…

基础数据结构——数组(动态数组,二维数组,缓存与局部性原理)

1.概述 在计算机科学中&#xff0c;数组是由一组元素&#xff08;值或变量&#xff09;组成的数据结构&#xff0c;每个元素有至少一个索引或键来标识 因为数组内的元素是连续存储的&#xff0c;所以数组中元素的地址&#xff0c;可以通过其索引计算出来&#xff0c;例如&…

生产力工具|vscode for mac的安装python库和使用虚拟环境(一)

一、在vscode中运行python代码&#xff08;mac或windows&#xff09; &#xff08;一&#xff09;在vscode中安装Python插件 若想在vscode中高效率的编辑Python代码&#xff0c;需要安装Python插件&#xff0c;点击下图中红框内的按钮&#xff1a; 然后在左上角的搜索框中输入…

Vue前端开发2.1 单文件组件

文章目录 一、单文件组件概念二、单文件组件构成1. 模板&#xff08;Template&#xff09;2. 样式&#xff08;Style&#xff09;3. 逻辑&#xff08;Script&#xff09; 三、单文件组件演示1. 创建Vue项目2. 启动Vue项目3. 用VS Code打开项目4. 清空样式文件代码5. 创建欢迎组…

【redis】热点key问题

【redis】热点key问题 【一】什么是热点key问题【二】什么样的key被称为热key【三】热点Key问题的危害【四】如何监控发现热点key【五】热点Key的解决方案【1】使用二级缓存【2】将热key分散到不同的服务器中【3】热key拆分【4】将核心/非核心业务做Redis的隔离 【六】业界已有…

Nature 正刊丨细菌免疫蛋白直接感知两种不同的噬菌体蛋白

01摘要 真核先天免疫系统使用模式识别受体通过检测病原体相关的分子模式来感知感染&#xff0c;然后触发免疫反应。细菌也进化出了类似的免疫蛋白&#xff0c;可以感知其病毒捕食者的某些成分&#xff0c;即噬菌体1,2,3,4,5,6。尽管不同的免疫蛋白可以识别不同的噬菌体编码的触…

log4j2.xml

log4j2.xml 1、log4j2.xml使用2、日志器的流程解析2.1、几个重要的类2.2、整体流程图 3、部分源码3.1、通过简单例子看源码3.2、log4j2.xml配置指导 如侵权&#xff0c;请联系&#xff0c;无心侵权&#xff5e; 如有错误&#xff0c;也请指正。 1、log4j2.xml使用 <?xml v…

Anaconda虚拟环境安装cuda和pytorch

首先电脑上要有Anaconda&#xff0c;使用conda创建一个虚拟环境,并激活 conda create yolov8 conda activate yolov8winR输入cmd&#xff0c;在命令窗口输入 NVIDIA-smi可以查看到自己电脑支持的cuda环境&#xff0c;如下图 再打开torch的官网 pytorch官网 查看目前支持的版…

目标检测——Cascade R-CNN算法解读

论文&#xff1a; Cascade R-CNN: Delving into High Quality Object Detection (2017.12.3) 链接&#xff1a;https://arxiv.org/abs/1712.00726 Cascade R-CNN: High Quality Object Detection and Instance Segmentation (2019.6.24) 链接&#xff1a;https://arxiv.org/abs…

Z 字形变换(6)

这道题之前一直不会做&#xff0c;明白他是什么意思&#xff0c;但是找不到方法或者方法过于繁琐 方法1&#xff1a; 这是我在力扣评论区看到的方法&#xff0c;太精彩了。 虽然我实现起来效率并不高&#xff0c;可能是我代码的问题&#xff0c;但是他的思路很巧妙。 字符串的…

Spring--1

spring是一个轻量级的&#xff0c;采用IOC与AOP编程思想的java后端开发框架&#xff0c;简化了企业级的应用开发。 Spring体系 数据访问层&#xff0c;Web层&#xff0c;配置中心&#xff0c;测试区 IOC 控制反转&#xff0c;将创建对象的控制权交由Spring框架&#xff0c;需…

音频分割:长语音音频 分割为 短语音音频 - python 实现

在做语音任务时&#xff0c;有是会用到的语音音频是长音频&#xff0c;这就需要我们将长音频分割为短音频。 该示例将声音的音量和静默时间结合作为语音的分割条件。 使用音量和静默时间结合的分割条件&#xff0c;能够比较好的进行自然断句&#xff0c;不会话语没有说完就切断…

Spring声明式事务管理:深入探索XML配置方式

前言 Spring的事务管理&#xff0c;无论是基于xml还是注解实现&#xff0c;本质上还是实现数据库的事务管理机制&#xff0c;因此要注意发送SQL的连接是否为同一个&#xff0c;这是实现声明式事务的关键。 以下案例和实现基于SSM整合框架完成&#xff0c;不知道如何整合SSM&…

【K8S系列】Kubernetes Pod 状态详细介绍及异常状态解决方案

在 Kubernetes 中&#xff0c;Pod 是最小的可调度单元&#xff0c;负责运行一个或多个容器。Pod 的状态能够反映其生命周期中的不同阶段&#xff0c;帮助用户了解当前的运行状况。本文将详细介绍 Kubernetes Pod 的各种状态及其可能的异常状态解决方案。 一、Pod 状态概览 Po…

查缺补漏----数据结构树高总结

① 对于平衡二叉树而言&#xff0c;树高的规律&#xff1a; 高度为h的平衡二叉树的含有的最少结点数&#xff08;所有非叶节点的平衡因子均为1&#xff09;&#xff1a; n01&#xff0c;n11&#xff0c;n22 含有的最多结点数&#xff1a; (高度为h的满二叉树含有的结点数) ②…

Flutter在 iOS 中实现无弹窗获取剪切板内容

前言 在最新的项目需求中&#xff0c;我们需要在获取剪切板内容时避免弹出授权提示。这一功能是基于竞品的实现&#xff0c;旨在优化用户体验&#xff0c;特别是在推广获取跳转链接的场景下非常有用。 解决方案 通过查阅资料&#xff0c;我们发现对于 iOS 16 及以上的系统&a…

Fusion创建一个简单的api脚本文件

我的Fusion版本&#xff1a;Fusion 2.0.20476 x86_64 脚本模块在实用程序->附加模型->脚本和附加模块&#xff0c;快捷键为shifts 里面有一些演示脚本&#xff0c;可以直接使用 也可以自己创建一个新的脚本 创建的脚本在此处—— 选择脚本文件&#xff0c;点击编辑&a…

Unity Mirror NetworkManager初识

文章目录 Network Manager网络管理器什么是网络管理器&#xff1f;通过Transports进行定制化网络连接管理自定义连接地址和端口号Game State Management游戏状态管理Network Manager HUD玩家预制体及其生成控制Spawn Prefabs其他预制体注册Scene Management场景管理 Network Ma…