“Yeah It’s on. ”
正文
MutationObserver
https://developer.mozilla.org/zh-CN/docs/Web/API/MutationObserver
MutationObserver接口提供了监视对DOM树所做更改的能力。它被设计为旧的Mutation Events功能的替代品,该功能是DOM3 Events规范的一部分。
document.currentScript
在学习vue-cli3时发现了currentScript这个知识点
在库模式中,项目的 publicPath 是根据主文件的加载路径动态设置的(用以支持动态的资源加载能力)。但是这个功能用到了 document.currentScript,而 IE 浏览器并不支持这一特性。所以如果网站需要支持 IE 的话,建议使用库之前先在页面上引入 current-script-polyfill。
https://cli.vuejs.org/zh/guide/build-targets.html#%E5%BA%94%E7%94%A8
https://developer.mozilla.org/zh-CN/docs/Web/API/Document/currentScript
返回其所包含的脚本中正在被执行的 <script>
元素.
var curScriptElement = document.currentScript;
下例演示了如何检测当前正在执行脚本的
if (document.currentScript.async) {
console.log("Executing asynchronously");
} else {
console.log("Executing synchronously");
}
Range
https://developer.mozilla.org/zh-CN/docs/Web/API/Range
Range对象代表页面上一段连续的区域,通过Range对象可以获取或者修改页面上任何区域的内容。也可以通过Range的方法进行复制和移动页面任何区域的元素。
Range
接口表示一个包含节点与文本节点的一部分的文档片段。
可以用 Document
对象的 Document.createRange
方法创建 Range,也可以用 Selection
对象的 getRangeAt
方法获取 Range。另外,还可以通过 Document
对象的构造函数 Range()
来得到 Range。
属性
返回一个表示 Range
的起始位置和终止位置是否相同的布尔值
。
Range.commonAncestorContainer
只读
返回完整包含 startContainer
和 endContainer
的、最深一级的节点
。
返回包含 Range
终点的节点
。
返回一个表示 Range
终点在 endContainer
中的位置的数字。
返回包含 Range
开始的节点
。
返回一个表示 Range
起点在 startContainer
中的位置的数字。
例子
<body>
<table id="mytable" border="1">
<tr>
<td>内容1</td>
<td>内容2</td>
</tr>
<tr>
<td>内容3</td>
<td>内容4</td>
</tr>
</table>
<button onclick="delrow()">删除第一行</button>
</body>
<script>
function delrow() {
var table = document.getElementById("mytable");
if (table.rows.length > 0) {
var row = table.rows[0];
var rangeObj = document.createRange();
rangeObj.setStartBefore(row);
rangeObj.setEndAfter(row);
rangeObj.deleteContents();
}
}
</script>
兼容性
https://www.caniuse.com/#search=createRange
createRange 兼容性非常好,可以使用
Selection
https://developer.mozilla.org/zh-CN/docs/Web/API/Selection
表示用户选择的文本范围或光标的当前位置。
const selection = window.getSelection() ;
selection
是一个Selection
对象。 如果想要将selection
转换为字符串,可通过连接一个空字符串(”“)或使用String.toString()
方法。
一般来说,插入光标的位置可通过 Selection 获取,这时它被标记为 Collapsed
,这表示选区被压缩至一点,即光标位置。但要注意它与 focus
事件或 Document.activeElement
等的值没有必然联系。
crossorigin属性
https://developer.mozilla.org/zh-CN/docs/Web/HTML/CORS_settings_attributes
https://www.chrisyue.com/what-the-hell-is-crossorigin-attribute-in-html-script-tag.html
在HTML5中,一些 HTML 元素提供了对 CORS 的支持, 例如<audio> <img> <link> <script> <video>
均有一个跨域属性 (crossOrigin
property),它允许你配置元素获取数据的 CORS 请求。
关键字 | 描述 |
---|---|
anonymous |
对此元素的 CORS 请求将不设置凭据标志。 |
use-credentials |
对此元素的CORS请求将设置凭证标志;这意味着请求将提供凭据。 |
"" |
设置一个空的值,如 crossorigin 或 crossorigin="" ,和设置 anonymous 的效果一样。 |
从谷歌的结果来看,比较一致的说法是,当引入跨域的脚本(比如用了 apis.google.com 上的库文件)时,如果这个脚本有错误,因为浏览器的限制(根本原因是协议的规定),是拿不到错误信息的。当本地尝试使用 window.onerror
去记录脚本的错误时,跨域脚本的错误只会返回 Script error
。
而 HTML5 新的规定,是可以允许本地获取到跨域脚本的错误信息的,但有两个条件:一是跨域脚本的服务器必须通过 Access-Control-Allow-Origin
头信息允许当前域名可以获取错误信息,二是网页里的 script
标签也必须指明 src
属性指定的地址是支持跨域的地址,也就是 crossorigin 属性。有了这两个条件,就可以获取跨域脚本的错误信息:
但事情还是不够明朗,看起来跨域脚本报个错也没什么啊,为什么浏览器(准确说是 HTTP 协议)这么轴,非要规定默认情况页面是不能获取跨域脚本错误信息的呢?
这其实跟网络安全有关,不妨举一个例子来说明。
我们先假设浏览器默认可以将跨域脚本的错误信息返回。
这个时候我在我的博客里写下如下代码:
<script src="http://某个银行/会员信息网址">
<script src="http://某个银行2/会员信息网址">
...
注意 src 里面提到地址,都是 HTML 页面的地址,当成 JS 来执行,肯定是会报错的。
因为我们假设浏览器能报具体错误,这个错误可能是类似于:
“请登录” is undefined. “您好” is undefined.
我们通过报错信息的不一致,可能可以推断出当前访问我博客的会员在某某银行是否有账号。虽然不是什么大问题,但隐私的确是泄漏了,如果我是攻击者我可能会通过判断会员在某家银行是否有账号,『精准』推送相关的钓鱼网站给他。
说清楚了来龙去脉,我们就可以更好的判断,我们是否真的需要给 script
标签加上 crossorigin 属性了。另外除了 script
,所有能引入跨域资源的标签包括 link
和 img
之类,都有一样的属性。
script
的 crossorigin 和其他标签的crossorigin
属性作用不一样,script
的跨域属性跟脚本错误有关,上面已经解释过了,但其他标签的跨域属性跟 Canvas 有关
另外补充一点:
加载本地静态文件根本就不牵涉跨域的问题,所以本地就不应该用 crossorigin
印象中如果 crossorigin 用在了本地文件上,浏览器报错信息会告诉你 crossorigin 只能支持 http/https/… 等协议上
integrity
<script src="https://code.jquery.com/jquery-3.2.1.slim.min.js"
integrity="sha384-KJ3o2DKtIkvYIK3UENzmM7KCkRr/rE9/Qpg6aAZGJwFDMVNA/GpGFF93hXpG5KkN"
crossorigin="anonymous">
</script>
CDN 虽好,但 CDN 有可能被劫持,导致下载的文件是被篡改过的(比如通过 DNS 劫持),有了 integrity 就可以检查文件是否是原版。但因为本地文件用的域名跟网页是同一个域名,不存在劫持的问题(或者劫持就连网站本身一起被劫持了,那就不是 integrity 能解决的问题了),所以本地静态文件没有太大必要用这个属性。
readyState
https://developer.mozilla.org/zh-CN/docs/Web/API/Document/readyState
-
loading
(正在加载)document
仍在加载。 -
interactive
(可交互)文档已被解析,”正在加载“状态结束,但是诸如图像,样式表和框架之类的子资源仍在加载。
-
complete
(完成)文档和所有子资源已完成加载。表示
load (en-US)
状态的事件即将被触发。
如果insert.js是一个公用组件,尤其以后会开源的那种,那势必要考虑到各种加载场景,页面的头部,页面底部,或者异步动态加载。
if (document.readyState != 'loading') {
init();
} else {
window.addEventListener("DOMContentLoaded", function () {
init();
});
}
a标签
rel属性
https://www.w3school.com.cn/tags/att_a_rel.asp
浏览器支持: 所有浏览器都支持 rel 属性。
虽然在理论上讲,浏览器可以利用 rel 和 rev 属性改变锚内容的外观、或者自动构建文档浏览菜单,其他工具也可以使用这些属性来构建特殊的链接集合、目录和索引,但是只有极少数浏览器才会利用这些属性来改变链接的外观。
提示:尽管浏览器不会以任何方式使用该属性,不过搜索引擎可以利用该属性获得更多有关链接的信息。
定义和用法
<a>
标签的 rel 属性用于指定当前文档与被链接文档的关系。
用于 <a>
标签的可选属性 rel 和 rev 分别表示源文档与目标文档之间正式的关系和方向。
rel 属性指定从源文档到目标文档的关系,而 rev 属性则指定从目标文档到源文档的关系。这两种属性可以在<a>
标签中同时使用。
<a href="part_12.html" rel="next" rev="prev">
重要rel=”noreferrer”
http://ju.outofmemory.cn/entry/172298
在一个页面A中,点击一个链接,浏览器会跳转到页面B。那么如何描述A和B的关系呢?我们把A定义为B的refer/referrer/referer
通俗地说,A是B的来源页面/引荐页面。从技术实现上,你从浏览器起发出的任何请求(不准确,先这么理解吧),例如打开新页面、请求静态资源、ajax、请求后端api等等,都会在请求头部添加
问题出现
最近在做node爬虫时,发现,有一些页面需要跳转到某网站(新开页面),但这个网站链接链过去会报403错误,如下图。但是,从浏览器输入地址,则没有问题。
如果链接链过去的时候不带referrer信息,则网页能够正常打开。
最简易直观的实现方式,如果一个a标签设置了rel=”noreferrer”,那么点击这个链接时不传递referrer信息。(注意,这个html5标准中和referrer是正确的拼法)
<a href="http://baidu.com" target="_blank" rel="noreferrer">跳转</a>
如果担心兼容性问题,可以使用 noreferrer.js
href
要实现点击a标签原地不动 不跳转 不刷新效果
href = “” 是不行的
解决方式:
**href=”javascript:void(0);” **
target
_blank
浏览器总在一个新打开、未命名的窗口中载入目标文档。
_self
这个目标的值对所有没有指定目标的 <a>
标签是默认目标,它使得目标文档载入并显示在相同的框架或者窗口中作为源文档。这个目标是多余且不必要的,除非和文档标题 <base>
标签中的 target 属性一起使用。
_parent
这个目标使得文档载入父窗口或者包含来超链接引用的框架的框架集。如果这个引用是在窗口或者在顶级框架中,那么它与目标 _self 等效。
_top
这个目标使得文档载入包含这个超链接的窗口,用 _top 目标将会清除所有被包含的框架并将文档载入整个浏览器窗口。
提示: 这些 target 的所有 4个值都以下划线开始。任何其他用一个下划线作为开头的窗口或者目标都会被浏览器忽略,因此,不要将下划线作为文档中定义的任何框架 name 或 id 的第一个字符。
补充
在新窗口中打开页面的坑
用a标签的target=”_blank”属性,或者window.open(url)在新窗口中打开页面时,会存在潜在的安全问题。为什么呢?这个锅是一个叫opener的全局对象的锅。
性能问题
为什么新窗口中的页面会影响父页面的线程呢?chrome不是每个标签页一个单独的进程?然后进程内包含若干线程吗?
确实,chrome有不同的标签页面使用不同进程和线程,但是有个例外,通过a标签的target=”_blank”属性,或者window.open(url)在新窗口中打开页面, 会与父窗口共用进程和线程。为什么呢?
还是因为opener。。。。因为opener里有DOM信息。两个进程中同时hold住了DOM信息,在多进程下很难道控制,所以干脆就放在一个进程里了。这个算是chrome的一个小缺陷(firefox也有,ie没有),不过chrome目前正在跟进和优化这里。
解决方案
- 如果是a标签要在新窗口中打开,添加noopener属性
- 如果是js中打开新窗口,手动将新窗口的opener置为null
<a target="_blank" href="http://keenwon.com/">点击</a>
在新打开的标签页通过window.oponer ,可以获得当前页面的window 。这样的话,http://keenwon.com(被打开的页面)将获得当前页面的部分控制权,即使新打开的页面是跨域的也照样可以(例如location 就不存在跨域问题)。
input
上传文件
https://juejin.im/post/5aacc64e6fb9a028c6757bab
<input class="addPicInput" type="file" ref="uploadFile"
@change="fileChange" accept="image/*" multiple>
multiple
在实际开发过程中,会遇到上传文件的一些需求。但是使用原生的<input type="file" />
在使用中存在一些问题
- 在未上传文件时,显示”no file choosen”,用户界面不友好,不可配置
- 上传同一个文件,不会触发change事件,即使该文件做过修改
- 用户如果在上传过程中点击了“取消”,已经上传的文件会被移除
解决思路
在阅读了一些源码之后,总结了如下的解决方案。有点偷梁换柱的意思:
- 将真正的
<input type="file" />
隐藏,使用自定义的button通过$refs去触发文件上传,实现自定义显示 - 文件上传之后,处理完文件,将
<input type="file" />
的value设置为null,这样下次即使上传的是同一个文件,仍然会触发change事件 - 使用上述方法,点击取消文件被移除,但是不影响页面展示
监听中文输入
https://juejin.cn/post/6844903950634713096
compositionstart和compositionend
The compositionstart event is fired when a text composition system such as an
input method editor starts a new composition session
.For example, this event could be fired after a user starts entering a Chinese character using a Pinyin IME.
The compositionend event is fired when a text composition system such as an
input method editor completes or cancels the current composition session
.For example, this event could be fired after a user finishes entering a Chinese character using a Pinyin IME.
positionstart
事件,当用户使用拼音输入法开始输入汉字时,这个事件就会被触发。compositionend
事件, 当文中文输入完成时, compositionend 事件将被触发。
audio
切换url播放
<audio id="Audio" src="https://demo.dj63.com//2016/串烧舞曲/20150926/全中文CLUB音乐情一动心就痛情感慢摇串烧.mp3" controls>11111</audio>
<button onclick="changeUrl()">切换audio url</button>
const audio = document.getElementById("Audio")
const url2 = "https://demo.dj63.com//2016/%E4%B8%B2%E7%83%A7%E8%88%9E%E6%9B%B2/20161123/[%E9%A3%8E%E5%90%B9%E9%BA%A6%E6%B5%AA]%E6%89%93%E9%80%A0%E4%B8%AD%E8%8B%B1%E6%96%87%E6%97%8B%E5%BE%8B%E5%BC%B9%E8%B7%B3CLUB%E8%88%9E%E6%9B%B2%E4%B8%B2%E7%83%A7.mp3"
function changeUrl(){
audio.src = url2
// 记得调用play()
audio.play()
}
iframe
https://developer.mozilla.org/zh-CN/docs/Web/HTML/Element/iframe
HTML内联框架元素 (<iframe>
) 表示嵌套的browsing context。它能够将另一个HTML页面嵌入到当前页面中。
每个嵌入的浏览上下文(embedded browsing context)都有自己的会话历史记录(session history)和DOM树。包含嵌入内容的浏览上下文称为父级浏览上下文。顶级浏览上下文(没有父级)通常是由 Window
对象表示的浏览器窗口。
优点
- 安全性:
iframe
提供了一个隔离的环境,可以保护主文档的内容不受到嵌入内容的直接影响,从而减少跨站脚本(XSS)攻击的风险。
- 简单的嵌入:
- 通过简单的HTML代码,用户可以方便地在网页中嵌入其他网站的内容,如视频、地图、外部应用等,而无需深度集成。
- 独立加载:
- 嵌入的
iframe
内容可以独立于主页面加载,这有助于提高页面性能,尤其是在加载重内容时。
- 嵌入的
- 跨域支持:
iframe
可以嵌入不同域名的内容,允许使用各种外部资源而不需要同源策略的问题。
- 易于更新和维护:
- 嵌入内容(如视频或应用)只需在提供者的服务器上更新,用户无需更改主页面代码。
- 响应式设计:
- 可以通过CSS和JavaScript对
iframe
进行样式调整,使其适配多种屏幕尺寸,支持响应式设计。
- 可以通过CSS和JavaScript对
- 功能扩展:
- 可以在iframe内部提供各种功能,如播社交分享和分析工具等,而不需要改动主页面的JavaScript或CSS。
缺点
- 性能问题:
iframe
会增加额外的HTTP请求,可能导致页面加载速度变慢,尤其是在嵌入多个iframe
时。 - SEO挑战:搜索引擎可能不优化
iframe
中的内容,导致这些内容未被索引,从而影响SEO性能。 - 跨域问题:由于浏览器的同源策略,
iframe
中的内容可能无法与主页面进行充分的交互,限制了功能。 - 用户体验:在某些情况下,
iframe
可能导致不一致的用户体验,尤其是在不同的设备和浏览器上。 - 维护复杂性:如果嵌入的内容更新频繁,可能需要额外的维护开销來保证内容的正常显示。
跨域问题
iframe标签本身是可以跨域的
如果是iframe不传递参数的话,是可以src嵌入不同的网页的,例如你嵌入一个baidu,只要没有数据交互也是可以的。
但是,当父子页面需要通信时就会存在跨域,父子页面需要同源才能通信。
性能问题
使用 <iframe>
加载内容确实可能会对页面性能产生一定影响,具体影响取决于多个因素。以下是一些可能影响性能的方面:
- 额外的网络请求
<iframe>
会导致浏览器发起额外的 HTTP 请求来加载<iframe>
中的内容。这可能会增加页面加载时间,特别是如果iframe
加载的是大文件或复杂的内容。
- 渲染和绘制
- 每个
<iframe>
都是一个独立的文档上下文,浏览器会为每个 iframe 渲染一个新的页面。这增加了浏览器的渲染负担,可能导致整体页面性能下降,尤其是当多个<iframe>
同时加载时。
- 阻塞主线程
- 如果
<iframe>
加载的内容包含执行 JavaScript 的代码,且这些代码阻塞了主线程,会影响主页面的性能。尤其是 synchronous 的脚本会阻塞页面的渲染。
- CSS 和 JavaScript
- 如果
<iframe>
包含大量的 CSS 和 JavaScript 文件,可能会导致页面加载时间延长和性能下降。
- 跨域资源共享
- 如果
iframe
加载的内容来自不同的域,可能会影响页面与iframe
交互的性能,尤其是在 API 调用和数据交流涉及跨域请求的情况下。
- 内存使用
- 多个
<iframe>
会使用额外的内存。这在资源受限的设备上(如移动设备)可能会导致性能瓶颈。
优化建议
- 延迟加载 (
lazy loading
): 对于不需要立即显示的<iframe>
,可以使用懒加载技术,等到用户滚动到该部分时再加载,从而减少初始加载时间。 - 使用
sandbox
: 如果iframe
不需要与父页面交互,使用sandbox
属性可以限制<iframe>
的功能,从而提高整体安全性和性能。 - 减少数量: 尽量减少页面上使用的
<iframe>
数量。 - 内容优化: 确保
<iframe>
中的内容经过优化,避免不必要的资源加载。
总结
总的来说,<iframe>
的使用会对性能产生一定的影响,尤其是在它加载外部内容时。了解这一点并采取相应的优化措施,可以帮助尽量减少性能上的损失。如果你有特定的使用场景或其他问题,可以进一步探讨!
sandbox
iframe
的 sandbox
属性用于增加安全性,限制嵌入内容的功能。sandbox
属性可以接受多个值,下面是一些常见的默认值和它们的功能:
- 默认行为(无任何值):使用
sandbox
属性但不指定任何值时,iframe 内的内容会受到一系列严格的限制,包括:- 禁止执行 JavaScript
- 禁止提交表单
- 禁止使用插件
- 禁止弹出新窗口
- 禁止访问父文档的 DOM
- 允许特定功能:你可以添加不同的关键字来解除这些限制。例如:
allow-same-origin
:允许 iframe 从相同源加载资源。allow-scripts
:允许 iframe 内部执行 JavaScript。allow-forms
:允许提交表单。allow-popups
:允许弹出窗口。
例如,<iframe sandbox="allow-scripts allow-same-origin">
将允许脚本和相同源的访问,但其他限制仍然适用。
X-Frame-Options
X-Frame-Options 响应头来限制其在 iframe 中的嵌套。X-Frame-Options 是一种安全策略,可以由网站的服务器设置,用于控制页面是否允许在 iframe 中嵌套。
- DENY: 完全不允许页面在任何
<iframe>
中加载。 - SAMEORIGIN: 允许页面在同源(即相同域名)的
<iframe>
中加载。 - ALLOW-FROM uri: 允许指定的 URI 加载该页面(虽然此选项在许多现代浏览器中已不再支持)。
Attributes and properties
https://javascript.info/dom-attributes-and-properties
总结
- Attributes – is what’s written in HTML.
- Properties – is what’s in DOM objects.
Properties | Attributes | |
---|---|---|
Type | Any value, standard properties have types described in the spec | A string |
Name | Name is case-sensitive | Name is not case-sensitive |
For most situations using DOM properties is preferable. We should refer to attributes only when DOM properties do not suit us, when we need exactly attributes, for instance:
- We need a non-standard attribute. But if it starts with
data-
, then we should usedataset
. - We want to read the value “as written” in HTML. The value of the DOM property may be different, for instance the
href
property is always a full URL, and we may want to get the “original” value.
When the browser loads the page, it “reads” (another word: “parses”) the HTML and generates DOM objects from it. For element nodes, most standard HTML attributes automatically become properties of DOM objects.
For instance, if the tag is <body id="page">
, then the DOM object has body.id="page"
.
But the attribute-property mapping is not one-to-one
We’ve already seen built-in DOM properties. There are a lot. But technically no one limits us, and if there aren’t enough, we can add our own.
DOM nodes are regular JavaScript objects. We can alter them.
For instance, let’s create a new property in document.body
:
document.body.myData = {
name: 'Caesar',
title: 'Imperator'
};
alert(document.body.myData.title); // Imperator
We can add a method as well:
document.body.sayTagName = function() {
alert(this.tagName);
};
document.body.sayTagName(); // BODY (the value of "this" in the method is document.body)
We can also modify built-in prototypes like Element.prototype
and add new methods to all elements:
Element.prototype.sayHi = function() {
alert(`Hello, I'm ${this.tagName}`);
};
document.documentElement.sayHi(); // Hello, I'm HTML
document.body.sayHi(); // Hello, I'm BODY
So, DOM properties and methods behave just like those of regular JavaScript objects:
- They can have any value.
- They are case-sensitive (write
elem.nodeType
, notelem.NoDeTyPe
).
In HTML, tags may have attributes. When the browser parses the HTML to create DOM objects for tags, it recognizes standard attributes and creates DOM properties from them.
在HTML中,标签可以有属性。当浏览器解析HTML以创建标记的DOM对象时,它识别标准属性并从中创建DOM属性。
So when an element has id
or another standard attribute, the corresponding property gets created. But that doesn’t happen if the attribute is non-standard.s
因此,当一个元素具有id或另一个标准属性时,相应的属性将被创建。但是如果属性是非标准的,就不会发生这种情况。
For instance:
<body id="test" something="non-standard">
<script>
alert(document.body.id); // test
// non-standard attribute does not yield a property
alert(document.body.something); // undefined
</script>
</body>
Please note that a standard attribute for one element can be unknown for another one. For instance, "type"
is standard for <input>
(HTMLInputElement), but not for <body>
(HTMLBodyElement). Standard attributes are described in the specification for the corresponding element class.
Here we can see it:
<body id="body" type="...">
<input id="input" type="text">
<script>
alert(input.type); // text
alert(body.type); // undefined: DOM property not created, because it's non-standard
</script>
</body>
So, if an attribute is non-standard, there won’t be a DOM-property for it. Is there a way to access such attributes?
Sure. All attributes are accessible by using the following methods:
elem.hasAttribute(name)
– checks for existence.elem.getAttribute(name)
– gets the value.elem.setAttribute(name, value)
– sets the value.elem.removeAttribute(name)
– removes the attribute.
These methods operate exactly with what’s written in HTML.
Also one can read all attributes using elem.attributes
: a collection of objects that belong to a built-in Attr class, with name
and value
properties.
Here’s a demo of reading a non-standard property:
<body something="non-standard">
<script>
alert(document.body.getAttribute('something')); // non-standard
</script>
</body>
HTML attributes have the following features:
- Their name is case-insensitive (
id
is same asID
). // 不区分大小写的 - Their values are always strings.
Here’s an extended demo of working with attributes:
<body>
<div id="elem" about="Elephant"></div>
<script>
alert( elem.getAttribute('About') ); // (1) 'Elephant', reading
elem.setAttribute('Test', 123); // (2), writing
alert( elem.outerHTML ); // (3), see if the attribute is in HTML (yes)
for (let attr of elem.attributes) { // (4) list all
alert( `${attr.name} = ${attr.value}` );
}
</script>
</body>
Please note:
getAttribute('About')
– the first letter is uppercase here, and in HTML it’s all lowercase. But that doesn’t matter: attribute names are case-insensitive.- We can assign anything to an attribute, but it becomes a string. So here we have
"123"
as the value. - All attributes including ones that we set are visible in
outerHTML
. - The
attributes
collection is iterable and has all the attributes of the element (standard and non-standard) as objects withname
andvalue
properties.
Property-attribute synchronization
When a standard attribute changes, the corresponding property is auto-updated, and (with some exceptions) vice versa.
In the example below id
is modified as an attribute, and we can see the property changed too. And then the same backwards:
<input>
<script>
let input = document.querySelector('input');
// attribute => property
input.setAttribute('id', 'id');
alert(input.id); // id (updated)
// property => attribute
input.id = 'newId';
alert(input.getAttribute('id')); // newId (updated)
</script>
But there are exclusions, for instance input.value
synchronizes only from attribute → to property, but not back:
<input>
<script>
let input = document.querySelector('input');
// attribute => property
input.setAttribute('value', 'text');
alert(input.value); // text
// NOT property => attribute
input.value = 'newValue';
alert(input.getAttribute('value')); // text (not updated!)
</script>
In the example above:
- Changing the attribute
value
updates the property. - But the property change does not affect the attribute.
That “feature” may actually come in handy, because the user actions may lead to value
changes, and then after them, if we want to recover the “original” value from HTML, it’s in the attribute.
DOM properties are not always strings. For instance, the input.checked
property (for checkboxes) is a boolean:
<input id="input" type="checkbox" checked> checkbox
<script>
alert(input.getAttribute('checked')); // the attribute value is: empty string
alert(input.checked); // the property value is: true
</script>
There are other examples. The style
attribute is a string, but the style
property is an object:
<div id="div" style="color:red;font-size:120%">Hello</div>
<script>
// string
alert(div.getAttribute('style')); // color:red;font-size:120%
// object
alert(div.style); // [object CSSStyleDeclaration]
alert(div.style.color); // red
</script>
Most properties are strings though.
All attributes starting with “data-” are reserved for programmers’ use. They are available in the dataset
property.
Multiword attributes like data-order-state
become camel-cased: dataset.orderState
.
Here’s a rewritten “order state” example:
<style>
.order[data-order-state="new"] {
color: green;
}
.order[data-order-state="pending"] {
color: blue;
}
.order[data-order-state="canceled"] {
color: red;
}
</style>
<div id="order" class="order" data-order-state="new">
A new order.
</div>
<script>
// read
alert(order.dataset.orderState); // new
// modify
order.dataset.orderState = "pending"; // (*)
</script>
Using data-*
attributes is a valid, safe way to pass custom data.
activeElement
当前获得焦点的元素:
var x = document.activeElement.tagName;
// BUTTON
一个别的例子:
document.activeElement.blur()
attribute中的dir
dir是一个指示元素中文本方向的枚举属性。它的取值如下:
-
ltr, 指从左到右,用于那种从左向右书写的语言(比如英语);
-
rtl, 指从右到左,用于那种从右向左书写的语言(比如阿拉伯语);
-
auto, 指由用户代理决定方向。它在解析元素中字符时会运用一个基本算法,直到发现一个具有强方向性的字符,然后将这一方向应用于整个元素。
const isRTL = document.documentElement.getAttribute('dir') == 'rtl'
fakeElem.style[ isRTL ? 'right' : 'left' ] = '-9999px'
Preload 和 Prefetch
最近在研究vue-cli3,发现用到了一个插件PreloadPlugin,所以在这里学习一下preload和prefetch的知识。
Preload
Preload 是一种新的web标准,旨在提高性能并为web开发人员提供更细粒度的加载控制
Preload is a web standard aimed at improving performance and granular loading of resources. It is a declarative fetch that can tell a browser to start fetching a source because a developer knows the resource will be needed soon.
<link rel="preload" ></link>
preload通常用于本页面要用到的关键资源,包括关键js、字体、css文件。preload将会把资源得下载顺序权重提高,使得关键数据提前下载好,优化页面打开速度。
preload 是声明式的 fetch,可以强制浏览器请求资源,同时不阻塞文档 onload 事件。
preload:当前页面使用,尽早下载,优先级较高;
Prefetch
<link rel="prefetch" ></link>
prefetch:首次渲染时不需要,之后可能需要。优先级较低,在浏览器空闲时才会下载。使用场景:比如当前页可能跳转的页面,或者条件加载的资源。
什么时候该用 <link rel=”preload”> ? 什么时候又该用 <link rel=”prefetch”> ?
建议:对于当前页面很有必要的资源使用 preload,对于可能在将来的页面中使用的资源使用 prefetch。
preload 是对浏览器指示预先请求当前页需要的资源(关键的脚本,字体,主要图片)。
prefetch 应用场景稍微又些不同 —— 用户将来可能在其他部分(比如视图或页面)使用到的资源。如果 A 页面发起一个 B 页面的 prefetch 请求,这个资源获取过程和导航请求可能是同步进行的,而如果我们用 preload 的话,页面 A 离开时它会立即停止。
preload 和 prefetch 都被存储在 HTTP 缓存中。
当一个资源被 preload 或者 prefetch 获取后,它可以从 HTTP 缓存移动至渲染器的内存缓存中。如果资源可以被缓存(比如说存在有效的cache-control 和 max-age),它被存储在 HTTP 缓存中可以被现在或将来的任务使用,如果资源不能被缓存在 HTTP 缓存中,作为代替,它被放在内存缓存中直到被使用。
Web worker 和 Service worker
https://tinyshare.cn/post/HpDVBvTWbUD
Web worker,service worker和worklet,这些都是“JavaScript Workers”,虽然它们在运行方式上有一些相似的地方,并且它们在使用上也有一些重叠的地方。
一般来说,一个worker是一个脚本在浏览器主线程之外的单独的线程上运行。如果你想要在HTML文档中引用一个<script>
标签的典型的JavaScript文件,它会运行在主线程上。如果主线程上有太多的计算,会拖慢网站的速度,造成交互卡顿和响应延迟。
Web workers
Web workers是最常用的worker类型。它不像另外两种,它们除了运行在主线程外的特性外,没有一个特殊的应用场景。所以,web worker可以用于减少主线程上大量的线程活动。
worker.js文件中的任何代码都会开始运行,web worker最有用的是用于减少那些可能需要花费很长时间的工序或者与其他线程上平行运行。一个很好的例子是图片线程web应用:Squoosh,它使用web worker来处理图片处理任务,可以让主线程有精力处理用户与应用的交互而不被打扰。
像所有的worker,web worker没有获取DOM的权限,这意味着任何需要的信息将被在worker和主脚本间传递,传递参数使用window.postMessage()。
/* main.js */线程
// 创建 worker
const myWorker = new Worker('worker.js');
// 向 worker 传递信息
myWorker.postMessage('Hello!');
// 接收从 worker 传递过来的信息
myWorker.onmessage = function(e) {
console.log(e.data);
}
在worker脚本中,我们可以监听主脚本中的消息,并将响应返回。
/* worker.js */
// 接收主文件的信息
self.onmessage = function(e) {
console.log(e.data);
// 向主文件发送信息
self.postMessage(workerResult);
}
Service workers
Service workers 是一种提供详细的作为浏览器和网络或缓存间的代理的服务。
与Web worker类似,service worker 是在主脚本文件中注册,引用到一个专门的service worker文件。
/* main.js */
navigator.serviceWorker.register('/service-worker.js');
与一般的web worker不同,service worker有一些额外的特性来实现代理的目的。只要它们被安装且被激活,service worker就可以拦截主文档中发起的任何网络请求。
/* service-worker.js */
// Install (安装)
self.addEventListener('install', function(event) {
// ...
});
// Activate (激活)
self.addEventListener('activate', function(event) {
// ...
});
// 监听主文档中的网络请求
self.addEventListener('fetch', function(event) {
// ...
});
只要一被拦截,service worker就可以返回一个缓存中的文档作为响应,而不用走网络请求,因此可以让应用离线运行。
/* service-worker.js */
self.addEventListener('fetch', function(event) {
// 返回缓存中的数据
event.respondWith(
caches.match(event.request);
);
});