目录
一、元素偏移量offset系列
1.1 offset概述
offset即偏移量,使用offset系列相关属性可以动态的得到该元素的位置(偏移)、大小等
- 获得元素距离带有定位父元素的位置
- 获得元素滋生大小(宽度高度)
- 注意:返回值都不带单位
offset系列属性 | 作用 |
element.offsetParent | 返回作为该元素带有定位的父级元素,如果父级元素没有定位则返回body |
element.offsetTop | 返回元素相对带有定位父级元素上方的偏移 |
element.offsetLeft | 返回元素相对带有定位父级元素左边框的偏移 |
element.offsetWidth | 返回自身包括padding、边框、内容区的宽度,返回值不带单位 |
element.offsetHeight | 返回自身包括padding、边框、内容区的高度,返回值不带单位 |
1.2 offset 与style 区别
offset:
- offset可以得到任意形式的样式值
- offset系列获取的数值是没有单位的
- offsetWidth包含padding+border+width
- offsetWidth等属性是只读属性,只能获取不能赋值
- 获取元素大小位置,offset更合适
style:
- style只能的到行内样式表中的样式值
- style.width获取的是带有单位的字符串
- style.width获得不包含padding和border的值
- style.width是可读写属性,可以获取也可以赋值
- 想要给元素更改值,则需要style改变
1.3 案例练习 - 获取鼠标在盒子内的坐标
案例分析:
- 在盒子内点击,获得鼠标距离盒子左右的距离
- 首先得到鼠标在页面中的坐标(e.pageX, e.pageY)
- 其次得到盒子在页面中的距离(box.offsetLeft, box.offsetTop)
- 用鼠标距离页面的坐标减去何盒子在页面中的距离,得到鼠标在盒子内的坐标
.box{
width: 300px;
height: 300px;
background-color: skyblue;
padding: 10px;
}
<div class="box"></div>
<script>
var box = document.querySelector('.box');
box.addEventListener('mousemove',function(e){
console.log(e.pageX);
console.log(e.pageY);
console.log(box.offsetLeft);
console.log(box.offsetTop);
var x= e.pageX - this.offsetLeft;
var y = e.pageY - this.offsetTop;
this.innerHTML = 'x坐标:' + x + ' y坐标是:' + y;
});
</script>
1.4 案例练习 - 模态框拖拽
弹出框,即模态框
- 点击弹出层,会弹出模态框,并且显示灰色半透明的遮挡层
- 点击关闭按钮,可以关闭模态框,并且同时关闭灰色半透明遮挡层
- 鼠标放到模态框最上面一行,可以按住鼠标拖拽模态框在页面中移动
- 鼠标松开,可以停止拖动模态框移动
案例分析:
- 点击弹出层,模态框和遮挡层就会显示出来 display: block;
- 点击关闭按钮,模态框和遮挡层就会隐藏起来 dispaly:none;
- 在页面中拖拽的原理:鼠标按下且移动,松开鼠标则停止
- 触发事件是鼠标按下mousedown,鼠标移动mousemove,鼠标松开mouseup
- 拖拽过程:鼠标移动过程中,获得最新的赋值给模态框的 left 和 top 值
- 鼠标按下触发事件源是最上面一行,title栏
- 鼠标的坐标 减去 鼠标在盒子内的坐标,才是模态框的真正位置
- 鼠标按下,得到鼠标在盒子内的坐标
- 鼠标移动,模态框的坐标设置为:鼠标坐标 减去 盒子坐标即可,移动事件需写在按下事件内
*{
margin: 0;
padding: 0;
}
a{
text-decoration: none;
color: black;
}
.login-header{
width: 100%;
text-align: center;
height: 30px;
font-size: 24px;
line-height: 30px;
}
.login {
display: none;
width: 500px;
height: 300px;
position: fixed;
border: 1px solid #ebebeb;
left: 50%;
top: 50%;
background-color: #fff;
box-shadow: 0px 0px 20px #ddd;
z-index: 99;
transform: translate(-50%, -50%);
}
.title{
width: 100%;
height: 40px;
margin: 10px 0px 0px 0px;
text-align: center;
line-height: 40px;
font-size: 18px;
position: relative;
cursor: move;
}
.login-input-content{
margin-top: 20px;
}
.login-button{
width: 50%;
margin: 30px auto 0px auto;
line-height: 40px;
font-size: 14px;
border: 1px solid #ebebeb;
text-align: center;
}
.login-bg{
display: none;
width: 100%;
height: 100%;
position: fixed;
top: 0px;
left: 0px;
background: rgba(0, 0, 0, .3);
}
.login-button a {
display: block;
}
.login-input input.list-input{
float: left;
height: 35px;
line-height: 35px;
width: 350px;
border: 1px solid #ebebeb;
text-indent: 5px;
}
.login-input{
overflow: hidden;
margin: 0px 0px 20px 0px;
}
.login-input label{
float: left;
width: 90px;
height: 35px;
padding-right: 10px;
text-align: right;
line-height: 35px;
font-size: 14px;
}
.title span {
position: absolute;
font-size: 12px;
right: -20px;
top: -30px;
background: #fff;
border: 1px solid #ebebeb;
width: 40px;
height: 40px;
border-radius: 20px;
}
<div class="login-header"><a id="link" href="javascript:;">点击,弹出登录框</a></div>
<div id="login" class="login">
<div id="title" class="title">登录框
<span><a href="javascrip:;" id="closeBtn">关闭</a></span>
</div>
<div class="login-input-content">
<div class="login-input">
<label for="">用户名:</label>
<input type="text" placeholder="请输入用户名" name="info[username]" id="username" class="list-input">
</div>
<div class="login-input">
<label for="">密码:</label>
<input type="password" placeholder="请输入登录密码" name="info[password]" id="password" class="list-input">
</div>
</div>
<div id="loginBtn" class="login-button"><a href="javascript:;" id="login-button-sumbit">登录</a></div>
</div>
<!-- 遮挡层 -->
<div id="bg" class="login-bg"></div>
<script>
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')
// 点击链接,显示登录框
link.addEventListener('click',function(){
mask.style.display = 'block';
login.style.display = 'block';
});
// 点击关闭按钮,隐藏登录框
closeBtn.addEventListener('click',function(){
mask.style.display = 'none';
login.style.display = 'none';
});
// 鼠标按下,获取鼠标在盒子内的坐标
title.addEventListener('mousedown',function(e){
var x = e.pageX - login.offsetLeft;
var y = e.pageY - login.offsetTop;
document.addEventListener('mousemove', move);
function move(e){
login.style.left = e.pageX - x + 'px';
login.style.top = e.pageY - y + 'px';
}
// 鼠标松开,移除鼠标移动事件
document.addEventListener('mouseup',function(){
document.removeEventListener('mousemove', move);
});
});
</script>
1.5 案例练习 - 仿京东放大镜效果页面
案例分析:
- 整个案例可以分为3个功能模块
- 鼠标经过小图片盒子,黄色的遮挡层 和大图片盒子显示,离开隐藏2个盒子功能
- 黄色的遮挡层跟随鼠标功能
- 移动黄色遮挡层,大图片跟随移动功能
*{
margin: 0;
padding: 0;
}
.img_content{
width: 300px;
height: 300px;
}
.imgs{
height: 300px;
position: relative;
border: 1px solid #ccc;
}
.mask {
display: none;
position: absolute;
top: 0;
left: 0;
width: 200px;
height: 200px;
background: #FEDE4F;
opacity: .5;
border: 1px solid #ccc;
cursor: move;
}
.img_big{
display: none;
position: absolute;
left: 320px;
top: 0;
/* background: skyblue; */
width: 600px;
height: 600px;
border:1px solid #ccc;
overflow: hidden;
}
.img_big img{
position: absolute;
top: 0;
left: 0;
}
<div class="img_content">
<div class="imgs">
<img src="img/zdsbo.JPG" alt="" width="300px" height="300px">
<div class="mask"></div>
<div class="img_big"><img src="img/zdsbo.JPG" width="900px" height="900px" alt="" class="imgBig"></div>
</div>
</div>
<script>
var imgs = document.querySelector('.imgs');
var mask = document.querySelector('.mask');
var img_big = document.querySelector('.img_big');
// 显示遮挡层和大图
imgs.addEventListener('mouseover',function(){
mask.style.display = 'block';
img_big.style.display = 'block';
});
imgs.addEventListener('mouseout',function(){
mask.style.display = 'none';
img_big.style.display = 'none';
});
// 遮挡层跟随鼠标移动
imgs.addEventListener('mousemove',function(e){
var x = e.pageX - this.offsetLeft;
var y = e.pageY - this.offsetTop;
var maskX = x - mask.offsetWidth / 2;
var maskY = y - mask.offsetHeight / 2;
// 限制遮挡层移动范围
var maskMax = imgs.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';
// 大图片的移动距离 = 遮挡层移动距离 * 大图片最大移动距离 / 遮挡层的最大移动距离
var imgBig = document.querySelector('.imgBig');
var bigMax = imgBig.offsetWidth - img_big.offsetWidth; // 大图片最大移动距离
var bigX = maskX * bigMax / maskMax;
var bigY = maskY * bigMax / maskMax;
imgBig.style.left = -bigX + 'px';
imgBig.style.top = -bigY + 'px';
});
</script>