前言
系列首发于公众号『前端进阶圈』,更多精彩内容敬请关注公众号最新消息。
JS 如何判断一个元素是否在可视区域内?
- 办法一:
offsetTop、scrollTop
- 办法二:
getBoundingClientRect()
- 办法三:
Intersection Observer
办法一:offsetTop、scrollTop
// 公式
el.offsetTop - document.documentElement.scrollTop <= viewPortHeight;
// 代码实现
function isInViewPortOfOne(el) {
// viewPortHeight 兼容所有浏览器写法
const viewPortHeight =
window.innerHeight ||
document.documentElement.clientHeight ||
document.body.clientHeight;
const offsetTop = el.offsetTop;
const scrollTop = document.documentElement.scrollTop;
const top = offsetTop - scrollTop;
return top <= viewPortHeight;
}
办法二:getBoundingClientRect
- 返回值是一个 DOMRect 对象,领有 left, top, right, bottom, x, y, width, 和 height 属性
const target = document.querySelector(".target");
const clientRect = target.getBoundingClientRect();
console.log(clientRect);
// {
// bottom: 556.21875,
// height: 393.59375,
// left: 333,
// right: 1017,
// top: 162.625,
// width: 684
// }
// A:
// 如果一个元素在视窗之内的话,那么它肯定满足上面四个条件:// top 大于等于 0
// left 大于等于 0
// bottom 小于等于视窗高度
// right 小于等于视窗宽度
// 代码实现
function isInViewPort(element) {
const viewWidth = window.innerWidth || document.documentElement.clientWidth;
const viewHeight =
window.innerHeight || document.documentElement.clientHeight;
const {top, right, bottom, left} = element.getBoundingClientRect();
return top >= 0 && left >= 0 && right <= viewWidth && bottom <= viewHeight;
}
办法三:Intersection Observer
- Intersection Observer 即重叠观察者,从这个命名就能够看出它用于判断两个元素是否重叠,因为不必进行事件的监听,性能方面相比 getBoundingClientRect 会好很多
- 应用步骤次要分为两步:创立观察者和传入被观察者
// 第一步:创立观察者
const options = {
// 示意重叠面积占被观察者的比例,从 0 - 1 取值,// 1 示意齐全被蕴含
threshold: 1.0,
root:document.querySelector('#scrollArea') // 必须是指标元素的父级元素
};
const callback = (entries, observer) => {....}
const observer = new IntersectionObserver(callback, options);
// 通过 new IntersectionObserver 创立了观察者 observer,传入的参数 callback 在重叠比例超过 threshold 时会被执行 `
// 上段代码中被省略的 callback
const callback = function(entries, observer) {
entries.forEach(entry => {
entry.time; // 触发的工夫
entry.rootBounds; // 根元素的地位矩形,这种状况下为视窗地位
entry.boundingClientRect; // 被观察者的地位举办
entry.intersectionRect; // 重叠区域的地位矩形
entry.intersectionRatio; // 重叠区域占被观察者面积的比例(被观察者不是矩形时也依照矩形计算)entry.target; // 被观察者
});
};
// 第二步:传入被观察者
const target = document.querySelector('.target');
observer.observe(target);
残缺代码
-
前两种办法
<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"> <script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.6.3/jquery.min.js" integrity="sha512-STof4xm1wgkfm7heWqFJVn58Hm3EtS31XFaagaa8VMReCXAkQnJZ+jEy8PCC/iT18dFy95WcExNHFTqLyp72eQ==" crossorigin="anonymous" referrerpolicy="no-referrer"></script> <title>Document</title> <style> .container { display: flex; flex-wrap: wrap; } .target { margin: 5px; width: 20px; height: 20px; background: red; } </style> </head> <body> <div class="container"></div> </body> </html> <script> (() => {const $container = $(".container"); function createTargets() {const htmlString = new Array(10000).fill('<div class="target"></div>').join("") $container.html(htmlString) } createTargets(); const $targets = $(".target"); function isInViewPort(el){ // 办法 1 // const viewPortHeight = window.innerHeight || document.documentElement.clientHeight || document.body.clientHeight; // const offsetTop = el.offsetTop; // const scollTop = document.documentElement.scrollTop; // return offsetTop-scollTop <= viewPortHeight // 办法 2 const viewPortHeight = window.innerHeight || document.documentElement.clientHeight || document.body.clientHeight; const viewPortWidth = window.innerWidth || document.documentElement.clientWidth || document.body.clientWidth; const {top,right,left,bottom} = el.getBoundingClientRect(); return top >= 0 && left >= 0 && bottom <= viewPortHeight && right <= viewPortWidth } // 事件监听 $(window).on("scroll",()=>{console.log("scroll!!"); $targets.each((index,element)=>{if(isInViewPort(element)){$(element).css("background-color","blue") } }) }) })(); </script>
-
第三种办法
<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"> <script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.6.3/jquery.min.js" integrity="sha512-STof4xm1wgkfm7heWqFJVn58Hm3EtS31XFaagaa8VMReCXAkQnJZ+jEy8PCC/iT18dFy95WcExNHFTqLyp72eQ==" crossorigin="anonymous" referrerpolicy="no-referrer"></script> <title>Document</title> <style> * { margin: 0; padding: 0; } .test { width: 200px; height: 1000px; background: orange; } .box { width: 150px; height: 150px; margin: 50px; background: red; } #sta { position: fixed; left: 40%; top: 40%; width: 200px; height: 100px; background: greenyellow; } </style> </head> <body> <div class="test">test</div> <div class="box">box</div> <div id="sta"> 初始化 </div> </body> </html> <script> (() => {var status_node=document.querySelector("#sta"); const box = document.querySelector('.box'); const intersectionObserver = new IntersectionObserver((entries) => {entries.forEach((item) => {if (item.isIntersecting) { box.innerText = '进入可视区域'; status_node.innerText = '进入可视区域'; console.log('进入可视区域'); }else{ box.innerText = '进来了'; status_node.innerText = '进来了'; } }) }); intersectionObserver.observe(box); })(); </script>
特殊字符形容:
- 问题标注
Q:(question)
- 答案标注
R:(result)
- 注意事项规范:
A:(attention matters)
- 详情形容标注:
D:(detail info)
- 总结标注:
S:(summary)
- 剖析标注:
Ana:(analysis)
-
提醒标注:
T:(tips)
往期举荐:
- 热点面试题:浏览器和 Node 的宏工作和微工作?
- 这是你了解的 CSS 选择器权重吗?
- 热点面试题:JS 中 call, apply, bind 概念、用法、区别及实现?
- 热点面试题:罕用位运算办法?
- Vue 数据监听 Object.definedProperty() 办法的实现
- 热点面试题:Virtual DOM 相干问题?
- 热点面试题:Array 中有哪些非破坏性办法?
-
热点面试题:协商缓存和强缓存的了解及区别?
最初:
- 欢送关注『前端进阶圈』公众号,一起摸索学习前端技术 ……
- 公众号回复 加群 或 扫码, 即可退出前端交流学习群,一起高兴摸鱼和学习 ……
- 公众号回复 加好友,即可添加为好友