[TOC]

七、Element对象

Element元素的继承关系:

+---------------+       +-----------+          +----------+
|  EventTarget  | ----> | Node      | =------> | Element  |
+---------------+       +-----------+          +----------+

Element对象对应网页的HTML元素。每个HTML元素在DOM树上都会转化成一个Element节点对象。

元素节点的nodeType属性都是1.

<p>ppppp</p>
<script>
    var p = document.querySelector('p');
    console.log(p.nodeName);  // "P"
    console.log(p.nodeType);  // 1
</script>

Element对象继承了Node接口,因此Node的属性和方法在Element对象都存在。此外,不同的HTML元素对应的元素节点是不一样的,浏览器使用不同的构造函数,生成不同的元素节点,比如<a>元素的节点对象由HTMLAnchorElement构造函数生成,<button>元素的节点对象由HTMLButtonElement构造函数生成。因此,元素节点不是一种对象,而是一组对象,这些对象除了继承Element的属性和方法,还有各自构造函数的属性和方法。

1. Element实例的属性

1.1 元素特性的相关属性

  1. Element.id

Element.id属性返回指定元素的id属性,该属性可读写。

注意,id属性的值是大小写敏感,即浏览器能正确识别<p id="foo"><p id="FOO">这两个元素的id属性,但是最好不要这样命名。

<p id="foo">ppppp</p>
<script>
    var p = document.querySelector('p');
    console.log(p.id);  // "foo"
    p.id = "one";   // 此时ID会立刻修改
    console.log(p.id);  // "one"
</script>
  1. Element.tagName

Element.tagName属性返回指定元素的大写标签名,与nodeName属性的值相等。

<span id="myspan">Hello</span>
<script>
    var span = document.getElementById('myspan');
    console.log(span.tagName);
    console.log(span.nodeName);
</script>
  1. Element.dir

Element.dir属性用于读写当前元素的文字方向,可能是从左到右("ltr"),也可能是从右到左("rtl")。

  1. Element.accessKey

Element.accessKey属性用于读写分配给当前元素的快捷键。

<button accesskey="h" id="btn">点击</button>
<script>
    var btn = document.getElementById('btn');
    console.log(btn.accessKey);
</script>

上面代码中,btn元素的快捷键是h,按下Alt + h就能将焦点转移到它上面。

  1. Element.draggable

    Element.draggable属性返回一个布尔值,表示当前元素是否可拖动。该属性可读写。

  2. Element.lang

    Element.lang属性返回当前元素的语言设置。该属性可读写。

    // HTML 代码如下
    // <html lang="en">
    document.documentElement.lang // "en"
    
  3. Element.tabIndex

    Element.tabIndex属性返回一个整数,表示当前元素在 Tab 键遍历时的顺序。该属性可读写。

    tabIndex属性值如果是负值(通常是-1),则 Tab 键不会遍历到该元素。如果是正整数,则按照顺序,从小到大遍历。如果两个元素的tabIndex属性的正整数值相同,则按照出现的顺序遍历。遍历完所有tabIndex为正整数的元素以后,再遍历所有tabIndex等于0、或者属性值是非法值、或者没有tabIndex属性的元素,顺序为它们在网页中出现的顺序。

  4. Element.title

    Element.title属性用来读写当前元素的 HTML 属性title。该属性通常用来指定,鼠标悬浮时弹出的文字提示框。

    <p title="我是p" id="btn">111111</p>
    <script>
        var btn = document.getElementById('btn');
        console.log(btn.title); // "我是p"
    </script>
    

1.2 元素状态的相关属性

  1. Element.hidden

    Element.hidden属性返回一个布尔值,表示当前元素的hidden属性,用来控制当前元素是否可见。该属性可读写。

    <p title="我是p" id="btn">111111</p>
    <div id="mydiv">我会变</div>
    <script>
        var btn = document.getElementById('btn');
        var mydiv = document.getElementById('mydiv');
        btn.addEventListener('click', function () {
            mydiv.hidden = !mydiv.hidden;
        }, false);  // 点击btn 会切换 显示与隐藏 mydiv
    </script>
    

    注意,该属性与 CSS 设置是互相独立的。CSS 对这个元素可见性的设置,Element.hidden并不能反映出来。也就是说,这个属性并不难用来判断当前元素的实际可见性。

    CSS 的设置高于Element.hidden。如果 CSS 指定了该元素不可见(display: none)或可见(display: hidden),那么Element.hidden并不能改变该元素实际的可见性。换言之,这个属性只在 CSS 没有明确设定当前元素的可见性时才有效。

  2. Element.contentEditable,Element.isContentEditable

    <div contenteditable id="one">123</div>
    <script>
        var one = document.getElementById('one');
        console.log(one.contentEditable); // "true"此时页面one元素可以编辑
        console.log(one.isContentEditable); // true 
        one.contentEditable = false;  
        console.log(one.contentEditable); // "false"  页面不能编辑了
        console.log(one.isContentEditable); // false
    </script>
    

    上面代码中,<div>元素有contenteditable属性,因此用户可以在网页上编辑这个区块的内容。

    Element.contentEditable属性返回一个字符串,表示是否设置了contenteditable属性,有三种可能的值。该属性可写。

    "true":元素内容可编辑

    "false":元素内容不可编辑

    "inherit":元素是否可编辑,继承了父元素的设置

    Element.isContentEditable属性返回一个布尔值,同样表示是否设置了contenteditable属性。该属性只读。

1.3 Element.attributes

Element.attributes属性返回一个类似数组的对象,成员是当前元素节点的所有属性节点

<p id="one" class="tt">123</p>
<script>
    var p = document.querySelector('p');
    var attrs = p.attributes;
    for (var i = attrs.length - 1; i >= 0; i--) {
        console.log(attrs[i].name + '->' + attrs[i].value);
    }
    // "class->tt"
    // "id->one"
</script>

上面代码遍历p元素的所有属性。

1.4 Element.className,Element.classList

className属性用来读写当前元素节点的class属性。它的值是一个字符串,每个class之间用空格分割。

classList属性返回一个类似数组的对象,当前元素节点的每个class就是这个对象的一个成员。

<div class="one two three" id="myDiv"></div>
<script>
    var div = document.getElementById('myDiv');
    console.log(div.className); // "one two three"
    console.log(div.classList); 
    // DOMTokenList(3) ["one", "two", "three", value: "one two three", length:3]
</script>

上面代码中,className属性返回一个空格分隔的字符串,而classList属性指向一个类似数组的对象,该对象的length属性(只读)返回当前元素的class数量。

classList对象有下列方法:

add():增加一个 class

remove():移除一个 class

contains():检查当前元素是否包含某个 class

toggle():将某个 class 移入或移出当前元素

item():返回指定索引位置的 class

toString():将 class 的列表转为字符串

<div class="one two three" id="myDiv"></div>
<script>
    var div = document.getElementById('myDiv');
    div.classList.add('myCssClass');
    div.classList.add('foo', 'bar');
    div.classList.remove('myCssClass');
    div.classList.toggle('myCssClass'); // 如果 myCssClass 不存在就加入,否则移除
    div.classList.contains('myCssClass'); // 返回 true 或者 false
    div.classList.item(0); // 返回第一个 Class
    div.classList.toString();
</script>

classNameclassList在添加和删除某个 class 时的写法:

<div class="one" id="foo"></div>
<script>
    var foo = document.getElementById('foo');
    // 添加class
    foo.className += 'bold';
    foo.classList.add('bold');
    // 删除class
    foo.classList.remove('bold');
    foo.className = foo.className.replace(/^bold$/, '');
</script>

toggle方法可以接受一个布尔值,作为第二个参数。如果为true,则添加该属性;如果为false,则去除该属性。

el.classList.toggle('abc', boolValue);

// 等同于
if (boolValue) {
  el.classList.add('abc');
} else {
  el.classList.remove('abc');
}

1.5 Element.dataset

网页元素可以自定义data-属性,用来添加数据。

Element.dataset属性返回一个对象,可以从这个对象读写data-属性。

<div id="foo" data-columns="3" data-index-number="12314" data-parent="cars"></div>
<script>
    var foo = document.getElementById('foo');
    console.log(foo.dataset.columns);   // "3"
    console.log(foo.dataset.indexNumber); // "12314"
    console.log(foo.dataset.parent);    // "cars"
</script>

注意,dataset上面的各个属性返回都是字符串。

HTML 代码中,data-属性的属性名,只能包含英文字母、数字、连词线(-)、点(.)、冒号(:)和下划线(_)。它们转成 JavaScript 对应的dataset属性名,规则如下:

开头的data-会省略

如果连词线后面跟了一个英文字母,那么连词线会取消,该字母变成大写

其他字符不变

因此,data-abc-def对应dataset.abcDefdata-abc-1对应dataset["abc-1"]

除了使用dataset读写data-属性,也可以使用Element.getAttribute()Element.setAttribute(),通过完整的属性名读写这些属性。

<div id="foo" data-columns="3" data-index-number="12314" data-parent="cars"></div>
<script>
    var foo = document.getElementById('foo');
    console.log(foo.getAttribute('data-columns'));    // "3"
    console.log(foo.getAttribute('data-index-number')); // "12314"
    console.log(foo.getAttribute('data-parent'));   // "cars"
</script>

1.6 Element.innerHTML

Element.innerHTML属性返回一个字符串,等同于该元素包含的所有 HTML 代码。该属性可读写,常用来设置某个节点的内容。它能改写所有元素节点的内容,包括<HTML><body>元素。

如果将innerHTML属性设为空,等于删除所有它包含的所有节点:el.innerHTML = '';,代码等于将el节点变成了一个空节点,el原来包含的节点被全部删除。

注意,读取属性值的时候,如果文本节点包含&、小于号(<)和大于号(>),innerHTML属性会将它们转为实体形式&amp;&lt;&gt;。如果想得到原文,建议使用element.textContent属性。

<p id="para"> 5 > 3 </p>
<script>
    var foo = document.getElementById('para')
    console.log(foo.innerHTML);   // " 5 &gt; 3 "
</script>

写入的时候,如果插入的文本包含 HTML 标签,会被解析成为节点对象插入 DOM。注意,如果文本之中含有<script>标签,虽然可以生成script节点,但是插入的代码不会执行。

<p id="one">one</p>
<script>
    var one = document.getElementById('one');
    var name = "<script>alert('haha')\<\/script>";
    one.innerHTML = name;
    console.log(one.innerHTML);
</script>

上面代码将脚本插入内容,脚本并不会执行。但是,innerHTML还是有安全风险的。

<p id="one">one</p>
<script>
    var one = document.getElementById('one');
    var name = "<img src=x onerror=alert(1)>";
    one.innerHTML = name; // 页面运行后 会弹出框
    console.log(one.innerHTML);
</script>

上面代码中,alert方法是会执行的。因此为了安全考虑,如果插入的是文本,最好用textContent属性代替innerHTML

1.7 Element.outerHTML

Element.outerHTML属性返回一个字符串,表示当前元素节点的所有 HTML 代码,包括该元素本身和所有子元素。

<div id="d"><p>Hello</p></div>
<script>
    var d = document.getElementById('d');
    console.log(d.innerHTML); // <p>Hello</p> 
    // "<p>Hello</p>"
    console.log(d.outerHTML); // <div id="d"><p>Hello</p></div>
    // "<div id=\"d\"><p>Hello</p></div>"
</script>

outerHTML属性是可读写的,对它进行赋值,等于替换掉当前元素。

<div id="container"><div id="d">Hello</div></div>
<script>
    var container = document.getElementById('container');
    var d = document.getElementById('d');
    console.log(container.firstChild.nodeName); // "DIV"
    console.log(d.nodeName);          // "DIV"
    d.outerHTML = '<p>Hello</p>';
    console.log(container.firstChild.nodeName); // "P"
    console.log(d.nodeName);          // "DIV"
</script>

上面代码中,变量d代表子节点,它的outerHTML属性重新赋值以后,内层的div元素就不存在了,被p元素替换了。但是,变量d依然指向原来的div元素,这表示被替换的DIV元素还存在于内存中。

注意,如果一个节点没有父节点,设置outerHTML属性会报错:

var div = document.createElement('div');
div.outerHTML = '<p>test</p>';
// DOMException: Failed to set the 'outerHTML' property on 'Element': This element has no parent node.

1.8 Element.clientHeight,Element.clientWidth

Element.clientHeight属性返回一个整数值,表示元素节点的 CSS 高度(单位像素),只对块级元素生效,对于行内元素返回0。如果块级元素没有设置 CSS 高度,则返回实际高度。

除了元素本身的高度,它还包括padding部分,但是不包括bordermargin。如果有水平滚动条,还要减去水平滚动条的高度。注意,这个值始终是整数,如果是小数会被四舍五入。

Element.clientWidth属性返回元素节点的 CSS 宽度,同样只对块级元素有效,也是只包括元素本身的宽度和padding,如果有垂直滚动条,还要减去垂直滚动条的宽度。

document.documentElementclientHeight属性,返回当前视口的高度(即浏览器窗口的高度),等同于window.innerHeight属性减去水平滚动条的高度(如果有的话)。document.body的高度则是网页的实际高度。一般来说,document.body.clientHeight大于document.documentElement.clientHeight

// 视口高度
document.documentElement.clientHeight
// 网页总高度
document.body.clientHeight

1.9 Element.clientLeft,Element.clientTop

Element.clientLeft属性等于元素节点左边框(left border)的宽度(单位像素),不包括左侧的paddingmargin。如果没有设置左边框,或者是行内元素(display: inline),该属性返回0。该属性总是返回整数值,如果是小数,会四舍五入。

Element.clientTop属性等于网页元素顶部边框的宽度(单位像素),其他特点都与clientTop相同。

<style>
    #tt {border: 1px solid}
    #one { border: 12px solid red; padding:10px; margin: 15px; }
</style>
<div id="tt">
    <div id="one">one</div>
</div>
<script>
    var one = document.getElementById('one');
    console.log(one.clientLeft);  // 12
    console.log(one.clientTop);   // 12
</script>

1.10 Element.scrollHeight,Element.scrollWidth

Element.scrollHeight属性返回一个整数值(小数会四舍五入),表示当前元素的总高度(单位像素),包括溢出容器、当前不可见的部分。它包括padding,但是不包括bordermargin以及水平滚动条的高度(如果有水平滚动条的话),还包括伪元素(::before::after)的高度。

Element.scrollWidth属性表示当前元素的总宽度(单位像素),其他地方都与scrollHeight属性类似。这两个属性只读。

整张网页的总高度可以从document.documentElementdocument.body上读取。

<style>
    #tt { border: 1px solid; height: 100px; width: 200px;  overflow: scroll; }
</style>
<div id="tt">
    <p>123ssssaaaaaaaaaaaaaaaaaaaaaa</p>
    <p>123</p><p>123</p><p>123</p>
    <p>123</p><p>123</p><p>123</p><p>123</p>
</div>
<script>
    var tt = document.getElementById('tt');
    console.log(tt.clientHeight); // 85
    console.log(tt.clientWidth);  // 185
    console.log(tt.scrollHeight); // 320
    console.log(tt.scrollWidth);  // 255
</script>

注意,如果元素节点的内容出现溢出,即使溢出的内容是隐藏的,scrollHeight属性仍然返回元素的总高度。

上面的代码中,即使tt元素的CSS高度只有100像素,且溢出部分不可见,但是scrollHeight仍然会返回该元素的原始高度。

1.11 Element.scrollLeft,Element.scrollTop

Element.scrollLeft属性表示当前元素的水平滚动条向右侧滚动的像素数量,Element.scrollTop属性表示当前元素的垂直滚动条向下滚动的像素数量。对于那些没有滚动条的网页元素,这两个属性总是等于0。

这两个属性都可读写,设置该属性的值,会导致浏览器将当前元素自动滚动到相应的位置。

如果要查看整张网页的水平的和垂直的滚动距离,要从document.documentElement元素上读取。

<style>
    #tt { border: 1px solid; height: 100px; width: 200px;  overflow: scroll; }
</style>
<div id="tt">
    <p>123ssssaaaaaaaaaaaaaaaaaaaaaa</p>
    <p>123</p><p>123</p><p>123</p>
    <p>123</p><p>123</p><p>123</p><p>123</p>
</div>
  <button>我是按钮</button>
  <script>
    var tt = document.getElementById('tt');
    var btn = document.querySelector("button");
    btn.onclick = function() {
      console.log(tt.scrollLeft); // 0 
      console.log(tt.scrollTop);  // 0
      tt.scrollLeft = "40";     // 执行后横向滚动条会移动到中间位置(大约)
    }
    // 读取整张网页的水平的和垂直的滚动距离
    console.log(document.documentElement.scrollLeft)
    console.log(document.documentElement.scrollLeft)
  </script>

1.12 Element.offsetParent

Element.offsetParent属性返回最靠近当前元素的、并且 CSS 的position属性不等于static的上层元素。

该属性主要用于确定子元素位置偏移的计算基准,Element.offsetTopElement.offsetLeft就是根据offsetParent元素的距离计算的。

如果该元素是不可见的(display属性为none),或者位置是固定的(position属性为fixed),则offsetParent属性返回null

如果某个元素的所有上层节点的position属性都是static,则Element.offsetParent属性指向<body>元素。

<style>
    #tt { border: 1px solid; height: 100px; width: 200px;  overflow: scroll; }
</style>

<div id="tt" style="position: relative;">
    <p>
        <span id="one">Hello</span>
    </p>
    <p>
        <span id="two" style="display: none;">Hello</span>
    </p>
</div>
<span id="three">333</span>
<script>
    var one = document.getElementById('one');
    var two = document.getElementById('two');
    var three = document.getElementById('three');
    console.log(one.offsetParent);  // <div  id="tt" ......
    console.log(two.offsetParent) // null
    console.log(three.offsetParent) // <body> ......
</script>

上面的代码中,id为one的元素的offsetParent属性就是id为ttdiv元素。

上面的代码中,id为twospan元素的offsetParent属性是null

上面的代码中,id为threespan元素的offsetParent属性是body元素。

1.13 Element.offsetHeight,Element.offsetWidth

Element.offsetHeight属性返回一个整数,表示元素的 CSS 垂直高度(单位像素),包括元素本身的高度、padding 和 border,以及水平滚动条(如果存在且渲染的话),不包含:before或:after等伪类元素的高度。

Element.offsetWidth属性表示元素的 CSS 水平宽度(单位像素),其他都与Element.offsetHeight一致。

这两个属性都是只读属性,只比Element.clientHeightElement.clientWidth多了边框的高度或宽度。如果元素的 CSS 设为不可见(比如display: none;),则返回0

<style>
    #tt { border: 5px solid; height: 100px; width: 200px; padding: 5px; overflow: scroll; }
</style>
<div id="tt">
    <p>123ssssaaaaaaaaaaaaaaaaaaaaaa</p>
    <p>123</p><p>123</p><p>123</p>
    <p>123</p><p>123</p><p>123</p><p>123</p>
</div>
<script>
    var tt = document.getElementById('tt');
    console.log(tt.clientHeight); // 95
    console.log(tt.clientWidth);  // 195
    console.log(tt.scrollHeight); // 330
    console.log(tt.scrollWidth);  // 260
    console.log(tt.offsetHeight); // 120
    console.log(tt.offsetWidth ); // 220
</script>

1.14 Element.offsetLeft,Element.offsetTop

Element.offsetLeft返回当前元素左上角相对于Element.offsetParent节点的水平位移,Element.offsetTop返回垂直位移,单位为像素。通常,这两个值是指相对于父节点的位移。

<style>
    body { margin: 0; }
    .one { border: 1px solid; height: 300px; width: 300px;
        overflow: scroll; position: relative;
    }
    p { margin: 30px 0; border: 1px solid red; }
</style>
<p>123</p>
<div class="one">
    <p>1</p><p>2</p><p>3</p> <p>4</p><p>5</p><p>6</p>
    <p>7</p><p>8</p><p>9</p> <p>10</p><p>11</p><p>12</p>
    <p>13</p>dddddddd <span class=f>ffffffffffffffffffffffffff</span><p>14</p>
    <p>15</p><p>16</p><p>17</p> <p>18</p><p>19</p><p>20</p>
</div>
<script>
    var one = document.querySelector(".one");
    var f = document.querySelector(".f");
    // one.scrollTop = f.offsetTop; 设置 直接滚动到指定位置
    console.log(f.offsetLeft) // 80
    console.log(f.offsetTop)  // 732
</script>

下面的代码可以算出元素左上角相对于整张网页的坐标:

function getElementPosition(e) {
  var x = 0;
  var y = 0;
  while (e !== null)  {
    x += e.offsetLeft;
    y += e.offsetTop;
    e = e.offsetParent; // 这个是重点,每次循环都找到offsetParent
  }
  return {x: x, y: y};
}

下面是函数测试:

<style>
    body { margin: 0; border: 1px solid;}    
    p { margin: 30px 0; border: 1px solid red; }
  </style>
  <p>123</p>
  <div> <p>内部</p>
    <div> <p>又一个内部</p
      <div> <p id="one">又一个</p>
      </div>
    </div>
  </div>
  <script>
      var one = document.querySelector("#one");
      console.log(one.offsetTop);
      function getElementPosition(e) {
        var x = 0; var y = 0;
        while (e !== null)  {
          x += e.offsetLeft; y += e.offsetTop; e = e.offsetParent;
        }
        return {x: x, y: y};
      }
    var tmp = getElementPosition(one);
    console.log(tmp.x)  // 1
    console.log(tmp.y)  // 193
  </script>

1.15 Element.style

每个元素节点都有style用来读写该元素的行内样式信息

style属性返回一个 CSSStyleDeclaration 对象,表示元素的 内联style 属性(attribute),但忽略任何样式表应用的属性。

由于 style 属性的优先级和通过style设置内联样式是一样的,并且在css层级样式中拥有最高优先级,因此在为特定的元素设置样式时很有用。

HTMLElement.style MDN

1.16 Element.children,Element.childElementCount

Element.children属性返回一个类似数组的对象(HTMLCollection实例),包括当前元素节点的所有子元素。如果当前元素没有子元素,则返回的对象包含零个成员。

Element.children属性与Node.childNodes属性的区别是,它只包括元素类型的子节点,不包括其他类型的子节点。

Element.childElementCount属性返回当前元素节点包含的子元素节点的个数,与Element.children.length的值相同。

<div id="one">
    <span>1 </span><span>2 </span><span>3 </span>
</div>
<div id="two"></div>
<script>
    var one = document.getElementById("one");
    var two = document.getElementById("two");
    if (one.children.length) {
        var children = one.children;
        for (var i = 0; i < children.length; i++){
            console.log(children[i]);
        }
    }
    // <span>1 </span>
    // <span>2 </span>
    // <span>3 </span>
    console.log(two.children);  // HTMLCollection []
</script>

1.17 Element.firstElementChild,Element.lastElementChild

Element.firstElementChild属性返回当前元素的第一个元素子节点。

Element.lastElementChild返回最后一个元素子节点。

如果没有元素子节点,这两个属性返回null

<div id="one">
    <span>1 </span><span>2 </span><span>3 </span>
  </div>
  <div id="two"></div>
  <script>
    var one = document.getElementById("one");
    var two = document.getElementById("two");
    console.log(one.firstElementChild); // <span>1 </span>
    console.log(one.lastElementChild);  // <span>3 </span>
    console.log(two.firstElementChild); // null
    console.log(two.lastElementChild);  // null
  </script>

1.18 Element.nextElementSibling,Element.previousElementSibling

Element.nextElementSibling属性返回当前元素节点的后一个同级元素节点,如果没有则返回null

Element.previousElementSibling属性返回当前元素节点的前一个同级元素节点,如果没有则返回null

<div id="tt">
    <div id="div-01">Here is div-01</div>
    <div id="div-02">Here is div-02</div>
  </div>
  <script>
    var one = document.getElementById('div-01');
    var two = document.getElementById("div-02");
    console.log(one.nextElementSibling);  // <div id="div-02">Here is div-02</div>
    console.log(two.nextElementSibling);  // null
    console.log(one.previousElementSibling);// null
    console.log(two.previousElementSibling);// <div id="div-01">Here is div-01</div>
  </script>

2. 实例方法

2.1 属性相关方法

元素节点提供六个方法,用来操作属性。

  • getAttribute():读取某个属性的值
  • getAttributeNames():返回当前元素的所有属性名
  • setAttribute():写入属性值
  • hasAttribute():某个属性是否存在
  • hasAttributes():当前元素是否有属性
  • removeAttribute():删除属性

这些方法的介绍请看《属性的操作》一章。

2.1 Element.querySelector()

Element.querySelector方法接受 CSS 选择器作为参数,返回父元素的第一个匹配的子元素。如果没有找到匹配的子元素,就返回null

<div id="tt">
    <div>123</div> <p>p</p> <p>2p</p>
</div>
<script>
    var one = document.getElementById('tt');
    var p = one.querySelector("p");
    console.log(p); // <p>p</p>
</script>

上面代码返回content节点的第一个p元素。

Element.querySelector方法可以接受任何复杂的 CSS 选择器。

document.body.querySelector("style[type='text/css'], style:not([type])");

注意,这个方法无法选中伪元素。

它可以接受多个选择器,它们之间使用逗号分隔。

element.querySelector('div, p');

上面代码返回element的第一个divp子元素。

需要注意的是,浏览器执行querySelector方法时,是先在全局范围内搜索给定的 CSS 选择器,然后过滤出哪些属于当前元素的子元素。因此,会有一些违反直觉的结果,下面是一段 HTML 代码。

<div>
    <blockquote id="outer">
        <p>Hello</p>
        <div id="inner">
            <p>World</p>
        </div>
    </blockquote>
</div>
<script>
    var outer = document.getElementById('outer');
    var p = outer.querySelector('div p')
    console.log(p); // <p>Hello</p>
</script>

上面的代码实际上返回的是第一个p元素,而不是第二个。

2.3 Element.querySelectorAll()

Element.querySelectorAll方法接受 CSS 选择器作为参数,返回一个NodeList实例,包含所有匹配的子元素。

该方法的执行机制与querySelector方法相同,也是先在全局范围内查找,再过滤出当前元素的子元素。因此,选择器实际上针对整个文档的。

它也可以接受多个 CSS 选择器,它们之间使用逗号分隔。如果选择器里面有伪元素的选择器,则总是返回一个空的NodeList实例。

<div id="one">
    <div class="tt"><p>tt</p></div>
    <div id="two">
        <p>1</p> <p>2</p> <p>3</p>
        <p>4</p> <p>5</p>
    </div>
</div>
<script>
    var one = document.getElementById('one');
    var p = one.querySelectorAll('div#two > p')
    console.log(p); // NodeList(5) [p, p, p, p, p]
</script>

2.4 Element.getElementsByClassName()

Element.getElementsByClassName方法返回一个HTMLCollection实例,成员是当前元素节点的所有具有指定 class 的子元素节点。该方法与document.getElementsByClassName方法的用法类似,只是搜索范围不是整个文档,而是当前元素节点。

注意,该方法的参数大小写敏感。

由于HTMLCollection实例是一个活的集合,document对象的任何变化会立刻反应到实例,下面的代码不会生效。

<div id="example">
    <p class="foo">123</p>
    <p class="foo">abc</p>
</div>
<script>
    var element = document.getElementById('example');
    var matches = element.getElementsByClassName('foo');
    for (var i = 0; i< matches.length; i++) {
        matches[i].classList.remove('foo');
        matches.item(i).classList.add('bar');
    }
    // 执行后,HTML 代码如下
    // <div id="example">
    //   <p></p>
    //   <p class="foo bar"></p>
    // </div>
</script>

上面代码中,matches集合的第一个成员,一旦被拿掉 class 里面的foo,就会立刻从matches里面消失,导致出现上面的结果。

2.5 Element.getElementsByTagName()

Element.getElementsByTagName方法返回一个HTMLCollection实例,成员是当前节点的所有匹配指定标签名的子元素节点。该方法与document.getElementsByClassName方法的用法类似,只是搜索范围不是整个文档,而是当前元素节点。

注意,该方法的参数是大小写不敏感的。

<div id="one">
    <p id="foo">123</p>
    <p class="foo">abc</p>
</div>
<script>
    var one = document.getElementById('one');
    var foo = document.getElementById("foo");
    var p = one.getElementsByTagName('p');
    console.log(p);
    one.removeChild(foo);
    console.log(p);
</script>

2.6 Element.closest()

Element.closest方法接受一个 CSS 选择器作为参数,返回匹配该选择器的、最接近当前节点的一个祖先节点(包括当前节点本身)。如果没有任何节点匹配 CSS 选择器,则返回null

<article>
    <div id="div-01">Here is div-01
        <div id="div-02">Here is div-02
            <div id="div-03">Here is div-03</div>
        </div>
    </div>
</article>
<script>
    var div03 = document.getElementById('div-03');
    console.log(div03.closest("#div-02"));    // div-02
    console.log(div03.closest("div div"));    // div-03
    console.log(div03.closest("article > div"));//div-01
    console.log(div03.closest(":not(div)"));  // article
</script>

上面代码中,由于closest方法将当前节点也考虑在内,所以第二个closest方法返回div-03

2.7 Element.matches()

Element.matches方法返回一个布尔值,表示当前元素是否匹配给定的 CSS 选择器。

<div class="one two" id="tt">one</div>
<script>
    var tt = document.getElementById('tt');
    console.log(tt.matches(".two"));  // true
    console.log(tt.matches(".three"));  // false
    console.log(tt.matches("#tt"));   // true
    console.log(tt.matches("#ee"));   // false
</script>

2.8 事件相关方法

以下三个方法与Element节点的事件相关。这些方法都继承自EventTarget接口,详见相关章节。

  • Element.addEventListener():添加事件的回调函数
  • Element.removeEventListener():移除事件监听函数
  • Element.dispatchEvent():触发事件
element.addEventListener('click', listener, false);
element.removeEventListener('click', listener, false);

var event = new Event('click');
element.dispatchEvent(event);

2.9 Element.scrollIntoView()

Element.scrollIntoView方法滚动当前元素,进入浏览器的可见区域,类似于设置window.location.hash的效果。

该方法可以接受一个布尔值作为参数:

如果为true,表示元素的顶部与当前区域的可见部分的顶部对齐(前提是当前区域可滚动);

如果为false,表示元素的底部与当前区域的可见部分的尾部对齐(前提是当前区域可滚动)。

如果没有提供该参数,默认为true

<style>
    body { margin: 0; }
    .one { border: 1px solid; height: 300px; width: 300px;
        overflow: scroll; position: relative;
    }
    p { margin: 30px 0; border: 1px solid red; }
</style>
<p>123</p>
<div class="one">
    <p>1</p><p>2</p><p>3</p> <p>4</p><p>5</p><p>6</p>
    <p>7</p><p>8</p><p>9</p> <p>10</p><p>11</p><p>12</p>
    <p>13</p>dddddddd <span class=f>ffffffffffffffffffffffffff</span><p>14</p>
    <p>15</p><p>16</p><p>17</p> <p>18</p><p>19</p><p>20</p>
</div>
<script>
    var one = document.querySelector(".one");
    var f = document.querySelector(".f");
    f.scrollIntoView();   // f移动到了one内部的可见区域顶部 
    f.scrollIntoView(false);// f移动到了one内部的可见区域底部
</script>

2.10 Element.getBoundingClientRect()

Element.getBoundingClientRect方法返回一个对象,提供当前元素节点的大小、位置等信息,基本上就是 CSS 盒状模型的所有信息。

<style>
    body { margin: 0; }
    .one { border: 1px solid; height: 300px; width: 300px;
    }
</style>
<p>ss</p>
<div class="one">
    <p>123</p>
</div>
<script>
    var one = document.querySelector(".one");
    var ret = one.getBoundingClientRect();
    console.log(ret); // DOMRect {x: 0, y: 54, width: 302, height: 302, top: 54, …}
    console.log(Object.keys(ret));  // []
</script>

上面代码中,getBoundingClientRect方法返回的rect对象,具有以下属性(全部为只读)。

  • x:元素左上角相对于视口的横坐标
  • y:元素左上角相对于视口的纵坐标
  • height:元素高度
  • width:元素宽度
  • left:元素左上角相对于视口的横坐标,与x属性相等
  • right:元素右边界相对于视口的横坐标(等于x + width
  • top:元素顶部相对于视口的纵坐标,与y属性相等
  • bottom:元素底部相对于视口的纵坐标(等于y + height

由于元素相对于视口(viewport)的位置,会随着页面滚动变化,因此表示位置的四个属性值,都不是固定不变的。如果想得到绝对位置,可以将left属性加上window.scrollXtop属性加上window.scrollY

注意,getBoundingClientRect方法的所有属性,都把边框(border属性)算作元素的一部分。也就是说,都是从边框外缘的各个点来计算。因此,widthheight包括了元素本身 + padding + border

另外,上面的这些属性,都是继承自原型的属性,Object.keys会返回一个空数组,这一点也需要注意。

上面代码中,ret对象没有自身属性,而Object.keys方法只返回对象自身的属性,所以返回了一个空数组。

2.11 Element.getClientRects()

Element.getClientRects方法返回一个类似数组的对象,里面是当前元素在页面上形成的所有矩形(所以方法名中的Rect用的是复数)。每个矩形都有bottomheightleftrighttopwidth六个属性,表示它们相对于视口的四个坐标,以及本身的高度和宽度。

对于盒状元素(比如<div><p>),该方法返回的对象中只有该元素一个成员。

对于行内元素(比如<span><a><em>),该方法返回的对象有多少个成员(经过chrome浏览器测试):

HTML代码中行内元素行数,加上实际网页行数减一,

比如代码中span元素写了三行,实际页面是2行,那么返回的对象就是4个

Element.getBoundingClientRect()方法对于行内元素总是返回一个矩形对象。

<style>
    HTML, body {margin: 0;}
    .one {
        width: 800px; border: 1px solid; font-size: 15px;
    }
</style>
<div class="one">
    <span id="inline">
        Hello World
        Element.getClientRects
        方法返回一个类似数组的对象Element.getClientRects方法返回一个类似数组的对象Ele
    </span>
</div>
<script>
    var el = document.getElementById('inline');
    console.log(el.getClientRects().length);// 4
    console.log(el.getClientRects());   // 1
    // DOMRectList {0: DOMRect, 1: DOMRect, 2: DOMRect, 3: DOMRect, length: 4}
    console.log(el.getBoundingClientRect());
    // DOMRect {x: 1, y: 1, width: 790.71875, height: 42, top: 1, …}
</script>

这个方法主要用于判断行内元素是否换行,以及行内元素的每一行的位置偏移。

2.12 Element.insertAdjacentElement()

Element.insertAdjacentElement方法在相对于当前元素的指定位置,插入一个新的节点。该方法返回被插入的节点,如果插入失败,返回null

该方法一共可以接受两个参数,第一个参数是一个字符串,表示插入的位置,第二个参数是将要插入的节点。第一个参数只可以取如下的值:

  • beforebegin:当前元素之前
  • afterbegin:当前元素内部的第一个子节点前面
  • beforeend:当前元素内部的最后一个子节点后面
  • afterend:当前元素之后

注意,beforebeginafterend这两个值,只在当前节点有父节点时才会生效。如果当前节点是由脚本创建的,没有父节点,那么插入会失败。

如果插入的节点是一个文档里现有的节点,它会从原有位置删除,放置到新的位置。

<p id="pp">我是pp</p>
<div id="one"> 123</div>
<script>
    var one = document.getElementById('one');
    var pp = document.getElementById('pp');
    var p1 = document.createElement('p');
    var p2 = document.createElement('p');
    p2.innerText = 'p2q';
    tt = p1.insertAdjacentElement('afterend', p2);
    dd = one.insertAdjacentElement('afterend', p2);
    ff = one.insertAdjacentElement('afterend', pp);// 这里会把 pp移动到div里面
    console.log(tt);// null
    console.log(dd);// <p>p2q</p>
    console.log(ff);// <p id="pp">我是pp</p>
</script>

2.13 Element.insertAdjacentHTML(),Element.insertAdjacentText()

Element.insertAdjacentHTML方法用于将一个 HTML 字符串,解析生成 DOM 结构,插入相对于当前节点的指定位置。

该方法接受两个参数,第一个是一个表示指定位置的字符串,第二个是待解析的 HTML 字符串。第一个参数只能设置下面四个值之一。

  • beforebegin:当前元素之前
  • afterbegin:当前元素内部的第一个子节点前面
  • beforeend:当前元素内部的最后一个子节点后面
  • afterend:当前元素之后

该方法只是在现有的 DOM 结构里面插入节点,这使得它的执行速度比innerHTML方法快得多。

注意,该方法不会转义 HTML 字符串,这导致它不能用来插入用户输入的内容,否则会有安全风险。

Element.insertAdjacentText方法在相对于当前节点的指定位置,插入一个文本节点,用法与Element.insertAdjacentHTML方法完全一致。

<div id="one">one</div>
<div id="two">two</div>
<script>
    var one = document.getElementById('one');
    var two = document.getElementById('two');
    one.insertAdjacentHTML('beforeend', '<p id="pt">pp</p>');
    one.insertAdjacentHTML('beforeend', '<>   1');
    two.insertAdjacentText('beforeend', '<p id="pt">pp</p>');
    two.insertAdjacentText('beforeend', '<>   1');
    
    // HTML 代码:<div id="one">one</div>
    var d1 = document.getElementById('one');
    d1.insertAdjacentHTML('afterend', '<div id="two">two</div>');
    // 执行后的 HTML 代码:
    // <div id="one">one</div><div id="two">two</div>
    
    // HTML 代码:<div id="one">one</div>
    var d1 = document.getElementById('one');
    d1.insertAdjacentText('afterend', 'two');
    // 执行后的 HTML 代码:
    // <div id="one">one</div>two
</script>

2.14 Element.remove()

Element.remove方法继承自 ChildNode 接口,用于将当前元素节点从它的父节点移除。

3<div id="one">one</div>4
<script>
    var one = document.getElementById('one');
    tt = one.remove();  // one从HTML中删除
    console.log(tt)   // undefined
</script>

2.15 Element.focus(),Element.blur()

Element.focus方法用于将当前页面的焦点,转移到指定元素上。

该方法可以接受一个对象作为参数。参数对象的preventScroll属性是一个布尔值,指定是否将当前元素停留在原始位置,而不是滚动到可见区域。

function getFocus() {
  document.getElementById('btn').focus({preventScroll:false});
}

上面代码会让btn元素获得焦点,并滚动到可见区域。

最后,从document.activeElement属性可以得到当前获得焦点的元素。

Element.blur方法用于将焦点从当前元素移除。

<div>123:<input id="one" type="text"></div>
<div>ppp:<input id="two" type="text"></div>
<script>
    var one = document.getElementById('one');
    one.focus();  // one 输入框会获取焦点
    console.log(document.activeElement);// 可以得到有焦点的元素
    //     one.blur();
</script>

2.16 Element.click()

Element.click方法用于在当前元素上模拟一次鼠标点击,相当于触发了click事件。

3. 参考资料

Element对象 阮一峰

Element 节点 阮一峰 网道

HTMLElement MDN

Element MDN

Last Updated: 4/18/2020, 4:44:55 PM