HTML中常见的各种位置距离(clientTop clientLeft clientWidth ClientHeight offsetleft offsettop offsetwidth offsetheight等等)以及dom中的坐标讨论

最近在学习JavaScript,特意买了一本犀牛角书来看看,尼玛一千多页,看的我头昏脑涨,翻到DOM这章节,突然记起平常在使用DOM时,碰到了好多的这个dom里面的各种宽度,高度,特意在此写一写,写的不好或者写错了,欢迎各位指正。好了废话不多说,开始进入主题。

这篇文章主要讨论两点:

一、DOM中各种宽度、高度

二、DOM中的坐标系

下面我们看看DOM中都有一些什么宽度、高度。

常见的


offsetWidth


clientWidth


scrollWidth


offsetHeight


clientHeight


scrollHeight


offsetLeft


clientLeft


scrollLeft


offsetTop


clientTop


scrollTop

下面我们来一一讲解

offsetleft:元素的边框的外边缘距离与已定位的父容器(offsetparent)的左边距离(不包括元素的边框和父容器的边框)。

offsettop:同理是指元素的边框的外边缘距离与已定位的父容器(offsetparent)的上边距离(不包括元素的边框和父容器的边框)。

offsetwidth:描述元素外尺寸宽度,是指 元素内容宽度+内边距宽度(左右两个)+边框(左右两个),不包括外边距和滚动条部分。

offsetheight:同理 描述元素外尺寸高度,是指 元素内容高度+内边距高度(上下两个)+边框(上下两个),不包括外边距和滚动条部分。

下面我们看一下一段代码

<div id="divParent" style="padding: 8px; background-color:#CCC; position: relative;">
        <div id="divChild" style="background-color:#C00; margin: 30px; padding: 10px;
            height: 200px; width: 200px; border: solid 10px  #0000CC;">
        </div>
    </div>

  

<script type="text/javascript">
    var div = document.getElementById(‘divChild‘);
    var offsetHeight = div.offsetHeight;
    var offsetWidth = div.offsetWidth;
    div.innerHTML += ‘offsetHeight: ‘ + offsetHeight + ‘<br />‘;
    div.innerHTML += ‘offsetWidth: ‘ + offsetWidth + ‘<br />‘;

    var offsetLeft = div.offsetLeft;
    var offsetTop = div.offsetTop;
    div.innerHTML += ‘offsetLeft: ‘ + offsetLeft + ‘<br />‘;
    div.innerHTML += ‘offsetTop: ‘ + offsetTop + ‘<br />‘;

    var offsetParent = div.offsetParent;
    div.innerHTML += ‘offsetParent: ‘ + offsetParent.id + ‘<br />‘;

</script>

  看一下效果

现在我们按照上面的说明比对一下

offsetleft:(div id=divChild)margin 30(外边距--距离父容器左边30px)+ (div id=divParent) padding 8 (父容器的内边左距离 8px)=38

offsettop:margin 30+ padding 8=38;

offsetwidth:本身的宽度(200)+内边距左右(10*2)+边框左右(10*2)=240;

offsetheight:同理;

下面我们来看下第二组

clientleft:元素的内边距的外边缘和边框的外边缘的距离,实际就是边框的左边框宽度

clienttop:同理边框的上边框的宽度

clientwidth:用于描述元素内尺寸宽度,是指 元素内容+内边距 大小,不包括边框、外边距、滚动条部分

clientheight:同理 用于描述元素内尺寸高度,是指 元素内容+内边距 大小,不包括边框、外边距、滚动条部分

我们只更改一下JavaScript的代码,DOM不改变

chrome 41 版本

clientleft:左边框宽度 10;

clienttop:10;

clientwidth:本身宽度(200)+内边距(10*2)=220;

clientheight:本身高度(200)+内边距(10*2)=220;

正确,下面我们来看下嵌套的div盒子模型图

div id=divChild

div id=divParent

符合我们上面所说的,下面我们看下

scrollwidth:内容区域尺寸加上内边距加上溢出尺寸,当内容正好和内容区域匹配没有溢出时,这些属性与clientWidth和clientHeight相等

scrollheight:同上

scrolltop:滚动条上方卷去的高度

scrollleft:滚动条左边卷去的宽度

好现在我们 看一段代码

 <div id="divParent"  style="padding: 8px; background-color: #aaa; height:200px; width:300px; overflow:auto" >
        <div id="divChild" style="background-color: #0f0;height: 400px; width: 500px; border: solid 10px #f00;">
        </div>
    </div>

  

<script type="text/javascript">
    var divParent= document.getElementById("divParent");
    var scrollwidth = divParent.scrollWidth;
    var scrollheight = divParent.scrollHeight;
    var scrolltop = divParent.scrollTop;
    var scrollleft = divParent.scrollLeft;
    divChild.innerHTML += ‘clientWidth: ‘ + scrollwidth + ‘<br />‘;
    divChild.innerHTML += ‘clientHeight: ‘ + scrollheight + ‘<br />‘;
</script>

  现在我们要计算id=divParent的scrollheight,和scrollwidth。根据上面的说明,我们知道应该按照下面的公式计算

scrollwidth=子div的宽度(500)+子div的边框(10*2)+父容器的padding(8)=528

scrollwidth=子div的高度(400)+子div的边框(10*2)+父容器的padding(8)=428

现在我们验证一下

我们发现在 ie8及之后的 浏览器 为428,firework 也为428;而 chrome Safari opera 都为436;

因此我们可以猜测 chrome和 Safari、opera 在计算 scrollheight时,加上了 父容器的下 padding(8) 即 428+8=436;

下面我们在看看scrolltop和scrollleft的值怎么样

测试了好几个浏览器都发现其值为零,这是肿么回事,尼玛,万能的百度、谷歌,原来我的滚动条一直在顶端和左端,没有卷走高度,我们修改一下代码,将显示内容绑定到onscroll事件

<script type="text/javascript">
    var divParent = document.getElementById("divParent");
    divParent.onscroll = function () {
        divChild.innerHTML = "";
        var scrollwidth = divParent.scrollWidth;
        var scrollheight = divParent.scrollHeight;
        var scrolltop = divParent.scrollTop;
        var scrollleft = divParent.scrollLeft;
        divChild.innerHTML += ‘clientWidth: ‘ + scrollwidth + ‘<br />‘;
        divChild.innerHTML += ‘clientHeight: ‘ + scrollheight + ‘<br />‘;
        divChild.innerHTML += ‘scrolltop: ‘ + scrolltop + ‘<br />‘;
        divChild.innerHTML += ‘scrollleft: ‘ + scrollleft + ‘<br />‘;
    }
</script>

终于发现值改变了

另外我们在使用window的pageXOffset,pageYOffset求浏览器滚动条的卷去的高度和左边的距离,同时scrolltop、scrollleft也可以,但是要注意

当我们网页页面申明了dtd(文档模型时), 使用document.documentElement.scrolltop scrollleft获取浏览器卷去上面的距离,左边卷去的距离。

如果网页没有申明dtd(怪异模式时),使用document.body.scrolltop  scrollleft获取上边卷去的距离、左边卷去的距离。

什么是dtd,这里简单解释一下,我们使用vs新建页面时,在每个网页的顶端 一般会有这样的申明

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">

这里申明了 网页的模型 ,再浏览器加载网页时,会选择对应的模式渲染网页。

现在我们来测试一下

 <div id="divParent"   style="padding: 8px; background-color: #aaa; height:1200px; width:300px; overflow:auto" >
        <div id="divChild" style="background-color: #0f0;height: 400px; width: 500px; border: solid 10px #f00; position:absolute">
        </div>
    </div>

  

<script type="text/javascript">
    var divParent = document.getElementsByTagName("body")[0];
    divParent.onscroll = function () {
        divChild.innerHTML = "";
        divChild.style.top = window.pageYOffset + "px";
        var scrollwidth = divParent.scrollWidth;
        var scrollheight = divParent.scrollHeight;
        var scrolltop = divParent.scrollTop;
        var scrollleft = divParent.scrollLeft;
        var pagexoffset = window.pageXOffset;
        var pageyoffset = window.pageYOffset;
        var documentelementtop = document.documentElement.scrollTop;
        var documentelementleft = document.documentElement.scrollLeft;
        var bodyscrolltop = document.body.scrollTop;
        var bodyscrollleft = document.body.scrollLeft;
        divChild.innerHTML += ‘clientWidth: ‘ + scrollwidth + ‘<br />‘;
        divChild.innerHTML += ‘clientHeight: ‘ + scrollheight + ‘<br />‘;
        divChild.innerHTML += ‘scrolltop: ‘ + scrolltop + ‘<br />‘;
        divChild.innerHTML += ‘scrollleft: ‘ + scrollleft + ‘<br />‘;
        divChild.innerHTML += ‘pagexoffset: ‘ + pagexoffset + ‘<br />‘;
        divChild.innerHTML += ‘pageyoffset: ‘ + pageyoffset + ‘<br />‘;
        divChild.innerHTML += ‘documentelementtop: ‘ + documentelementtop + ‘<br />‘;
        divChild.innerHTML += ‘documentelementleft: ‘ + documentelementleft + ‘<br />‘;
        divChild.innerHTML += ‘bodyscrolltop: ‘ + bodyscrolltop + ‘<br />‘;
        divChild.innerHTML += ‘bodyscrollleft: ‘ + bodyscrollleft + ‘<br />‘;
    }
</script>

  chrome测试结果,我们发现 pagexoffset  document.body.scrolltop 可以用,但是注意 我当前的申明了 html5的<!DOCTYPE html>,说明document.documentelement可以用的。

在来看看 firefox的测试效果,同样申明了文档类型 这里 document.documentelement可以用,尼玛真是日了狗了,难道是html5 文档类型特殊一点,我们测试一下其他类型的文档类型

但是发现结果和上面的一样,好吧,不管上面那些了,下面总结如何兼容scrolltop如果是求 scrolltop 可以取 body.scrolltop 和documen.documentelement.scrolltop的max值,这样不管如何总能取一个正确的值。

我们在上面的章节知道了如何取scrolltop值,接下来我们就来介绍一下dom中的窗口坐标和文档坐标,

窗口坐标:当前显示可见的页面的左上角的坐标(如果出现滚动条,且滚动条发生滚动,则 窗口坐标和文档坐标不一致,因为窗口坐标只显示当前显示的页面的部分,而文档可能因为滚动条的缘故遮盖了);

文档坐标:垂直滚动条在最上方,没有滚动,水平滚动条在最左边,没有滚动时,时的左上方的坐标

如果没有滚动条时,窗口坐标和文档坐标一致。

那如何在窗口坐标和文档坐标之间进行转换呢,这里我们就要用到上面的滚动条卷去的高度和坐标的距离,假设,有一个元素在文档中的y坐标(即垂直方向的)是200px;但是我们通过滚动条向下滚动了75px,那么当前的窗口坐标就为125px;

因此如果scrolltop的值大于0的话  我们窗口的坐标y=文档的坐标y-scrolltop,

同时视口坐标也有对应的方法可以使用

可以通过调用元素getBoundingClientRect方法。方法返回一个有left、right、top、bottom属性的对象,分别表示元素四个位置的相对于视口的坐标。getBoundingClientRect所返回的坐标包含元素的内边距和边框,不包含外边距。兼容性很好,非常好用。

下面贴一段犀牛角上的代码

得到滚动条值

function getScrollOffsets(w) {
            var w = w || window;           //除ie8及更早版本,其他浏览器都能使用
            if (w.pageXoffset != null) {
                return { x: w.pageXoffset, y: pageYoffset };
            }            //标注模式ie(或任何浏览器)
            var d = w.document;
            if (document.compatMode == "CSS1Compat")
                return { x: d.documentElement.scrollLeft, y: d.documentElement.scrollTop };                //怪异模式下(没有申明dtd)
            return { x: d.body.scrollLeft, y: d.body.scrollTop };
        }通过使用这个函数发现也不是很兼容 在chrome中一直是0,说明即使在标准模型中,,即申明了dtd chrome 也不能取documentElement.scrollLeft值。opera也同样不能取值,firefox能取值。因此早平常的使用中我还是 使用math.max(docuemnt.body.scrolltop,docuemnt.documentelement.scrolltop)来使用。好了今天的就介绍到这里。如有错误,欢迎各位博客园朋友指正
时间: 2024-10-09 03:15:18

HTML中常见的各种位置距离(clientTop clientLeft clientWidth ClientHeight offsetleft offsettop offsetwidth offsetheight等等)以及dom中的坐标讨论的相关文章

js和jq中常见的各种位置距离之offsetLeft和position().left的区别(四)

offsetLeft:元素的边框的外边缘距离与已定位的父容器(offsetparent)的左边距离(不包括元素的边框和父容器的边框).position().left:使用position().left方法时事实上是把该元素当绝对定位来处理,获取的是该元素相当于最近的一个拥有绝对或者相对定位的父元素的偏移位置. 附上调试代码: 1 <style> 2 *{margin:0;padding:0;} 3 #parent{ position: relative; padding: 10px; marg

js和jq中常见的各种位置距离之offset和offset()的区别(三)

offsetLeft:元素的边框的外边缘距离与已定位的父容器(offsetparent)的左边距离(不包括元素的边框和父容器的边框). offset().left:返回的是相对于当前文档的坐标,使用offset()方法不管该元素如何定位,也不管其父元素如何定位,都是获取的该元素相对于当前窗口的偏移坐标 附上调试代码: 1 <style> 2 *{margin:0;padding:0;} 3 #parent{ position: relative; padding: 10px; margin:3

ios开发——开发总结篇&amp;开发中常见错误和警告总结(五)

开发中常见错误和警告总结(五) 一.Missing file xxx 如果你在finder中删除了工程里面的文件,xcode上会出现一个警告,Missing file xxx, 有个警告在那恨事不自在.上网着了下,发现了如下解决方法: 1.打开terminal, cd 到刚才你删除的文件的文件夹,就是xcode提示你missing的文件夹 2.执行 svn delete missFileName  二.LLVM GCC Warning 4.3默认使用的是ARC编译,不是原来的GCC编译,所以你编

js事件对象--DOM中的事件对象/IE中的事件对象/跨浏览器的事件对象

事件对象    在触发DOM上的某个事件时,会产生一个事件对象event,这个对象中包含着所有与事件有关的信息.包括导致事件的元素.事件的类型,以及其他与特定事件相关的信息.例如,鼠标操作导致的事件对象中,会包含鼠标位置的信息,而键盘操作导致的事件对象中,会包含与按下的键有关的信息.所有浏览器对象都支持event对象,但支持方式不同. DOM中的事件对象 兼容DOM的浏览器会将一个event对象传入到事件处理程序中.无论指定事件处理程序时使用什么方法(DOM0级或DOM2级),都会传入event

页游中常见的游戏元素位置自适应浏览器大小变化之解决方案

在玩页游中,很常见的一个功能是,当你改变浏览器的大小时,页游中一些元素位置比如聊天框等会自动随着浏览器变化而变化,该功能如何实现了? 哈,解决方式是:监听舞台变化,获取变化后的舞台宽高(stage.stageWidth,stage,stageHeight),然后根据新的舞台宽高,再改变舞台上各个元素的位置. 上代码测试: <?xml version="1.0" encoding="utf-8"?> <s:Application xmlns:fx=&

JS中的位置和宽度:clientWidth、offsetWidth、scrollWidth,clientTop和clientLeft等区别

下面说的这三种宽,都是基于元素被加入到DOW渲染树后,也就是被添加到页面中以后,才可以获取.并且图片这类后加载元素无法获取宽. clientWidth  客户宽   offsetWidth  偏移宽 scrollWidth  滚动宽 直接获取样式的宽度得到的是样式值,带有px单位. 通过 div.clientWidth 获取到的是样式的数值,是没有单位的.他们一个是字符串,一个是数值. div.clientWidth 的值是 width+padding ,有滚动条的时候:width+paddin

C程序中常见的内存操作错误

对C/C++程序员来说,管理和使用虚拟存储器可能是个困难的, 容易出错的任务.与存储器有关的错误属于那些令人惊恐的错误, 因为它们在时间和空间上, 经常是在距错误源一段距离之后才表现出来. 将错误的数据写到错误的位置, 你的程序可能在最终失败之前运行了好几个小时,且使程序中止的位置距离错误的位置已经很远啦.而避免这种噩梦的最好方法就是防范于未然. 幸好<深入理解计算机系统>中有一段讲: C程序中常见的内存操作有关的10种典型编程错误,十分经典, 因此抄写在此, 以便以后随时查看,复习. 把优秀

【转】《深入理解计算机系统》C程序中常见的内存操作有关的典型编程错误

原文地址:http://blog.csdn.net/slvher/article/details/9150597 对C/C++程序员来说,内存管理是个不小的挑战,绝对值得慎之又慎,否则让由上万行代码构成的模块跑起来后才出现内存崩溃,是很让人痛苦的.因为崩溃的位置在时间和空间上,通常是在距真正的错误源一段距离之后才表现出来.前几天线上模块因堆内存写越界1个字节引起各种诡异崩溃,定位问题过程中的折腾仍历历在目,今天读到<深入理解计算机系统>第9章-虚拟存储器,发现书中总结了C程序中常见的内存操作有

2016年4月27日_JAVA学习笔记_JAVA中常见的API(一)String

1.String在JAVA中是一个单独的类,只不过是一种特殊的,专门用来表示字符串的类.之前接触到的创建方式很简单,就是跟C语言中创建变量一样, String aString = "This is a String."; //变量类型为String,变量名为aString,内容为"This is a String.". 在学习API时,接触到了一种特别的创建方式.因为String是一个类,那么就肯定可以用其构造器方法来创建相应的对象. String aString