mirror of
https://github.com/jellyfin/jellyfin-web
synced 2025-03-30 19:56:21 +00:00
171 lines
5.7 KiB
JavaScript
171 lines
5.7 KiB
JavaScript
|
|
import { playbackManager } from '../playback/playbackmanager';
|
|
import layoutManager from '../layoutManager';
|
|
import template from './subtitlesync.template.html';
|
|
import './subtitlesync.scss';
|
|
|
|
let player;
|
|
let subtitleSyncSlider;
|
|
let subtitleSyncTextField;
|
|
let subtitleSyncCloseButton;
|
|
let subtitleSyncContainer;
|
|
|
|
function init(instance) {
|
|
const parent = document.createElement('div');
|
|
document.body.appendChild(parent);
|
|
parent.innerHTML = template;
|
|
|
|
subtitleSyncSlider = parent.querySelector('.subtitleSyncSlider');
|
|
subtitleSyncTextField = parent.querySelector('.subtitleSyncTextField');
|
|
subtitleSyncCloseButton = parent.querySelector('.subtitleSync-closeButton');
|
|
subtitleSyncContainer = parent.querySelector('.subtitleSyncContainer');
|
|
|
|
if (layoutManager.tv) {
|
|
subtitleSyncSlider.classList.add('focusable');
|
|
// HACK: Delay to give time for registered element attach (Firefox)
|
|
setTimeout(function () {
|
|
subtitleSyncSlider.enableKeyboardDragging();
|
|
}, 0);
|
|
}
|
|
|
|
subtitleSyncContainer.classList.add('hide');
|
|
|
|
subtitleSyncTextField.updateOffset = function (offset) {
|
|
this.textContent = offset + 's';
|
|
};
|
|
|
|
subtitleSyncTextField.addEventListener('click', function () {
|
|
// keep focus to prevent fade with osd
|
|
this.hasFocus = true;
|
|
});
|
|
|
|
subtitleSyncTextField.addEventListener('keydown', function (event) {
|
|
if (event.key === 'Enter') {
|
|
// if input key is enter search for float pattern
|
|
let inputOffset = /[-+]?\d+\.?\d*/g.exec(this.textContent);
|
|
if (inputOffset) {
|
|
inputOffset = inputOffset[0];
|
|
inputOffset = parseFloat(inputOffset);
|
|
|
|
subtitleSyncSlider.updateOffset(inputOffset);
|
|
} else {
|
|
this.textContent = (playbackManager.getPlayerSubtitleOffset(player) || 0) + 's';
|
|
}
|
|
this.hasFocus = false;
|
|
event.preventDefault();
|
|
} else {
|
|
// keep focus to prevent fade with osd
|
|
this.hasFocus = true;
|
|
if (event.key.match(/[+-\d.s]/) === null) {
|
|
event.preventDefault();
|
|
}
|
|
}
|
|
|
|
// FIXME: TV layout will require special handling for navigation keys. But now field is not focusable
|
|
event.stopPropagation();
|
|
});
|
|
|
|
subtitleSyncTextField.blur = function () {
|
|
// prevent textfield to blur while element has focus
|
|
if (!this.hasFocus && this.prototype) {
|
|
this.prototype.blur();
|
|
}
|
|
};
|
|
|
|
function updateSubtitleOffset() {
|
|
const value = parseFloat(subtitleSyncSlider.value);
|
|
// set new offset
|
|
playbackManager.setSubtitleOffset(value, player);
|
|
// synchronize with textField value
|
|
subtitleSyncTextField.updateOffset(value);
|
|
}
|
|
|
|
subtitleSyncSlider.updateOffset = function (sliderValue) {
|
|
// default value is 0s = 0ms
|
|
this.value = sliderValue === undefined ? 0 : sliderValue;
|
|
|
|
updateSubtitleOffset();
|
|
};
|
|
|
|
subtitleSyncSlider.addEventListener('change', () => updateSubtitleOffset());
|
|
|
|
subtitleSyncSlider.getBubbleHtml = function (_, value) {
|
|
return '<h1 class="sliderBubbleText">'
|
|
+ (value > 0 ? '+' : '') + parseFloat(value) + 's'
|
|
+ '</h1>';
|
|
};
|
|
|
|
subtitleSyncCloseButton.addEventListener('click', function () {
|
|
playbackManager.disableShowingSubtitleOffset(player);
|
|
SubtitleSync.prototype.toggle('forceToHide');
|
|
});
|
|
|
|
instance.element = parent;
|
|
}
|
|
|
|
class SubtitleSync {
|
|
constructor(currentPlayer) {
|
|
player = currentPlayer;
|
|
init(this);
|
|
}
|
|
|
|
destroy() {
|
|
SubtitleSync.prototype.toggle('forceToHide');
|
|
if (player) {
|
|
playbackManager.disableShowingSubtitleOffset(player);
|
|
playbackManager.setSubtitleOffset(0, player);
|
|
}
|
|
const elem = this.element;
|
|
if (elem) {
|
|
elem.parentNode.removeChild(elem);
|
|
this.element = null;
|
|
}
|
|
}
|
|
|
|
toggle(action) {
|
|
if (action && !['hide', 'forceToHide'].includes(action)) {
|
|
console.warn('SubtitleSync.toggle called with invalid action', action);
|
|
return;
|
|
}
|
|
|
|
if (player && playbackManager.supportSubtitleOffset(player)) {
|
|
if (!action) {
|
|
// if showing subtitle sync is enabled and if there is an external subtitle stream enabled
|
|
if (playbackManager.isShowingSubtitleOffsetEnabled(player) && playbackManager.canHandleOffsetOnCurrentSubtitle(player)) {
|
|
// if no subtitle offset is defined or element has focus (offset being defined)
|
|
if (!(playbackManager.getPlayerSubtitleOffset(player) || subtitleSyncTextField.hasFocus)) {
|
|
// set default offset to '0' = 0ms
|
|
subtitleSyncSlider.value = '0';
|
|
subtitleSyncTextField.textContent = '0s';
|
|
playbackManager.setSubtitleOffset(0, player);
|
|
}
|
|
// show subtitle sync
|
|
subtitleSyncContainer.classList.remove('hide');
|
|
return;
|
|
}
|
|
} else if (action === 'hide' && subtitleSyncTextField.hasFocus) {
|
|
// do not hide if element has focus
|
|
return;
|
|
}
|
|
|
|
subtitleSyncContainer.classList.add('hide');
|
|
}
|
|
}
|
|
|
|
update(offset) {
|
|
this.toggle();
|
|
|
|
const value = parseFloat(subtitleSyncSlider.value) + offset;
|
|
subtitleSyncSlider.updateOffset(value);
|
|
}
|
|
|
|
incrementOffset() {
|
|
this.update(+subtitleSyncSlider.step);
|
|
}
|
|
|
|
decrementOffset() {
|
|
this.update(-subtitleSyncSlider.step);
|
|
}
|
|
}
|
|
|
|
export default SubtitleSync;
|