實(shí)現(xiàn)一個(gè)簡(jiǎn)化版的jQuery的API

代碼地址:https://github.com/aaronyxu/slider-demo-2

實(shí)現(xiàn)目的

本文的目的是通過(guò)模仿jQuery實(shí)現(xiàn)兩個(gè)簡(jiǎn)化的jQuery的API:

$nodes.addClass(classes)
$nodes.setText('text') 
  1. $nodes 是我們模仿jQuery建立的一個(gè)簡(jiǎn)化版的jQuery對(duì)象漫玄。

  2. $nodes.addClass(classes) 給所有的$nodes 添加指定的class坡倔。

其中參數(shù) classes可以是一個(gè)字符串表示一個(gè)樣式類名兵琳,也可以是個(gè)數(shù)組柳刮,表示多個(gè)樣式類名懊昨,同時(shí)添加多個(gè)classes爽蝴。

  1. $nodes.setText('text') 把所有的 $nodestextcontent 變?yōu)橹付ǖ奈谋緝?nèi)容急波。

    參數(shù) 'text' 是一個(gè)字符串从铲。

驗(yàn)證方法

為了驗(yàn)證API的準(zhǔn)確性,我們要求API能實(shí)現(xiàn)以下功能:

var $div = $('div') // 接收一個(gè)包含CSS選擇器的字符串澄暮,創(chuàng)建一個(gè)由簡(jiǎn)化版jQuery包裝的DOM元素
$div.addClass('red') // 可將所有 div 的 class 添加一個(gè) red
$div.setText('hi') // 可將所有 div 的 textContent 變?yōu)?hi
$div.addClass(['small', 'circle']) // 可將所有 div 的 class 同時(shí)添加一個(gè) small 和 circle

HTML 代碼

<!DOCTYPE html>
<html>
<head>
    <title>jQuery API Demo</title>
    <script src="./jQuery-API.js"></script>
    <link rel="stylesheet" type="text/css" href="./style.css">
</head>
<body>
    <div class="demo"></div>
    <div class="demo"></div>
    <div class="demo"></div>
    <div class="demo"></div>
</body>
</html>

CSS 代碼

.demo {
    width: 400px;
    height: 300px;
    border: 10px solid black;
    font-size: 20px;
    display: inline-block;
}
.red {
    background: red;
}
.circle {
    border-radius: 50%;
}
.small {
    width: 200px;
    height: 150px;
}

命名空間和返回對(duì)象結(jié)構(gòu)統(tǒng)一

首先我們要寫一個(gè)函數(shù)名段,使得傳入函數(shù)的節(jié)點(diǎn)或者是一個(gè)包含CSS選擇器的字符串,都返回一個(gè)經(jīng)過(guò)我們包裝過(guò)的一個(gè)DOM對(duì)象泣懊,這個(gè)新對(duì)象必須保持形式統(tǒng)一伸辟,并且同時(shí)含有 addClasssetText 兩個(gè)屬性,這樣我們就能直接在新對(duì)象調(diào)用我們所實(shí)現(xiàn)的兩個(gè)API馍刮。

window.jQuery = function(nodeOrSelector) {
    let nodes = {}
    if (typeof nodeOrSelector === 'string') {
        let temp = document.querySelectorAll(nodeOrSelector)  
        for (let i = 0; i < temp.length; i++) {
            nodes[i] = temp[i]
        }
        nodes.length = temp.length
    } else if (nodeOrSelector instanceof Node) {
        nodes = {
            0: nodeOrSelector,
            length: 1
        }
    }

    nodes.addClass = function(classes) {
    }
    nodes.setText = function(text) {
    }

    return nodes
}
window.$ = jQuery
  1. 為了使得通過(guò)包含CSS選擇器的字符串返回得到的對(duì)象保持純凈(不包含DOM對(duì)象的原型鏈)信夫,我們通過(guò)遍歷 document.querySelectorAll(nodeOrSelector) 得到一個(gè)純凈的偽數(shù)組。

  2. 同時(shí)為了使得由字符串和節(jié)點(diǎn)傳入返回得到的對(duì)象保持結(jié)構(gòu)一致卡啰,我們把由傳入節(jié)點(diǎn)返回的對(duì)象也包裝成一個(gè)純凈的偽數(shù)組忙迁。

    nodes = {
        0: nodeOrSelector,
        length: 1
    }
    
  3. 同時(shí)我們所返回的對(duì)象也必須含有 addClasssetText 屬性。

    nodes.addClass = function(classes) {
    }
    nodes.setText = function(text) {
    }
    
  4. 為了方便書寫碎乃,使得 $ 縮寫來(lái)返回得到一個(gè)jQuery對(duì)象

    window.$ = jQuery
    

addClass屬性

為了簡(jiǎn)化操作姊扔,我們只在這里考慮兩種參數(shù)類型。

  1. 字符串

    表示一個(gè)指定添加的class名稱

  2. 數(shù)組

    數(shù)組的value表示一組指定添加的class名稱

所以我們需要先判斷傳入的參數(shù)是哪種類型

    nodes.addClass = function(classes) {
        if (classes instanceof Array) {
            classes.forEach((value) => {
                for (let i = 0; i < nodes.length; i++) {
                    nodes[i].classList.add(value)
                }
            })
        } else if (typeof classes === 'string') {
            for (let i = 0; i < nodes.length; i++) {
                nodes[i].classList.add(classes)
            }
        }

    }
  • 對(duì)于數(shù)組我們首先調(diào)用數(shù)組的 forEach 遍歷 classes 再通過(guò) for 循環(huán)遍歷 nodes 把所有樣式類名添加到所有 nodes 中梅誓。
  • 對(duì)于字符串恰梢,我們直接使用 for 循環(huán)遍歷 nodes

setText屬性

    nodes.setText = function(text) {
        for (let i = 0; i < nodes.length; i++) {
            nodes[i].textContent = text
        }
    }

我們通過(guò) for 循環(huán)遍歷 nodes 調(diào)用DOM元素的 textContent 函數(shù) 給所有 nodes 添加指定的文本內(nèi)容

驗(yàn)證結(jié)果

我們?cè)跒g覽器的console界面輸入以下代碼

var $div = $('div') // 接收一個(gè)包含CSS選擇器的字符串梗掰,創(chuàng)建一個(gè)由簡(jiǎn)化版jQuery包裝的DOM元素
$div
瀏覽器console界面截圖

返回得到了一個(gè)純凈的偽數(shù)組嵌言,4個(gè)div,length 屬性 以及我們需要調(diào)用的2個(gè)函數(shù)addClasssetText及穗。

$div.addClass('red') // 可將所有 div 的 class 添加一個(gè) red

CSS 代碼:

.red {
    background: red;
}

我們可以看到所有div的背景變紅了摧茴。

$div.setText('hi') // 可將所有 div 的 textContent 變?yōu)?hi

我們可以看到所有div都被添加了文本內(nèi)容:hi

$div.addClass(['small', 'circle']) // 可將所有 div 的 class 同時(shí)添加一個(gè) small 和 circle

CSS 代碼:

.circle {
    border-radius: 50%;
}
.small {
    width: 200px;
    height: 150px;
}

我們可以看到所有div的class同時(shí)被添加了 small 和 circle埂陆。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末苛白,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子焚虱,更是在濱河造成了極大的恐慌购裙,老刑警劉巖,帶你破解...
    沈念sama閱讀 222,104評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件鹃栽,死亡現(xiàn)場(chǎng)離奇詭異躏率,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,816評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門薇芝,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)蓬抄,“玉大人,你說(shuō)我怎么就攤上這事夯到〕ǎ” “怎么了?”我有些...
    開封第一講書人閱讀 168,697評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵黄娘,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我克滴,道長(zhǎng)逼争,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 59,836評(píng)論 1 298
  • 正文 為了忘掉前任劝赔,我火速辦了婚禮誓焦,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘着帽。我一直安慰自己杂伟,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,851評(píng)論 6 397
  • 文/花漫 我一把揭開白布仍翰。 她就那樣靜靜地躺著赫粥,像睡著了一般。 火紅的嫁衣襯著肌膚如雪予借。 梳的紋絲不亂的頭發(fā)上越平,一...
    開封第一講書人閱讀 52,441評(píng)論 1 310
  • 那天,我揣著相機(jī)與錄音灵迫,去河邊找鬼秦叛。 笑死,一個(gè)胖子當(dāng)著我的面吹牛瀑粥,可吹牛的內(nèi)容都是我干的挣跋。 我是一名探鬼主播,決...
    沈念sama閱讀 40,992評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼狞换,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼避咆!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起修噪,我...
    開封第一講書人閱讀 39,899評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤牌借,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后割按,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體膨报,經(jīng)...
    沈念sama閱讀 46,457評(píng)論 1 318
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,529評(píng)論 3 341
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了现柠。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片院领。...
    茶點(diǎn)故事閱讀 40,664評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖够吩,靈堂內(nèi)的尸體忽然破棺而出比然,到底是詐尸還是另有隱情,我是刑警寧澤周循,帶...
    沈念sama閱讀 36,346評(píng)論 5 350
  • 正文 年R本政府宣布强法,位于F島的核電站,受9級(jí)特大地震影響湾笛,放射性物質(zhì)發(fā)生泄漏饮怯。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,025評(píng)論 3 334
  • 文/蒙蒙 一嚎研、第九天 我趴在偏房一處隱蔽的房頂上張望蓖墅。 院中可真熱鬧,春花似錦临扮、人聲如沸论矾。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,511評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)贪壳。三九已至,卻和暖如春蚜退,著一層夾襖步出監(jiān)牢的瞬間寥袭,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,611評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工关霸, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留传黄,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 49,081評(píng)論 3 377
  • 正文 我出身青樓队寇,卻偏偏與公主長(zhǎng)得像膘掰,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子佳遣,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,675評(píng)論 2 359

推薦閱讀更多精彩內(nèi)容