@cloudnest/redxplyr 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.editorconfig +10 -0
- package/.gitpod.yml +6 -0
- package/.node-version +1 -0
- package/.prettierrc +7 -0
- package/.stickler.yml +5 -0
- package/.stylelintrc.json +26 -0
- package/CHANGELOG.md +16 -0
- package/CONTRIBUTING.md +34 -0
- package/CONTROLS.md +49 -0
- package/Dockerfile +32 -0
- package/LICENSE.md +22 -0
- package/README.md +194 -0
- package/cspell.json +48 -0
- package/dist/redxplyr.css +1 -0
- package/dist/redxplyr.js +8801 -0
- package/dist/redxplyr.min.js +2 -0
- package/dist/redxplyr.min.js.map +1 -0
- package/dist/redxplyr.min.mjs +1 -0
- package/dist/redxplyr.min.mjs.map +1 -0
- package/dist/redxplyr.mjs +8793 -0
- package/dist/redxplyr.polyfilled.js +9294 -0
- package/dist/redxplyr.polyfilled.min.js +2 -0
- package/dist/redxplyr.polyfilled.min.js.map +1 -0
- package/dist/redxplyr.polyfilled.min.mjs +1 -0
- package/dist/redxplyr.polyfilled.min.mjs.map +1 -0
- package/dist/redxplyr.polyfilled.mjs +9286 -0
- package/dist/redxplyr.svg +1 -0
- package/eslint.config.mjs +39 -0
- package/gulpfile.js +8 -0
- package/package.json +114 -0
- package/pnpm-workspace.yaml +8 -0
- package/src/js/captions.js +411 -0
- package/src/js/config/defaults.js +459 -0
- package/src/js/config/states.js +10 -0
- package/src/js/config/types.js +34 -0
- package/src/js/console.js +28 -0
- package/src/js/controls.js +1870 -0
- package/src/js/fullscreen.js +305 -0
- package/src/js/html5.js +148 -0
- package/src/js/listeners.js +854 -0
- package/src/js/media.js +61 -0
- package/src/js/plugins/ads.js +647 -0
- package/src/js/plugins/preview-thumbnails.js +706 -0
- package/src/js/plugins/vimeo.js +443 -0
- package/src/js/plugins/youtube.js +451 -0
- package/src/js/plyr.d.ts +729 -0
- package/src/js/plyr.js +1291 -0
- package/src/js/plyr.polyfilled.js +13 -0
- package/src/js/source.js +155 -0
- package/src/js/storage.js +70 -0
- package/src/js/support.js +100 -0
- package/src/js/ui.js +297 -0
- package/src/js/utils/animation.js +33 -0
- package/src/js/utils/arrays.js +23 -0
- package/src/js/utils/browser.js +21 -0
- package/src/js/utils/elements.js +263 -0
- package/src/js/utils/events.js +116 -0
- package/src/js/utils/fetch.js +45 -0
- package/src/js/utils/i18n.js +47 -0
- package/src/js/utils/is.js +81 -0
- package/src/js/utils/load-image.js +19 -0
- package/src/js/utils/load-script.js +14 -0
- package/src/js/utils/load-sprite.js +77 -0
- package/src/js/utils/numbers.js +17 -0
- package/src/js/utils/objects.js +43 -0
- package/src/js/utils/promise.js +14 -0
- package/src/js/utils/strings.js +80 -0
- package/src/js/utils/style.js +148 -0
- package/src/js/utils/time.js +36 -0
- package/src/js/utils/urls.js +40 -0
- package/src/sass/base.scss +69 -0
- package/src/sass/components/badges.scss +12 -0
- package/src/sass/components/captions.scss +58 -0
- package/src/sass/components/control.scss +52 -0
- package/src/sass/components/controls.scss +65 -0
- package/src/sass/components/menus.scss +205 -0
- package/src/sass/components/poster.scss +27 -0
- package/src/sass/components/progress.scss +107 -0
- package/src/sass/components/sliders.scss +99 -0
- package/src/sass/components/times.scss +20 -0
- package/src/sass/components/tooltips.scss +91 -0
- package/src/sass/components/volume.scss +18 -0
- package/src/sass/lib/animation.scss +31 -0
- package/src/sass/lib/css-vars.scss +103 -0
- package/src/sass/lib/functions.scss +3 -0
- package/src/sass/lib/mixins.scss +82 -0
- package/src/sass/plugins/ads.scss +53 -0
- package/src/sass/plugins/preview-thumbnails/index.scss +121 -0
- package/src/sass/plugins/preview-thumbnails/settings.scss +17 -0
- package/src/sass/plyr.scss +46 -0
- package/src/sass/settings/badges.scss +7 -0
- package/src/sass/settings/breakpoints.scss +9 -0
- package/src/sass/settings/captions.scss +10 -0
- package/src/sass/settings/colors.scss +18 -0
- package/src/sass/settings/controls.scss +30 -0
- package/src/sass/settings/cosmetics.scss +5 -0
- package/src/sass/settings/helpers.scss +7 -0
- package/src/sass/settings/menus.scss +13 -0
- package/src/sass/settings/progress.scss +18 -0
- package/src/sass/settings/sliders.scss +39 -0
- package/src/sass/settings/tooltips.scss +11 -0
- package/src/sass/settings/type.scss +16 -0
- package/src/sass/states/fullscreen.scss +15 -0
- package/src/sass/types/audio.scss +61 -0
- package/src/sass/types/video.scss +170 -0
- package/src/sass/utils/animation.scss +7 -0
- package/src/sass/utils/hidden.scss +28 -0
- package/src/sprite/plyr-airplay.svg +8 -0
- package/src/sprite/plyr-captions-off.svg +7 -0
- package/src/sprite/plyr-captions-on.svg +7 -0
- package/src/sprite/plyr-download.svg +8 -0
- package/src/sprite/plyr-enter-fullscreen.svg +4 -0
- package/src/sprite/plyr-exit-fullscreen.svg +4 -0
- package/src/sprite/plyr-fast-forward.svg +3 -0
- package/src/sprite/plyr-logo-vimeo.svg +6 -0
- package/src/sprite/plyr-logo-youtube.svg +6 -0
- package/src/sprite/plyr-muted.svg +8 -0
- package/src/sprite/plyr-pause.svg +8 -0
- package/src/sprite/plyr-pip.svg +6 -0
- package/src/sprite/plyr-play.svg +5 -0
- package/src/sprite/plyr-restart.svg +5 -0
- package/src/sprite/plyr-rewind.svg +3 -0
- package/src/sprite/plyr-settings.svg +5 -0
- package/src/sprite/plyr-volume.svg +11 -0
- package/tasks/build.js +226 -0
- package/tasks/deploy.js +216 -0
- package/tasks/utils/publish.js +34 -0
|
@@ -0,0 +1,854 @@
|
|
|
1
|
+
// ==========================================================================
|
|
2
|
+
// Plyr Event Listeners
|
|
3
|
+
// ==========================================================================
|
|
4
|
+
|
|
5
|
+
import controls from './controls';
|
|
6
|
+
import ui from './ui';
|
|
7
|
+
import { repaint } from './utils/animation';
|
|
8
|
+
import browser from './utils/browser';
|
|
9
|
+
import { getElement, getElements, matches, toggleClass } from './utils/elements';
|
|
10
|
+
import { off, on, once, toggleListener, triggerEvent } from './utils/events';
|
|
11
|
+
import is from './utils/is';
|
|
12
|
+
import { silencePromise } from './utils/promise';
|
|
13
|
+
import { getAspectRatio, getViewportSize, supportsCSS } from './utils/style';
|
|
14
|
+
|
|
15
|
+
class Listeners {
|
|
16
|
+
constructor(player) {
|
|
17
|
+
this.player = player;
|
|
18
|
+
this.lastKey = null;
|
|
19
|
+
this.focusTimer = null;
|
|
20
|
+
this.lastKeyDown = null;
|
|
21
|
+
|
|
22
|
+
this.handleKey = this.handleKey.bind(this);
|
|
23
|
+
this.toggleMenu = this.toggleMenu.bind(this);
|
|
24
|
+
this.firstTouch = this.firstTouch.bind(this);
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
// Handle key presses
|
|
28
|
+
handleKey(event) {
|
|
29
|
+
const { player } = this;
|
|
30
|
+
const { elements } = player;
|
|
31
|
+
const { key, type, altKey, ctrlKey, metaKey, shiftKey } = event;
|
|
32
|
+
const pressed = type === 'keydown';
|
|
33
|
+
const repeat = pressed && key === this.lastKey;
|
|
34
|
+
|
|
35
|
+
// Bail if a modifier key is set
|
|
36
|
+
if (altKey || ctrlKey || metaKey || shiftKey) {
|
|
37
|
+
return;
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
// If the event is bubbled from the media element
|
|
41
|
+
// Firefox doesn't get the key for whatever reason
|
|
42
|
+
if (!key) {
|
|
43
|
+
return;
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
// Seek by increment
|
|
47
|
+
const seekByIncrement = (increment) => {
|
|
48
|
+
// Divide the max duration into 10th's and times by the number value
|
|
49
|
+
player.currentTime = (player.duration / 10) * increment;
|
|
50
|
+
};
|
|
51
|
+
|
|
52
|
+
// Handle the key on keydown
|
|
53
|
+
// Reset on keyup
|
|
54
|
+
if (pressed) {
|
|
55
|
+
// Check focused element
|
|
56
|
+
// and if the focused element is not editable (e.g. text input)
|
|
57
|
+
// and any that accept key input http://webaim.org/techniques/keyboard/
|
|
58
|
+
const focused = document.activeElement;
|
|
59
|
+
if (is.element(focused)) {
|
|
60
|
+
const { editable } = player.config.selectors;
|
|
61
|
+
const { seek } = elements.inputs;
|
|
62
|
+
|
|
63
|
+
if (focused !== seek && matches(focused, editable)) {
|
|
64
|
+
return;
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
if (event.key === ' ' && matches(focused, 'button, [role^="menuitem"]')) {
|
|
68
|
+
return;
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
// Which keys should we prevent default
|
|
73
|
+
const preventDefault = [
|
|
74
|
+
' ',
|
|
75
|
+
'ArrowLeft',
|
|
76
|
+
'ArrowUp',
|
|
77
|
+
'ArrowRight',
|
|
78
|
+
'ArrowDown',
|
|
79
|
+
|
|
80
|
+
'0',
|
|
81
|
+
'1',
|
|
82
|
+
'2',
|
|
83
|
+
'3',
|
|
84
|
+
'4',
|
|
85
|
+
'5',
|
|
86
|
+
'6',
|
|
87
|
+
'7',
|
|
88
|
+
'8',
|
|
89
|
+
'9',
|
|
90
|
+
|
|
91
|
+
'c',
|
|
92
|
+
'f',
|
|
93
|
+
'k',
|
|
94
|
+
'l',
|
|
95
|
+
'm',
|
|
96
|
+
];
|
|
97
|
+
|
|
98
|
+
// If the key is found prevent default (e.g. prevent scrolling for arrows)
|
|
99
|
+
if (preventDefault.includes(key)) {
|
|
100
|
+
event.preventDefault();
|
|
101
|
+
event.stopPropagation();
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
switch (key) {
|
|
105
|
+
case '0':
|
|
106
|
+
case '1':
|
|
107
|
+
case '2':
|
|
108
|
+
case '3':
|
|
109
|
+
case '4':
|
|
110
|
+
case '5':
|
|
111
|
+
case '6':
|
|
112
|
+
case '7':
|
|
113
|
+
case '8':
|
|
114
|
+
case '9':
|
|
115
|
+
if (!repeat) {
|
|
116
|
+
seekByIncrement(Number.parseInt(key, 10));
|
|
117
|
+
}
|
|
118
|
+
break;
|
|
119
|
+
|
|
120
|
+
case ' ':
|
|
121
|
+
case 'k':
|
|
122
|
+
if (!repeat) {
|
|
123
|
+
silencePromise(player.togglePlay());
|
|
124
|
+
}
|
|
125
|
+
break;
|
|
126
|
+
|
|
127
|
+
case 'ArrowUp':
|
|
128
|
+
player.increaseVolume(0.1);
|
|
129
|
+
break;
|
|
130
|
+
|
|
131
|
+
case 'ArrowDown':
|
|
132
|
+
player.decreaseVolume(0.1);
|
|
133
|
+
break;
|
|
134
|
+
|
|
135
|
+
case 'm':
|
|
136
|
+
if (!repeat) {
|
|
137
|
+
player.muted = !player.muted;
|
|
138
|
+
}
|
|
139
|
+
break;
|
|
140
|
+
|
|
141
|
+
case 'ArrowRight':
|
|
142
|
+
player.forward();
|
|
143
|
+
break;
|
|
144
|
+
|
|
145
|
+
case 'ArrowLeft':
|
|
146
|
+
player.rewind();
|
|
147
|
+
break;
|
|
148
|
+
|
|
149
|
+
case 'f':
|
|
150
|
+
player.fullscreen.toggle();
|
|
151
|
+
break;
|
|
152
|
+
|
|
153
|
+
case 'c':
|
|
154
|
+
if (!repeat) {
|
|
155
|
+
player.toggleCaptions();
|
|
156
|
+
}
|
|
157
|
+
break;
|
|
158
|
+
|
|
159
|
+
case 'l':
|
|
160
|
+
player.loop = !player.loop;
|
|
161
|
+
break;
|
|
162
|
+
|
|
163
|
+
default:
|
|
164
|
+
break;
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
// Escape is handle natively when in full screen
|
|
168
|
+
// So we only need to worry about non native
|
|
169
|
+
if (key === 'Escape' && !player.fullscreen.usingNative && player.fullscreen.active) {
|
|
170
|
+
player.fullscreen.toggle();
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
// Store last key for next cycle
|
|
174
|
+
this.lastKey = key;
|
|
175
|
+
}
|
|
176
|
+
else {
|
|
177
|
+
this.lastKey = null;
|
|
178
|
+
}
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
// Toggle menu
|
|
182
|
+
toggleMenu(event) {
|
|
183
|
+
controls.toggleMenu.call(this.player, event);
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
// Device is touch enabled
|
|
187
|
+
firstTouch = () => {
|
|
188
|
+
const { player } = this;
|
|
189
|
+
const { elements } = player;
|
|
190
|
+
|
|
191
|
+
player.touch = true;
|
|
192
|
+
|
|
193
|
+
// Add touch class
|
|
194
|
+
toggleClass(elements.container, player.config.classNames.isTouch, true);
|
|
195
|
+
};
|
|
196
|
+
|
|
197
|
+
// Global window & document listeners
|
|
198
|
+
global = (toggle = true) => {
|
|
199
|
+
const { player } = this;
|
|
200
|
+
|
|
201
|
+
// Keyboard shortcuts
|
|
202
|
+
if (player.config.keyboard.global) {
|
|
203
|
+
toggleListener.call(player, window, 'keydown keyup', this.handleKey, toggle, false);
|
|
204
|
+
}
|
|
205
|
+
|
|
206
|
+
// Click anywhere closes menu
|
|
207
|
+
toggleListener.call(player, document.body, 'click', this.toggleMenu, toggle);
|
|
208
|
+
|
|
209
|
+
// Detect touch by events
|
|
210
|
+
once.call(player, document.body, 'touchstart', this.firstTouch);
|
|
211
|
+
};
|
|
212
|
+
|
|
213
|
+
// Container listeners
|
|
214
|
+
container = () => {
|
|
215
|
+
const { player } = this;
|
|
216
|
+
const { config, elements, timers } = player;
|
|
217
|
+
|
|
218
|
+
// Keyboard shortcuts
|
|
219
|
+
if (!config.keyboard.global && config.keyboard.focused) {
|
|
220
|
+
on.call(player, elements.container, 'keydown keyup', this.handleKey, false);
|
|
221
|
+
}
|
|
222
|
+
|
|
223
|
+
// Toggle controls on mouse events and entering fullscreen
|
|
224
|
+
on.call(
|
|
225
|
+
player,
|
|
226
|
+
elements.container,
|
|
227
|
+
'mousemove mouseleave touchstart touchmove enterfullscreen exitfullscreen',
|
|
228
|
+
(event) => {
|
|
229
|
+
const { controls: controlsElement } = elements;
|
|
230
|
+
|
|
231
|
+
// Remove button states for fullscreen
|
|
232
|
+
if (controlsElement && event.type === 'enterfullscreen') {
|
|
233
|
+
controlsElement.pressed = false;
|
|
234
|
+
controlsElement.hover = false;
|
|
235
|
+
}
|
|
236
|
+
|
|
237
|
+
// Show, then hide after a timeout unless another control event occurs
|
|
238
|
+
const show = ['touchstart', 'touchmove', 'mousemove'].includes(event.type);
|
|
239
|
+
let delay = 0;
|
|
240
|
+
|
|
241
|
+
if (show) {
|
|
242
|
+
ui.toggleControls.call(player, true);
|
|
243
|
+
// Use longer timeout for touch devices
|
|
244
|
+
delay = player.touch ? 3000 : 2000;
|
|
245
|
+
}
|
|
246
|
+
|
|
247
|
+
// Clear timer
|
|
248
|
+
clearTimeout(timers.controls);
|
|
249
|
+
|
|
250
|
+
// Set new timer to prevent flicker when seeking
|
|
251
|
+
timers.controls = setTimeout(() => ui.toggleControls.call(player, false), delay);
|
|
252
|
+
},
|
|
253
|
+
);
|
|
254
|
+
|
|
255
|
+
// Set a gutter for Vimeo
|
|
256
|
+
const setGutter = () => {
|
|
257
|
+
if (!player.isVimeo || player.config.vimeo.premium) {
|
|
258
|
+
return;
|
|
259
|
+
}
|
|
260
|
+
|
|
261
|
+
const target = elements.wrapper;
|
|
262
|
+
const { active } = player.fullscreen;
|
|
263
|
+
const [videoWidth, videoHeight] = getAspectRatio.call(player);
|
|
264
|
+
const useNativeAspectRatio = supportsCSS(`aspect-ratio: ${videoWidth} / ${videoHeight}`);
|
|
265
|
+
|
|
266
|
+
// If not active, remove styles
|
|
267
|
+
if (!active) {
|
|
268
|
+
if (useNativeAspectRatio) {
|
|
269
|
+
target.style.width = null;
|
|
270
|
+
target.style.height = null;
|
|
271
|
+
}
|
|
272
|
+
else {
|
|
273
|
+
target.style.maxWidth = null;
|
|
274
|
+
target.style.margin = null;
|
|
275
|
+
}
|
|
276
|
+
return;
|
|
277
|
+
}
|
|
278
|
+
|
|
279
|
+
// Determine which dimension will overflow and constrain view
|
|
280
|
+
const [viewportWidth, viewportHeight] = getViewportSize();
|
|
281
|
+
const overflow = viewportWidth / viewportHeight > videoWidth / videoHeight;
|
|
282
|
+
|
|
283
|
+
if (useNativeAspectRatio) {
|
|
284
|
+
target.style.width = overflow ? 'auto' : '100%';
|
|
285
|
+
target.style.height = overflow ? '100%' : 'auto';
|
|
286
|
+
}
|
|
287
|
+
else {
|
|
288
|
+
target.style.maxWidth = overflow ? `${(viewportHeight / videoHeight) * videoWidth}px` : null;
|
|
289
|
+
target.style.margin = overflow ? '0 auto' : null;
|
|
290
|
+
}
|
|
291
|
+
};
|
|
292
|
+
|
|
293
|
+
// Handle resizing
|
|
294
|
+
const resized = () => {
|
|
295
|
+
clearTimeout(timers.resized);
|
|
296
|
+
timers.resized = setTimeout(setGutter, 50);
|
|
297
|
+
};
|
|
298
|
+
|
|
299
|
+
on.call(player, elements.container, 'enterfullscreen exitfullscreen', (event) => {
|
|
300
|
+
const { target } = player.fullscreen;
|
|
301
|
+
|
|
302
|
+
// Ignore events not from target
|
|
303
|
+
if (target !== elements.container) {
|
|
304
|
+
return;
|
|
305
|
+
}
|
|
306
|
+
|
|
307
|
+
// If it's not an embed and no ratio specified
|
|
308
|
+
if (!player.isEmbed && is.empty(player.config.ratio)) {
|
|
309
|
+
return;
|
|
310
|
+
}
|
|
311
|
+
|
|
312
|
+
// Set Vimeo gutter
|
|
313
|
+
setGutter();
|
|
314
|
+
|
|
315
|
+
// Watch for resizes
|
|
316
|
+
const method = event.type === 'enterfullscreen' ? on : off;
|
|
317
|
+
method.call(player, window, 'resize', resized);
|
|
318
|
+
});
|
|
319
|
+
};
|
|
320
|
+
|
|
321
|
+
// Listen for media events
|
|
322
|
+
media = () => {
|
|
323
|
+
const { player } = this;
|
|
324
|
+
const { elements } = player;
|
|
325
|
+
|
|
326
|
+
// Time change on media
|
|
327
|
+
on.call(player, player.media, 'timeupdate seeking seeked', event => controls.timeUpdate.call(player, event));
|
|
328
|
+
|
|
329
|
+
// Display duration
|
|
330
|
+
on.call(player, player.media, 'durationchange loadeddata loadedmetadata', event =>
|
|
331
|
+
controls.durationUpdate.call(player, event));
|
|
332
|
+
|
|
333
|
+
// Handle the media finishing
|
|
334
|
+
on.call(player, player.media, 'ended', () => {
|
|
335
|
+
// Show poster on end
|
|
336
|
+
if (player.isHTML5 && player.isVideo && player.config.resetOnEnd) {
|
|
337
|
+
// Restart
|
|
338
|
+
player.restart();
|
|
339
|
+
|
|
340
|
+
// Call pause otherwise IE11 will start playing the video again
|
|
341
|
+
player.pause();
|
|
342
|
+
}
|
|
343
|
+
});
|
|
344
|
+
|
|
345
|
+
// Check for buffer progress
|
|
346
|
+
on.call(player, player.media, 'progress playing seeking seeked', event =>
|
|
347
|
+
controls.updateProgress.call(player, event));
|
|
348
|
+
|
|
349
|
+
// Handle volume changes
|
|
350
|
+
on.call(player, player.media, 'volumechange', event => controls.updateVolume.call(player, event));
|
|
351
|
+
|
|
352
|
+
// Handle play/pause
|
|
353
|
+
on.call(player, player.media, 'playing play pause ended emptied timeupdate', event =>
|
|
354
|
+
ui.checkPlaying.call(player, event));
|
|
355
|
+
|
|
356
|
+
// Loading state
|
|
357
|
+
on.call(player, player.media, 'waiting canplay seeked playing', event => ui.checkLoading.call(player, event));
|
|
358
|
+
|
|
359
|
+
// Click video
|
|
360
|
+
if (player.supported.ui && player.config.clickToPlay && !player.isAudio) {
|
|
361
|
+
// Re-fetch the wrapper
|
|
362
|
+
const wrapper = getElement.call(player, `.${player.config.classNames.video}`);
|
|
363
|
+
|
|
364
|
+
// Bail if there's no wrapper (this should never happen)
|
|
365
|
+
if (!is.element(wrapper)) {
|
|
366
|
+
return;
|
|
367
|
+
}
|
|
368
|
+
|
|
369
|
+
// On click play, pause or restart
|
|
370
|
+
on.call(player, elements.container, 'click', (event) => {
|
|
371
|
+
const targets = [elements.container, wrapper];
|
|
372
|
+
|
|
373
|
+
// Ignore if click if not container or in video wrapper
|
|
374
|
+
if (!targets.includes(event.target) && !wrapper.contains(event.target)) {
|
|
375
|
+
return;
|
|
376
|
+
}
|
|
377
|
+
|
|
378
|
+
// Touch devices will just show controls (if hidden)
|
|
379
|
+
if (player.touch && player.config.hideControls) {
|
|
380
|
+
return;
|
|
381
|
+
}
|
|
382
|
+
|
|
383
|
+
if (player.ended) {
|
|
384
|
+
this.proxy(event, player.restart, 'restart');
|
|
385
|
+
this.proxy(
|
|
386
|
+
event,
|
|
387
|
+
() => {
|
|
388
|
+
silencePromise(player.play());
|
|
389
|
+
},
|
|
390
|
+
'play',
|
|
391
|
+
);
|
|
392
|
+
}
|
|
393
|
+
else {
|
|
394
|
+
this.proxy(
|
|
395
|
+
event,
|
|
396
|
+
() => {
|
|
397
|
+
silencePromise(player.togglePlay());
|
|
398
|
+
},
|
|
399
|
+
'play',
|
|
400
|
+
);
|
|
401
|
+
}
|
|
402
|
+
});
|
|
403
|
+
}
|
|
404
|
+
|
|
405
|
+
// Disable right click
|
|
406
|
+
if (player.supported.ui && player.config.disableContextMenu) {
|
|
407
|
+
on.call(
|
|
408
|
+
player,
|
|
409
|
+
elements.wrapper,
|
|
410
|
+
'contextmenu',
|
|
411
|
+
(event) => {
|
|
412
|
+
event.preventDefault();
|
|
413
|
+
},
|
|
414
|
+
false,
|
|
415
|
+
);
|
|
416
|
+
}
|
|
417
|
+
|
|
418
|
+
// Volume change
|
|
419
|
+
on.call(player, player.media, 'volumechange', () => {
|
|
420
|
+
// Save to storage
|
|
421
|
+
player.storage.set({
|
|
422
|
+
volume: player.volume,
|
|
423
|
+
muted: player.muted,
|
|
424
|
+
});
|
|
425
|
+
});
|
|
426
|
+
|
|
427
|
+
// Speed change
|
|
428
|
+
on.call(player, player.media, 'ratechange', () => {
|
|
429
|
+
// Update UI
|
|
430
|
+
controls.updateSetting.call(player, 'speed');
|
|
431
|
+
|
|
432
|
+
// Save to storage
|
|
433
|
+
player.storage.set({ speed: player.speed });
|
|
434
|
+
});
|
|
435
|
+
|
|
436
|
+
// Quality change
|
|
437
|
+
on.call(player, player.media, 'qualitychange', (event) => {
|
|
438
|
+
// Update UI
|
|
439
|
+
controls.updateSetting.call(player, 'quality', null, event.detail.quality);
|
|
440
|
+
});
|
|
441
|
+
|
|
442
|
+
// Update download link when ready and if quality changes
|
|
443
|
+
on.call(player, player.media, 'ready qualitychange', () => {
|
|
444
|
+
controls.setDownloadUrl.call(player);
|
|
445
|
+
});
|
|
446
|
+
|
|
447
|
+
// Proxy events to container
|
|
448
|
+
// Bubble up key events for Edge
|
|
449
|
+
const proxyEvents = player.config.events.concat(['keyup', 'keydown']).join(' ');
|
|
450
|
+
|
|
451
|
+
on.call(player, player.media, proxyEvents, (event) => {
|
|
452
|
+
let { detail = {} } = event;
|
|
453
|
+
|
|
454
|
+
// Get error details from media
|
|
455
|
+
if (event.type === 'error') {
|
|
456
|
+
detail = player.media.error;
|
|
457
|
+
}
|
|
458
|
+
|
|
459
|
+
triggerEvent.call(player, elements.container, event.type, true, detail);
|
|
460
|
+
});
|
|
461
|
+
};
|
|
462
|
+
|
|
463
|
+
// Run default and custom handlers
|
|
464
|
+
proxy = (event, defaultHandler, customHandlerKey) => {
|
|
465
|
+
const { player } = this;
|
|
466
|
+
const customHandler = player.config.listeners[customHandlerKey];
|
|
467
|
+
const hasCustomHandler = is.function(customHandler);
|
|
468
|
+
let returned = true;
|
|
469
|
+
|
|
470
|
+
// Execute custom handler
|
|
471
|
+
if (hasCustomHandler) {
|
|
472
|
+
returned = customHandler.call(player, event);
|
|
473
|
+
}
|
|
474
|
+
|
|
475
|
+
// Only call default handler if not prevented in custom handler
|
|
476
|
+
if (returned !== false && is.function(defaultHandler)) {
|
|
477
|
+
defaultHandler.call(player, event);
|
|
478
|
+
}
|
|
479
|
+
};
|
|
480
|
+
|
|
481
|
+
// Trigger custom and default handlers
|
|
482
|
+
bind = (element, type, defaultHandler, customHandlerKey, passive = true) => {
|
|
483
|
+
const { player } = this;
|
|
484
|
+
const customHandler = player.config.listeners[customHandlerKey];
|
|
485
|
+
const hasCustomHandler = is.function(customHandler);
|
|
486
|
+
|
|
487
|
+
on.call(
|
|
488
|
+
player,
|
|
489
|
+
element,
|
|
490
|
+
type,
|
|
491
|
+
event => this.proxy(event, defaultHandler, customHandlerKey),
|
|
492
|
+
passive && !hasCustomHandler,
|
|
493
|
+
);
|
|
494
|
+
};
|
|
495
|
+
|
|
496
|
+
// Listen for control events
|
|
497
|
+
controls = () => {
|
|
498
|
+
const { player } = this;
|
|
499
|
+
const { elements } = player;
|
|
500
|
+
// IE doesn't support input event, so we fallback to change
|
|
501
|
+
const inputEvent = browser.isIE ? 'change' : 'input';
|
|
502
|
+
|
|
503
|
+
// Play/pause toggle
|
|
504
|
+
if (elements.buttons.play) {
|
|
505
|
+
Array.from(elements.buttons.play).forEach((button) => {
|
|
506
|
+
this.bind(
|
|
507
|
+
button,
|
|
508
|
+
'click',
|
|
509
|
+
() => {
|
|
510
|
+
silencePromise(player.togglePlay());
|
|
511
|
+
},
|
|
512
|
+
'play',
|
|
513
|
+
);
|
|
514
|
+
});
|
|
515
|
+
}
|
|
516
|
+
|
|
517
|
+
// Pause
|
|
518
|
+
this.bind(elements.buttons.restart, 'click', player.restart, 'restart');
|
|
519
|
+
|
|
520
|
+
// Rewind
|
|
521
|
+
this.bind(
|
|
522
|
+
elements.buttons.rewind,
|
|
523
|
+
'click',
|
|
524
|
+
() => {
|
|
525
|
+
// Record seek time so we can prevent hiding controls for a few seconds after rewind
|
|
526
|
+
player.lastSeekTime = Date.now();
|
|
527
|
+
player.rewind();
|
|
528
|
+
},
|
|
529
|
+
'rewind',
|
|
530
|
+
);
|
|
531
|
+
|
|
532
|
+
// Rewind
|
|
533
|
+
this.bind(
|
|
534
|
+
elements.buttons.fastForward,
|
|
535
|
+
'click',
|
|
536
|
+
() => {
|
|
537
|
+
// Record seek time so we can prevent hiding controls for a few seconds after fast forward
|
|
538
|
+
player.lastSeekTime = Date.now();
|
|
539
|
+
player.forward();
|
|
540
|
+
},
|
|
541
|
+
'fastForward',
|
|
542
|
+
);
|
|
543
|
+
|
|
544
|
+
// Mute toggle
|
|
545
|
+
this.bind(
|
|
546
|
+
elements.buttons.mute,
|
|
547
|
+
'click',
|
|
548
|
+
() => {
|
|
549
|
+
player.muted = !player.muted;
|
|
550
|
+
},
|
|
551
|
+
'mute',
|
|
552
|
+
);
|
|
553
|
+
|
|
554
|
+
// Captions toggle
|
|
555
|
+
this.bind(elements.buttons.captions, 'click', () => player.toggleCaptions());
|
|
556
|
+
|
|
557
|
+
// Download
|
|
558
|
+
this.bind(
|
|
559
|
+
elements.buttons.download,
|
|
560
|
+
'click',
|
|
561
|
+
() => {
|
|
562
|
+
triggerEvent.call(player, player.media, 'download');
|
|
563
|
+
},
|
|
564
|
+
'download',
|
|
565
|
+
);
|
|
566
|
+
|
|
567
|
+
// Fullscreen toggle
|
|
568
|
+
this.bind(
|
|
569
|
+
elements.buttons.fullscreen,
|
|
570
|
+
'click',
|
|
571
|
+
() => {
|
|
572
|
+
player.fullscreen.toggle();
|
|
573
|
+
},
|
|
574
|
+
'fullscreen',
|
|
575
|
+
);
|
|
576
|
+
|
|
577
|
+
// Picture-in-Picture
|
|
578
|
+
this.bind(
|
|
579
|
+
elements.buttons.pip,
|
|
580
|
+
'click',
|
|
581
|
+
() => {
|
|
582
|
+
player.pip = 'toggle';
|
|
583
|
+
},
|
|
584
|
+
'pip',
|
|
585
|
+
);
|
|
586
|
+
|
|
587
|
+
// Airplay
|
|
588
|
+
this.bind(elements.buttons.airplay, 'click', player.airplay, 'airplay');
|
|
589
|
+
|
|
590
|
+
// Settings menu - click toggle
|
|
591
|
+
this.bind(
|
|
592
|
+
elements.buttons.settings,
|
|
593
|
+
'click',
|
|
594
|
+
(event) => {
|
|
595
|
+
// Prevent the document click listener closing the menu
|
|
596
|
+
event.stopPropagation();
|
|
597
|
+
event.preventDefault();
|
|
598
|
+
|
|
599
|
+
controls.toggleMenu.call(player, event);
|
|
600
|
+
},
|
|
601
|
+
null,
|
|
602
|
+
false,
|
|
603
|
+
); // Can't be passive as we're preventing default
|
|
604
|
+
|
|
605
|
+
// Settings menu - keyboard toggle
|
|
606
|
+
// We have to bind to keyup otherwise Firefox triggers a click when a keydown event handler shifts focus
|
|
607
|
+
// https://bugzilla.mozilla.org/show_bug.cgi?id=1220143
|
|
608
|
+
this.bind(
|
|
609
|
+
elements.buttons.settings,
|
|
610
|
+
'keyup',
|
|
611
|
+
(event) => {
|
|
612
|
+
if (![' ', 'Enter'].includes(event.key)) {
|
|
613
|
+
return;
|
|
614
|
+
}
|
|
615
|
+
|
|
616
|
+
// Because return triggers a click anyway, all we need to do is set focus
|
|
617
|
+
if (event.key === 'Enter') {
|
|
618
|
+
controls.focusFirstMenuItem.call(player, null, true);
|
|
619
|
+
return;
|
|
620
|
+
}
|
|
621
|
+
|
|
622
|
+
// Prevent scroll
|
|
623
|
+
event.preventDefault();
|
|
624
|
+
|
|
625
|
+
// Prevent playing video (Firefox)
|
|
626
|
+
event.stopPropagation();
|
|
627
|
+
|
|
628
|
+
// Toggle menu
|
|
629
|
+
controls.toggleMenu.call(player, event);
|
|
630
|
+
},
|
|
631
|
+
null,
|
|
632
|
+
false, // Can't be passive as we're preventing default
|
|
633
|
+
);
|
|
634
|
+
|
|
635
|
+
// Escape closes menu
|
|
636
|
+
this.bind(elements.settings.menu, 'keydown', (event) => {
|
|
637
|
+
if (event.key === 'Escape') {
|
|
638
|
+
controls.toggleMenu.call(player, event);
|
|
639
|
+
}
|
|
640
|
+
});
|
|
641
|
+
|
|
642
|
+
// Set range input alternative "value", which matches the tooltip time (#954)
|
|
643
|
+
this.bind(elements.inputs.seek, 'mousedown mousemove', (event) => {
|
|
644
|
+
const rect = elements.progress.getBoundingClientRect();
|
|
645
|
+
const scrollLeft = event.pageX - event.clientX;
|
|
646
|
+
const percent = (100 / rect.width) * (event.pageX - rect.left - scrollLeft);
|
|
647
|
+
event.currentTarget.setAttribute('seek-value', percent);
|
|
648
|
+
});
|
|
649
|
+
|
|
650
|
+
// Pause while seeking
|
|
651
|
+
this.bind(elements.inputs.seek, 'mousedown mouseup keydown keyup touchstart touchend', (event) => {
|
|
652
|
+
const seek = event.currentTarget;
|
|
653
|
+
const attribute = 'play-on-seeked';
|
|
654
|
+
|
|
655
|
+
if (is.keyboardEvent(event) && !['ArrowLeft', 'ArrowRight'].includes(event.key)) {
|
|
656
|
+
return;
|
|
657
|
+
}
|
|
658
|
+
|
|
659
|
+
// Record seek time so we can prevent hiding controls for a few seconds after seek
|
|
660
|
+
player.lastSeekTime = Date.now();
|
|
661
|
+
|
|
662
|
+
// Was playing before?
|
|
663
|
+
const play = seek.hasAttribute(attribute);
|
|
664
|
+
// Done seeking
|
|
665
|
+
const done = ['mouseup', 'touchend', 'keyup'].includes(event.type);
|
|
666
|
+
|
|
667
|
+
// If we're done seeking and it was playing, resume playback
|
|
668
|
+
if (play && done) {
|
|
669
|
+
seek.removeAttribute(attribute);
|
|
670
|
+
silencePromise(player.play());
|
|
671
|
+
}
|
|
672
|
+
else if (!done && player.playing) {
|
|
673
|
+
seek.setAttribute(attribute, '');
|
|
674
|
+
player.pause();
|
|
675
|
+
}
|
|
676
|
+
});
|
|
677
|
+
|
|
678
|
+
// Fix range inputs on iOS
|
|
679
|
+
// Super weird iOS bug where after you interact with an <input type="range">,
|
|
680
|
+
// it takes over further interactions on the page. This is a hack
|
|
681
|
+
if (browser.isIos) {
|
|
682
|
+
const inputs = getElements.call(player, 'input[type="range"]');
|
|
683
|
+
Array.from(inputs).forEach(input => this.bind(input, inputEvent, event => repaint(event.target)));
|
|
684
|
+
}
|
|
685
|
+
|
|
686
|
+
// Seek
|
|
687
|
+
this.bind(
|
|
688
|
+
elements.inputs.seek,
|
|
689
|
+
inputEvent,
|
|
690
|
+
(event) => {
|
|
691
|
+
const seek = event.currentTarget;
|
|
692
|
+
// If it exists, use seek-value instead of "value" for consistency with tooltip time (#954)
|
|
693
|
+
let seekTo = seek.getAttribute('seek-value');
|
|
694
|
+
|
|
695
|
+
if (is.empty(seekTo)) {
|
|
696
|
+
seekTo = seek.value;
|
|
697
|
+
}
|
|
698
|
+
|
|
699
|
+
seek.removeAttribute('seek-value');
|
|
700
|
+
|
|
701
|
+
player.currentTime = (seekTo / seek.max) * player.duration;
|
|
702
|
+
},
|
|
703
|
+
'seek',
|
|
704
|
+
);
|
|
705
|
+
|
|
706
|
+
// Seek tooltip
|
|
707
|
+
this.bind(elements.progress, 'mouseenter mouseleave mousemove', event =>
|
|
708
|
+
controls.updateSeekTooltip.call(player, event));
|
|
709
|
+
|
|
710
|
+
// Preview thumbnails plugin
|
|
711
|
+
// TODO: Really need to work on some sort of plug-in wide event bus or pub-sub for this
|
|
712
|
+
this.bind(elements.progress, 'mousemove touchmove', (event) => {
|
|
713
|
+
const { previewThumbnails } = player;
|
|
714
|
+
|
|
715
|
+
if (previewThumbnails && previewThumbnails.loaded) {
|
|
716
|
+
previewThumbnails.startMove(event);
|
|
717
|
+
}
|
|
718
|
+
});
|
|
719
|
+
|
|
720
|
+
// Hide thumbnail preview - on mouse click, mouse leave, and video play/seek. All four are required, e.g., for buffering
|
|
721
|
+
this.bind(elements.progress, 'mouseleave touchend click', () => {
|
|
722
|
+
const { previewThumbnails } = player;
|
|
723
|
+
|
|
724
|
+
if (previewThumbnails && previewThumbnails.loaded) {
|
|
725
|
+
previewThumbnails.endMove(false, true);
|
|
726
|
+
}
|
|
727
|
+
});
|
|
728
|
+
|
|
729
|
+
// Show scrubbing preview
|
|
730
|
+
this.bind(elements.progress, 'mousedown touchstart', (event) => {
|
|
731
|
+
const { previewThumbnails } = player;
|
|
732
|
+
|
|
733
|
+
if (previewThumbnails && previewThumbnails.loaded) {
|
|
734
|
+
previewThumbnails.startScrubbing(event);
|
|
735
|
+
}
|
|
736
|
+
});
|
|
737
|
+
|
|
738
|
+
this.bind(elements.progress, 'mouseup touchend', (event) => {
|
|
739
|
+
const { previewThumbnails } = player;
|
|
740
|
+
|
|
741
|
+
if (previewThumbnails && previewThumbnails.loaded) {
|
|
742
|
+
previewThumbnails.endScrubbing(event);
|
|
743
|
+
}
|
|
744
|
+
});
|
|
745
|
+
|
|
746
|
+
// Polyfill for lower fill in <input type="range"> for webkit
|
|
747
|
+
if (browser.isWebKit) {
|
|
748
|
+
Array.from(getElements.call(player, 'input[type="range"]')).forEach((element) => {
|
|
749
|
+
this.bind(element, 'input', event => controls.updateRangeFill.call(player, event.target));
|
|
750
|
+
});
|
|
751
|
+
}
|
|
752
|
+
|
|
753
|
+
// Current time invert
|
|
754
|
+
// Only if one time element is used for both currentTime and duration
|
|
755
|
+
if (player.config.toggleInvert && !is.element(elements.display.duration)) {
|
|
756
|
+
this.bind(elements.display.currentTime, 'click', () => {
|
|
757
|
+
// Do nothing if we're at the start
|
|
758
|
+
if (player.currentTime === 0) {
|
|
759
|
+
return;
|
|
760
|
+
}
|
|
761
|
+
|
|
762
|
+
player.config.invertTime = !player.config.invertTime;
|
|
763
|
+
|
|
764
|
+
controls.timeUpdate.call(player);
|
|
765
|
+
});
|
|
766
|
+
}
|
|
767
|
+
|
|
768
|
+
// Volume
|
|
769
|
+
this.bind(
|
|
770
|
+
elements.inputs.volume,
|
|
771
|
+
inputEvent,
|
|
772
|
+
(event) => {
|
|
773
|
+
player.volume = event.target.value;
|
|
774
|
+
},
|
|
775
|
+
'volume',
|
|
776
|
+
);
|
|
777
|
+
|
|
778
|
+
// Update controls.hover state (used for ui.toggleControls to avoid hiding when interacting)
|
|
779
|
+
this.bind(elements.controls, 'mouseenter mouseleave', (event) => {
|
|
780
|
+
elements.controls.hover = !player.touch && event.type === 'mouseenter';
|
|
781
|
+
});
|
|
782
|
+
|
|
783
|
+
// Also update controls.hover state for any non-player children of fullscreen element (as above)
|
|
784
|
+
if (elements.fullscreen) {
|
|
785
|
+
Array.from(elements.fullscreen.children)
|
|
786
|
+
.filter(c => !c.contains(elements.container))
|
|
787
|
+
.forEach((child) => {
|
|
788
|
+
this.bind(child, 'mouseenter mouseleave', (event) => {
|
|
789
|
+
if (elements.controls) {
|
|
790
|
+
elements.controls.hover = !player.touch && event.type === 'mouseenter';
|
|
791
|
+
}
|
|
792
|
+
});
|
|
793
|
+
});
|
|
794
|
+
}
|
|
795
|
+
|
|
796
|
+
// Update controls.pressed state (used for ui.toggleControls to avoid hiding when interacting)
|
|
797
|
+
this.bind(elements.controls, 'mousedown mouseup touchstart touchend touchcancel', (event) => {
|
|
798
|
+
elements.controls.pressed = ['mousedown', 'touchstart'].includes(event.type);
|
|
799
|
+
});
|
|
800
|
+
|
|
801
|
+
// Show controls when they receive focus (e.g., when using keyboard tab key)
|
|
802
|
+
this.bind(elements.controls, 'focusin', () => {
|
|
803
|
+
const { config, timers } = player;
|
|
804
|
+
|
|
805
|
+
// Skip transition to prevent focus from scrolling the parent element
|
|
806
|
+
toggleClass(elements.controls, config.classNames.noTransition, true);
|
|
807
|
+
|
|
808
|
+
// Toggle
|
|
809
|
+
ui.toggleControls.call(player, true);
|
|
810
|
+
|
|
811
|
+
// Restore transition
|
|
812
|
+
setTimeout(() => {
|
|
813
|
+
toggleClass(elements.controls, config.classNames.noTransition, false);
|
|
814
|
+
}, 0);
|
|
815
|
+
|
|
816
|
+
// Delay a little more for mouse users
|
|
817
|
+
const delay = this.touch ? 3000 : 4000;
|
|
818
|
+
|
|
819
|
+
// Clear timer
|
|
820
|
+
clearTimeout(timers.controls);
|
|
821
|
+
|
|
822
|
+
// Hide again after delay
|
|
823
|
+
timers.controls = setTimeout(() => ui.toggleControls.call(player, false), delay);
|
|
824
|
+
});
|
|
825
|
+
|
|
826
|
+
// Mouse wheel for volume
|
|
827
|
+
this.bind(
|
|
828
|
+
elements.inputs.volume,
|
|
829
|
+
'wheel',
|
|
830
|
+
(event) => {
|
|
831
|
+
// Detect "natural" scroll - supported on OS X Safari only
|
|
832
|
+
// Other browsers on OS X will be inverted until support improves
|
|
833
|
+
const inverted = event.webkitDirectionInvertedFromDevice;
|
|
834
|
+
// Get delta from event. Invert if `inverted` is true
|
|
835
|
+
const [x, y] = [event.deltaX, -event.deltaY].map(value => (inverted ? -value : value));
|
|
836
|
+
// Using the biggest delta, normalize to 1 or -1 (or 0 if no delta)
|
|
837
|
+
const direction = Math.sign(Math.abs(x) > Math.abs(y) ? x : y);
|
|
838
|
+
|
|
839
|
+
// Change the volume by 2%
|
|
840
|
+
player.increaseVolume(direction / 50);
|
|
841
|
+
|
|
842
|
+
// Don't break page scrolling at max and min
|
|
843
|
+
const { volume } = player.media;
|
|
844
|
+
if ((direction === 1 && volume < 1) || (direction === -1 && volume > 0)) {
|
|
845
|
+
event.preventDefault();
|
|
846
|
+
}
|
|
847
|
+
},
|
|
848
|
+
'volume',
|
|
849
|
+
false,
|
|
850
|
+
);
|
|
851
|
+
};
|
|
852
|
+
}
|
|
853
|
+
|
|
854
|
+
export default Listeners;
|