本文是对jQuery的起源的初步探索。先通过两个函数来扩展原生DOM的操作,然后引入命名空间以及对其重构,接着将该命名空间扩大到Node上,改造一个自己的Node2,引出jQuery。
引子
- 首先,我有一个需求===========> 要获得一个
<li>
标签的所有兄弟元素。
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>JS Bin</title>
</head>
<body>
<ul>
<li id="item1">选项1</li>
<li id="item2">选项2</li>
<li id="item3">选项3</li>
<li id="item4">选项4</li>
<li id="item5">选项5</li>
<li id="item6">选项6</li>
</ul>
</body>
</html>
此时你刚学完原生DOM操作,知道有nextSibling previousSibling parentNode
。你发现貌似没有直接一下子获得全部兄弟元素的API啊,身为一个优秀的90后,你果断手写一个函数实现这个需求啊。
function getSiblings(node){
var allChild = item2.parentNode.children
var childObj = {length: 0}
for (let i = 0; i < allChild.length; i++){
if (allChild[i] !== node){
childObj[childObj.length] = allChild[i]
childObj.length += 1
}
}
return childObj
}
好了,以上的函数就能满足需求了,它接受你传入的某个元素,返回包含这个元素所有兄弟元素的伪数组。
注意: 要用item2.parentNode.children
这样子才不会获得文本节点。所以你想获得item2
的所有兄弟,只需要getSiblings(item2)
获得所有兄弟的演示地址============>demo
- 你刚解决了一个问题,领导又给你提了一个需求,让你给
<li>
添加一个类
领导还没说完,你立马想到了,直接
item2.classList.add('类名')
啊,哈哈,我好聪明啊,不愧是优秀的90后。给你任意一个元素要直接加上这个类名,别给我的一个一个的加,太二了,如果元素原来有一个不应该存在的类名,给我删了,领导接着说完全部的需求。
这…看来不能
item1.classList.add('类名')
item2.classList.add('类名')
item3.classList.add('类名')
这么弱智的干了啊,那我还用函数嘛,你灵机一动。嗯,不愧是善于思考的90后
function addClass(node, classes){
for (var key in classes){
var value = classes[key]
if (value){
node.classList.add(key)
} else{
node.classList.remove(key)
}
}
}
上图是为添加元素的时候的item2
的模样,记住它,待会和下图对比。
可以看到,执行方法后,item2
的类名变为b、c,这是因为你是addClass(item2, {a: 0, b: 1, c: true})
这么调用的,意思是类名不应该有a,删除a
,并加上b c
。
以上对象的遍历并取值用到了falsey
值
复习一下,js的6个falsey
值
0
NaN
''
null
undefined
false
除此之外,其他的全是true。
不过你想的太美了,领导看到你的代码中的这个片段,直接抓狂了……
if (value){
node.classList.add(key)
} else{
node.classList.remove(key)
}
}
这段代码给我优化了,明明就是一句话的事。
你回去想了一会,可以这么优化
var methodName = value ? 'add' : 'remove'
node.classList[methodName](key)
最后你把如下代码提交。
function addClass(node, classes){
for (var key in classes){
var value = classes[key]
var methodName = value ? 'add' : 'remove'
node.classList[methodName](key)
}
}
- 注意一点上述代码不能用点运算符,要用
[]
运算符,classList['add'] === classList.add
给任一元素添加类名==========================>demo
命名空间
你完成了上面的两个需求后,领导本着锻炼你的原则,又给你提了新的需求。
- 少林那,你看你这两函数写的挺好的,如果分开放,每次还要在找,不如放到一起把。
- 我擦嘞,让我放到一起。好吧……
- 回来后,少林冥思苦想,突然想起了昨晚刚去的月坛西街的天府超市的送货车
- 我把那俩函数当成两种商品,我开个超市,把它俩收起来呗。
var shaolinDom = {} //少林开的超市
shaolinDom.addClass = addClass //把addClass这个商品收进来
shaolin.getSibling = getSiblings //把getSiblings这个商品收进来
那我咋用呢,该咋用就咋用呗。
shaolinDom.addClass(item5, {a: true, b: false, c: 0}) //把item5上原本的b c类名删掉,加上 a类名
shaolinDom.getSiblings(item6) //获得item6的所有兄弟元素
- 看着上面的代码,我就在想啊,还是自己由优化一下吧,免得回来改。
var shaolinDom = {}
shaolinDom.addClass = function(node, classes){
for (var key in classes){
var value = classes[key]
var methodName = value ? 'add':'remove'
node.classList[methodName](key)
}
}
shaolinDom.getSiblings = function (node){
var allChild = node.parentNode.children
var childObj = {length: 0}
for (let i = 0; i< allChild.length; i++){
if (allChild[i] !== node){
childObj[childObj.length] = allChild[i]
childObj.length += 1
}
}
return childObj
}
shaolinDom.addClass(item5, {a: true, b: false, c: 0})
var allSiblings = shaolinDom.getSiblings(item6)
console.log(allSiblings)
引入命名空间=======================>demo
命名空间的优化=====================>demo
- 你提交了新的需求后,领导对你刮目相看啊,今年的新员工还是不错的
- 你刚想松口气,领导接着说,少林那,你看你这两函数啊只能在
shaolinDom
用啊,而且我每次要把item5
传到函数里面,每次好麻烦的啦,你改进一下,让我的元素可以直接调用方法呗,比如item5.getSiblings()
这样多好。这样子操作的话,item5
拥有自主权,就像你买东西,你想去买那个东西你就去买那个东西嘛,而不是东西去选择你啊。 - 我擦擦嘞,想想经理分析还是很有道理的,果然还是要继续优化啊……
- 回来会,我在工位上想,你不是要操作DOM吗,还想这么
item5.getSiblings()
操作,那我这次直接给你干到Node的原型上;
Node.prototype.addClass = function(classes){
for (var key in classes){
var value = classes[key]
var methodName = value ? 'add':'remove'
this.classList[methodName](key)
}
}
Node.prototype.getSiblings = function (){
var allChild = this.parentNode.children
var childObj = {length: 0}
for (let i = 0; i< allChild.length; i++){
if (allChild[i] !== this){
childObj[childObj.length] = allChild[i]
childObj.length += 1
}
}
return childObj
}
- 丫的,我写出这些代码之后,我瞬间感觉自己两米了呢,赶紧去找领导
- 领导一看,吸了一口冷气,心想,挺牛 啊,不过我要问问我这小子
this
的知识 - 少林那,你这函数里面
this
是啥,再给我讲讲咋用呗 - 我屮艸芔茻,轮到我牛了吧,我把袖子已撸,是这么回事,巴拉巴拉
- 当然,我要先给经理讲一下怎么用
item5.addClass({a: true, b: false, c: 0}) //既然Node原型都有了这两函数,item5是node类型,直接用呗
console.log(item6.getSiblings())
- 我刚写完,领导就叫,这
addClass
函数里面这就一个参数啊,getSiblings
函数怎么没参数啊 - 哦,这个啊,我讲了this后就明白了。不过讲这个
this
之前先要讲一讲这个call()
,方便理解
//上面的代码等同于以下代码
item5.addClass.call(item5, {a: true, b: false, c: 0}) //call()方法的第一个参数就是this
console.log(item6.getSiblings.call(item6))
- 如果你把call()省了,直接用
()
去调用函数,自己脑补call()
就好啦,自然也就知道this是谁啦。 - 这小子还可以啊,不错。不过要继续引导一下啊
进一步升级,绑定Node
的原型链上==================>demo
用call()
方便理解this
================================>demo
自己写一个构造函数
没多久,领导的考验又来了
- 少林那,你看你上次吧你自己写的函数绑到Node上了,看似挺好,但是其他人不一定用你的这两函数啊,你绑到Node上,多占地啊,而且你可以绑到Node原型上,别人也可以写一个同名函数绑到Node原型上,万一给你覆盖了,你用的时候,不就懵逼了嘛。你自己写一个全局函数实现一下相同的需求吧。
- 呦,这次经理说的很对啊,我的错,我改进一下
- 突然联想到以前的各种构造函数,
String() Number() Array()
可以直接返回一个对象,我也这么干吧
window.Node2 = function(node){
return {
getSiblings: function(){
var allChild = node.parentNode.children
var childObj = {length: 0}
for (let i = 0; i< allChild.length; i++){
if (allChild[i] !== node){
childObj[childObj.length] = allChild[i]
childObj.length += 1
}
}
return childObj
},
addClass: function(classes){
for (var key in classes){
var value = classes[key]
var methodName = value ? 'add':'remove'
node.classList[methodName](key) //闭包的使用
}
}
}
}
- 实现了一个全局构造函数,返回一个对象,该对象里面有两个key(getSiblings、addClass),value分别又是两个函数(又体现了函数是第一公民的地位),而且还用到了闭包。
var node2 = Node2(item3) //node2就是用Node2()构造函数构造的返回的对象
node2.getSiblings() //对象的点运算符去去操作属性啊
node2.addClass({'a': 0, 'b': true, 'c': true})
- 领导一看,嗯,是时候让他见识真正的
jQuery
了
自己实现一个构造函数去理解=======================>demo
jQuery的雏形
- 这次领导没再提需求,而是自己改起了代码
window.jQuery = function(node){
return {
getSiblings: function(){
var allChild = node.parentNode.children
var childObj = {length: 0}
for (let i = 0; i< allChild.length; i++){
if (allChild[i] !== node){
childObj[childObj.length] = allChild[i]
childObj.length += 1
}
}
return childObj
},
addClass: function(classes){
for (var key in classes){
var value = classes[key]
var methodName = value ? 'add':'remove'
node.classList[methodName](key) //闭包的使用
}
}
}
}
- 你看看我改了一个位置,你看着这像啥
- 我看了一会这难道是传说中的
jQuery
? - 这是它的雏形,大概意思你已经一步一步写出来了,
jQuey
就是一个构造函数,它返回一个对象,这个对象有很多key,对应的value又是一些函数。 - 那怎么还用
$
这个操作呢 - 哈哈,一个语法糖吗,你看
window.$ = jQuery
- 给你出个题吧,你用现在的知识用
jQuery
实现把某个元素变红,最好验证一下,你的参数是node还是一个选择器,提示一下,可以用querySelector()
,querySelector
会返回文档中匹配指定的选择器组的第一个元素 - 我想了一会,写出如下代码
window.JQuery = function(nodeOrSelector){
let node
/*判断一下nodeOrSelector是node还是一个选择器
var node2 = JQuery(item3)这是nodej节点
var node2 = JQuery('#item3')这是Id选择器
*/
if(typeof nodeOrSelector === 'string'){
node = document.querySelector(nodeOrSelector)
} else{
node = nodeOrSelector
}
return {
getSiblings: function(){
var allChild = node.parentNode.children
var childObj = {length: 0}
for (let i = 0; i< allChild.length; i++){
if (allChild[i] !== node){
childObj[childObj.length] = allChild[i]
childObj.length += 1
}
}
return childObj
},
addClass: function(classes){
for (var key in classes){
var value = classes[key]
var methodName = value ? 'add':'remove'
node.classList[methodName](key)
}
}
}
}
所以
//var node2 = JQuery('#item3')与下列代码作用相同,把item3变红
var node2 = JQuery('ul > li:nth-child(3)')
jQuery的雏形======================>demo
使用一下自己的jQuery,写一个API
- 少林那,你也会用初级的jQuery了,咱们练习一下,你把修改一下所有的
<li>
的内容吧,可以使用querySelectorAll() 返回一个NodeList的伪数组
- 我听完之后,思考了一会,写下了如下代码
window.jQuery = function(nodeOrSelector){
let nodes = {}
if(typeof nodeOrSelector === 'string'){
let temp = document.querySelectorAll(nodeOrSelector) //NodeList
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){
classes.forEach((value) => {
for (let i = 0; i < nodes.length; i++){
nodes[i].classList.add(value)
}
})
}
nodes.getText = function(){
var texts = []
for (let i = 0; i < nodes.length; i++){
texts.push(nodes[i].textContent)
}
return texts
}
nodes.setText = function(text){
for (let i = 0; i < nodes.length; i++){
nodes[i].textContent = text
}
}
return nodes
}
- 经理看了之后,告诉我其实
jQuery
不喜欢get set
的方法,所以它把两个方法合成一个方法,用if
判断你传参的类型就能知道用户想获得还是要设置text
。把getText setText
属性改造成如下的text
属性
//等同于get、set方法
nodes.text = function(text){
if(text === undefined){
var texts = []
for (let i = 0; i < nodes.length; i++){
texts.push(nodes[i].textContent)
}
return texts
} else {
for (let i = 0; i < nodes.length; i++){
nodes[i].textContent = text
}
}
}
控制多个<li>
的内容================================>demo
一个很低级的错误
在上述的过程中,我把class这个关键字竟然当成了变量名,还纳闷哪里出错了,耽误了很多时间。
情景再现
- 如果你只是想给每一个
<li>
元素添加红色,直接传red
就可以了
//addClass属性简写成 nodes.addClass = function(className){ for (let i = 0; i < nodes.length; i++){ nodes[i].classList.add(className) } }
- 如果你只是想给每一个
- 可是一开始却是这样
变量命名一定要规范啊
由jQuery构造的变量尽量`#### 来表示
比如
window.$ = jQuery
var $div = $('div') //$div变量 是 所有div元素的伪数组
最终,少林在经理的循循善诱下,开始探索jQuery
的道路。虽然jQuery
使用量在下降,但是依然有60%的web开发人员在用。
以上并不是完全真实的jQuery的推导,只是大约是那个意思,可以帮助我更好的理解而已。真正的JQuery必须去看文档,英文文档,中文文档
总之,jQuery我来啦~去探索真正强大的jQuery
吧,去理解write less, do more
的含义吧,去体会一句顶一万句
的力量吧。
最后安利一波刘震云的小说一句顶一万句