jQuery 源碼解析(二十四) DOM操作模塊 包裹元素 詳解

来源:https://www.cnblogs.com/greatdesert/archive/2019/11/15/11820454.html
-Advertisement-
Play Games

本節說一下DOM操作模塊里的包裹元素子模塊,該模塊可將當前匹配的元素替換指定的DOM元素,有如下方法: wrap(html) ;在每個匹配元素的外層添加一層DOM元素 ;該方法會遍歷匹配元素集合,在每個元素上調用.wrapAll()方法 ;不同於wrapAll()的是該方法會在每個匹配元素外面都套一 ...


本節說一下DOM操作模塊里的包裹元素子模塊,該模塊可將當前匹配的元素替換指定的DOM元素,有如下方法:

  • wrap(html)               ;在每個匹配元素的外層添加一層DOM元素                                ;該方法會遍歷匹配元素集合,在每個元素上調用.wrapAll()方法        ;不同於wrapAll()的是該方法會在每個匹配元素外面都套一層html元素。
  • wrapAll(html)            ;會將html轉化為一個DOM節點並放在第一個匹配元素的前面,再把其他匹配元素也依次放進去    ;html可以是html片段、選擇器表達式、jQuery對象、DOM元素或函數,下同。
  • wrapInner(html)        ;在每個匹配元素的內容前後包裹HTML元素    ;該方法會遍歷匹配元素集合,並通過調用方法.wrapAll()為每個匹配元素的所有內容包裹一段HTML結構。
  • unwrap()                  ;移除匹配元素集合中每個元素的父標簽,並把匹配元素留在父元素的位置上

舉個慄子:

 writer by:大沙漠 QQ:22969969

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Document</title>
    <script src="http://libs.baidu.com/jquery/1.7.1/jquery.min.js"></script>
</head>
<body>
    <p>你好</p>
    <p>Hello World</p>
    <div>
        <i>
            <span>測試文本</span>
        </i>
    </div>

    <button id="b1">按鈕1</button> <br/>
    <button id="b2">按鈕2</button><button id="b3">按鈕3</button> <br/>
    <button id="b4">按鈕4</button><br/><button id="b5">按鈕5</button>
    <script>
        b1.onclick=function(){$('p').wrap('<div></div>')}              //內部將<div></div>轉化為jQuery對象放到第一個匹配元素<p>你好</p>之前,再將匹配元素移動到該DOM節點內部

        b2.onclick=function(){$('p').wrapAll('<div></div>')}           //內部將<div></div>轉化為jQuery對象放到第一個匹配元素<p>你好</p>之前,再將匹配元素移動到該DOM節點內部
        b3.onclick=function(){$('p').wrapAll('<div><p></p></div>')}    //如果含有子節點,則會將匹配元素移動到子節點裡面

        b4.onclick=function(){$('p').wrapInner('<div></div>')}         //在每個匹配元素的內容前後添加一層DOM節點(包裹層)

        b5.onclick=function(){$('span').unwrap() }                     //移除每個匹配元素的父元素,並讓匹配元素占有該節點位置
    </script>
</body>
</html>

渲染如下:

 對應的DOM樹如下:

 點擊按鈕1會在所有的P標簽上加一個div父節點,如下:

 點擊按鈕2將在第一個p標簽前添加一個div,然後把所有p標簽放到div之下,如下:

 點擊按鈕3將在第一個p標簽前添加一個div>p雙層DOM,然後把所有p標簽放到div之下,如下:

 點擊按鈕4將在p標簽內最外層嵌套一層div標簽,如下:

 點擊按鈕5將會去除 span的上一層DOM節點,如下:

 如果再次點擊,會將span的上一層DOM繼續移除,直到遇到body節點為止

 

源碼分析


wrapInner和wrap都是基於wrapAll實現的,wrapAll實現如下:

jQuery.fn.extend({
    wrapAll: function( html ) {                //在匹配的元素外面放置html元素。html參數可以是html片段、選擇器表達式、jQuery對象、DOM元素或函數。
        if ( jQuery.isFunction( html ) ) {        //如果html是函數
            return this.each(function(i) {
                jQuery(this).wrapAll( html.call(this, i) );    //遍歷匹配元素,在每個匹配元素上執行html函數,並用該函數的返回值作為參數迭代調用.wrapAll()函數。
            });
        }

        if ( this[0] ) {                        //如果當前有匹配元素
            // The elements to wrap the target around
            var wrap = jQuery( html, this[0].ownerDocument ).eq(0).clone(true);        //將html轉化為一個jQuery對象

            if ( this[0].parentNode ) {                                                //如果當前第一個匹配元素有父元素,
                wrap.insertBefore( this[0] );                                            //則把創建的包裹元素插入第一個匹配元素之前。
            }

            wrap.map(function() {                            //遍歷wrap元素
                var elem = this;                                //elem是創建的包裹元素的引用

                while ( elem.firstChild && elem.firstChild.nodeType === 1 ) {    //如果html里含有一個子節點
                    elem = elem.firstChild;                                            //則重置elem為html的子節點,上面的按鈕3會執行到這裡
                }

                return elem;
            }).append( this );                                    //這一行的this是當前匹配的jQuery對象,把每個匹配元素移動到插入的元素之後
        }

        return this;
    },
})

wrapAll首先會把參數轉化為一個jQuery對象,然後插入到當前第一個匹配元素的前面,最後以生成的jQuery對象為主句,調用append()將當前匹配匹配的所有元素添加到新生成的jQuery對象對應的DOM節點內部。對應上面的按鈕2

wrap()實現如下:

jQuery.fn.extend({
    wrap: function( html ) {                //在每個匹配元素的外層添加一層DOM元素
        var isFunction = jQuery.isFunction( html );        //在每個匹配元素前後包裹一段HTML結構,該方法會遍歷匹配元素集合,在每個元素上調用.wrapAll()方法。
        return this.each(function(i) {
            jQuery( this ).wrapAll( isFunction ? html.call(this, i) : html );    //依次調用wrapALl()函數
        });
    },
})

wrapInner的實現如下:

jQuery.fn.extend({
    wrapInner: function( html ) {            //用於在匹配元素集合中每個元素的內容前後包裹一段HTML結構,該方法會遍歷匹配元素集合
        if ( jQuery.isFunction( html ) ) {                //如果html是函數
            return this.each(function(i) {
                jQuery(this).wrapInner( html.call(this, i) );    //遍歷匹配元素,在每個匹配元素上執行html函數,並用該函數的返回值作為參數迭代調用.wrapInner()函數。
            });
        }

        return this.each(function() {                    //遍歷匹配元素集合
            var self = jQuery( this ),
                contents = self.contents();                    //先獲取所有子節點

            if ( contents.length ) {                        //如果有子節點
                contents.wrapAll( html );                        //調用wrapAll(html)為當前元素的所有內容包裹一段HTML代碼。
            } else {
                self.append( html );                        //如果當前元素沒有內容,則直接將參數html插入當前內容。
            }
        });
    },
})

unwrap的實現如下:

Query.fn.extend({
    unwrap: function() {                        //移除匹配元素集合中每個元素的父標簽,並把匹配元素留在父元素的位置上
        return this.parent().each(function() {            //先遍歷父節點
            if ( !jQuery.nodeName( this, "body" ) ) {        //如果不是body元素
                jQuery( this ).replaceWith( this.childNodes );    //則調用replaceWith將this.childNodes替換為this,註意,這裡的this上下文是父節點
            }
        }).end();
    }
})

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

-Advertisement-
Play Games
更多相關文章
  • 1、啟動頁面:index.html <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <meta name="viewport" content="width=device-width,initial-scale=1.0"> <title>hc ...
  • 元素的顯示與隱藏 1. 本質:讓一個元素在頁面中隱藏或顯示出來 2. display:顯示隱藏 用於設置一個元素應如何顯示 display:none;隱藏對象 display:block;隱藏對象 【註意】 display隱藏元素後,不在占有原來的位置 3. visibility:顯示隱藏 visi ...
  • HTTP狀態碼分類 1XX ——信息,伺服器收到請求,需要請求者繼續執行操作 2XX——成功,操作被成功接收並處理 3XX——重定向,需要進一步的操作以完成請求 4XX——客戶端錯誤,請求包含語法錯誤或者無法完成請求 5XX——伺服器錯誤,伺服器在處理請求的過程中發生了錯誤 2XX狀態碼 | 狀態碼 ...
  • 事件發生時會在元素節點之間按照特定的順序傳播,這個傳播過程即DOM事件流。 DOM事件流分為三個階段,分別為: 捕獲階段:事件從Document節點自上而下向目標節點傳播的階段; 目標階段:真正的目標節點正在處理事件的階段; 冒泡階段:事件從目標節點自上而下向Document節點傳播的階段。 捕獲階 ...
  • 一、載入 ui element vue add element 載入過程及成功結果如下 > vue add element> �� Installing vue-cli-plugin-element... > + [email protected] > added 1 pack ...
  • 在安裝vue-cli之前,要先安裝node.js這個大家百度一下就可以了 1、安裝 vue-cli npm install -g @vue/cli-init 2、初始化一個項目,名為 hcmanage ,並選擇使用 webpack 打包方式 vue init webpack hcmanage 3、切 ...
  • 一、環境安裝 Node.js 安裝包及源碼下載地址為: "https://nodejs.org/en/download/" 。 在該頁面你可以根據不同平臺系統選擇你需要的 Node.js 安裝包。 Node.js裡面集成了npm,npm是一種包管理工具,允許用戶從NPM伺服器下載別人編寫的包或命令行 ...
  • 動畫是CSS3中具有顛覆性的特征之一,可通過設置多個節點來精確控制一個或一組動畫,常用來實現複雜的動畫效果。 語法格式: animation:動畫名稱 花費時間 運動曲線 何時開始 播放次數 是否反方向; 關於幾個值,除了名字,動畫時間,延時有嚴格順序要求其它隨意r @keyframes 動畫名稱 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...