1
0
Fork 0
mirror of https://github.com/FrankerFaceZ/FrankerFaceZ.git synced 2025-06-27 21:05:53 +00:00
FrankerFaceZ/src/sites/twitch-twilight/modules/player.jsx
SirStendec 1c0f7edab5 4.12.3
* Changed: Prevent the video quality from dropping when using Picture-in-Picture. (This is handled via an override to the Page Visibility API, preventing the player from detecting the page is hidden when PIP is active. This override *only* functions when Picture-in-Picture is in use.)
* Fixed: Recalculate the position of the navigation bar underline when showing/hiding the Discover link.
* Fixed: Don't load our color normalization CSS if the user isn't making use of a custom theme.
2019-09-28 23:52:50 -04:00

929 lines
No EOL
25 KiB
JavaScript

'use strict';
// ============================================================================
// Twitch Player
// ============================================================================
import Module from 'utilities/module';
import {createElement, on, off} from 'utilities/dom';
export const PLAYER_ROUTES = ['front-page', 'user', 'video', 'user-video', 'user-clip', 'user-videos', 'user-clips', 'user-collections', 'user-events', 'user-followers', 'user-following', 'dash'];
const STYLE_VALIDATOR = createElement('span');
export default class Player extends Module {
constructor(...args) {
super(...args);
this.should_enable = true;
this.inject('settings');
this.inject('site.fine');
this.inject('site.web_munch');
this.inject('site.css_tweaks');
this.inject('site.router');
this.inject('i18n');
this.Player = this.fine.define(
'twitch-player',
n => n.player && n.onPlayerReady,
PLAYER_ROUTES
);
this.PersistentPlayer = this.fine.define(
'twitch-player-persistent',
n => n.renderMiniHoverControls && n.togglePause,
PLAYER_ROUTES
);
this.SquadStreamBar = this.fine.define(
'squad-stream-bar',
n => n.shouldRenderSquadBanner && n.props && n.props.triggerPlayerReposition,
PLAYER_ROUTES
);
this.settings.add('player.volume-scroll', {
default: false,
ui: {
path: 'Player > General >> Volume',
title: 'Adjust volume by scrolling with the mouse wheel.',
description: '*This setting will not work properly on streams with visible extensions when mouse interaction with extensions is allowed.*',
component: 'setting-check-box'
},
changed: val => {
for(const inst of this.Player.instances)
this.updateVolumeScroll(inst, val);
}
});
this.settings.add('player.button.reset', {
default: true,
ui: {
path: 'Player > General >> General',
title: 'Add a `Reset Player` button to the player controls.',
description: "Double-clicking the Reset Player button destroys and recreates Twitch's player, potentially fixing playback issues without a full page refresh.",
component: 'setting-check-box'
},
changed: () => {
for(const inst of this.Player.instances)
this.addResetButton(inst);
}
});
if ( document.pictureInPictureEnabled )
this.settings.add('player.button.pip', {
default: true,
ui: {
path: 'Player > General >> General',
title: 'Add a `Picture-in-Picture` button to the player controls.',
description: "Clicking the PiP button attempts to toggle Picture-in-Picture mode for the player's video.",
component: 'setting-check-box'
},
changed: () => {
for(const inst of this.Player.instances)
this.addPiPButton(inst);
}
});
this.settings.add('player.volume-scroll-steps', {
default: 0.1,
ui: {
path: 'Player > General >> Volume',
title: 'Volume scroll amount',
description: 'How much the volume level is changed per individual scroll input.',
component: 'setting-select-box',
data: [
{value: 0.1, title: '10%'},
{value: 0.05, title: '5%'},
{value: 0.02, title: '2%'},
{value: 0.01, title: '1%'}
]
}
});
this.settings.add('player.captions.font-size', {
default: '',
ui: {
path: 'Player > Closed Captioning >> Font',
title: 'Font Size',
description: 'How large should captions be. This can be a percentage, such as `10%`, or a pixel value, such as `50px`.',
component: 'setting-text-box'
},
changed: () => this.updateCaptionsCSS()
});
this.settings.add('player.captions.font-family', {
default: '',
ui: {
path: 'Player > Closed Captioning >> Font',
title: 'Font Family',
description: 'Override the font used for displaying Closed Captions.',
component: 'setting-text-box'
},
changed: () => this.updateCaptionsCSS()
});
this.settings.add('player.captions.custom-position', {
default: false,
ui: {
path: 'Player > Closed Captioning >> Position',
sort: -1,
title: 'Enable overriding the position and alignment of closed captions.',
component: 'setting-check-box'
},
changed: () => this.updateCaptionsCSS()
});
this.settings.add('player.captions.vertical', {
default: '10%',
ui: {
path: 'Player > Closed Captioning >> Position',
title: 'Vertical Position',
component: 'setting-text-box',
description: 'Override the position for Closed Captions. This can be a percentage, such as `10%`, or a pixel value, such as `50px`.'
},
changed: () => this.updateCaptionsCSS()
});
this.settings.add('player.captions.horizontal', {
default: '50%',
ui: {
path: 'Player > Closed Captioning >> Position',
title: 'Horizontal Position',
component: 'setting-text-box',
description: 'Override the position for Closed Captions. This can be a percentage, such as `10%`, or a pixel value, such as `50px`.'
},
changed: () => this.updateCaptionsCSS()
});
this.settings.add('player.captions.alignment', {
default: 32,
ui: {
path: 'Player > Closed Captioning >> Position',
title: 'Alignment',
component: 'setting-select-box',
data: [
{value: 11, title: 'Top Left'},
{value: 12, title: 'Top Center'},
{value: 13, title: 'Top Right'},
{value: 21, title: 'Middle Left'},
{value: 22, title: 'Middle Center'},
{value: 23, title: 'Middle Right'},
{value: 31, title: 'Bottom Left'},
{value: 32, title: 'Bottom Center'},
{value: 33, title: 'Bottom Right'}
]
},
changed: () => this.updateCaptionsCSS()
});
this.settings.add('player.theatre.no-whispers', {
default: false,
requires: ['whispers.show'],
process(ctx, val) {
if ( ! ctx.get('whispers.show') )
return true;
return val;
},
ui: {
path: 'Player > General >> Theatre Mode',
title: 'Hide whispers when Theatre Mode is enabled.',
component: 'setting-check-box'
},
changed: val => this.css_tweaks.toggle('theatre-no-whispers', val)
});
this.settings.add('player.theatre.metadata', {
default: false,
ui: {
path: 'Player > General >> Theatre Mode',
title: 'Show metadata when mousing over the player.',
component: 'setting-check-box'
},
changed: val => this.css_tweaks.toggle('theatre-metadata', val)
});
this.settings.add('player.theatre.auto-enter', {
default: false,
ui: {
path: 'Player > General >> Theatre Mode',
title: 'Automatically open Theatre Mode when visiting a channel.',
component: 'setting-check-box'
}
});
this.settings.add('player.ext-hide', {
default: 0,
ui: {
path: 'Player > General >> Extensions',
title: 'Show Overlay Extensions',
component: 'setting-select-box',
data: [
{value: 2, title: 'Never'},
{value: 1, title: 'With Controls'},
{value: 0, title: 'Always'}
]
},
changed: val => this.updateHideExtensions(val)
});
this.settings.add('player.ext-interaction', {
default: true,
ui: {
path: 'Player > General >> Extensions',
title: 'Allow mouse interaction with overlay extensions.',
component: 'setting-check-box'
},
changed: val => this.css_tweaks.toggle('player-ext-mouse', !val)
})
this.settings.add('player.home.autoplay', {
default: true,
ui: {
path: 'Player > General >> Playback',
title: 'Auto-play featured broadcasters on the front page.',
component: 'setting-check-box'
},
});
this.settings.add('player.no-autoplay', {
default: false,
ui: {
path: 'Player > General >> Playback',
title: 'Do not automatically start playing videos or streams.',
description: 'Note: This feature does not apply when navigating directly from channel to channel.',
component: 'setting-check-box'
}
});
this.settings.add('player.vod.autoplay', {
default: true,
ui: {
path: 'Player > General >> Playback',
title: 'Auto-play the next recommended video after a video finishes.',
component: 'setting-check-box'
}
});
this.settings.add('player.volume-always-shown', {
default: false,
ui: {
path: 'Player > General >> Volume',
title: 'Keep the volume slider expanded at all times.',
component: 'setting-check-box'
},
changed: val => this.css_tweaks.toggle('player-volume', val)
});
this.settings.add('player.hide-event-bar', {
default: false,
ui: {
path: 'Channel > Appearance >> General',
title: 'Hide the Event Bar',
description: 'Hide the Event Bar which appears above the player when there is an ongoing event for the current channel.',
component: 'setting-check-box'
},
changed: val => {
this.css_tweaks.toggleHide('player-event-bar', val);
this.PersistentPlayer.forceUpdate();
}
});
this.settings.add('player.hide-rerun-bar', {
default: false,
ui: {
path: 'Channel > Appearance >> General',
title: 'Hide the Rerun Bar',
description: 'Hide the Rerun Bar which appears above the player when the current channel is playing a video rather than live content.',
component: 'setting-check-box'
},
changed: val => {
this.css_tweaks.toggleHide('player-rerun-bar', val);
this.PersistentPlayer.forceUpdate();
}
});
this.settings.add('player.hide-squad-banner', {
default: false,
ui: {
path: 'Channel > Appearance >> General',
title: 'Hide the Squad Streaming Bar',
component: 'setting-check-box'
},
changed: () => this.SquadStreamBar.forceUpdate()
});
this.settings.add('player.hide-mouse', {
default: true,
ui: {
path: 'Player > General >> General',
title: "Hide mouse when controls aren't visible.",
component: 'setting-check-box'
},
changed: val => this.css_tweaks.toggle('player-hide-mouse', val)
});
}
updateHideExtensions(val) {
if ( val === undefined )
val = this.settings.get('player.ext-hide');
this.css_tweaks.toggleHide('player-ext-hover', val === 1);
this.css_tweaks.toggleHide('player-ext', val === 2);
}
updateCaptionsCSS() {
// Font
const font_size = this.settings.get('player.captions.font-size');
let font_family = this.settings.get('player.captions.font-family');
if ( font_family.indexOf(' ') !== -1 && font_family.indexOf(',') === -1 && font_family.indexOf('"') === -1 && font_family.indexOf("'") === -1 )
font_family = `"${font_family}"`;
STYLE_VALIDATOR.style.fontSize = '';
STYLE_VALIDATOR.style.fontFamily = '';
STYLE_VALIDATOR.style.fontSize = font_size;
STYLE_VALIDATOR.style.fontFamily = font_family;
const font_out = [];
if ( STYLE_VALIDATOR.style.fontFamily )
font_out.push(`font-family: ${STYLE_VALIDATOR.style.fontFamily} !important;`);
if ( STYLE_VALIDATOR.style.fontSize )
font_out.push(`font-size: ${STYLE_VALIDATOR.style.fontSize} !important;`);
if ( font_out.length )
this.css_tweaks.set('captions-font', `.player-captions {
${font_out.join('\n\t')}
}`)
else
this.css_tweaks.delete('captions-font');
// Position
const enabled = this.settings.get('player.captions.custom-position'),
vertical = this.settings.get('player.captions.vertical'),
horizontal = this.settings.get('player.captions.horizontal'),
alignment = this.settings.get('player.captions.alignment');
if ( ! enabled ) {
this.css_tweaks.delete('captions-position');
return;
}
const out = [],
align_horizontal = alignment % 10,
align_vertical = Math.floor(alignment / 10);
let custom_top = false,
custom_left = false;
STYLE_VALIDATOR.style.top = '';
STYLE_VALIDATOR.style.top = vertical;
if ( STYLE_VALIDATOR.style.top ) {
out.push(`${align_vertical === 3 ? 'bottom' : 'top'}: ${STYLE_VALIDATOR.style.top} !important;`)
out.push(`${align_vertical === 3 ? 'top' : 'bottom'}: unset !important;`);
custom_top = true;
}
STYLE_VALIDATOR.style.top = '';
STYLE_VALIDATOR.style.top = horizontal;
if ( STYLE_VALIDATOR.style.top ) {
out.push(`${align_horizontal === 3 ? 'right' : 'left'}: ${STYLE_VALIDATOR.style.top} !important;`);
out.push(`${align_horizontal === 3 ? 'left' : 'right'}: unset !important;`);
custom_left = true;
}
if ( align_horizontal !== 2 )
out.push(`width: unset !important;`);
out.push(`transform: translate(${(!custom_left || align_horizontal === 2) ? '-50%' : '0'}, ${(!custom_top || align_vertical === 2) ? '-50%' : '0'})`);
this.css_tweaks.set('captions-position', `.player-captions-container {
${out.join('\n\t')};
}`);
}
onEnable() {
this.css_tweaks.toggle('player-volume', this.settings.get('player.volume-always-shown'));
this.css_tweaks.toggle('player-ext-mouse', !this.settings.get('player.ext-interaction'));
this.css_tweaks.toggle('theatre-no-whispers', this.settings.get('player.theatre.no-whispers'));
this.css_tweaks.toggle('theatre-metadata', this.settings.get('player.theatre.metadata'));
this.css_tweaks.toggle('player-hide-mouse', this.settings.get('player.hide-mouse'));
this.css_tweaks.toggleHide('player-event-bar', this.settings.get('player.hide-event-bar'));
this.css_tweaks.toggleHide('player-rerun-bar', this.settings.get('player.hide-rerun-bar'));
this.updateHideExtensions();
this.updateCaptionsCSS();
this.installVisibilityHook();
const t = this;
this.SquadStreamBar.ready(cls => {
const old_should_render = cls.prototype.shouldRenderSquadBanner;
cls.prototype.shouldRenderSquadBanner = function(...args) {
if ( t.settings.get('player.hide-squad-banner') )
return false;
return old_should_render.call(this, ...args);
}
this.SquadStreamBar.forceUpdate();
this.updateSquadContext();
});
this.SquadStreamBar.on('mount', this.updateSquadContext, this);
this.SquadStreamBar.on('update', this.updateSquadContext, this);
this.SquadStreamBar.on('unmount', this.updateSquadContext, this);
this.Player.on('mount', this.onMount, this);
this.Player.on('unmount', this.onUnmount, this);
this.Player.ready((cls, instances) => {
const old_init = cls.prototype.initializePlayer;
if ( old_init ) {
cls.prototype.initializePlayer = function(...args) {
const ret = old_init.call(this, ...args);
t.process(this);
return ret;
}
} else {
this.Player.on('will-mount', this.overrideInitialize, this);
}
for(const inst of instances) {
if ( ! old_init )
this.overrideInitialize(inst);
this.onMount(inst);
this.process(inst);
}
});
this.on('i18n:update', () => {
for(const inst of this.Player.instances) {
this.addPiPButton(inst);
this.addResetButton(inst);
}
});
}
installVisibilityHook() {
if ( ! document.pictureInPictureEnabled ) {
this.log.info('Skipping visibility hook. Picture-in-Picture is not available.');
return;
}
try {
Object.defineProperty(document, 'hidden', {
configurable: true,
get() {
// If Picture in Picture is active, then we should not
// drop quality. Therefore, we need to trick Twitch
// into thinking the document is still active.
if ( document.pictureInPictureElement != null )
return false;
return document.visibilityState === 'hidden';
}
});
} catch(err) {
this.log.warning('Unable to install document visibility hook.', err);
}
}
updateSquadContext() {
this.settings.updateContext({
squad_bar: this.hasSquadBar()
});
}
hasSquadBar() {
const inst = this.SquadStreamBar.first;
return inst ? inst.shouldRenderSquadBanner(inst.props) : false
}
overrideInitialize(inst) {
const t = this,
old_init = inst.initializePlayer;
inst.initializePlayer = function(...args) {
const ret = old_init.call(inst, ...args);
t.process(inst);
return ret;
}
}
onMount(inst) {
if ( this.settings.get('player.theatre.auto-enter') && inst.onTheatreChange )
inst.onTheatreChange(true);
if ( this.settings.get('player.no-autoplay') || (! this.settings.get('player.home.autoplay') && this.router.current.name === 'front-page') ) {
if ( inst.player )
this.disableAutoplay(inst);
else {
const wrapped = inst.onPlayerReady;
inst.onPlayerReady = () => {
const ret = wrapped.call(inst);
this.disableAutoplay(inst);
return ret;
}
}
}
}
onUnmount(inst) { // eslint-disable-line class-methods-use-this
this.cleanup(inst);
}
process(inst) {
this.addResetButton(inst);
this.addPiPButton(inst);
this.addEndedListener(inst);
this.addStateTags(inst);
this.addControlVisibility(inst);
this.updateVolumeScroll(inst);
}
cleanup(inst) { // eslint-disable-line class-methods-use-this
const p = inst.player,
pr = inst.playerRef,
video = pr && pr.querySelector('video'),
reset = pr && pr.querySelector('.ffz--player-reset'),
pip = pr && pr.querySelector('.ffz--player-pip');
if ( reset )
reset.remove();
if ( pip )
pip.remove();
if ( video && video._ffz_pip_enter ) {
video.removeEventListener('enterpictureinpicture', video._ffz_pip_enter);
video.removeEventListener('leavepictureinpicture', video._ffz_pip_exit);
video._ffz_pip_enter = null;
video._ffz_pip_exit = null;
}
if ( inst._ffz_on_ended ) {
p && off(p, 'ended', inst._ffz_on_ended);
inst._ffz_on_ended = null;
}
if ( inst._ffz_visibility_handler ) {
if ( pr ) {
off(pr, 'mousemove', inst._ffz_visibility_handler);
off(pr, 'mouseleave', inst._ffz_visibility_handler);
}
inst._ffz_visibility_handler = null;
}
if ( inst._ffz_scroll_handler ) {
pr && off(pr, 'wheel', inst._ffz_scroll_handler);
inst._ffz_scroll_handler = null;
}
if ( inst._ffz_autoplay_handler ) {
if ( p ) {
off(p, 'play', inst._ffz_autoplay_handler);
off(p, 'playing', inst._ffz_autoplay_handler);
off(p, 'contentShowing', inst._ffz_autoplay_handler);
}
inst._ffz_autoplay_handler = null;
}
if ( inst._ffz_on_state ) {
if ( p ) {
off(p, 'ended', inst._ffz_on_state);
off(p, 'pause', inst._ffz_on_state);
off(p, 'playing', inst._ffz_on_state);
off(p, 'error', inst._ffz_on_state);
}
inst._ffz_on_state = null;
}
}
addEndedListener(inst) {
let p = inst.player;
if ( ! p )
return;
if ( p.player )
p = p.player;
if ( inst._ffz_on_ended )
off(p, 'ended', inst._ffz_on_ended);
on(p, 'ended', inst._ffz_on_ended = async () => {
if ( this.settings.get('player.vod.autoplay') )
return;
try {
(await this.parent.awaitElement('.pl-rec__cancel', inst.playerRef, 1000)).click();
} catch(err) { /* do nothing~ */ }
});
}
addControlVisibility(inst) { // eslint-disable-line class-methods-use-this
const p = inst.playerRef;
if ( ! p )
return;
if ( inst._ffz_visibility_handler ) {
off(p, 'mousemove', inst._ffz_visibility_handler);
off(p, 'mouseleave', inst._ffz_visibility_handler);
}
let timer;
const c = () => { p.dataset.controls = false };
const f = inst._ffz_visibility_handler = e => {
clearTimeout(timer);
if ( e.type === 'mouseleave' )
return c();
timer = setTimeout(c, 5000);
p.dataset.controls = true;
};
on(p, 'mousemove', f);
on(p, 'mouseleave', f);
}
addStateTags(inst) {
let p = inst.player;
if ( ! p )
return;
if ( p.player )
p = p.player;
if ( inst._ffz_on_state ) {
off(p, 'ended', inst._ffz_on_state);
off(p, 'pause', inst._ffz_on_state);
off(p, 'playing', inst._ffz_on_state);
off(p, 'error', inst._ffz_on_state);
}
const f = inst._ffz_on_state = () => this.updateStateTags(inst);
on(p, 'ended', f);
on(p, 'pause', f);
on(p, 'playing', f);
on(p, 'error', f);
f();
}
updateStateTags(inst) { // eslint-disable-line class-methods-use-this
const p = inst.playerRef;
let player = inst.player;
if ( ! p || ! player )
return;
if ( player.player )
player = player.player;
p.dataset.ended = player.ended;
p.dataset.paused = player.paused;
}
disableAutoplay(inst) {
let p = inst.player;
if ( ! p )
return this.log.warn('disableAutoplay() called without Player');
if ( p.player )
p = p.player;
if ( p.readyState > 0 ) {
this.log.info('Player already playing. Pausing.');
return p.pause();
}
if ( ! inst._ffz_autoplay_handler ) {
const listener = inst._ffz_autoplay_handler = () => {
setTimeout(() => {
this.log.info('Pausing due to playback.');
inst._ffz_autoplay_handler = null;
p.pause();
setTimeout(() => {
off(p, 'play', listener);
off(p, 'playing', listener);
off(p, 'contentShowing', listener);
}, 250);
});
}
on(p, 'play', listener);
on(p, 'playing', listener);
on(p, 'contentShowing', listener);
}
}
updateVolumeScroll(inst, enabled) {
if ( enabled === undefined )
enabled = this.settings.get('player.volume-scroll');
const pr = inst.playerRef;
if ( ! pr )
return;
if ( ! enabled && inst._ffz_scroll_handler ) {
off(pr, 'wheel', inst._ffz_scroll_handler);
inst._ffz_scroll_handler = null;
} else if ( enabled && ! inst._ffz_scroll_handler ) {
on(pr, 'wheel', inst._ffz_scroll_handler = e => {
const delta = e.wheelDelta || -(e.deltaY || e.detail || 0);
let player = inst.player;
if ( player.player )
player = player.player;
if ( player ) {
const amount = this.settings.get('player.volume-scroll-steps'),
volume = Math.max(0, Math.min(1, player.getVolume() + (delta > 0 ? amount : -amount)));
player.setVolume(volume);
if ( volume !== 0 )
player.setMuted(false);
}
e.preventDefault();
return false;
});
}
}
addPiPButton(inst, tries = 0) {
const t = this,
el = inst.playerRef && inst.playerRef.querySelector('.player-buttons-right .pl-flex'),
container = el && el.parentElement,
has_pip = document.pictureInPictureEnabled && this.settings.get('player.button.pip');
if ( ! container ) {
if ( ! has_pip )
return;
if ( tries < 5 )
return setTimeout(this.addPiPButton.bind(this, inst, (tries||0) + 1), 250);
return this.log.warn('Unable to find container element for PiP Button');
}
let tip, btn = container.querySelector('.ffz--player-pip');
if ( ! has_pip ) {
if ( btn )
btn.remove();
return;
}
if ( ! btn ) {
btn = (<button
class="player-button player-button--pip ffz--player-pip ffz-i-window-restore"
type="button"
onClick={t.pipPlayer.bind(t, inst)} // eslint-disable-line react/jsx-no-bind
>
{tip = <span class="player-tip js-control-tip" />}
</button>);
container.insertBefore(btn, el.nextSibling);
} else
tip = btn.querySelector('.player-tip');
const pip_active = !!document.pictureInPictureElement
btn.classList.toggle('ffz-i-window-restore', ! pip_active);
btn.classList.toggle('ffz-i-window-maximize', pip_active);
tip.dataset.tip = this.i18n.t('player.pip_button', 'Click to Toggle Picture-in-Picture');
}
pipPlayer(inst) {
const video = inst.playerRef && inst.playerRef.querySelector('video');
if ( ! video || ! document.pictureInPictureEnabled )
return;
if ( ! video._ffz_pip_enter ) {
video.addEventListener('enterpictureinpicture', video._ffz_pip_enter = () => {
this.addPiPButton(inst);
});
video.addEventListener('leavepictureinpicture', video._ffz_pip_exit = () => {
this.addPiPButton(inst);
});
}
if ( document.pictureInPictureElement )
document.exitPictureInPicture();
else
video.requestPictureInPicture();
}
addResetButton(inst, tries = 0) {
const t = this,
el = inst.playerRef && inst.playerRef.querySelector('.player-buttons-right .pl-flex'),
container = el && el.parentElement;
if ( ! this.settings.get('player.button.reset') ) {
if ( container ) {
const btn = container.querySelector('.ffz--player-reset');
if ( btn )
btn.remove();
}
return;
}
if ( ! container ) {
if ( tries < 5 )
return setTimeout(this.addResetButton.bind(this, inst, (tries||0) + 1), 250);
return this.log.warn('Unable to find container element for Reset Button');
}
let tip = container.querySelector('.ffz--player-reset .player-tip');
if ( ! tip )
container.insertBefore(<button
class="player-button player-button--reset ffz--player-reset ffz-i-cancel"
type="button"
onDblClick={t.resetPlayer.bind(t, inst)} // eslint-disable-line react/jsx-no-bind
>
{tip = <span class="player-tip js-control-tip" />}
</button>, el.nextSibling);
tip.dataset.tip = this.i18n.t('player.reset_button', 'Double-Click to Reset Player');
}
resetPlayer(inst) {
// Player shutdown logic copied from componentWillUnmount
inst.checkPlayerDependencyAnimationFrame && cancelAnimationFrame(inst.checkPlayerDependencyAnimationFrame);
inst.checkPlayerDependencyAnimationFrame = null;
inst.maybeDetachFromWindow();
const ES = this.web_munch.getModule('extension-service');
if ( ES )
ES.extensionService.unregisterPlayer();
this.cleanup(inst);
const klass = inst.player.constructor;
inst.player.destroy();
inst.playerRef.innerHTML = '';
inst.initializePlayer(klass);
}
getInternalPlayer(inst) {
if ( ! inst )
inst = this.Player.first;
const node = this.fine.getChildNode(inst),
el = node && node.querySelector('.player-ui');
if ( ! el || ! el._reactRootContainer )
return null;
return this.fine.searchTree(el, n => n.props && n.props.player && n.context && n.context.store);
}
get current() {
// There should only ever be one player instance, but might change
// when they re-add support for the mini player.
for(const inst of this.Player.instances)
if ( inst && inst.player )
return inst.player;
return null;
}
}