深入理解元素視圖的3個方法

来源:http://www.cnblogs.com/xiaohuochai/archive/2016/09/02/5832712.html
-Advertisement-
Play Games

[1]getBoundingClientRect [2]getClientRects [3]elementFromPoint ...


×
目錄
[1]BoundingClient [2]ClientRects [3]elementFromPoint

前面的話

  前面介紹了offset偏移client可視區scroll滾動,這三部分主要從屬性的角度來對元素尺寸信息進行獲取和修改。本文主要介紹元素視圖的三個方法,包括getBoundingClientRect()、getClientRects()和elementFromPoint()

 

getBoundingClientRect()

  判斷一個元素的尺寸和位置是簡單的方法就是使用getBoundingClientRect()

  Element.getBoundingClientRect()方法返回一個對象,該對象提供當前元素節點的大小、它相對於視口(viewport)的位置等信息。但是,各個瀏覽器返回的對象包含的屬性不相同

firefox: top left right bottom width height x y(其中,x=left,y=top)
chrome/safari/IE9+:top left right bottom width height
IE8-:  top left right bottom

  問題來了,該方法返回的width和height是可視寬高client,還是滾動寬高scroll,或者是偏移寬高offset,或者是設置寬高呢

<div id="test" style="width: 100px;height: 100px;padding: 10px;line-height: 200px;border:1px solid black;overflow:scroll">內容</div>    
<script>
//chrome/safari: 220(10+200+10)
//firefox/IE: 210(10+200)
console.log(test.scrollHeight)
//103(100+10+10-17)
console.log(test.clientHeight)
//122(100+10+10+1+1)
console.log(test.offsetHeight)
//122(100+10+10+1+1)
console.log(test.getBoundingClientRect().height)
</script>

  由代碼結果看出,該方法返回的寬高是偏移寬高offset

Element.getBoundingClientRect().width =  border-left-width + padding-left + width + padding-right + border-right-width

Element.getBoundingClientRect().height =  border-top-width + padding-top + height + padding-bottom + border-bottom-width

  下麵來分析top、left、right、bottom這四個值

  top:   元素頂部相對於視口的縱坐標

  left:  元素左邊界相對視口的橫坐標

  right: 元素右邊界相對視口的橫坐標

  bottom:元素底部相對於視口的縱坐標

bottom = top + height
right = left + width

  [註意]該方法的所有屬性值都沒有單位,且給定的是元素在頁面中相對於視口的位置

  問題又來了,相對於視口和相對於頁面有什麼區別。理論上,與absolute和fixed的區別類似,但表現上與它們正相反。發生滾動時,fixed元素保持不動是為了保持與視口的原始距離;而發生滾動時,getBoundingClientRect()方法的top、left、right、bottom這四個值相應的發生改變,是因為元素位置移動了,與視口距離自然也改變了

bug

  IE7-瀏覽器把視口的左上角坐標設置為(2,2),其他瀏覽器則將(0,0)作為起點坐標

<body style="margin:0">
<div id="test" style="width: 100px;height: 50px;padding: 10px;line-height: 200px;overflow:scroll;border:1px solid black">內容</div>    
<script>
//chrome/firefox/safari/IE8+ 0 72(50+10+10+1+1)
//IE7- 2 74(72+2)
console.log(test.getBoundingClientRect().top,test.getBoundingClientRect().bottom)

//chrome/firefox/safari/IE8+ 0 122(100+10+10+1+1)
//IE71 2 124(122+2)
console.log(test.getBoundingClientRect().left,test.getBoundingClientRect().right)
</script>
</body>

相容

  可以利用IE7-瀏覽器中特性節點的specified屬性實現瀏覽器識別

function getBoundingClientRect(obj){
    var temp = obj.getBoundingClientRect();
    //IE7-瀏覽器
    if(Boolean(obj.attributes[0]) && !obj.attributes[0].specified){
        return{
            left: temp.left -2,
            top: temp.top -2,
            right: temp.right -2,
            bottom: temp.bottom -2
        }
    }else{
        return temp;
    }    
}

 

getClientRects()

  getClientRects()方法與getBoundingClientRect()不同,該方法是一個返回元素的數個矩形區域的類數組對象。每個類數組對象的參數與getBoundingClientRect()方法相同,每個矩形都有bottom、height、left、right、top和width六個屬性,表示它們相對於視口的四個坐標,以及本身的高度和寬度

  如果應用於塊級元素,則getClientRects()[0]和getBoundingClientRect()的屬性返回相同的值,且IE7-瀏覽器在getClientRects()方法中,同樣存在 視口左上角坐標被設置為(2,2)的bug

<body style="margin:0">
<div id="test" style="width: 100px;height: 50px;padding: 10px;line-height: 200px;overflow:scroll;border:1px solid black">內容</div>    
<script>
//其他瀏覽器返回0 0, IE7-瀏覽器返回 2 2 
console.log(test.getClientRects()[0].top,test.getBoundingClientRect().top)
//其他瀏覽器返回0 0, IE7-瀏覽器返回 2 2 
console.log(test.getClientRects()[0].left,test.getBoundingClientRect().left)
//72(50+10+10+1+1) 72
console.log(test.getClientRects()[0].height,test.getBoundingClientRect().height)
</script>

  實際上,該方法主要用於內聯元素,內聯元素有多少行,該方法返回的對象有多少個成員。這個方法主要用於判斷行內元素是否換行,以及行內元素的每一行的位置偏移

<body style="margin:0">
<div style="width:100px;"><span id="el">
Hello World
Hello World
Hello World
</span></div>
<script>
console.log(el.getClientRects().length); // 3
console.log(el.getClientRects()[0].left); // 0
console.log(el.getClientRects()[0].right); // 88
console.log(el.getClientRects()[0].bottom); // 17
console.log(el.getClientRects()[0].height); // 16
console.log(el.getClientRects()[0].width); // 88
</script>
</body>

 

elementFromPoint()

  getBoundingClientRect(x,y)方法使我們能在視口中判定元素的位置。但有時我們想反過來,判定在視口中的指定位置上有什麼元素。這可以用Document對象的elementFromPoint()方法來判定。傳遞X和Y坐標(相對於視口),該方法選擇在指定坐標的最上層和最裡層的Element對象。如果指定的點在視口以外,elementFromPoint()返回null

  [註意]最上層是指z-index最大的元素;最裡層是指最裡層的子元素

  這個方法可以用來檢測元素是否發生重疊或是碰撞

<body style="margin:0">
<div id="test" style="width: 100px;height: 100px;">
    <span id="span1">123</span>
</div>
<script>
console.log(document.elementFromPoint(2,2).id); //span1
</script>

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • turbine介紹:附7 turbine turbine部署:第二章 部署war包到tomcat 一、總體圖 前提: 假設有兩個服務A/B分別布在兩台機器上(這裡為了簡單,使用一個服務myboot來代替),A、B都訪問myboot2的兩個介面getHotelInfo和getHotelInfo2(這兩 ...
  • ApplicationContext作為資源載入器;ApplicationContext作為事件發佈者; Java原生提供了事件發佈機制 EventObject對象作為發佈的事件,EventListener作為處理髮布事件的監聽器。但是其並沒有提供發佈者的角色來橋接EventObject和Event ...
  • 希望你看了此小隨 可以實現自己的MVC框架 也祝所有的程式員身體健康一切安好 ——久伴深海丶默 1.什麼是前端控制器(font controller)。Java Web中的前端控制器是應用的門面,簡單的說所有的請求都會經過這個前端控制器,由前端控制器根據請求的內容來決定如何處理並將處理的結果返回給瀏 ...
  • 一、定義 策略模式定義了一系列的演算法,並將每一個演算法封裝起來,而且使它們還可以相互替換。策略模式讓演算法獨立於使用它的客戶而獨立變化。 二、UML圖:(來自百度百科) 根據上面的UML圖,我們需要的類有: 1)Content類 2)Strategy抽象類 3)若幹ConcreteStrategy類 三 ...
  • 從Tomcat源碼提煉出設計模式-門面設計模式: 概念 外部訪問內部,耦合度增加,不利於擴展。而門面模式在內部基礎上進行再度封裝,只提供外部想要的方法。這時訪問方式由“外部 內部”變為了“外部 門面對象 內部”。 目的 1 有些內部對象的方法,不想對外部得知使用,使用內部對象Facade模式,只提供 ...
  • 100萬條以上數據,想要做0.01秒以內響應的中文全文索引怎麼辦,SQL Server,MySQL不精通?阿裡雲開放搜索服務打開了新思路... ...
  • 本文為原創文章,轉載請註明出處,謝謝 分散式鎖 1、原理 建立表示鎖的父節點(圖中locker節點) 每個爭搶鎖的伺服器在locker節點下創建有序的臨時節點 判斷自己是否搶到鎖:獲取locker下所有子節點,併進行從小到大排序,判斷自己創建的臨時節點是否是最小節點。 是最小節點,說明搶到鎖,執行相 ...
  • 顯示上面照片效果css ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...