成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

原生 JS 獲取元素的尺寸和位置

simon_chen / 3452人閱讀

摘要:如何獲取元素的真實(shí)尺寸呢通過(guò)下,介紹。這個(gè)函數(shù)主要提供給我們?cè)睾蛯挾仍趦?nèi)的一系列值仍然不要妄想通過(guò)獲得,加上原先的,就可以減去和的值獲得元素的真實(shí)尺寸。

關(guān)于元素的尺寸和位置,這原本是 CSS 干的事,但更多的時(shí)候需要用 JavaScript 來(lái)獲取這些參數(shù),比如一個(gè)很好的例子 js 實(shí)現(xiàn)的圖片瀑布流。

在介紹 JS 中的例子之前,先來(lái)說(shuō)明一下 css 中的元素尺寸。

CSS 中的 width 和 height

先開(kāi)個(gè)頭吧,一個(gè)元素所占據(jù)的物理尺寸包括以下幾個(gè)部分,由內(nèi)到外分別是內(nèi)容,padding,border,margin,這些值加到一起才算是一個(gè)元素真實(shí)尺寸。這里面并沒(méi)有把滾動(dòng)條的寬度算上,因?yàn)闈L動(dòng)條時(shí)占用 padding 的寬度的,如果 padding 寬度小于滾動(dòng)條,那么滾動(dòng)條多出來(lái)的部分將占用內(nèi)容的寬度。padding與滾動(dòng)條關(guān)系。

比如下面的這個(gè)例子:

// html
// css style .test{ width:100px; height: 100px; padding:10px; border:2px solid black; margin: 5px; }

上圖是 chrome 調(diào)試下的 styles,所以 .test 的實(shí)際寬度應(yīng)該是 100px + 20px + 4px + 10px = 134px,這里把 margin 也算進(jìn)去,高度的計(jì)算同理。知道這一點(diǎn)很重要,當(dāng)我們需要精確設(shè)定元素寬度的時(shí)候,就不會(huì)因?yàn)槌叽邕^(guò)大而把元素?cái)D到下一行。

不過(guò),這是入門(mén)級(jí)的 CSS。除此之外,還需要知道一個(gè)非常重要的 CSS 樣式,即 box-sizing,可參考 MDN 上的介紹。

/* 關(guān)鍵字值 */
box-sizing: content-box;
box-sizing: border-box;

box-sizing 有兩個(gè)關(guān)鍵字(據(jù)說(shuō)還有一個(gè) padding-box,反正我在 chrome 上測(cè)試不成功),content-box 是默認(rèn)值,此時(shí) width 只表示內(nèi)容 content,border-box 表示元素的 width 等于 content + padding + border 三者之和。border-box 非常有用,尤其當(dāng)我們?cè)谑褂?100% 來(lái)規(guī)定寬高的時(shí)候,如果元素存在 border 或 padding,將直接導(dǎo)致元素的實(shí)際大小大于 100%,估計(jì)還有人記得 calc 帶來(lái)的痛苦。

修改上面的例子:

.test{
  width:100px;
  height: 100px;
  padding:10px;
  border:2px solid black;
  margin: 5px;
  box-sizing: border-box;
}

76 + 20 + 4 = 100px,此時(shí)的 width 表示三者之和,而內(nèi)容的寬度只有 76px 了。

JS 獲取元素尺寸

千萬(wàn)不要嘗試用 element.style.width 或 element.style.height 來(lái)獲得元素的高度和寬度,它們的默認(rèn)值都是 0,除非你在 html 元素里面設(shè)置,否則 js 是無(wú)法獲得 css 的樣式的,必須要用其他的方法。比如下面這段代碼 element.style.width 的值才是 100px:

JS 中 element 對(duì)象提供 offsetHeight, scrollHeight, clientHeight(每個(gè)都對(duì)應(yīng) width),其中:

offsetHeight 可以用來(lái)計(jì)算元素的物理空間,此空間包括內(nèi)容,padding 和 border(還包括滾動(dòng)條的寬度,但大多時(shí)候滾動(dòng)條的寬度是計(jì)算到 padding 和內(nèi)容中的)。

var test = document.getElementsByClassName("test")[0];
test.offsetHeight // 100

scrollHeight 用來(lái)計(jì)算可滾動(dòng)容器的大小,包括不可見(jiàn)的部分,比如一個(gè) 300*300 的容器放入一個(gè) 600*600 的圖片,此時(shí) scrollHeight 為 600,當(dāng)然,scrollHeight 的值需要加上 padding 的值。

clientHeight 表示可視區(qū)域,包括內(nèi)容和 padding ,如果有滾動(dòng)條,還需要減去滾動(dòng)條的寬度

舉個(gè)例子,還是之前那個(gè) test,加入 test2:

//css .test{ overflow: auto; //新增 } .test2{ width: 150px; height: 150px; background-color: gray; }

來(lái)看一看 test 的輸出值是多少:

var test = document.getElementsByClassName("test")[0];
test.offsetHeight // 100
test.scrollHeight // 170
test.clientHeight // 79

此時(shí)滾動(dòng)條的寬度是 17px,根據(jù)前面的介紹,滾動(dòng)條時(shí)占用 padding 和 content 寬度的,而 17px 大于 padding 的 10px,故還有 7px 會(huì)占據(jù) content。

分析一下,offsetHeight 的值是 100,padding 10px,滾動(dòng)條雖然存在,但是占了 padding 和內(nèi)容的空間,offsetHeight 的值是 4+20+76 = 100px。scrollHeight 的值是可滾動(dòng)的范圍加上padding 值,同樣不包括滾動(dòng)條,即 150+20 = 170px。clientHeight 的值是可見(jiàn)區(qū)域,但是不包括滾動(dòng)條的值(滾動(dòng)條。。。),所以20+76-17 = 79px。

其實(shí)也不是非常復(fù)雜。這個(gè)時(shí)候可以得出滾動(dòng)條寬度的計(jì)算:offsetHeight 減去 border 和 clientHeight 的和就是滾動(dòng)條寬度。

獲取元素內(nèi)容的尺寸

剛說(shuō)了半天,還是無(wú)法獲得元素內(nèi)容的尺寸,最接近內(nèi)容寬度的是 clientHeight,在沒(méi)有滾動(dòng)條的情況下,減去 padding 值就是內(nèi)容的尺寸。

如何獲取元素的真實(shí)尺寸呢?

通過(guò) getComputedStyle (IE 下 currentStyle),MDN 介紹。

getComputedStyle 這個(gè)函數(shù)主要提供給我們?cè)?border 和 padding 寬度在內(nèi)的一系列值(仍然不要妄想通過(guò) element.style.border-width 獲得),加上原先的 offsetHeight,就可以減去 border 和 padding 的值獲得元素的真實(shí)尺寸。

// 考慮 IE 的兼容性
function getStyle(el) { 
  if(window.getComputedStyle) { 
    return window.getComputedStyle(el, null); 
  }else{ 
    return el.currentStyle; 
  } 
} 
function getWH(el, name) { 
  var val = name === "width" ? el.offsetWidth : el.offsetHeight, 
  which = name === "width" ? ["Left", "Right"] : ["Top", "Bottom"]; 
  // display is none 
  if(val === 0) { 
    return 0; 
  } 
  var style = getStyle(el);
  // 左右或上下兩邊的都減去
  for(var i = 0, a; a = which[i++];) { 
    val -= parseFloat( style["border" + a + "Width"]) || 0; 
    val -= parseFloat( style["padding" + a ] ) || 0; 
  } 
  return val; 
}
// 測(cè)試,正確
getWH(test, "width"); // 76
獲取元素的位置

在這里先隆重推出一個(gè)重量級(jí)嘉賓函數(shù),即 getBoundingClientRect,貼上 MDN 鏈接。

element.getBoundingClientRect() 會(huì)返回一個(gè)數(shù)組,比如:

test.getBoundingClientRect();
  bottom:108
  height:100
  left:13
  right:113
  top:8
  width:100

其中,width 和 height 跟 element.offset 的值是一致的,left bottom 等值則表示距離瀏覽器窗口的距離,如果要獲得元素的位置,只需要得到 left 和 top 的值即可,

var X= test.getBoundingClientRect().left;
var Y =test.getBoundingClientRect().top;
//再加上滾動(dòng)距離,就可以得到絕對(duì)位置
var X= test.getBoundingClientRect().left+document.body.scrollLeft;
var Y =test.getBoundingClientRect().top+document.body.scrollTop;

此方法之外,還有其他方法。比如每個(gè)元素都有 offsetTop 和 offsetLeft 屬性,表示距離父容器左、上角的邊距,offsetParent 表示父容器,先得到距離父容器的距離,依次累加,得到絕對(duì)位置。

function getPosition(element, name){
  name = name.toLowerCase().replace("left", "Left").replace("top", "Top");
  var offset = "offset" + name;
  var actualLeft = element[offset];
  var current = element.offsetParent;
  while (current !== null){
    actualLeft += current[offset];
    current = current.offsetParent;
  }
  return actualLeft;
}
getPosition(test,"left") // 13
getPosition(test,"top") // 8

結(jié)果和 getBoundingClientRect() 值一樣,有時(shí)候需要考慮是相對(duì)于屏幕的位置還是絕對(duì)位置,然后再做進(jìn)一步的計(jì)算。

總結(jié)

感覺(jué)最近的文章越來(lái)越水,主要是因?yàn)樽罱荚谧セA(chǔ),學(xué)習(xí) nodejs 和 ES6 的基本語(yǔ)法,沒(méi)時(shí)間去看一下比較流行的框架。白天又特別忙,學(xué)校里又一大堆事情,各種煩躁。共勉!

參考

用Javascript獲取頁(yè)面元素的位置
關(guān)于元素的尺寸(dimensions) 說(shuō)明
height、clientHeight、scrollHeight、offsetHeight區(qū)別

歡迎來(lái)我博客交流。

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/91246.html

相關(guān)文章

  • 從GGEditor一個(gè)案例看JS原生拖拽功能

    摘要:涉及部分的,不會(huì)對(duì)理解全局產(chǎn)生干擾。在上監(jiān)聽(tīng)事件,當(dāng)為畫(huà)布時(shí),通過(guò)創(chuàng)建一個(gè)藍(lán)色虛線框移動(dòng)的時(shí)候,更新的位置在上監(jiān)聽(tīng)事件,落在畫(huà)布時(shí),創(chuàng)建一個(gè)的節(jié)點(diǎn)從而完成整個(gè)拖拽添加元素的功能。 showImg(https://segmentfault.com/img/remote/1460000019564977);showImg(https://segmentfault.com/img/remot...

    PingCAP 評(píng)論0 收藏0
  • 從GGEditor一個(gè)案例看JS原生拖拽功能

    摘要:涉及部分的,不會(huì)對(duì)理解全局產(chǎn)生干擾。在上監(jiān)聽(tīng)事件,當(dāng)為畫(huà)布時(shí),通過(guò)創(chuàng)建一個(gè)藍(lán)色虛線框移動(dòng)的時(shí)候,更新的位置在上監(jiān)聽(tīng)事件,落在畫(huà)布時(shí),創(chuàng)建一個(gè)的節(jié)點(diǎn)從而完成整個(gè)拖拽添加元素的功能。 showImg(https://segmentfault.com/img/remote/1460000019564977);showImg(https://segmentfault.com/img/remot...

    taowen 評(píng)論0 收藏0
  • 瀑布流插件Masonry中文文檔【翻譯】

    摘要:本位為官方文檔翻譯,原始鏈接安裝下載下載壓縮或未壓縮的壓縮未壓縮在直接飲用文件。排列未加載完成的圖片時(shí)會(huì)導(dǎo)致元素的重疊,可以解決這個(gè)問(wèn)題。布局組件尺寸尺寸配置項(xiàng)和可以可以設(shè)置組件的列寬和間距。增加移除控件在瀑布流末尾增加新控件并重排。 本位為Masonry官方文檔翻譯,原始鏈接 安裝Install 下載 下載壓縮或未壓縮的masonry masonry.pkgd.min.js (壓縮...

    soasme 評(píng)論0 收藏0
  • 【前端性能優(yōu)化】高性能JavaScript整理總結(jié)

    摘要:然后執(zhí)行環(huán)境會(huì)創(chuàng)建一個(gè)活動(dòng)對(duì)象,活動(dòng)對(duì)象作為函數(shù)運(yùn)行的變量對(duì)象,包含所有局部變量命名參數(shù)參數(shù)集合和,當(dāng)執(zhí)行環(huán)境銷(xiāo)毀,活動(dòng)對(duì)象也被銷(xiāo)毀。 高性能JavaScript整理總結(jié) 關(guān)于前端性能優(yōu)化:首先想到的是雅虎軍規(guī)34條然后最近看了《高性能JavaScript》大概的把書(shū)中提到大部分知識(shí)梳理了下并加上部分個(gè)人理解這本書(shū)有參考雅虎特別性能小組的研究成果,所以跟34 軍規(guī)有很多相似之處有不當(dāng)之...

    zzbo 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<