更好的選擇器策略
讓我們從基礎的例子開始。下面的HTML代碼表示的是可以開合的導航菜單。
<button class="nav-menu-toggle">Toggle Nav Menu</button> <nav> <ul> <li><a href="/">West Philadelphia</a></li> <li><a href="/cab">Cab Whistling</a></li> <li><a href="/throne">Throne Sitting</a></li> </ul> </nav>
下面這個是點擊按鈕之後控制導航菜單開合的javascript代碼
$('.nav-menu-toggle').on('click',function(){$('nav').toggle();});
這可能是最常用的實現方式。它能夠使用,但是比較脆。javascript代碼依賴了按鈕的類名nav-menu-toggle。很可能在未來其他開發者或者健忘的你在重搆代碼的時候會刪除或者重命名這個類名。
問題的核心是我們同時在表現和交互中使用了CSS的類名。這違反了關注點分離的原則,讓維護更容易出錯。
讓我們用一個不同的方法來實現
<button data-hook="nav-menu-toggle">Toggle Nav Menu</button> <nav data-hook="nav-menu"> <ul> <li> <a href="/">West Philadelphia</a> </li> <li> <a href="/cab">Cab Whistling</a> </li> <li> <a href="/throne">Throne Sitting</a> </li> </ul> </nav>
這次我們使用這個data屬性(data-hook)來選擇元素。任何對CSS類的改變將不會影響到javascript,讓我們能夠實現關注點分離以及更加穩定的代碼。
下面我們用data-hook屬性來選擇對應的元素:
$('[data-hook="nav-menu-toggle"]').on('click',function(){ $('[data-hook="nav-menu"]').toggle(); });
需要注意的是,我也使用data-hook作爲nav元素的選擇器。你不一定需要,但是我喜歡這裡面包含的思想:任何使用你看到data-hook,你會知道這個元素在javascript中引用到啦。
一些語法糖
我必須承認data-hook選擇器並不是很漂亮。讓我們通過擴展jQuery實現一個自定義的函數:
$.extend({hook:function(hookName){ var selector; if(!hookName || hookName === '*'){ // select all data-hooks selector='[data-hook]' }else{ // select specific data-hook selector='[data-hook*="'+hookName+'"]'; } return $(selector); } });
上面準備完畢,我們來重寫一下javascript。
$.hook('nav-menu-toggle').on('',function(){ $.hook('nav-menu').toggle(); });
更好的是,我們甚至可以把一系列以空格分開的hook名字放在一個元素上。
<button data-hook="nav-menu-toggle video-pause click-track">Toggle Nav Menu</button>
我們可以找到裡面的任意個hook名字:
$.hook('click-track'); // returns the button as expected
我們也能夠找到頁面上所有的hook元素
// both are equivalent $.hook(); $.hook('*');
防止函數表達式
到目前爲止,我們在事件處理中使用的都是匿名函數。讓我們重寫一下,使用聲明的函數來代替它。
function toggleNavMenu(){ $.hook('nav-menu').toggle(); } $.hook('nav-menu-toggle').on('click',toggleNavMenu);
這讓事件綁定的代碼更加易讀。這個toggleNavMenu函數名表達了意圖,是代碼自我注釋的好例子。
我們同時也獲得了可複用的能力,因爲其他地方可能需要使用toggleNavMenu函數。
最後,這對於自動化測試來說是意見大喜事,因爲聲明的函數的單元測試要比匿名函數單元測試容易的多。
同時使用多個事件
jQuery提供了一個簡單方便的語法來處理多事件的問題。比如,你可以爲一系列空格隔開的事件列表綁定同一個事件處理函數。
$.hook('nav-menu-toggle').on('click keydown mouseenter',trackAction);
如果你需要爲不同的事件綁定不同的處理函數,你可以使用對象表達方式:
$.hook('nav-menu-toggle').on({'click':trackClick,'keydown':tranckKeyDown,'mouseenter':trackMouseEnter});
反過來,你可以同時取消多個事件的綁定:
// unbinds keydown and mouseenter $.hook('nav-menu-toggle').off('keydown mouseenter'); // nuclear options:unbinds everything $.hook('nav-menu-toggle').off();
你可以想象到的是,不小心的取消事件綁定可能會導致嚴重的我們不想要的副作用。繼續看我們可以通過哪些技巧來減輕這個問題。
小心的取消事件綁定
一般情況下我們不會在一個元素的同一事件類型綁定多個事件處理函數。讓我們再看一下之前的那個按鈕:
<button data-hook="nav-menu-toggle video-pause click-track">Toggle Nav Menu</button>
不同的代碼區域可能會在同一個元素的同一事件綁定不同的事件處理函數:
// somewhere in the nav code $.hook('nav-menu-toggle').on('click',toggleNavMenu); // somewhere in the video playback code $.hook('video-pause').on('click',pauseCarltonDanceVideo); // somewhere in the analytics code $.hook('click-track').on('click',trackClick);
盡管我們使用了不同的選擇器,但是這個元素現在有三個事件處理函數啦。假如我們的分析代碼不在關心這個按鈕:
// no good$.hook('click-track').off('click');
糟糕的是,上面的代碼實際上回刪除所有的點擊事件處理函數,不僅僅是trackClick。我們應該實用更加有辨別力的方式來指定我們需要刪除的事件處理函數:
$.hook('click-track').off('click',trackClick);
另一種方式是使用命名空間。任何事件都有資格使用一個命名空間來實現綁定和取消綁定,這樣你就可以更好的控制事件綁定和取消綁定。
// binds a click event in the "analytics" namespace $.hook('click-track').on('click.analytics', trackClick); // unbinds only click events in the "analytics" namespace $.hook('click-track').off('click.analytics');
你也可以使用多個命名空間:
// binds a click event in both the "analytics" and "usability" namespaces $.hook('click-track').on('click.analytics.usability',trackClick); // unbinds any events in either the "analytics" OR "usability" namespaces $.hook('click-track').off('.usability .analytics'); // unbinds any events in both the "analytics" AND "usability" namespaces $.hook('click-track').off('.usability.analytics');
需要注意的是,命名空間的順序是沒有關系的,因爲命名空間不是層級式的。
如果你有一個複襍的功能需要多個元素綁定多個事件,那麽使用命名空間是一種簡單的把他們組織起來然後快速清除的方式:
// free all elements on the page of any "analytics" event handling $('*').off('.analytics');
命名空間在寫插件的時候尤其有用,因爲這樣你就能保証衹會取消自己命名空間範圍内的事件處理函數的綁定。