jQuery鏈式調用與show知識淺析
來源:易賢網 閱讀:936 次 日期:2016-07-01 14:24:23
溫馨提示:易賢網小編為您整理了“jQuery鏈式調用與show知識淺析”,方便廣大網友查閱!

這篇文章主要介紹了jQuery的XX如何實現?——2.show與鏈式調用 的相關資料,非常具有參考借鑒價值,感興趣的朋友一起學習吧

jQuery使用許久了,但是有一些API的實現實在想不通。下面將使用簡化的代碼來介紹,主要關注jQuery的實現思想。

相較于上一篇,代碼更新了:21~78

(function(window, undefined){

function jQuery(sel){

return new jQuery.prototype.init(sel);

}

jQuery.prototype = {

constructor: jQuery,

init: function(sel){

if(typeof sel === 'string'){

var that = this;

var nodeList = document.querySelectorAll(sel);

Array.prototype.forEach.call(nodeList, function(val, i){

that[i] = val;

})

this.selector = sel;

this.length = nodeList.length;

}

},

show: function(){

Array.prototype.forEach.call(this, function(node){

//if(node.style) continue; //textnode沒有style

//刪除style上的display:none

var display = node.style.display;

if(display === 'none'){

//dispaly置為空后,css如果有display則css的生效

//否則默認的生效

node.style.display = '';

}

//元素display值為非默認值情況,需要還原為oldDisplay:div->display:inline-block

//或 檢測css上的display是否為none

if(node.style.display==='' || isHidden(node)){

//有oldDispaly則設置

if(node.oldDisplay) node.style.display = node.oldDisplay;

//沒有則設置為元素默認值或元素當前值

else node.style.display = getDisplay(node);

}

})

//鏈式調用

return this;

},

hide: function(){

Array.prototype.forEach.call(this, function(node){

if(!isHidden(node)) {

//jQuery使用其cache機制存儲信息,這里簡化一下

//直接掛載在對應的dom下

node.oldDisplay = getDisplay(node);

node.style.display = 'none';

}

})

return this;

}

}

function getDisplay(node){

var display = window.getComputedStyle(node, null).getPropertyValue('display');

if(display === 'none'){

var dom = document.createElement(node.nodeName);

//插入到body中

document.body.appendChild(dom);

//即可獲取到元素display的默認值

var display = window.getComputedStyle(dom, null).getPropertyValue('display');

document.body.removeChild(dom);

}

return display;

}

function isHidden(node) {

//忽略未append進document的元素這種隱藏情況:$('<div>block</div>')未append

return window.getComputedStyle(node, null).getPropertyValue('display') === 'none';

}

jQuery.prototype.init.prototype = jQuery.prototype;

window.$ = jQuery;

})(window);

先拿hide函數熱身一下。如上篇提到的,jQuery會將獲取到的nodeList處理成數組,所以一上來,我們用forEach處理數組里的每一個node節(jié)點。

接下來,我們只需要將每一個節(jié)點的style.display置為'none'即可隱藏。很簡單,對吧?(⊙0⊙) 。oldDisplay和return this先不管╰( ̄▽ ̄)╮

hide: function(){

Array.prototype.forEach.call(this, function(node){

if(!isHidden(node)) {

//jQuery使用其cache機制存儲信息,這里簡化一下

//直接掛載在對應的dom下

node.oldDisplay = getDisplay(node);

node.style.display = 'none';

}

})

return this;

}

其中isHidden是判斷該元素是否隱藏:已經隱藏的元素就沒必要再去處理了,直接跳過

function isHidden(node) {

//忽略未append進document的元素這種隱藏情況:$('<div>block</div>')未append

return window.getComputedStyle(node, null).getPropertyValue('display') === 'none';

}

--------------------------

接下來,來個稍繁瑣的show。先拋出一個問題來引發(fā)一系列問題:

hide某個元素只需要將display:none,那么show呢?

display:block不就行了嗎?這樣確實可以將元素顯示出來。但是萬一元素原來的值是display:inline呢?

那在hide處保存原來的值不就行了嗎?就像以下的代碼:

node.oldDisplay = getDisplay(node);

要是執(zhí)行show前沒有不執(zhí)行hide呢?比如下面這種情況,不就沒有oldDisplay了嗎(⊙0⊙)

<style>

div{ display:none; }

</style>

<div>display:none</div>$('div').show()

好,關鍵的地方到了:我們獲取元素display的默認值就可以了吧?比如div默認是block,span默認是inline。

思路有了,那么接下來的問題是:如何獲取元素display的默認值?

嘿嘿嘿,想不到吧?這里需要用點小技巧,大體思路如下:通過nodeName創(chuàng)建一個新的標簽,再獲取。

有個地方可以再優(yōu)化一下,getDisplay獲取到元素display默認值后,可以使用jQuery的cache機制存起來(實際上jQuery也是這么做了)。

function getDisplay(node){

var display = window.getComputedStyle(node, null).getPropertyValue('display');

if(display === 'none'){

var dom = document.createElement(node.nodeName);

//插入到body中

document.body.appendChild(dom);

//即可獲取到元素display的默認值

var display = window.getComputedStyle(dom, null).getPropertyValue('display');

document.body.removeChild(dom);

}

return display;

}

然后,綜合這兩種情況:

//有oldDispaly則設置

if(node.oldDisplay) node.style.display = node.oldDisplay;

//沒有則設置為元素默認值或元素當前值

else node.style.display = getDisplay(node);

以為這樣就結束了?NO,show函數的情況還是挺復雜的,我們大致要應對這幾種情況:

<style>

#none,#none2{ display: none; }

</style>

<body>

<div id="div">默認值為block</div>

<span id="span">默認值為inline</span>

<div id="div2" style="display:inline-block;">修改為inline-block</div>

<div id="none">通過css隱藏了</div>

<div id="none2" style="display:none">通過css和style隱藏了</div>

</body>

最終,show函數變成了這鬼樣ψ(╰_╯)。大致思路如下:

名單

show: function(){

Array.prototype.forEach.call(this, function(node){

//if(node.style) continue; //textnode沒有style

//刪除style上的display:none

var display = node.style.display;

if(display === 'none'){

//dispaly置為空后,css如果有display則css的生效

//否則默認的生效

node.style.display = '';

}

//元素display值為非默認值情況,需要還原為oldDisplay:div->display:inline-block

//或 檢測css上的display是否為none

if(node.style.display==='' || isHidden(node)){

//有oldDispaly則設置

if(node.oldDisplay) node.style.display = node.oldDisplay;

//沒有則設置為元素默認值或當前值

else node.style.display = getDisplay(node);

}

})

}

--------------------------

鏈式調用就是類似這種情況:

$('div').show().hide().css('height','300px').toggle()

實現起來非常簡單,只要在每個函數后面return this即可

--------------------------

有同學說:喂!這個show,hide不對吧?是不是漏了時間參數? 用setTimeOut自己實現吧~>_<~+。

本節(jié)最主要是讓大家知道jQuery需要考慮的情況非常多(很多臟活)。即時簡化了代碼,依然還是這么長。

寫完后,發(fā)現show還有一種情況沒考慮:

div{ display:none !important; }

<div>大家自己開腦洞,怎么處理吧(⊙0⊙)</div>

更多信息請查看網絡編程
易賢網手機網站地址:jQuery鏈式調用與show知識淺析

2026上岸·考公考編培訓報班

  • 報班類型
  • 姓名
  • 手機號
關于我們 | 聯(lián)系我們 | 人才招聘 | 網站聲明 | 網站幫助 | 非正式的簡要咨詢 | 簡要咨詢須知 | 新媒體/短視頻平臺 | 手機站點 | 投訴建議
工業(yè)和信息化部備案號:滇ICP備2023014141號-1 云南省教育廳備案號:云教ICP備0901021 滇公網安備53010202001879號 人力資源服務許可證:(云)人服證字(2023)第0102001523號
聯(lián)系電話:0871-65099533/13759567129 獲取招聘考試信息及咨詢關注公眾號:hfpxwx
咨詢QQ:1093837350(9:00—18:00)版權所有:易賢網