不少初學者對於JavaScript中的offset、scroll、client一直弄不明白,雖然網上處處均可以看一張圖(圖1),但這張圖太多太雜,而且因爲瀏覽器差別性,圖示也不徹底正確。css
圖一html
不知道你們看到這張圖的第一感受如何,反正個人感受就是「此次第,怎一個亂字了得」。瀏覽器
既然我認爲上圖太多太亂,那麼我就把offset、scroll、client分開說,但願能讓你們完全弄清楚,今天只說offset。測試
w3中offset相關頁面是:http://www.w3.org/TR/cssom-view/#extensions-to-the-htmlelement-interfacecode
在這裏咱們能夠看到,關於offset共有5個東西須要弄清楚:htm
一、offsetParent對象
二、offsetTopblog
三、offsetLeftip
四、offsetWidthci
五、offsetHeight
咱們根據難易程度把以上5點分爲三類來說解。
在分析以前,先來看段測試代碼:
<body> <style type="text/css"> body { border:20px solid #CCC; margin:10px; padding:40px; background:#EEE; } #test { width:400px; height:200px; padding:20px; background:#F60; border:5px solid #888; } </style> <div id="test"></div> <script> var test = document.getElementById("test"); test.innerHTML = "<p>Browser:" + navigator.userAgent + "</p>" + "<p>offsetWidth:" + test.offsetWidth + "</p>" + "<p>offsetHeight:"+test.offsetHeight+"</p>"+ "<p>offsetLeft:"+test.offsetLeft+"</p>"+ "<p>offsetTop:"+test.offsetTop+"</p>"; </script> </body>
這段代碼在各個瀏覽器中的效果如圖:
圖二(IE6/7)
圖三(IE8/9/10)
圖四(Firefox)
圖五(Chrome)
你們能夠看到,上面圖二~圖五中的共同點是 offsetWidth與offsetHeight是一致的,所以這裏放到地起講。
MDN中對offsetWidth的概述和描述是:
Returns the layout width of an element.
Typically, an element's
offsetWidth
is a measurement which includes the element borders, the element horizontal padding, the element vertical scrollbar (if present, if rendered) and the element CSS width.
也就是元素的可視寬度,這個寬度包括元素的邊框(border),水平padding,垂直滾動條寬度,元素自己寬度等。
offsetHeight跟offsetWidth相似,只是方向改成垂直方向上的。
只是咱們的示例中沒有水平和垂直滾動條。另外通過測試能夠發現,即便元素加上水平或垂直滾動條,offsetWidth跟offsetHeight的值是不會更改的,由於瀏覽器渲染時把滾動條的寬度(或高度)算在了元素自己的寬度(或高度)中了。
經過代碼及圖中數值,咱們不難看出:
offsetWidth=(border-width)*2+(padding-left)+(width)+(padding-right)
offsetHeight=(border-width)*2+(padding-top)+(height)+(padding-bottom)
對這兩個概念就總結到這裏,你們如今弄明白了嗎?
offsetWidth與offsetHeight有個特色,就是這兩個屬性的值只與該元素有關,與周圍元素(父級和子級元素無關)。
然而,offsetLeft與offsetTop卻不是這樣,這兩個屬性與offsetParent有關,但在咱們講到offsetParent以前,咱們先無論offsetParent是什麼及怎麼判斷,咱們只要知道offsetLeft和offsetTop與offsetParent有關就好了,上面的示例中offsetParent就是body。
MSDN上對offsetLeft的定義是:
Retrieves the calculated left position of the object relative to the layout or coordinate parent, as specified by the offsetParent property
也就是返回對象元素邊界的左上角頂點相對於offsetParent的左上角頂點的水平偏移量。從這個定義中咱們能夠明確地知道offsetLeft與當前元素的margin-left和offsetParent的padding-left有關。也就是說應該是:
offsetLeft=(offsetParent的padding-left)+(中間元素的offsetWidth)+(當前元素的margin-left)。
offsetTop=(offsetParent的padding-top)+(中間元素的offsetHeight)+(當前元素的margin-top)。
但經過上面的例子咱們能夠看到,當offsetParent爲body時,對於offsetLeft與offsetTop的值有三種,分別是:IE6/7中的40,IE8/9/10 和 Chrome中的70,以及FireFox中的50。
經過這些數值咱們能夠知道,當offsetParent爲body時狀況比較特殊:
在IE8/9/10及Chrome中,offsetLeft = (body的margin-left)+(body的border-width)+(body的padding-left)+(當前元素的margin-left)。
在FireFox中,offsetLeft = (body的margin-left)+(body的padding-left)+(當前元素的margin-left)。
終於到offsetParent了。
offsetParent屬性返回一個對象的引用,這個對象是距離調用offsetParent的元素最近的(在包含層次中最靠近的),而且是已進行過CSS定位的容器元素。 若是這個容器元素未進行CSS定位, 則offsetParent屬性的取值爲根元素的引用。
總的來講兩條規則:
一、若是當前元素的父級元素沒有進行CSS定位(position爲absolute或relative),offsetParent爲body。
二、若是當前元素的父級元素中有CSS定位(position爲absolute或relative),offsetParent取最近的那個父級元素。
上面的示例就是第1條說的狀況,咱們來驗證一下:
咱們把JS改成(添加了一行代碼:紅色部分):
var test = document.getElementById("test"); test.innerHTML = "<p>Browser:" + navigator.userAgent + "</p>" + "<p>offsetParent:" + test.offsetParent.tagName + "</p>" + "<p>offsetWidth:" + test.offsetWidth + "</p>" + "<p>offsetHeight:"+test.offsetHeight+"</p>"+ "<p>offsetLeft:"+test.offsetLeft+"</p>"+ "<p>offsetTop:"+test.offsetTop+"</p>";
FireFox下的效果爲:
圖六
在其餘瀏覽器中效果相同,都是body。
咱們再來驗證一下第2條,測試HTML以下:
<!DOCTYPE html> <html> <head> <title>Demo</title> </head> <body> <style type="text/css"> body { margin:0; padding:0; background:#EEE; } div,ul,li { margin:0; } li { height:20px; line-height:20px; } #test { width:400px; height:250px; padding:20px; background:#F60; border:10px solid #888; } #divtest { margin:30px; position:relative; left:50px; top:70px; padding:20px; } </style> <div id="divtest"> <ul> <li>Test</li> <li>Test</li> </ul> <div id="test"> </div> </div> <script> var test = document.getElementById("test"); test.innerHTML = "<p>Browser:" + navigator.userAgent + "</p>" + "<p>offsetParent:" + test.offsetParent.tagName + "</p>" + "<p>offsetWidth:" + test.offsetWidth + "</p>" + "<p>offsetHeight:"+test.offsetHeight+"</p>"+ "<p>offsetLeft:"+test.offsetLeft+"</p>"+ "<p>offsetTop:"+test.offsetTop+"</p>"; </script> </body> </html>
在FireFox中效果爲:
圖七
在其餘瀏覽器中offsetParent也是一致的。
在這裏咱們也能夠看到,第三點中給出的offsetLeft的計算公式是適用的。
以上的總結但願能對你們有所幫助,在看完本文內容後再回過頭來看文章開頭部分的那張圖(只看offset)部分,是否是清楚了不少?
最後,對於offsetParent爲body的狀況,如今的主流瀏覽器IE8/9/10和Chrome及Firefox都跟定義
offsetLeft=(offsetParent的padding-left)+(中間元素的offsetWidth)+(當前元素的margin-left)。offsetTop=(offsetParent的padding-top)+(中間元素的offsetHeight)+(當前元素的margin-top)。
的不同,對於這一點我也沒有弄明白,若是有朋友知道請不吝賜教。
轉自:http://www.cnblogs.com/jscode/archive/2012/09/03/2669299.html