@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.
Files changed (127) hide show
  1. package/.editorconfig +10 -0
  2. package/.gitpod.yml +6 -0
  3. package/.node-version +1 -0
  4. package/.prettierrc +7 -0
  5. package/.stickler.yml +5 -0
  6. package/.stylelintrc.json +26 -0
  7. package/CHANGELOG.md +16 -0
  8. package/CONTRIBUTING.md +34 -0
  9. package/CONTROLS.md +49 -0
  10. package/Dockerfile +32 -0
  11. package/LICENSE.md +22 -0
  12. package/README.md +194 -0
  13. package/cspell.json +48 -0
  14. package/dist/redxplyr.css +1 -0
  15. package/dist/redxplyr.js +8801 -0
  16. package/dist/redxplyr.min.js +2 -0
  17. package/dist/redxplyr.min.js.map +1 -0
  18. package/dist/redxplyr.min.mjs +1 -0
  19. package/dist/redxplyr.min.mjs.map +1 -0
  20. package/dist/redxplyr.mjs +8793 -0
  21. package/dist/redxplyr.polyfilled.js +9294 -0
  22. package/dist/redxplyr.polyfilled.min.js +2 -0
  23. package/dist/redxplyr.polyfilled.min.js.map +1 -0
  24. package/dist/redxplyr.polyfilled.min.mjs +1 -0
  25. package/dist/redxplyr.polyfilled.min.mjs.map +1 -0
  26. package/dist/redxplyr.polyfilled.mjs +9286 -0
  27. package/dist/redxplyr.svg +1 -0
  28. package/eslint.config.mjs +39 -0
  29. package/gulpfile.js +8 -0
  30. package/package.json +114 -0
  31. package/pnpm-workspace.yaml +8 -0
  32. package/src/js/captions.js +411 -0
  33. package/src/js/config/defaults.js +459 -0
  34. package/src/js/config/states.js +10 -0
  35. package/src/js/config/types.js +34 -0
  36. package/src/js/console.js +28 -0
  37. package/src/js/controls.js +1870 -0
  38. package/src/js/fullscreen.js +305 -0
  39. package/src/js/html5.js +148 -0
  40. package/src/js/listeners.js +854 -0
  41. package/src/js/media.js +61 -0
  42. package/src/js/plugins/ads.js +647 -0
  43. package/src/js/plugins/preview-thumbnails.js +706 -0
  44. package/src/js/plugins/vimeo.js +443 -0
  45. package/src/js/plugins/youtube.js +451 -0
  46. package/src/js/plyr.d.ts +729 -0
  47. package/src/js/plyr.js +1291 -0
  48. package/src/js/plyr.polyfilled.js +13 -0
  49. package/src/js/source.js +155 -0
  50. package/src/js/storage.js +70 -0
  51. package/src/js/support.js +100 -0
  52. package/src/js/ui.js +297 -0
  53. package/src/js/utils/animation.js +33 -0
  54. package/src/js/utils/arrays.js +23 -0
  55. package/src/js/utils/browser.js +21 -0
  56. package/src/js/utils/elements.js +263 -0
  57. package/src/js/utils/events.js +116 -0
  58. package/src/js/utils/fetch.js +45 -0
  59. package/src/js/utils/i18n.js +47 -0
  60. package/src/js/utils/is.js +81 -0
  61. package/src/js/utils/load-image.js +19 -0
  62. package/src/js/utils/load-script.js +14 -0
  63. package/src/js/utils/load-sprite.js +77 -0
  64. package/src/js/utils/numbers.js +17 -0
  65. package/src/js/utils/objects.js +43 -0
  66. package/src/js/utils/promise.js +14 -0
  67. package/src/js/utils/strings.js +80 -0
  68. package/src/js/utils/style.js +148 -0
  69. package/src/js/utils/time.js +36 -0
  70. package/src/js/utils/urls.js +40 -0
  71. package/src/sass/base.scss +69 -0
  72. package/src/sass/components/badges.scss +12 -0
  73. package/src/sass/components/captions.scss +58 -0
  74. package/src/sass/components/control.scss +52 -0
  75. package/src/sass/components/controls.scss +65 -0
  76. package/src/sass/components/menus.scss +205 -0
  77. package/src/sass/components/poster.scss +27 -0
  78. package/src/sass/components/progress.scss +107 -0
  79. package/src/sass/components/sliders.scss +99 -0
  80. package/src/sass/components/times.scss +20 -0
  81. package/src/sass/components/tooltips.scss +91 -0
  82. package/src/sass/components/volume.scss +18 -0
  83. package/src/sass/lib/animation.scss +31 -0
  84. package/src/sass/lib/css-vars.scss +103 -0
  85. package/src/sass/lib/functions.scss +3 -0
  86. package/src/sass/lib/mixins.scss +82 -0
  87. package/src/sass/plugins/ads.scss +53 -0
  88. package/src/sass/plugins/preview-thumbnails/index.scss +121 -0
  89. package/src/sass/plugins/preview-thumbnails/settings.scss +17 -0
  90. package/src/sass/plyr.scss +46 -0
  91. package/src/sass/settings/badges.scss +7 -0
  92. package/src/sass/settings/breakpoints.scss +9 -0
  93. package/src/sass/settings/captions.scss +10 -0
  94. package/src/sass/settings/colors.scss +18 -0
  95. package/src/sass/settings/controls.scss +30 -0
  96. package/src/sass/settings/cosmetics.scss +5 -0
  97. package/src/sass/settings/helpers.scss +7 -0
  98. package/src/sass/settings/menus.scss +13 -0
  99. package/src/sass/settings/progress.scss +18 -0
  100. package/src/sass/settings/sliders.scss +39 -0
  101. package/src/sass/settings/tooltips.scss +11 -0
  102. package/src/sass/settings/type.scss +16 -0
  103. package/src/sass/states/fullscreen.scss +15 -0
  104. package/src/sass/types/audio.scss +61 -0
  105. package/src/sass/types/video.scss +170 -0
  106. package/src/sass/utils/animation.scss +7 -0
  107. package/src/sass/utils/hidden.scss +28 -0
  108. package/src/sprite/plyr-airplay.svg +8 -0
  109. package/src/sprite/plyr-captions-off.svg +7 -0
  110. package/src/sprite/plyr-captions-on.svg +7 -0
  111. package/src/sprite/plyr-download.svg +8 -0
  112. package/src/sprite/plyr-enter-fullscreen.svg +4 -0
  113. package/src/sprite/plyr-exit-fullscreen.svg +4 -0
  114. package/src/sprite/plyr-fast-forward.svg +3 -0
  115. package/src/sprite/plyr-logo-vimeo.svg +6 -0
  116. package/src/sprite/plyr-logo-youtube.svg +6 -0
  117. package/src/sprite/plyr-muted.svg +8 -0
  118. package/src/sprite/plyr-pause.svg +8 -0
  119. package/src/sprite/plyr-pip.svg +6 -0
  120. package/src/sprite/plyr-play.svg +5 -0
  121. package/src/sprite/plyr-restart.svg +5 -0
  122. package/src/sprite/plyr-rewind.svg +3 -0
  123. package/src/sprite/plyr-settings.svg +5 -0
  124. package/src/sprite/plyr-volume.svg +11 -0
  125. package/tasks/build.js +226 -0
  126. package/tasks/deploy.js +216 -0
  127. 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;