HTML文档坐标和视口坐标

2024-05-16 14:58

本文主要是介绍HTML文档坐标和视口坐标,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!


文档坐标和视口坐标

  视口坐标是相对于窗口的坐标,而文档坐标是相对于整个文档而言。例如,在文档坐标中如果一个元素的相对于文档的Y坐标是200px,并且用户已经把浏览器向下滚动了75px,那么视口坐标中元素的Y坐标为200px – 75px = 125px。

  如何获取浏览器滚动条的位置?Window对象的pageXOffset和pageYOffset属性在所有浏览器中提供这些值,除IE8以及更早的版本。IE和所有现代浏览器也可以通过scrollLeft和scrollTop属性获取滚动条位置。

  下面代码的getScrollOffsets方法获取滚动条位置:

view source
print ?
01//以一个对象的x和y属性放回滚动条的位置
02function getScrollOffsets(w){
03    w = w || window;
04    //除了IE 8以及更早的版本以外,其他浏览器都支持
05    if(w.pageXOffset != null) return {x: w.pageXOffset, y: w.pageYOffset};
06    //对标准模式下的IE
07    var d = w.document;
08    if(document.compatMode == "CSS1Compat")
09        return {x: d.documentElement.scrollLeft, y: d.documentElement.scrollTop};
10    //对怪异模式下的浏览器
11    return { x: d.body.scrollLeft, y: d.body.scrollTop};
12}

  有时候能够判定视口的尺寸也是非常有用的,下面的代码简便地查询视口尺寸:

view source
print ?
01//作为一个对象的w和h属性返回视口的尺寸
02function getViewportSize(w){
03    //使用指定的窗口, 如果不带参数则使用当前窗口
04    w = w || window;
05 
06    //除了IE8及更早的版本以外,其他浏览器都能用
07    if(w.innerWidth != null)
08        return {w: w.innerWidth, h: w.innerHeight};
09 
10    //对标准模式下的IE(或任意浏览器)
11    var d = w.document;
12    if(document.compatMode == "CSS1Compat")
13        return {w: d.documentElement.clientWidth, h: d.documentElement.clientHeight};
14 
15    //对怪异模式下的浏览器
16    return {w: d.body.clientWidth, h: d.body.clientHeight};
17}

  上面的两个例子已经使用到scrollLeft、scrollTop、clientWidth、clientHeight。 scrollLeft和scrollTop获取滚动条位置,而clientWidth和clientHeight获取对象的尺寸。


  查询元素的几何尺寸


  判定一个元素的尺寸和位置最简单的方法是调用它的getBoundingClientRect()方法。该方法是在IE5中引入的,而现在当前的所有浏览器都实现了。它不需要参数,返回left、right、top、bottom属性的对象。

  这个方法返回元素在视口坐标中的位置。为了转换为甚至用户滚动浏览器窗口以后任然有效的文档坐标,需要加上滚动偏移量:

view source
print ?
1//元素相对于文档的坐标位置
2function getElementRect(e){
3    var box = e.getBoundingClientRect();
4    var offsets = getScrollOffsets();
5    var x = box.left + offsets.x;
6    var y = box.top + offsets.y;
7 
8    return {x:x, y: y};
9}

  在很多浏览器中,getBoundingClientRect()返回的对象还包括width和height属性。但在原始的IE中未实现。可以这样计算元素的width和height:

view source
print ?
1//元素尺寸
2function getElementSize(e){
3    var box = getElementRect(e);
4    var w = box.width || box.right - box.left;
5    var h = box.height || box.bottom - box.top;
6 
7    return {w: w, h: h};
8}

  滚动元素


  之前的getScrollOffsets方法可以查询滚动条的位置。该例子的scrollLeft和scrollTop属性可以用来设置让浏览器滚动,但有一种更简单的方法从Javascript最早时期开始支持的。Window对象的scrollTop()方法接口一个点的X和Y坐标(文档坐标),并作为滚动条的偏移量设置它们。下面代码滚动浏览器到文档最下面的页面可见:

view source
print ?
1//滚动到浏览器最底部
2function scrollToBottom(){
3    //获取文档和视口的高度
4    var documentHeight = document.documentElement.offsetHeight;
5    var viewportHeight = window.innerHeight; //或使用上面的getViewPortSize()方法
6 
7    //然后,滚动让最后一页在视口中可见
8    window.scrollTo(0, documentHeight - viewportHeight);
9}

  Window的scrollBy方法和scroll()和scrollTo()类似,但是它的参数是相对的,并在当前滚动条的偏移量上增加。例如,快速阅读者可能会喜欢这样:

view source
print ?
1javascript:void setInterval(function(){scrollBy(0, 10)}, 200);

  如果想让某个元素在文档中可见,可以利用getBoundingClientRect()计算元素的位置,并转换为文档坐标,然后使用scrollTo()方法达到目的。但在需要显示Html元素上调用scrollIntoView()方法更方便。

  scrollIntoView()的行为与设置window.location.hash为一个命名锚点的名字后浏览器产生的行为类似。


  元素尺寸、位置和溢出


  任何HTML元素的只读属性offsetWidth和offsetHeight以CSS像素返回它的屏幕尺寸。返回的尺寸包含元素的边框和内边距,除去了外边距。

  所有HTML元素拥有offsetLeft和offsetTop属性来返回元素的X和Y坐标。这些值是文档坐标,并直接指定元素的位置。当对于已定位元素的后代元素和一些其他元素,这些属性返回的坐标是相对于祖先元素的而非文档。

  offsetParent属性指定这些属性所相对的父元素。如果offsetParent为null,这些属性都是文档坐标,因此,一般来说,用offsetLeft和offsetTop来计算元素e的位置需要一个循环:

view source
print ?
01//计算元素位置
02function getElementPosition(e){
03    var x = 0, y = 0;
04    while(e != null){
05        x += e.offsetLeft;
06        y += e.offsetTop;
07        e = e.offsetParent;
08    }
09 
10    return {x: x, y: y };
11}

  getElementPosition函数也不总是计算正确的值,下面看如何修复它。除了这些名字以offset开头的属性以外,所有的文档元素定义了其他两组属性,名字一组以client开头,另一组以scroll开头。即每个元素都有以下这些属性:

offsetWidth

clientWidth scrollWidth
offsetHeight clientHeight crollHeight
offsetLeft clientLeft scrollLeft
offsetTop clientTop scrollTop
offsetParent    

  为了理解client和scroll属性,你需要知道元素的实际内容可能比分配用来容纳的盒子更大,因此单个元素可能有滚动条。内容区域是视口,就像浏览器窗口,当实际内容比视口大,需要把元素滚动套位置考虑进去。

  clientWidth和clientHeight类似offsetWidth和offsetHeight,区别在于它们不包含边框大小。只包含内容和内边距。同时,如果浏览器在内边距和边框之间添加了滚动条,clientWidth和clientHeight不包含滚动条尺寸。在文档的根元素上查询这些属性时,它们的返回值和窗口的innerWidth和innerHeight属性值相等。

  clientLeft和clientTop属性没什么用:它们返回元素的内边距的外边框和它的边框的外边缘之间的水平距离和垂直距离。

  scrollWidth和scrollHeight是元素的内容区域加上它的内边距再加上任何溢出内容的尺寸。当内容正好和内容区域匹配没溢出时,这些属性与clientWidth和clientHeight相等。有溢出时,包含了溢出的内容尺寸。

  scollLeft和scrollTop指定元素滚动条的位置。在getScrollOffsets()方法中查询过它们。注意,scrollLeft和scrollTop是可写的,通过设置它们来让元素中的内容滚动(HTML元素并没有类似Window对象的scrollTo()方法。


  DEMO


  下面代码介绍了前面几个函数的使用:

view source
print ?
001<!DOCTYPE html>
002<html lang="en">
003    <head>
004        <meta charset="UTF-8">
005        <title>Title</title>
006        <script type="text/javascript">
007            //以一个对象的x和y属性放回滚动条的位置
008            function getScrollOffsets(w){
009                w = w || window;
010                //除了IE 8以及更早的版本以外,其他浏览器都支持
011                if(w.pageXOffset != null) return {x: w.pageXOffset, y: w.pageYOffset};
012                //对标准模式下的IE
013                var d = w.document;
014                if(document.compatMode == "CSS1Compat")
015                    return {x: d.documentElement.scrollLeft, y: d.documentElement.scrollTop};
016                //对冠以模式下的浏览器
017                return { x: d.body.scrollLeft, y: d.body.scrollTop};
018            }
019 
020            //作为一个对象的w和h属性返回视口的尺寸
021            function getViewportSize(w){
022                //使用指定的窗口, 如果不带参数则使用当前窗口
023                w = w || window;
024 
025                //除了IE8及更早的版本以外,其他浏览器都能用
026                if(w.innerWidth != null)
027                    return {w: w.innerWidth, h: w.innerHeight};
028 
029                //对标准模式下的IE(或任意浏览器)
030                var d = w.document;
031                if(document.compatMode == "CSS1Compat")
032                    return {w: d.documentElement.clientWidth, h: d.documentElement.clientHeight};
033 
034                //对怪异模式下的浏览器
035                return {w: d.body.clientWidth, h: d.body.clientHeight};
036            }
037 
038            //元素相对于文档的坐标位置
039            function getElementRect(e){
040                var box = e.getBoundingClientRect();
041                var offsets = getScrollOffsets();
042                var x = box.left + offsets.x;
043                var y = box.top + offsets.y;
044 
045                return {x:x, y: y};
046            }
047 
048            //元素尺寸
049            function getElementSize(e){
050                var box = getElementRect(e);
051                var w = box.width || box.right - box.left;
052                var h = box.height || box.bottom - box.top;
053 
054                return {w: w, h: h};
055            }
056 
057            //滚动到浏览器最底部
058            function scrollToBottom(){
059                //获取文档和视口的高度
060                var documentHeight = document.documentElement.offsetHeight;
061                var viewportHeight = window.innerHeight; //或使用上面的getViewPortSize()方法
062 
063                //然后,滚动让最后一页在视口中可见
064                window.scrollTo(0, documentHeight - viewportHeight);
065            }
066 
067            //计算元素位置
068            function getElementPosition(e){
069                var x = 0, y = 0;
070                while(e != null){
071                    x += e.offsetLeft;
072                    y += e.offsetTop;
073                    e = e.offsetParent;
074                }
075 
076            return {x: x, y: y };
077            }
078        </script>
079    </head>
080    <body>
081        <button id="scrolltoBottomBtn">滚动到浏览器</button>
082        <div style="height: 400px; background: red;">
083 
084        </div>
085        <button id="btn">获取滚动条位置</button>
086        <button id="viewportBtn">获取视口尺寸</button>
087        <button id="eleRectBtn">元素文档坐标</button>
088        <script type="text/javascript">
089            var btn = document.getElementById("btn");
090            btn.onclick = function(event){
091            console.log(getScrollOffsets());
092            }
093 
094            var viewportBtn = document.getElementById("viewportBtn");
095            viewportBtn.onclick = function(event){
096            console.log(getViewportSize());
097            }
098 
099            var eleRectBtn = document.getElementById("eleRectBtn");
100            eleRectBtn.onclick = function(eevent){
101            console.log(getElementRect(this));
102            }
103 
104            var scrolltoBottomBtn = document.getElementById("scrolltoBottomBtn");
105            scrolltoBottomBtn.onclick = function(){
106            scrollToBottom();
107            }
108        </script>
109    </body>
110</html>

这篇关于HTML文档坐标和视口坐标的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/995238

相关文章

vue解决子组件样式覆盖问题scoped deep

《vue解决子组件样式覆盖问题scopeddeep》文章主要介绍了在Vue项目中处理全局样式和局部样式的方法,包括使用scoped属性和深度选择器(/deep/)来覆盖子组件的样式,作者建议所有组件... 目录前言scoped分析deep分析使用总结所有组件必须加scoped父组件覆盖子组件使用deep前言

VUE动态绑定class类的三种常用方式及适用场景详解

《VUE动态绑定class类的三种常用方式及适用场景详解》文章介绍了在实际开发中动态绑定class的三种常见情况及其解决方案,包括根据不同的返回值渲染不同的class样式、给模块添加基础样式以及根据设... 目录前言1.动态选择class样式(对象添加:情景一)2.动态添加一个class样式(字符串添加:情

React实现原生APP切换效果

《React实现原生APP切换效果》最近需要使用Hybrid的方式开发一个APP,交互和原生APP相似并且需要IM通信,本文给大家介绍了使用React实现原生APP切换效果,文中通过代码示例讲解的非常... 目录背景需求概览技术栈实现步骤根据 react-router-dom 文档配置好路由添加过渡动画使用

SpringBoot3集成swagger文档的使用方法

《SpringBoot3集成swagger文档的使用方法》本文介绍了Swagger的诞生背景、主要功能以及如何在SpringBoot3中集成Swagger文档,Swagger可以帮助自动生成API文档... 目录一、前言1. API 文档自动生成2. 交互式 API 测试3. API 设计和开发协作二、使用

使用Vue.js报错:ReferenceError: “Vue is not defined“ 的原因与解决方案

《使用Vue.js报错:ReferenceError:“Vueisnotdefined“的原因与解决方案》在前端开发中,ReferenceError:Vueisnotdefined是一个常见... 目录一、错误描述二、错误成因分析三、解决方案1. 检查 vue.js 的引入方式2. 验证 npm 安装3.

vue如何监听对象或者数组某个属性的变化详解

《vue如何监听对象或者数组某个属性的变化详解》这篇文章主要给大家介绍了关于vue如何监听对象或者数组某个属性的变化,在Vue.js中可以通过watch监听属性变化并动态修改其他属性的值,watch通... 目录前言用watch监听深度监听使用计算属性watch和计算属性的区别在vue 3中使用watchE

python解析HTML并提取span标签中的文本

《python解析HTML并提取span标签中的文本》在网页开发和数据抓取过程中,我们经常需要从HTML页面中提取信息,尤其是span元素中的文本,span标签是一个行内元素,通常用于包装一小段文本或... 目录一、安装相关依赖二、html 页面结构三、使用 BeautifulSoup javascript

基于C#实现将图片转换为PDF文档

《基于C#实现将图片转换为PDF文档》将图片(JPG、PNG)转换为PDF文件可以帮助我们更好地保存和分享图片,所以本文将介绍如何使用C#将JPG/PNG图片转换为PDF文档,需要的可以参考下... 目录介绍C# 将单张图片转换为PDF文档C# 将多张图片转换到一个PDF文档介绍将图片(JPG、PNG)转

Vue3 的 shallowRef 和 shallowReactive:优化性能

大家对 Vue3 的 ref 和 reactive 都很熟悉,那么对 shallowRef 和 shallowReactive 是否了解呢? 在编程和数据结构中,“shallow”(浅层)通常指对数据结构的最外层进行操作,而不递归地处理其内部或嵌套的数据。这种处理方式关注的是数据结构的第一层属性或元素,而忽略更深层次的嵌套内容。 1. 浅层与深层的对比 1.1 浅层(Shallow) 定义

这15个Vue指令,让你的项目开发爽到爆

1. V-Hotkey 仓库地址: github.com/Dafrok/v-ho… Demo: 戳这里 https://dafrok.github.io/v-hotkey 安装: npm install --save v-hotkey 这个指令可以给组件绑定一个或多个快捷键。你想要通过按下 Escape 键后隐藏某个组件,按住 Control 和回车键再显示它吗?小菜一碟: <template