04fbde49ec
Co-authored-by: import <>
158 lines
3.7 KiB
JavaScript
158 lines
3.7 KiB
JavaScript
/**
|
|
* @file touchOverlay.js
|
|
* Touch UI component
|
|
*/
|
|
|
|
import videojs from 'video.js/dist/alt/video.core.novtt.min.js';
|
|
import window from 'global/window';
|
|
|
|
const Component = videojs.getComponent('Component');
|
|
const dom = videojs.dom || videojs;
|
|
|
|
/**
|
|
* The `TouchOverlay` is an overlay to capture tap events.
|
|
*
|
|
* @extends Component
|
|
*/
|
|
class TouchOverlay extends Component {
|
|
/**
|
|
* Creates an instance of the this class.
|
|
*
|
|
* @param {Player} player
|
|
* The `Player` that this class should be attached to.
|
|
*
|
|
* @param {Object} [options]
|
|
* The key/value store of player options.
|
|
*/
|
|
constructor(player, options) {
|
|
super(player, options);
|
|
|
|
this.seekSeconds = options.seekSeconds;
|
|
this.tapTimeout = options.tapTimeout;
|
|
|
|
// Add play toggle overlay
|
|
this.addChild('playToggle', {});
|
|
|
|
// Clear overlay when playback starts or with control fade
|
|
player.on(['playing', 'userinactive'], e => {
|
|
if (!this.player_.paused()) {
|
|
this.removeClass('show-play-toggle');
|
|
}
|
|
});
|
|
|
|
// A 0 inactivity timeout won't work here
|
|
if (this.player_.options_.inactivityTimeout === 0) {
|
|
this.player_.options_.inactivityTimeout = 5000;
|
|
}
|
|
|
|
this.enable();
|
|
}
|
|
|
|
/**
|
|
* Builds the DOM element.
|
|
*
|
|
* @return {Element}
|
|
* The DOM element.
|
|
*/
|
|
createEl() {
|
|
const el = dom.createEl('div', {
|
|
className: 'vjs-touch-overlay',
|
|
// Touch overlay is not tabbable.
|
|
tabIndex: -1,
|
|
});
|
|
|
|
return el;
|
|
}
|
|
|
|
/**
|
|
* Debounces to either handle a delayed single tap, or a double tap
|
|
*
|
|
* @param {Event} event
|
|
* The touch event
|
|
*
|
|
*/
|
|
handleTap(event) {
|
|
// Don't handle taps on the play button
|
|
if (event.target !== this.el_) {
|
|
return;
|
|
}
|
|
|
|
event.preventDefault();
|
|
|
|
if (this.firstTapCaptured) {
|
|
this.firstTapCaptured = false;
|
|
if (this.timeout) {
|
|
window.clearTimeout(this.timeout);
|
|
}
|
|
this.handleDoubleTap(event);
|
|
} else {
|
|
this.firstTapCaptured = true;
|
|
this.timeout = window.setTimeout(() => {
|
|
this.firstTapCaptured = false;
|
|
this.handleSingleTap(event);
|
|
}, this.tapTimeout);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Toggles display of play toggle
|
|
*
|
|
* @param {Event} event
|
|
* The touch event
|
|
*
|
|
*/
|
|
handleSingleTap(event) {
|
|
this.removeClass('skip');
|
|
this.toggleClass('show-play-toggle');
|
|
}
|
|
|
|
/**
|
|
* Seeks by configured number of seconds if left or right part of video double tapped
|
|
*
|
|
* @param {Event} event
|
|
* The touch event
|
|
*
|
|
*/
|
|
handleDoubleTap(event) {
|
|
const rect = this.el_.getBoundingClientRect();
|
|
const x = event.changedTouches[0].clientX - rect.left;
|
|
|
|
// Check if double tap is in left or right area
|
|
if (x < rect.width * 0.4) {
|
|
this.player_.currentTime(Math.max(0, this.player_.currentTime() - this.seekSeconds));
|
|
this.addClass('reverse');
|
|
} else if (x > rect.width - rect.width * 0.4) {
|
|
this.player_.currentTime(Math.min(this.player_.duration(), this.player_.currentTime() + this.seekSeconds));
|
|
this.removeClass('reverse');
|
|
} else {
|
|
return;
|
|
}
|
|
|
|
// Remove play toggle if showing
|
|
this.removeClass('show-play-toggle');
|
|
|
|
// Remove and readd class to trigger animation
|
|
this.removeClass('skip');
|
|
window.requestAnimationFrame(() => {
|
|
this.addClass('skip');
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Enables touch handler
|
|
*/
|
|
enable() {
|
|
this.firstTapCaptured = false;
|
|
this.on('touchend', this.handleTap);
|
|
}
|
|
|
|
/**
|
|
* Disables touch handler
|
|
*/
|
|
disable() {
|
|
this.off('touchend', this.handleTap);
|
|
}
|
|
}
|
|
|
|
Component.registerComponent('TouchOverlay', TouchOverlay);
|
|
export default TouchOverlay;
|