Javascript DOM 笔记

DOM简单来说就是一套js访问Document对象的一套增删改查规则,文档由许多节点构成,需要重点关注的有Element、Document、Text、Attribute四种节点。

每种节点具有一些公共的属性,包括nodeType、nodeName、nodeValue、parentNode、nextSibling、previousSibling、nextElementSibling、previousElementSibling、ownerDocument、childNodes(类数组对象),其中 parentNode、nextSibling、previousSibling、firstChild、lastChild、nextElementSibling、previousElementSibling、firstElementChild、lastElementChild 如果不存在则为 null

var parent1 = document.getElementById("parent1");
var div1 = document.getElementById("div1");
var div2 = document.getElementById("div2");

console.log(div1.nodeName);        //DIV,自动转为大写
console.log(div1.nodeType);        //1
console.log(div1.nodeValue);       //null

每个节点都有一个childNodes属性,保存着一个NodeList对象。NodeList对象是一个类数组对象,动态的、实时的

  console.log(parent1.childNodes[1]);             //<div id="div1">This is a div</div>,在chrome中换行也被当做了一个节点,所以在获取node节点时,最好先判断一下nodeType
  //上面代码等价于下面代码
  console.log(parent1.childNodes.item(1));        //<div id="div1">This is a div</div>

  //非IE8以下的写法
  var arr = Array.prototype.slice.call(someNode.childNodes, 0);       //childNodes转化为数组对象
  var newarr = Array.prototype.slice.call(arguments, 0);          //arguments类数组转化为数组对象

  //可以用下面的代码将一个nodeList转化为一个数组对象,兼容IE8
  function convertToArray(nodes1) {
    var array1 = null;
    try {
      array1 = Array.prototype.slice.call(nodes1, 0);         //针对非IE浏览器
    } catch (ex) {
      array1 = new Array();
      for (var i = 0; i < array1.length; i++) {
        array1.push(nodes1[i]);
      }
    }
    return array1;
  }

  console.log(div1.nextSibling);          //#text

操作节点的方法有

someNode.appendChild(newNode) 末尾追加节点,返回追加的节点

someNode.insertBefore(newNode,null) 在parent节点的子节点的某个位置插入节点,返回插入的节点,如果后一个参数为null,则效果等同appendChild()

someNode.replaceChild(newNode, someNode.firstChild) 在parent节点的子节点中用新节点替换子节点中的一项

someNode.removeChild(someNode.firstChild) 在parent节点的子节点中移除

cloneNode(true) 克隆节点,不克隆事件,IE会复制事件,所以在复制节点前,先移除事件;当参数为true时,复制子节点

  var div3 = document.createElement("div");
  var returnedNode1 = div1.appendChild(div3);
  console.log(returnedNode1 == div3);        //true
  console.log(div1.lastChild == div3);       //true

传入文档的新节点是现在文档的一部分,则结果是将该节点从原来的位置转移到新位置。任何DOM节点不能同时出现在文档的多个位置上。

  var returnedNode2 = parent1.appendChild(parent1.firstChild);
  console.log(returnedNode2 == parent1.firstChild);        //false
  console.log(returnedNode2 == parent1.lastChild);        //true

  var returnedNode3 = parent1.insertBefore(div3, null);    //插入成为最后一个节点
  console.log(div3 == parent1.lastChild);             //true

  var returnedNode4 = parent1.insertBefore(div3, parent1.firstChild);         //返回插入的元素
  console.log(returnedNode4 == div3);            //true;
  console.log(div3 == parent1.firstChild);       //true

  //替换节点replaceChild() 移除节点removeChild()
  var returnedNode5 = parent1.removeChild(div3, parent1.firstChild);
  console.log(returnedNode5);            //<div></div>
  var returnedNode6 = parent1.removeChild(parent1.lastChild);
  console.log(returnedNode6);            //<div id="div1">This is a div</div>


  var ul1 = document.getElementById("ul1");
  var deepList = ul1.cloneNode(true);
  console.log(deepList);             //<ul id="ul1"><li>item1</li><li>item2</li><li>item3</li></ul>

  var shallowList = ul1.cloneNode();
  console.log(shallowList);          //<ul id="ul1"></ul>

Document类型
document实例 》 HTMLDocument类型 》 Document类型
JS通过Document类型表示文档。在浏览器中,document对象是HTMLDocument(继承自Document类型)的一个实例,表示整个HTML页面
document也是window的一个属性,可以直接访问

  var html1 = document.documentElement;          //获取对<html>的引用
  var html2 = document.childNodes[0];            //通过childNodes间接获取
  var body1 = document.body;                     //直接获取body
  var doctype1 = document.doctype;               //取得对<!DOCTYPE>的引用,有兼容性问题
  var originTitle1 = document.title;
  var url1 = document.URL;               //获取完整url
  var domain1 = document.domain;         //获取一级域名例:www.baidu.com
  var referrer1 = document.referrer;

  document.write();
  document.writeIn();

Element元素
HTML元素 》 HTML子类型 》 HTMLElement类型 》 Element类型

查找元素的方法
docuement.getElementById();
document.getElementsByTagName(); 返回一个HTMLCollection对象,也是一个动态集合。HTMLCollection有length属性、item()、namedItem()方法

特性
所有HTML元素都由HTMLElement类型表示,不是直接通过这个类型,也是通过它的子类型来表示。HTMLElement类型直接继承自Element并添加了一些属性。添加的这
些属性分别对应于每个HTML元素中都存在的下列标准特性。
id
title
lang
dir
className

取得特性
getAttribute(); 可以取得自定义特性。

setAttribute(); 设置的特性名会自动转化为小写

removeAttribute();

根据HTML5规范,自定义特性应该加上data-前缀以便验证。最上面的五种直接用js的方法来操作比较快,其他自定义属性用getAttribute 或者setAttribute等方法

通过js的方法来操作属性

  var div = document.getElementById("div1");
  console.log(div.id, div.className, div.title, div.lang, div.dir);
  //设置
  div.id = "newDiv";

  //通过node节点的方式来获取
  div.getAttribute("id");
  div.getAttribute("class");
  //获取自定义特性
  div.getAttribute("data-selfAttribute");

有两类特殊的属性,他们虽然有对应的属性名,但是属性的值与通过getAttribute()返回的值并不相同,即style和onclick处理函数
getAttribute()返回的值是字符串;
.style返回的是一个对象

创建元素

  var div = document.createElement("div");           //传入标签名
  var div = document.createElement("<div class=\"aaaa\">hello world!</div>");           //传入一段html文本

  //遍历时的注意事项
  for (var i = 0; var len = element.childNodes.length;i < len;i++){
    if (element.childNodes[i].nodeType == 1) {
      //do something ,先判断类型再执行操作
    }
  }

操作文本节点的方法有

appendData(text):将text添加到节点末尾
deleteData(offset,count):从offset指定的位置开始删除count个字符
insertData(offset,text):在offset处插入指定的字符串
replaceData(offset,count,text):用text替换从fosset指定的位置开始到offset+count出的字符串
splitText(offset):从offset指定的位置将当前文本节点分为两个文本节点。
substringData(offset,count):提取从offset指定的位置开始到fooset+count位置处的字符串

//创建文本节点并添加到div中

  var div = document.createElement("div");
  div.className = "aaa";

  var text = document.createTextNode("Hello world!");
  div.appendChild(text);

  document.body.appendChild(div);

通过normallize和splitText两个方法来拼合文本节点和打断文本节点

  var element = document.createElement("div");
  element.className = "message";

  var textNode1 = document.createTextNode("This is the first node, ");
  element.appendChild(textNode1);

  var textNode2 = document.createTextNode("This is the another node");
  element.appendChild(textNode2);

  document.body.appendChild(element);

  console.log(element.childNodes.length);             //2

  element.normalize();
  console.log(element.childNodes.length);                 //1
  console.log(element.firstChild.nodeValue);              //This is the first node, This is the another node,

  element.firstChild.splitText(10);                       //注意操作的是子节点
  console.log(element.childNodes.length);                 //2
  console.log(element.firstChild);                        //This is th


  document.querySelector();
  document.querySelectorAll()

//跨浏览器遍历所有元素

  var i,
    len,
    child = element.firstChild;
  while (child != element.lastChild) {
    if (child.nodeType === 1) {
      processChild(child);
    }
    child = child.nextSibling;
  }

//使用element traversal遍历

  var i,
    len,
    child = element.firstElementChild;
  while (child != element.lastElementChild) {
    processChild(child);
    child = child.nextElementSibling;
  }

HTML5新增一种该操作类名的方式classList,它是一个类似于nodelist的对象,有属性,有方法
length
add(value):将给定的字符串值添加到类表中,如果值已经存在了,就不添加了
contains(value):表示列表中是否存在给定的值,如果存在则返回true,否则返回false。
remove(value):从列表中删除给定的字符串。
toggle(value):

  div.classList.remove("disabled");
  div.classList.add("current");
  div.classList.toggle("user");

确定元素中是否包含既定的类名

  if (div.classList.contains("bd") && !div.classList.contains("disabled")) {
    //执行操作
  }
  //迭代类名
  for (var i = 0; len = div.classList.length; i < len;i++){
    doSomthing(div.classList[i]);
  }
  //动态脚本设置
  function loadScript(url){
    var script = document.createElement("script");
    script.type = "text/javascript";
    script.src = url;
    document.body.appendChild(script);
  }
  loadScript("client.js");          //不知道何时加载完毕

  //动态样式表
  function loadStyles(url) {
    var link = document.createElement("link");
    link.rel = "stylesheet";
    link.type = "text/css";
    link.href = url;
    var head = document.getElementsByTagName("head")[0];
    head.appendChild(link);
  }
  loadStyles("client.css")          //不知道何时加载完毕

HTML5单独增加的API

document.querySelector() 根据css规则来选择元素,返回第一个匹配元素,没有则返回null

document.querySelectorAll() 根据css规则来选择元素,返回一个 NodeList 对象,没有则返回空的 NodeList

matchesSelector() 如果调用元素与该选择符匹配,则返回true,否则,返回false;例子:

if(document.body.matchesSelector("body.page1")){ //true}

getElementsByClassName("className1 className2 ...")         通过className选取元素,返回nodeList对象

classList属性 一个class的属性对象。拥有add(value) contains(value) remove(value) toggle(value) 四个方法,方便操作class

document.activeElement属性 始终指向DOM当中当前获得了焦点的元素
默认情况下,文档刚刚加载完成时,document.activeElement中保存的是document.body元素的引用。文档加载期间,document.activeElement的值为null。

document.hasFocus() 确定文档是否获得了焦点,页面中的元素获得了焦点,再调用document.hasFocus() 返回true,代表用户正在与页面交互

readyState属性 文档是否已经加载完毕的指示器,其值有两个loading和complete 示例:

if(document.readyState == "complete"){ //... }

compatMode兼容模式属性 标准模式其值为CSS1Compat,混杂模式为BackCompat 示例:

if(document.compatMode == "CSS1Compat"){//...}else {/...}

document.head属性 直接获取head引用,示例:

var head = document.head || document.getElementsByTagName("head")[0]

document.charset字符集属性

data- 自定义数据属性,目的是提供与渲染无关的是信息,或者提供语义信息。 设置好之后,可以通过元素的dataset属性来访问
如果是驼峰写法,则会自动将驼峰分开来,转变成小写。

innerHTML 对应元素的子元素,设置值时只会替换子节点的内容,尽量把所有元素拼接成字符串后再append

outterHTML 设置时,会替换节点本身

scrollIntoView()方法 针对容器 该方法可以在所有HTML元素上调用,通过滚动浏览器窗口或某个容器元素,调用元素就可以出现再是口中。如果传入true,则调用元素会尽可能顶部与视口平齐。如果传入false,则调用元素会近可能全部出现在视口中,顶部不一定平齐。

contains方法 判断某个节点是不是另一个节点的后代 示例:

  * alert(document.documentElement.contains(document.body));        //true

下面三个方法,safari 和 chrome实现了
scrollIntoViewIfNeeded(alignCenter) 针对容器 只在当前元素再视口中公布可见的情况下,才滚动浏览器窗口或容器元素 alignCenter 设置为true时,会尽可能出现在视口中部

scrollByLines(lineCount) 针对元素 将元素的内容滚动指定的行高,正值和负值均可

scrollByPages(pageCount) 针对元素 将元素的内容滚动指定的页面高度,具体高度有元素的高度决定。

myDiv.style 获取元素样式,其中style里面包含的只有行间样式,没有嵌套和引用的

document.defaultView.getComputedStyle(element,null).color 获取计算后的样式,包括浏览器默认样式, 第二个参数通常设置为null

cuttentStyle属性 IE中不支持getComputedStyle 方法,但是有相同的属性来获取计算后的样式,注意border不能一次性获取,因为有四条边

offsetHeight 包括上下border padding
offsetWidth
offsetTop 元素的左外边框至包含元素的左内边框之间的像素距离,其中offsetLeft 和 offsetTop 属性与包含元素有关,
包含元素的引用保存再offsetParent属性中。offsetParent 属性不一定与parentNode的值相等。例如td元素的offsetParent 是
作为其祖先元素的table,因为table是DOM层次中距离td最近的一个具有大小的元素
offsetLeft

clientHeight 包含内容、上下padding,不包含上下border
clientWidth

scrollHeight 不包含滚动条,元素内容的总高度
scrollWidth
scrollLeft 被隐藏在padding左边的内容宽度,包含border
scrollTop

  //自定义属性示例:
  var div = document.getElementById("mydiv");
    //设置自定义属性
  div.dataset.appId = "45124";

  //获取自定义属性
  var appId = div.dataset.appId;
  var myName = div.dataset.myName;

焦点管理
HTML5添加了辅助管理DOM焦点的功能。首先就是,这个属性始终会引用DOM中当前获得焦点的元素。
元素获得焦点的方式有页面载入、用户输入和在代码中调用focus()方法

  var button = document.getElementById("myButton");
  button.focus();
  alert(document.activeElement === button);           //true

  var button = document.getElementById("myButton");
  button.focus();
  alert(document.hasFocus());             //true,证明用户正在与页面交互

  function getElementLeft(element) {
    var actualLeft = element.offsetLeft;
    var current = element.offsetParent;            //offsetParent相邻父元素

    while (current !== null) {
      actualLeft += current.offsetLeft;
      current = current.offsetParent;
    }
    return actualLeft;
  }

  function getElementTop(element) {
    var actualTop = element.offsetTop;
    var current = element.offsetParent;

    while (current !== null) {
      actualTop += current.offsetTop;
      current = current.offsetParent;
    }
    return actualTop;
  }

  /*获取客户区大小*/
  function getViewport() {
    if (document.compatMode == "BackCompat") {
      return {
        width: document.body.clientWidth,
        height: document.body.clientHeight
      }
    } else {
      return {
        width: document.documentElement.clientWidth,
        height: document.documentElement.clientHeight
      }
    }
  }

  //确定文档总高度
  var docHeight = Math.max(document.documentElement.scrollHeight, document.documentElement.clientHeight);
  var docWidth = Math.max(document.documentElement.scrollWidth, document.documentElement.clientWidth);

  function scrollToTop(element) {
    if (element.screenTop != 0) {
      element.screenTop = 0;
    }
  }


  //确定元素大小
  function getBoundingClientRect(element) {
    var scrollTop = document.documentElement.scrollTop;
    var scrollLeft = document.documentElement.scrollLeft;
    if (element.getBoundingClientRect) {
      if (typeof arguments.callee.offset != "number") {
        var temp = document.createElement("div");
        temp.style.cssText = "position: absolute; left: 0; top: 0;";
        document.body.appendChild(temp);
        arguments.callee.offset = -temp.getBoundingClientRect().top - scrollTop;
        document.body.removeChild(temp);
        temp = null;
      }
      var rect = element.getBoundingClientRect();
      var offset = arguments.callee.offset;

      return {
        left: rect.left + offset,
        right: rect.right + offset,
        top: rect.top + offset,
        bottom: rect.bottom + offset
      }
    } else {
      var actualLeft = getElementLeft(element);
      var actualTop = getElementTop(element);
      return {
        left: actualLeft - scrollLeft,
        right: actualLeft + element.offsetWidth - scrollLeft,
        top: actualTop - scrollTop,
        bottom: actualTop + element.offsetHeight - scrollTop
      }
    }
  }
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,384评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,845评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,148评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,640评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,731评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,712评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,703评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,473评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,915评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,227评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,384评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,063评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,706评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,302评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,531评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,321评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,248评论 2 352

推荐阅读更多精彩内容