# 進度事件
## 進度事件的種類
進度事件用來描述資源加載的進度,主要由 AJAX 請求、`<img>`、`<audio>`、`<video>`、`<style>`、`<link>`等外部資源的加載觸發,繼承了`ProgressEvent`接口。它主要包含以下幾種事件。
- `abort`:外部資源中止加載時(比如用戶取消)觸發。如果發生錯誤導致中止,不會觸發該事件。
- `error`:由于錯誤導致外部資源無法加載時觸發。
- `load`:外部資源加載成功時觸發。
- `loadstart`:外部資源開始加載時觸發。
- `loadend`:外部資源停止加載時觸發,發生順序排在`error`、`abort`、`load`等事件的后面。
- `progress`:外部資源加載過程中不斷觸發。
- `timeout`:加載超時時觸發。
注意,除了資源下載,文件上傳也存在這些事件。
下面是一個例子。
```javascript
image.addEventListener('load', function (event) {
image.classList.add('finished');
});
image.addEventListener('error', function (event) {
image.style.display = 'none';
});
```
上面代碼在圖片元素加載完成后,為圖片元素添加一個`finished`的 Class。如果加載失敗,就把圖片元素的樣式設置為不顯示。
有時候,圖片加載會在腳本運行之前就完成,尤其是當腳本放置在網頁底部的時候,因此有可能`load`和`error`事件的監聽函數根本不會執行。所以,比較可靠的方式,是用`complete`屬性先判斷一下是否加載完成。
```javascript
function loaded() {
// ...
}
if (image.complete) {
loaded();
} else {
image.addEventListener('load', loaded);
}
```
由于 DOM 的元素節點沒有提供是否加載錯誤的屬性,所以`error`事件的監聽函數最好放在`<img>`元素的 HTML 代碼中,這樣才能保證發生加載錯誤時百分之百會執行。
```html
<img src="/wrong/url" onerror="this.style.display='none';" />
```
`loadend`事件的監聽函數,可以用來取代`abort`事件、`load`事件、`error`事件的監聽函數,因為它總是在這些事件之后發生。
```javascript
req.addEventListener('loadend', loadEnd, false);
function loadEnd(e) {
console.log('傳輸結束,成功失敗未知');
}
```
`loadend`事件本身不提供關于進度結束的原因,但可以用它來做所有加載結束場景都需要做的一些操作。
另外,`error`事件有一個特殊的性質,就是不會冒泡。所以,子元素的`error`事件,不會觸發父元素的`error`事件監聽函數。
## ProgressEvent 接口
`ProgressEvent`接口主要用來描述外部資源加載的進度,比如 AJAX 加載、`<img>`、`<video>`、`<style>`、`<link>`等外部資源加載。進度相關的事件都繼承了這個接口。
瀏覽器原生提供了`ProgressEvent()`構造函數,用來生成事件實例。
```javascript
new ProgressEvent(type, options)
```
`ProgressEvent()`構造函數接受兩個參數。第一個參數是字符串,表示事件的類型,這個參數是必須的。第二個參數是一個配置對象,表示事件的屬性,該參數可選。配置對象除了可以使用`Event`接口的配置屬性,還可以使用下面的屬性,所有這些屬性都是可選的。
- `lengthComputable`:布爾值,表示加載的總量是否可以計算,默認是`false`。
- `loaded`:整數,表示已經加載的量,默認是`0`。
- `total`:整數,表示需要加載的總量,默認是`0`。
`ProgressEvent`具有對應的實例屬性。
- `ProgressEvent.lengthComputable`
- `ProgressEvent.loaded`
- `ProgressEvent.total`
如果`ProgressEvent.lengthComputable`為`false`,`ProgressEvent.total`實際上是沒有意義的。
下面是一個例子。
```javascript
var p = new ProgressEvent('load', {
lengthComputable: true,
loaded: 30,
total: 100,
});
document.body.addEventListener('load', function (e) {
console.log('已經加載:' + (e.loaded / e.total) * 100 + '%');
});
document.body.dispatchEvent(p);
// 已經加載:30%
```
上面代碼先構造一個`load`事件,拋出后被監聽函數捕捉到。
下面是一個實際的例子。
```javascript
var xhr = new XMLHttpRequest();
xhr.addEventListener('progress', updateProgress, false);
xhr.addEventListener('load', transferComplete, false);
xhr.addEventListener('error', transferFailed, false);
xhr.addEventListener('abort', transferCanceled, false);
xhr.open();
function updateProgress(e) {
if (e.lengthComputable) {
var percentComplete = e.loaded / e.total;
} else {
console.log('不能計算進度');
}
}
function transferComplete(e) {
console.log('傳輸結束');
}
function transferFailed(evt) {
console.log('傳輸過程中發生錯誤');
}
function transferCanceled(evt) {
console.log('用戶取消了傳輸');
}
```
上面是下載過程的進度事件,還存在上傳過程的進度事件。這時所有監聽函數都要放在`XMLHttpRequest.upload`對象上面。
```javascript
var xhr = new XMLHttpRequest();
xhr.upload.addEventListener('progress', updateProgress, false);
xhr.upload.addEventListener('load', transferComplete, false);
xhr.upload.addEventListener('error', transferFailed, false);
xhr.upload.addEventListener('abort', transferCanceled, false);
xhr.open();
```
- 前言
- 入門篇
- 導論
- 歷史
- 基本語法
- 數據類型
- 概述
- null,undefined 和布爾值
- 數值
- 字符串
- 對象
- 函數
- 數組
- 運算符
- 算術運算符
- 比較運算符
- 布爾運算符
- 二進制位運算符
- 其他運算符,運算順序
- 語法專題
- 數據類型的轉換
- 錯誤處理機制
- 編程風格
- console 對象與控制臺
- 標準庫
- Object 對象
- 屬性描述對象
- Array 對象
- 包裝對象
- Boolean 對象
- Number 對象
- String 對象
- Math 對象
- Date 對象
- RegExp 對象
- JSON 對象
- 面向對象編程
- 實例對象與 new 命令
- this 關鍵字
- 對象的繼承
- Object 對象的相關方法
- 嚴格模式
- 異步操作
- 概述
- 定時器
- Promise 對象
- DOM
- 概述
- Node 接口
- NodeList 接口,HTMLCollection 接口
- ParentNode 接口,ChildNode 接口
- Document 節點
- Element 節點
- 屬性的操作
- Text 節點和 DocumentFragment 節點
- CSS 操作
- Mutation Observer API
- 事件
- EventTarget 接口
- 事件模型
- Event 對象
- 鼠標事件
- 鍵盤事件
- 進度事件
- 表單事件
- 觸摸事件
- 拖拉事件
- 其他常見事件
- GlobalEventHandlers 接口
- 瀏覽器模型
- 瀏覽器模型概述
- window 對象
- Navigator 對象,Screen 對象
- Cookie
- XMLHttpRequest 對象
- 同源限制
- CORS 通信
- Storage 接口
- History 對象
- Location 對象,URL 對象,URLSearchParams 對象
- ArrayBuffer 對象,Blob 對象
- File 對象,FileList 對象,FileReader 對象
- 表單,FormData 對象
- IndexedDB API
- Web Worker
- 附錄:網頁元素接口
- a
- img
- form
- input
- button
- option
- video,audio