js计算元素距离顶部的高度及元素是否在可视区判断
前言:
在业务当中,咱们常常要计算元素的大小和元素在页面的位置信息。好比说,在一个滚动区域内,我要知道元素A是在可视区内,仍是在隐藏内容区(滚动到外边看不到了)。有时还要进一步知道,元素是所有都显示在可视区,仍是有部分在可视区部分在隐藏内容区。有时还要进一步知道,在隐藏内容区的那一部分是占多大的大小。so,来聊聊如何获取元素的大小和位置信息。浏览器
计算元素距离顶部的高度:偏移量
在二维的世界里,能够想象成一个二维坐标系。每个元素在坐标系内都有两个基本的属性:大小和位置。框架
大小:
dom元素在页面的大小有两个属性:offsetWidth 、offsetHeight,dom
offsetHeight:元素在垂直方向上的占用空间大小,以像素计。包括元素的高度、(可见的)水平滚动条的高度、上边框高度和下边框高度。(个人理解:盒模型包括border内的高度总和)函数
offsetWidth: (同理) 布局
位置:spa
offsetParent 是一个只读属性,返回一个指向最近的(closest,指包含层级上的最近)包含该元素的定位元素。若是没有定位的元素,则 offsetParent 为最近的 table, table cell 或根元素
offsetTop:元素相对于其 offsetParent 元素的顶部的距离code
offsetLeft:(同理)blog
元素大小和位置信息图解element
原理:
计算元素距离顶部的高度:将元素的offsetTop与其offsetParent的相同属性相加,如此循环直至根元素,就能够获得一个基本准确的值。封装以下函数(开箱即用,函数返回值即为元素距离顶部高度)get
function getElementTop (el) {
var actualTop = el.offsetTop
var current = el.offsetParent
while (current !== null) {
actualTop += current.offsetTop
current = current.offsetParent
}
return actuanlTop
}
分析一下代码:
计算元素距离左部的高度:(同理可得)
元素是否在可视区判断:结合scrollTop
假如这样的业务场景:有一个wrap滚动容器,wrap的内容区content的高度超过wrap的高度,则出现纵向滚动条。随意拖动滚动条到某个位置,要判断content里面的子元素input输入框是否在可视区内,若不在可视区内,自动拖动滚动条,使其进入可视区。这样的业务场景其实常常有遇到。
原理:
首先,根据以上getElementTop(domInput)函数,获得元素input距离顶部的高度elementTop。再结合滚动容器domWrap的scrollTop属性获得滚动条高度scrollTop(被隐藏在内容区域上方的像素数),进行比较,便可判断
scrollTop > elementTop: 滚动条高度大于元素离顶部高度,说明元素进入了隐藏内容区,进入的量为 scrollTop - elementTop
scrollTop < elementTop:滚动条高度小于元素的离顶部高度,说明元素还没进入上方的隐藏内容区,如要保证元素在可视区内,则必须同时知足条件,元素不在下方的隐藏内容区: elementTop - scrollTop < document.documentElemnt.clientHeight
结论:
因此,元素在可视区的初步判断条件为:scrollTop < elementTop && elementTop - scrollTop < document.documentElemnt.clientHeight
以上判断还不太严谨,若是wrap还同时存在横向滚动条,还得再判断是否元素在横向的可视区内,若是要判断元素是否彻底在可视区,还得加上自身的高度值,即为:scrollTop < elementTop && elementTop + input.offsetHeight - scrollTop < document.documentElemnt.clientHeight。
经过设置scrollTop属性的值domWrap.scrollTop = elementTop 可让滚动条自动滚动,且input元素恰好在可视区的上方。
ps:书上是这么说的,对于简单的CSS布局的页面,getElementTop函数能够获得很是精确的结果。对于使用表格和内嵌框架布局的页面,因为不一样浏览器实现这些元素的方式不一样,所以获得的结果就不太精确。在个人业务中,我没有用到表格和内嵌框架的布局,计算结果确实是精确的,可放心使用。若是使用表格和内嵌框架布局的页面,Keeping a eye on it。