|
| 1 | +# 11 自定义视频播放器 |
| 2 | + |
| 3 | +> 作者:©[Xing Liu](https://github.com/S1ngS1ng) |
| 4 | +> 简介:[JavaScript30](https://javascript30.com) 是 [Wes Bos](https://github.com/wesbos) 推出的一个 30 天挑战。项目免费提供了 30 个视频教程、30 个挑战的起始文档和 30 个挑战解决方案源代码。目的是帮助人们用纯 JavaScript 来写东西,不借助框架和库,也不使用编译器和引用。现在你看到的是这系列指南的第 10 篇。完整指南在 [GitHub](https://github.com/soyaine/JavaScript30),喜欢请 Star 哦♪(^∇^*) |
| 5 | +
|
| 6 | +> 创建时间:2017-02-06 |
| 7 | +最后更新:2017-04-16 |
| 8 | + |
| 9 | +## 实现目标 |
| 10 | +* 为 `video` 元素添加自定义样式的播放控制面板 |
| 11 | + * 可滑动调节音量、播放速度 |
| 12 | + * 可通过按钮快进、回退 |
| 13 | + * 可点击视频画面或按钮播放或暂停视频播放 |
| 14 | + * 可点击或拖动进度条选择视频播放进度 |
| 15 | + |
| 16 | +## 解决思路 |
| 17 | +0. 首先,我们已经有了 HTML 文件,里面包含各种播放器用到的元素,格式也已经在 CSS 文件中帮我们设置好了 |
| 18 | +1. 在 JS 中选择我们需要添加功能的 HTML 元素,建立好变量 |
| 19 | +2. 用 JS 写好播放器的功能 |
| 20 | +3. 给第一步中获取的元素加上事件监听和回调,即可实现功能 |
| 21 | + |
| 22 | +## 知识点 |
| 23 | +* `video` 对象的各种属性、方法和事件 |
| 24 | + * `paused` |
| 25 | + * `play()` |
| 26 | + * `pause()` |
| 27 | + * `currentTime` |
| 28 | + |
| 29 | +## 过程指南 |
| 30 | + |
| 31 | +### 开始 |
| 32 | + |
| 33 | +HTML 元素中,`video` 标签是我们的视频,而下面的 `player__controls` 就是我们自己的控制面板 |
| 34 | + |
| 35 | +```html |
| 36 | +<div class="player"> |
| 37 | + <video class="player__video viewer" src="https://player.vimeo.com/external/194837908.sd.mp4?s=c350076905b78c67f74d7ee39fdb4fef01d12420&profile_id=164"></video> |
| 38 | + |
| 39 | + <div class="player__controls"> |
| 40 | + <div class="progress"> |
| 41 | + <div class="progress__filled"></div> |
| 42 | + </div> |
| 43 | + <button class="player__button toggle" title="Toggle Play">►</button> |
| 44 | + <input type="range" name="volume" class="player__slider" min="0" max="1" step="0.05" value="1"> |
| 45 | + <input type="range" name="playbackRate" class="player__slider" min="0.5" max="2" step="0.1" value="1"> |
| 46 | + <button data-skip="-10" class="player__button">« 10s</button> |
| 47 | + <button data-skip="25" class="player__button">25s »</button> |
| 48 | + </div> |
| 49 | + </div> |
| 50 | +``` |
| 51 | + |
| 52 | +开始之前我们先把所有需要用到的元素节点先取到: |
| 53 | + |
| 54 | +```javascript |
| 55 | +/* 获取元素 */ |
| 56 | +const player = document.querySelector('.player'); |
| 57 | +const video = player.querySelector('.viewer'); |
| 58 | +const progress = player.querySelector('.progress'); |
| 59 | +const progressBar = player.querySelector('.progress__filled'); |
| 60 | +const toggle = player.querySelector('.toggle'); |
| 61 | +const skipButtons = player.querySelectorAll('[data-skip]'); |
| 62 | +const ranges = player.querySelectorAll('.player__slider'); |
| 63 | +``` |
| 64 | + |
| 65 | +### **播放/暂停** |
| 66 | +#### 功能实现 |
| 67 | +视频最主要的功能自然就是播放和暂停了,而且其他功能也需要视频播放之后才能看出效果,所以我们先来实现这个功能。 |
| 68 | +`video` 对象有一个叫 `paused` 的属性来判断视频是否在播放;另外还提供了两个方法来进行播放和暂停的操作:`.play()` 方法可以播放视频,`.pause()` 方法暂停播放 |
| 69 | + |
| 70 | +那么只需要在点击的视频的时候进行这两个操作就可以了,我们可以写一个 `togglePlay` 方法,根据视频的播放状态来判断该执行哪个: |
| 71 | + |
| 72 | +```javascript |
| 73 | +function togglePlay() { |
| 74 | + if (video.paused) { |
| 75 | + video.play(); |
| 76 | + } else ( |
| 77 | + video.pause(); |
| 78 | + ) |
| 79 | +} |
| 80 | + |
| 81 | +video.addEventListener('click', togglePlay); |
| 82 | +``` |
| 83 | + |
| 84 | +对于这种二选一的判断用三元操作符会更简洁,我们可以直接在一行里写完,像这样: |
| 85 | + |
| 86 | +```javascript |
| 87 | +function tooglePlay() { |
| 88 | + video.paused ? video.play() : video.pause() |
| 89 | +} |
| 90 | +``` |
| 91 | +或者我们可以用字符串来执行属性方法: |
| 92 | + |
| 93 | +```javascript |
| 94 | +function tooglePlay() { |
| 95 | + const method = vidoe.paused ? 'play' : 'pause'; |
| 96 | + video[method](); |
| 97 | + |
| 98 | +} |
| 99 | +``` |
| 100 | +这种方法可能会觉得看起来别扭,但在有些情况下是挺有用的。现在我们点击视频的话已经能够正常切换播放和暂停的状态了。 |
| 101 | + |
| 102 | +#### **图标切换** |
| 103 | +为了让用户知晓播放器状态,我们需要直观地通过图标来展示。当然,我们可以在 `togglePlay()` 方法中处理,不过更好的方式是给播放器加上另一个事件监听,用视频本身的播放状态来判断。 |
| 104 | + |
| 105 | +这是因为,除了点击播放/暂停按钮以外,我们还可以通过比如键盘快捷键、第三方插件甚至耳机上的操作按钮等其他方式来控制。因此,通过视频本身的播放状态来判断是最不容易出错的。代码如下: |
| 106 | + |
| 107 | +```javascript |
| 108 | +// 逻辑 |
| 109 | +function updateButton() { |
| 110 | + const icon = this.paused ? '►' : '❚ ❚'; |
| 111 | + toggle.textContent = icon; |
| 112 | +} |
| 113 | +// 事件 |
| 114 | +video.addEventListener('play', updateButton); |
| 115 | +video.addEventListener('pause', updateButton); |
| 116 | +``` |
| 117 | + |
| 118 | +上面的代码中,我们使用了关键字 `this`。其实在调用 `updateButton` 的时候,这个方法已经被绑定在了 `addEventListener` 的调用者上,也就是绑定到了 `video` 上。因此,`this.paused` 在这里就相当于 `video.paused`。 |
| 119 | + |
| 120 | +### **快退/快进** |
| 121 | +在 HTML 中,我们已经给快退/快进键添加了一个 `data-skip` 属性,对应的值即为快退/快进的秒数。 |
| 122 | + |
| 123 | +我们先来写事件处理。首先要有一个回调函数,叫 `skip`。事件监听的对象,当然是 `skipButtons`,对应的就是快退/快进两个按键。可以尝试一下,如果我们直接在命令行输出这个 `skipButtons`,会得到一个数组。因此,我们需要用 `forEach` 来遍历一下,给每一个按钮都添加上事件监听: |
| 124 | + |
| 125 | +```javascript |
| 126 | +// 逻辑 |
| 127 | +function skip() { |
| 128 | + |
| 129 | +} |
| 130 | +// 事件 |
| 131 | +skipButtons.forEach(button => button.addEventListener('click', skip)); |
| 132 | +``` |
| 133 | + |
| 134 | +然后我们来处理逻辑部分。`video` 有一个属性叫 `currentTime`,可以用来设置视频当前的时间。我们只要修改这个属性就可以了: |
| 135 | + |
| 136 | +```javascript |
| 137 | +function skip() { |
| 138 | + video.currentTime += parseFloat(this.dataset.skip); |
| 139 | +} |
| 140 | +``` |
| 141 | + |
| 142 | +要注意的是,这里就不能用 `this` 来访问 `video` 对象了,因为在这里面,`this` 指向的是遍历得到的每一个 `button`,而我们是要修改 `video` 的 `currentTime` 属性。 |
| 143 | + |
| 144 | +`data-**` 这样的属性以前提到过了,在 JavaScript 中需要通过 `.dataset.**` 来访问。因为我们获取到的是字符串,所以要通过 `parseFloat` 来转换一下。 |
| 145 | + |
| 146 | +### **音量和播放速度** |
| 147 | +接下来我们实现通过控制面板上两个滑动条来控制视频的音量和播放速度。这两个滑动条是 `range` 类型的 `input` 元素,在元素属性中我们指定了他们各自的最大、最小值和调节的“步值”。 |
| 148 | + |
| 149 | +其中需要注意的是,他们分别有一个 `volume` 和 `playbackRate` 的 `name` 属性,我们起这两个名字是因为他们是 `video` 对象里对应音量和播放速度的两个属性名。这样起名并不是必须的,但可以让我们后面 js 的操作更精简。 |
| 150 | + |
| 151 | +通过监听两个 `input` 元素的 `change` 事件,我们就可以通过其 `value` 值改变视频属性了: |
| 152 | + |
| 153 | +```javascript |
| 154 | +function handleRangeUpdate() { |
| 155 | + video[this.name] = this.value; |
| 156 | +} |
| 157 | + |
| 158 | +//遍历 ranges 给两个滑动条都绑定事件 |
| 159 | +ranges.forEach(range => range.addEventListener('change', handleRangeUpdate)); |
| 160 | +``` |
| 161 | +因为我们上面说过,`input` 的 `name` 值和 `video` 对象中的属性名是一样的,可以看到在 `handleRangeUpdate` 函数中我们利用了 `this.name` 的写法来代表属性,,这里的 `this` 一样是 `addEventListener` 的调用者,即 `range`。 |
| 162 | + |
| 163 | +现在调节两个滑动条我们已经可以改变视频相应属性了,美中不足就是滑块的调节并不是实时的,而要等我们放开鼠标才会生效,这是因为 `change` 事件只在 `blur`,也就是元素失去焦点的时候才会触发。要解决这个问题我们可以把 `change` 事件改为 `input` 事件;另一种比较传统的方法是同时监听鼠标在该元素上的 `mousemove` 事件来执行更新的操作, |
| 164 | +在原来的代码下加上一行: |
| 165 | + |
| 166 | +```javascript |
| 167 | +ranges.forEach(range => range.addEventListener('mousemove', handleRangeUpdate)); |
| 168 | +``` |
| 169 | + |
| 170 | +这样鼠标在这滑动条上移动的时候也会更新视频属性了,只不过只有在鼠标拖动滑块的时候才会有值的改变。 |
| 171 | + |
| 172 | +### **进度条操作** |
| 173 | +我们的进度条需要能在鼠标点击和拖动的时候改变视频播放的进度。我们先实现进度条随着视频播放更新进度的功能。 |
| 174 | + |
| 175 | +进度条显示进度的原理很简单,`progress__filled` 这个元素是一个 `flex` 定位的元素,我们改变其 `flex-basis` 的百分比值就可以调节它所占父元素的宽度。`flex-basis` 值代表 `flex` 元素在**主轴**方向上的初始尺寸。关于 `flex-basis` 的更多信息请参考 [MDN](https://developer.mozilla.org/zh-CN/docs/Web/CSS/flex-basis) |
| 176 | + |
| 177 | +```javascript |
| 178 | +// 根据当前播放时间来调节进度条 |
| 179 | +function handleProgress() { |
| 180 | + const percent = (video.currentTime / video.duration) * 100; |
| 181 | + progressBar.style.flexBasis = `${percent}%`; |
| 182 | +} |
| 183 | +``` |
| 184 | +现在只要运行 `handleProgress` 这个函数就能够更新对应的进度条,但我们需要的是自动执行这个操作。也许你会想到利用 `setInterval` 设置一个定时器,其实 `video` 元素给我们提供了更好的方法—— `timeupdate` 事件。这个事件会在媒体文件的 `currentTime` 属性改变的时触发,更多信息请参考 [MDN](https://developer.mozilla.org/zh-CN/docs/Web/Events/timeupdate) |
| 185 | + |
| 186 | +事件操作如下: |
| 187 | + |
| 188 | +```javascript |
| 189 | +video.addEventListener('timeupdate', handleProgress); |
| 190 | +``` |
| 191 | + |
| 192 | +现在随着视频的播放,进度条也会更新进度了。 |
| 193 | + |
| 194 | +接着我们需要点击进度条时调整播放进度,那么我们改变进度,或者说宽度就需要得到鼠标点击的位置,这可以通过事件对象的 `offsetX` 属性来找到,该属性代表鼠标点击位置相对于该元素的水平偏移。得到偏移之后计算出该位置的百分比,那么也就知道了进度的百分比: |
| 195 | + |
| 196 | +```javascript |
| 197 | +... |
| 198 | +// 根据点击位置设置播放时间 |
| 199 | +function scrub(e) { |
| 200 | + const scrubTime = (e.offsetX / progress.offsetWidth) * vidoe.duration; |
| 201 | + video.currentTime = scrubTime; |
| 202 | +} |
| 203 | + |
| 204 | +// 点击事件监听 |
| 205 | +progress.addEventListener('click', scrub); |
| 206 | +``` |
| 207 | + |
| 208 | +进度条还要求可以拖动,这个操作我们可以通过设置一个*标志*来判断当前是否出于拖动状态,然后配合 `mousedown`、`mouseup` 事件来更新这个标志: |
| 209 | + |
| 210 | +```javascript |
| 211 | +... |
| 212 | +let mousedown = false; |
| 213 | + |
| 214 | +// 鼠标在 progress 上移动时更新进度 |
| 215 | +progress.addEventListener('mousemove', (e) => { |
| 216 | + |
| 217 | + // 若处于拖拽状态则执行更新 |
| 218 | + if (mousedown) { |
| 219 | + scrub(e); |
| 220 | + } |
| 221 | +}); |
| 222 | + |
| 223 | +// 鼠标按下改变标志 |
| 224 | +progress.addEventListener('mousedown', () => mousedown = true); |
| 225 | + |
| 226 | +// 鼠标抬起恢复标志 |
| 227 | +progress.addEventListener('mouseup', () => mousedown = false); |
| 228 | +``` |
| 229 | + |
| 230 | +这样就实现了拖拽进度条时改变播放进度的功能,实际使用的时候会发现拖拽和视频的更新并不是实时的,会有一定延迟,这是因为 `mousemove` 事件触发的频率非常高,视频更新的速度跟不上。 |
| 231 | + |
| 232 | +对于 `mousemove` 的回调函数其实我们可以写得更简洁: |
| 233 | + |
| 234 | +```javascript |
| 235 | +progress.addEventListener('mousemove', (e) => mousedown && scrub(e)); |
| 236 | +``` |
| 237 | + |
| 238 | +我们利用逻辑和操作符 `&&` 的**短路**特性来实现 “只有当 `mousedown` 为 `true`,或可类型转换为 `true` 时才执行 `scrub(e)`” 的判断操作,由于逻辑和的判断必须两个都为真时才成立,所以若第一项不为真,那么 js 就不会去管第二项是什么,因此也就不会执行 `scrub(e)`。这种写法在实际项目中是挺常见的,算是一个小技巧,希望大家可以熟悉并使用。 |
| 239 | + |
| 240 | +## **结语** |
| 241 | +至此,我们已经实现了控制面板的绝大部分功能,最后一个留给大家自己尝试的功能是全屏播放: |
| 242 | +* 在控制面板中添加一个全屏按钮 |
| 243 | +* 点击该按钮后可以进入/退出全屏模式 |
0 commit comments