javascript比較文檔位置
更新時(shí)間:2008年04月08日 20:57:25 作者:
一個(gè)很棒的 blog 文章,是 PPK 兩年前寫的,文章中解釋了 contains() 和 compareDocumentPosition() 方法運(yùn)行在他們各自的瀏覽器上。從那起,我已經(jīng)對(duì)這些方法做了大量的研究,并且已經(jīng)在很多場(chǎng)合使用他們。在很多任務(wù)中,他們被證明是非常有用的(特別關(guān)于結(jié)構(gòu)的抽象 DOM 選擇器)。
1、DOMElement.contains(DOMNode)
這個(gè)方法起先用在 IE ,用來確定 DOM Node 是否包含在另一個(gè) DOM Element 中。
當(dāng)嘗試優(yōu)化 CSS 選擇器遍歷(像:“#id1 #id2”),這個(gè)方法很有用。你可以通過 getElementById 得到元素,然后使用 .contains() 確定 #id1 實(shí)際上是否包含 #id2。
注意點(diǎn):如果 DOM Node 和 DOM Element 相一致,.contains() 將返回 true ,雖然,一個(gè)元素不能包含自己。
這里有一個(gè)簡(jiǎn)單的執(zhí)行包裝,可以運(yùn)行在:Internet Explorer, Firefox, Opera, and Safari。
function contains(a, b) {
return a.contains ? a != b && a.contains(b) : !!(a.compareDocumentPosition(arg) & 16);
}
2、NodeA.compareDocumentPosition(NodeB)
這個(gè)方法是 DOM Level 3 specification 的一部分,允許你確定 2 個(gè) DOM Node 之間的相互位置。這個(gè)方法比 .contains() 強(qiáng)大。這個(gè)方法的一個(gè)可能應(yīng)用是排序 DOM Node 成一個(gè)詳細(xì)精確的順序。
使用這個(gè)方法你可以確定關(guān)于一個(gè)元素位置的一連串的信息。所有的這些信息將返回一個(gè)比特碼(Bit,比特,亦稱二進(jìn)制位)。
對(duì)于那些,人們知之甚少。比特碼是將多重?cái)?shù)據(jù)存儲(chǔ)為一個(gè)簡(jiǎn)單的數(shù)字(譯者注:0 或 1)。你最終打開 / 關(guān)閉個(gè)別數(shù)目(譯者注:打開/關(guān)閉對(duì)應(yīng) 0 /1),將給你一個(gè)最終的結(jié)果。
這里是從 NodeA.compareDocumentPosition(NodeB) 返回的結(jié)果,包含你可以得到的信息。
Bits Number Meaning
000000 0 元素一致
000001 1 節(jié)點(diǎn)在不同的文檔(或者一個(gè)在文檔之外)
000010 2 節(jié)點(diǎn) B 在節(jié)點(diǎn) A 之前
000100 4 節(jié)點(diǎn) A 在節(jié)點(diǎn) B 之前
001000 8 節(jié)點(diǎn) B 包含節(jié)點(diǎn) A
010000 16 節(jié)點(diǎn) A 包含節(jié)點(diǎn) B
100000 32 瀏覽器的私有使用
現(xiàn)在,這意味著一個(gè)可能的結(jié)果類似于:
<div id="a">
<div id="b"></div>
</div>
<script>
alert( document.getElementById("a").compareDocumentPosition(document.getElementById("b")) == 20);
</script>
一旦一個(gè)節(jié)點(diǎn) A 包含另一個(gè)節(jié)點(diǎn) B,包含 B(+16) 且在 B 之前(+4),則最后的結(jié)果是數(shù)字 20 。如果你查看比特發(fā)生的變化,將增加你的理解。
000100 (4) + 010000 (16) = 010100 (20)
這個(gè),毫無疑問,有助于理解單個(gè)最混亂的 DOM API 方法。當(dāng)然,他的價(jià)值當(dāng)之無愧的。
現(xiàn)在,DOMNode.compareDocumentPosition 在 Firefox 和 Opera 中是可用的。然而,有一些技巧,我們可以用來在 IE 中執(zhí)行他。
// Compare Position - MIT Licensed, John Resig
function comparePosition(a, b){
return a.compareDocumentPosition ?
a.compareDocumentPosition(b) :
a.contains ?
( a != b && a.contains(b) && 16 ) +
( a != b && b.contains(a) && 8 ) +
( a.sourceIndex >= 0 && b.sourceIndex >= 0 ?
(a.sourceIndex < b.sourceIndex && 4 ) +
(a.sourceIndex > b.sourceIndex && 2 ) :
1 ) :
0;
}
IE 提供給我們一些可以使用的方法和屬性。開始,使用 .contains() 方法(如我們前面所討論的),以便給我們包含(+16)或者被包含(+8)的結(jié)果。IE 還有一個(gè) .sourceIndex 屬性在所有的 DOM Element 對(duì)應(yīng)著元素在文檔中的位置,例如:document.documentElement.sourceIndex == 0。因?yàn)槲覀冇羞@個(gè)信息,我們可以完成兩個(gè) compareDocumentPosition 難題:在前面(+2)和在后面(+4)。另外,如果一個(gè)元素不在當(dāng)前的文檔,.sourceIndex 將等于 -1,這個(gè)給我們另外一個(gè)回答(+1)。最后,通過這個(gè)過程的推斷,我們可以確定如果一個(gè)元素等于他本身,返回一個(gè)空的比特碼(+0)。
這個(gè)函數(shù)可以在 Internet Explorer、Firefox 和 Opera 中運(yùn)行。但在 Safari 中卻有殘缺功能(因?yàn)樗挥?nbsp;contains() 方法,而沒有 .sourceIndex 屬性。我們只能得到 包含(+16),被包含(+8),其他的所有結(jié)果都將返回(+1)代表一個(gè)斷開)。
PPK 提供了一個(gè)關(guān)于通過創(chuàng)建一個(gè) getElementsByTagNames 方法使新功能可以被使用的很棒的例子。讓我們改編他到我們的新方法中:
// Original by PPK quirksmode.org
function getElementsByTagNames(list, elem) {
elem = elem || document;
var tagNames = list.split(','), results = [];
for ( var i = 0; i < tagNames.length; i++ ) {
var tags = elem.getElementsByTagName( tagNames[i] );
for ( var j = 0; j < tags.length; j++ )
results.push( tags[j] );
}
return results.sort(function(a, b){
return 3 - (comparePosition(a, b) & 6);
});
}
我們現(xiàn)在可以使用他來按次序構(gòu)建一個(gè)站點(diǎn)的目錄:
getElementsByTagNames("h1, h2, h3");
雖然 Firefox 和 Opera 都采取了一些主動(dòng)落實(shí)這一方法。我依然期待看到更多的瀏覽器進(jìn)入,以幫助向前推動(dòng)。
這個(gè)方法起先用在 IE ,用來確定 DOM Node 是否包含在另一個(gè) DOM Element 中。
當(dāng)嘗試優(yōu)化 CSS 選擇器遍歷(像:“#id1 #id2”),這個(gè)方法很有用。你可以通過 getElementById 得到元素,然后使用 .contains() 確定 #id1 實(shí)際上是否包含 #id2。
注意點(diǎn):如果 DOM Node 和 DOM Element 相一致,.contains() 將返回 true ,雖然,一個(gè)元素不能包含自己。
這里有一個(gè)簡(jiǎn)單的執(zhí)行包裝,可以運(yùn)行在:Internet Explorer, Firefox, Opera, and Safari。
function contains(a, b) {
return a.contains ? a != b && a.contains(b) : !!(a.compareDocumentPosition(arg) & 16);
}
2、NodeA.compareDocumentPosition(NodeB)
這個(gè)方法是 DOM Level 3 specification 的一部分,允許你確定 2 個(gè) DOM Node 之間的相互位置。這個(gè)方法比 .contains() 強(qiáng)大。這個(gè)方法的一個(gè)可能應(yīng)用是排序 DOM Node 成一個(gè)詳細(xì)精確的順序。
使用這個(gè)方法你可以確定關(guān)于一個(gè)元素位置的一連串的信息。所有的這些信息將返回一個(gè)比特碼(Bit,比特,亦稱二進(jìn)制位)。
對(duì)于那些,人們知之甚少。比特碼是將多重?cái)?shù)據(jù)存儲(chǔ)為一個(gè)簡(jiǎn)單的數(shù)字(譯者注:0 或 1)。你最終打開 / 關(guān)閉個(gè)別數(shù)目(譯者注:打開/關(guān)閉對(duì)應(yīng) 0 /1),將給你一個(gè)最終的結(jié)果。
這里是從 NodeA.compareDocumentPosition(NodeB) 返回的結(jié)果,包含你可以得到的信息。
Bits Number Meaning
000000 0 元素一致
000001 1 節(jié)點(diǎn)在不同的文檔(或者一個(gè)在文檔之外)
000010 2 節(jié)點(diǎn) B 在節(jié)點(diǎn) A 之前
000100 4 節(jié)點(diǎn) A 在節(jié)點(diǎn) B 之前
001000 8 節(jié)點(diǎn) B 包含節(jié)點(diǎn) A
010000 16 節(jié)點(diǎn) A 包含節(jié)點(diǎn) B
100000 32 瀏覽器的私有使用
現(xiàn)在,這意味著一個(gè)可能的結(jié)果類似于:
<div id="a">
<div id="b"></div>
</div>
<script>
alert( document.getElementById("a").compareDocumentPosition(document.getElementById("b")) == 20);
</script>
一旦一個(gè)節(jié)點(diǎn) A 包含另一個(gè)節(jié)點(diǎn) B,包含 B(+16) 且在 B 之前(+4),則最后的結(jié)果是數(shù)字 20 。如果你查看比特發(fā)生的變化,將增加你的理解。
000100 (4) + 010000 (16) = 010100 (20)
這個(gè),毫無疑問,有助于理解單個(gè)最混亂的 DOM API 方法。當(dāng)然,他的價(jià)值當(dāng)之無愧的。
現(xiàn)在,DOMNode.compareDocumentPosition 在 Firefox 和 Opera 中是可用的。然而,有一些技巧,我們可以用來在 IE 中執(zhí)行他。
// Compare Position - MIT Licensed, John Resig
function comparePosition(a, b){
return a.compareDocumentPosition ?
a.compareDocumentPosition(b) :
a.contains ?
( a != b && a.contains(b) && 16 ) +
( a != b && b.contains(a) && 8 ) +
( a.sourceIndex >= 0 && b.sourceIndex >= 0 ?
(a.sourceIndex < b.sourceIndex && 4 ) +
(a.sourceIndex > b.sourceIndex && 2 ) :
1 ) :
0;
}
IE 提供給我們一些可以使用的方法和屬性。開始,使用 .contains() 方法(如我們前面所討論的),以便給我們包含(+16)或者被包含(+8)的結(jié)果。IE 還有一個(gè) .sourceIndex 屬性在所有的 DOM Element 對(duì)應(yīng)著元素在文檔中的位置,例如:document.documentElement.sourceIndex == 0。因?yàn)槲覀冇羞@個(gè)信息,我們可以完成兩個(gè) compareDocumentPosition 難題:在前面(+2)和在后面(+4)。另外,如果一個(gè)元素不在當(dāng)前的文檔,.sourceIndex 將等于 -1,這個(gè)給我們另外一個(gè)回答(+1)。最后,通過這個(gè)過程的推斷,我們可以確定如果一個(gè)元素等于他本身,返回一個(gè)空的比特碼(+0)。
這個(gè)函數(shù)可以在 Internet Explorer、Firefox 和 Opera 中運(yùn)行。但在 Safari 中卻有殘缺功能(因?yàn)樗挥?nbsp;contains() 方法,而沒有 .sourceIndex 屬性。我們只能得到 包含(+16),被包含(+8),其他的所有結(jié)果都將返回(+1)代表一個(gè)斷開)。
PPK 提供了一個(gè)關(guān)于通過創(chuàng)建一個(gè) getElementsByTagNames 方法使新功能可以被使用的很棒的例子。讓我們改編他到我們的新方法中:
// Original by PPK quirksmode.org
function getElementsByTagNames(list, elem) {
elem = elem || document;
var tagNames = list.split(','), results = [];
for ( var i = 0; i < tagNames.length; i++ ) {
var tags = elem.getElementsByTagName( tagNames[i] );
for ( var j = 0; j < tags.length; j++ )
results.push( tags[j] );
}
return results.sort(function(a, b){
return 3 - (comparePosition(a, b) & 6);
});
}
我們現(xiàn)在可以使用他來按次序構(gòu)建一個(gè)站點(diǎn)的目錄:
getElementsByTagNames("h1, h2, h3");
雖然 Firefox 和 Opera 都采取了一些主動(dòng)落實(shí)這一方法。我依然期待看到更多的瀏覽器進(jìn)入,以幫助向前推動(dòng)。
您可能感興趣的文章:
- JavaScript顯示當(dāng)前文檔最后修改日期的方法
- Jsoup解析HTML實(shí)例及文檔方法詳解
- JS獲取整個(gè)頁面文檔的實(shí)現(xiàn)代碼
- JavaScript 解析讀取XML文檔 實(shí)例代碼
- javascript json 新手入門文檔
- javaScript 讀取和設(shè)置文檔元素的樣式屬性
- 用javascript實(shí)現(xiàn)讀取txt文檔的腳本
- javascript獲取元素離文檔各邊距離的方法
- JavaScript利用HTML DOM進(jìn)行文檔操作的方法
- JS簡(jiǎn)單實(shí)現(xiàn)查看文檔創(chuàng)建日期、修改日期和文檔大小的方法示例
相關(guān)文章
Javascript中判斷變量是數(shù)組還是對(duì)象(array還是object)
怎樣判斷一個(gè)JavaScript變量是array還是obiect,或許有很多初學(xué)者對(duì)此不是很清楚吧,下面為大家詳細(xì)解答下,希望對(duì)大家有所幫助2013-08-08微信小程序methods中定義的方法互相調(diào)用的實(shí)例代碼
這篇文章主要介紹了微信小程序methods中定義的方法互相調(diào)用的實(shí)例代碼,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2018-08-08JavaScript實(shí)現(xiàn)經(jīng)緯度轉(zhuǎn)換成地址功能
這篇文章主要介紹了JavaScript實(shí)現(xiàn)經(jīng)緯度轉(zhuǎn)換成地址,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2017-03-03淺談JavaScript中面向?qū)ο蠹夹g(shù)的模擬
淺談JavaScript中面向?qū)ο蠹夹g(shù)的模擬...2006-09-09JavaScript實(shí)現(xiàn)的開關(guān)燈泡點(diǎn)擊切換特效示例
這篇文章主要介紹了JavaScript實(shí)現(xiàn)的開關(guān)燈泡點(diǎn)擊切換特效,涉及javascript事件響應(yīng)及頁面元素屬性動(dòng)態(tài)操作相關(guān)實(shí)現(xiàn)技巧,需要的朋友可以參考下2019-07-07uniapp中uni.request(OBJECT)接口請(qǐng)求封裝實(shí)例代碼
在開發(fā)的時(shí)候經(jīng)常會(huì)用到前端請(qǐng)求后端接口,每次的請(qǐng)求都會(huì)出現(xiàn)地址不一樣,參數(shù)不一樣,方式不一樣等等情況,下面這篇文章主要給大家介紹了關(guān)于uniapp中uni.request(OBJECT)接口請(qǐng)求封裝的相關(guān)資料,需要的朋友可以參考下2022-12-12