您现在的位置: 万盛学电脑网 >> 程序编程 >> 脚本专题 >> javascript >> 正文

jQuery内核 DOM详细操作

作者:佚名    责任编辑:admin    更新时间:2022-06-22

 domManip是什么

dom即Dom元素,Manip是Manipulate的缩写,连在一起就是Dom操作的意思。

jQuery针对DOM操作的插入的方法有大概10种

append、prepend、before、after、replaceWith

appendTo、prependTo、insertBefore、insertAfter、replaceAll

分2组,上下对照,实现同样的功能。主要的不同是语法——特别是内容和目标的位置

依赖的domManip,buildFragment模块在之前就分析过了

在匹配元素集合中的每个元素后面插入参数所指定的内容,作为其兄弟节点

对于 .after(), 选择表达式在函数的前面,参数是将要插入的内容。

对于.insertAfter(), 刚好相反,内容在方法前面,它将被放在参数里元素的后面。

after

after: function() {    return this.domManip( arguments, function( elem ) 
{        if ( this.parentNode )
 {            this.parentNode.insertBefore( elem, this.nextSibling );
        }
    });
},

之前提过了所有的方法靠this.domManip合并参数处理,内部通过buildFragment模块构建文档碎片

然后把每一个方法的具体执行通过回调的方式提供出来处理

DOM操作并未提供一个直接可以在当前节点后插入一个兄弟节点的方法,但是提供了一个类似的方法

insertBefore() 方法:可在已有的子节点前插入一个新的子节点。语法 :insertBefore(newchild,refchild)

看看jQuery如何处理的

例如

inner.after('<p>Test</p>');

内部就会把  '<p>Test</p>' 通过buildFragment构建出文档elem

然后通过  this.parentNode.insertBefore( elem, this.nextSibling );

这里的this 就是对应着inner ,elem就是‘<p>Test</p>’

看到这里就很好理解了after的实现了

用原生方法简单模拟

var  inner = document.getElementsByClassName('inner')for(var i =0 ; i<inner.length;i++)
{    var elem = inner[i]    var div = document.createElement('div')
    div.innerHTML = 'aaaa'
    elem.parentNode.insertBefore(div,elem.nextSibling)
}

insertAfter

jQuery代码的设计者很聪明的,都尽可能的合并相似功能的方法,代码更加精炼美观

jQuery.each({
    appendTo: "append",
    prependTo: "prepend",
    insertBefore: "before",
    insertAfter: "after",
    replaceAll: "replaceWith"}, function( name, original ) {
    jQuery.fn[ name ] = function( selector ) {
                           
    };
});

DEMO

$('<p>Test</p>').insertAfter('.inner');

通过$('<p>Test</p>')构建一个文档,对象通过insertAfter方法插入到所有class等于inner的节点后

表达的意思与after是一样的,主要的不同是语法——特别是内容和目标的位置

jQuery.fn[ name ] = function( selector ) {        var elems,
            ret = [],
            insert = jQuery( selector ),
            last = insert.length - 1,
            i = 0;        for ( ; i <= last; i++ ) {
            elems = i === last ? this : this.clone( true );
            jQuery( insert[ i ] )[ original ]( elems );  
          // Support: QtWebKit
            // .get() because core_push.apply(_, arraylike) throws 
           core_push.apply( ret, elems.get() );
        }        return this.pushStack( ret );
    };

看具体的实现方法中.insertAfter('.inner');inner其实就被当作selector传入进来了

selector可能只是字符串选择器内部就需要转化,insert = jQuery( selector ),

$('<p>Test</p>')就是构建出来的文档碎片节点,那么如果赋给insert有多个的时候就需要完全克隆一份副本了,所以就直接赋给

elems = i === last ? this : this.clone( true ); jQuery( insert[ i ] )[ original ]( elems );

依旧是执行after

jQuery( insert[ i ] )[ original ]( elems );

最终还需要返回这个构建的新节点

收集构建的节点

core_push.apply( ret, elems.get() );

构建一个新jQuery对象,以便实现链式

this.pushStack( ret );

可见after 与 insertAfter 本质本质其实都是一样的,只是通过不同的方式调用

before()

根据参数设定,在匹配元素的前面插入内容

before: function() {        return this.domManip( arguments, function( elem )
 {            if ( this.parentNode )
 {                this.parentNode.insertBefore( elem, this );
            }
        });
    },

类似after只是替换了第二个参数,改变插入的位置

append()

在每个匹配元素里面的末尾处插入参数内容

append: function() {    return this.domManip( arguments, function( elem )

{        if ( this.nodeType === 1 || this.nodeType === 11 || this.nodeType === 9 )

{            var target = manipulationTarget( this, elem );

            target.appendChild( elem );
        }
    });
},

内部增加节点,直接可以调用appendChild方法

prepend()

将参数内容插入到每个匹配元素的前面(元素内部)

prepend: function() {    return this.domManip( arguments, function( elem )
 {        if ( this.nodeType === 1 || this.nodeType === 11 || this.nodeType === 9 )
 {            var target = manipulationTarget( this, elem );
            target.insertBefore( elem, target.firstChild );
        }
    });
},

类似after只是替换了第二个参数,改变插入的位置

replaceWith()

用提供