7 javascript DOM

来源:互联网 发布:怎样申请淘宝小二介入 编辑:程序博客网 时间:2024/06/16 12:47
一.概述
由于HTML文档被浏览器解析后就是一颗DOM树,要改变HTML结构,就需要通过JavaScript来操作DOM
始终记住DOM是一个树形结构。操作一个DOM借点实际上就是这么几个操作:
更新:更新该DOM节点的内容,相当于更新了该DOM节点表示的HTML内容
遍历:遍历该DOOM节点下的子节点,以便进行进一步的操作
添加:在该DOM节点下新增一个子节点,相当于动态增加了一个HTML节点
删除:将该节点从HTML中删除,相当于删掉了该DOM节点的内容以及它包含的所有子节点

在操作一个DOM节点前,我们需要通过各种方式先拿到这个DOM节点。最常用的方法是document.getElementById()和document.getElementsByTagName(),
以及CSS选择器document.getElementByClassName()。


由于ID在HTML文档中是唯一的,所以document.getElementById()可以直接定位唯一的一个DOM节点。document.getElementsByTagName()和document.getElementByClassName()
总是返回一组DOM节点。要精确的选择DOM,可以先定位父节点,再从父节点开始选择,以缩小范围

// 返回ID为'test'的节点
var test = document.getElementById("test");

// 先定位ID为'teset-table'的节点,再返回内部所有tr节点
var trs = document.getElementById("test-table").getElementByTagName("tr");

// 先定位ID为'test-div'的节点,再返回其内部所有class包含red的节点
var reds = document.getElementById('test-div').getElementByClassName("red");

// 获取节点test下的所有直属子节点
var cs = test.children;

// 获取节点test下第一个、最后一个子节点
var first = test.firstElementChild;
var last = test.lastElementChild;

1.选择符API
1.querySelector()
querySelector()方法接受一个CSS选择符,返回与该模式匹配的第一个元素,如果没有找到匹配的元素,返回null

// 取得body元素
var body = document.querySelector("body");

// 取得ID为"myDiv"的元素
var myDiv = document.querySelector("#myDiv");

// 取得类为"button"的第一个图像元素
var img = document.querySelector("img.button");

2.querySelectorAll()
参数与querySelect一样,返回的是一个NodeList实例
// 通过querySelectorAll获取q1节点内的符合条件的所有节点
var ps = q1.querySelectorAll("div.highlighted > p");

2.更新DOM
拿到一个DOM节点后,我们可以对它进行更新
可以直接修改节点的文本,方法有两种:
一种是修改innertHtml属性,这个方式非常强大,不但可以修改一个DOM节点的内容,还可以直接通过HTML片段修改DOM节点内部的子树:

// 获取<p id="p-id">...</p>
var p = document.getElementById('p-id');
// 设置文本为abc
p.innerHTML = 'ABC';// <p id="p-id">ABC</p>
//设置HTML
p.innerHTML = 'ABC <span style="color:red">Red</span> XYZ';
// <p>...</p>的内部结构已修改
用innerHtml时要注意,是否需要写入HTML。

第二种是修改innerText或textContent属性,这样可以自动对字符串进行HTML编码,保证无法设置任何HTML标签
// 获取<p id="p-id">...</p>
var p = document.getElementById("p-id");
// 设置文本
p.innerText = '<script>alert("hi")</script>';
// HTML被自动编码,无法设置一个<script>节点

两者的区别在于读取属性时,innerText不返回隐藏元素的文本,而testContent返回所有文本。

修改CSS也是经常需要的操作。DOM节点的style属性对应的CSS,可以直接获取或设置。因为CSS允许font-size这样的名称,但它并非
javascript有效的属性名,所以需要在javascript中改写为驼峰是命名为fontSize
// 获取<p id="p-id">...</p>
var p = document.getElementById("p-id");
// 设置CSS
p.style.color = "#ff0000";
p.style.fontSize = "20px";
p.style.paddingTop = "2em";

3.插入DOM
当我们获得了某个DOM节点,想在这个DOM节点内插入新的DOM,应该如何做?
如果这个DOM节点是空的,例如,<div></div> ,那么,直接使用innerHtml = "<span>child</span>"就可以修改DOM节点的内容,相当于“插入”
了新的DOM节点
如果这个DOM节点不是空的,那就不能这么做,因为innerHtml会直接替换掉原来的所有子节点。

1.有两个办法可以插入新的节点。一个是使用appendChild,把一个子节点添加到父节点的最后一个子节点。例如:
// HTML结构
<p id="js">JavaScript</p>
<div id="list">
<p id="java">Java</p>
<p id="python">Python</p>
<p id="scheme">Scheme</p>
</div>
把<p id="js">JavaScript</p>添加到<div id="list">的最后一项:
var 
js = document.getElementById("js"),
list = document.getElementById("list");
list.appenChild(js);
现在,HTML结构变成了这样:
<!-- HTML结构 -->
<div id="list">
<p id="java">Java</p>
<p id="python">Python</p>
<p id="scheme">Scheme</p>
<p id="js">JavaScript</p>
</div>
因为我们插入的js的节点已经存在于当前的文档树,因此这个节点首先会从原先的位置删除,再插入到新的位置

更多的时候我们会从零创建一个新的节点,然后插入到指定位置:
var 
list = document.getElementById("list"),
hashell = document.createElement("p");
hashell.id = "hashell";
hashell.innerText = "Hashell";
list.appendChild(hashell);
这样我们就动态添加了一个新的节点:
<!-- HTML结构 -->
<div id="list">
<p id="java">Java</p>
<p id="python">Python</p>
<p id="scheme">Scheme</p>
<p id="haskell">Haskell</p>
</div>

动态创建一个节点然后添加到DOM树中,可以实现很多功能。举个例子,下面的代码动态创建了一个<style>节点,然后把它们添加到<head>节点的末尾,这样就动态地
给文档添加了新的CSS定义:
var d = document.createElement("style");
d.setAttribute("type", "text/css");
d.innerHTML = "p {color : red}";
document.getElementByTagName("head")[0].appenChild(d);

2.如果我们要把子节点插入到指定位置怎么办?可以使用parentElement.insertBefore(newElement, referenceElement);,子节点会插入到referenceElement之前
还是以上面的HTML为例,假定我们要把Haskell插入到Python之前:
<!-- HTML结构 -->
<div id="list">
<p id="java">Java</p>
<p id="python">Python</p>
<p id="scheme">Scheme</p>
</div>
可以这么写:
var 
list = document.getElementById("list");
ref = document.getElementById("python");
haskell = document.getElementById("p");
haskell.id = "haskell";
haskell.innerText = "Haskell";
list.insertBefore(haskell, ref);
新的HTML结构如下:
<!-- HTML结构 -->
<div id="list">
<p id="java">Java</p>
<p id="haskell">Haskell</p>
<p id="python">Python</p>
<p id="scheme">Scheme</p>
</div>
可见,使用insertBefore重点是要拿到一个“参考子节点”的引用。很多时候,循环一个父节点的所有子节点,可以通过迭代childer属性来实现:
var 
i, c,
list = document.getElementById("list");
for (i = 0; i < list.children.length; ++i){
c = list.children[i];// 拿到第i个子节点
}

4.删除DOM
要删除一个DOM节点就比插入要容易得多
要删除一个节点,首先要获得该节点本身以及它的父节点,然后,调用父节点的removeChild把自己删除:

// 拿到待删除的节点
var self = document.getElementById("to-be-removed");
// 拿到父节点
var parent = self.parentElement;
// 删除
var removed = parent.removeChild(self);
removed === self;// true

注意,虽然删除后的节点不在文档树中,但其实它还在内存中,可以随时再次被添加到别的位置

当你遍历一个父节点的子节点并进行删除操作时,要注意,children属性是一个只读属性,并且他在子节点变化时会实时更新
例如,对于如下HTML结构:
<div id="parent">
<p>First</p>
<p>Second</p>
</div>
当我们用如下代码删除子节点时:
var parent = document.getElementById("parent");
parent.removeChild(parent.children[0]);
parent.removeChild(parent.children[1]);// 浏览器报错
0 0