lbry-desktop/ui/component/viewers/videoViewer/internal/plugins/videojs-hls-quality-selector/plugin.js

403 lines
12 KiB
JavaScript
Raw Normal View History

2021-02-18 13:45:39 -05:00
/* eslint-disable */
import videojs from 'video.js';
2021-02-18 13:45:39 -05:00
import { version as VERSION } from './package.json';
import ConcreteButton from './ConcreteButton';
import ConcreteMenuItem from './ConcreteMenuItem';
2022-04-20 15:40:21 -03:00
import * as QUALITY_OPTIONS from 'constants/player';
// Default options for the plugin.
const defaults = {};
// Cross-compatibility for Video.js 5 and 6.
const registerPlugin = videojs.registerPlugin || videojs.plugin;
/**
* VideoJS HLS Quality Selector Plugin class.
*/
class HlsQualitySelectorPlugin {
/**
* Plugin Constructor.
*
* @param {Player} player - The videojs player instance.
* @param {Object} options - The plugin options.
*/
constructor(player, options) {
this.player = player;
this.config = options;
2021-02-17 19:32:05 -08:00
// Ensure dependencies are met
if (!this.player.qualityLevels) {
2021-02-17 19:32:05 -08:00
console.error(`Error: Missing video.js quality levels plugin (required) - videojs-hls-quality-selector`);
return;
}
this.setupPlugin();
}
setupPlugin() {
// Create the quality button.
this.createQualityButton();
2021-03-08 03:35:04 -08:00
// Hide quality selector by default
this._qualityButton.hide();
// Bind event listeners
this.bindPlayerEvents();
// Listen for source changes
this.player.on('loadedmetadata', (e) => {
2022-04-26 21:00:34 -03:00
// if there was a quality option selected to default to, set it using the setQuality function
// as if it was being clicked on, on loadedmetadata
if (this.player.qualityToSet && !this.player.switchedFromDefaultQuality) {
this.setQuality(this.player.qualityToSet);
// Add this attribute to the video player so later it can be checked and avoid switching again
// Since this is only for initial load, based on the default quality setting
this.player.switchedFromDefaultQuality = true;
}
this.updatePlugin();
});
}
updatePlugin() {
if (this.player.claimSrcVhs || this.player.isLivestream) {
this._qualityButton.show();
} else {
this._qualityButton.hide();
}
}
/**
* Deprecated, returns VHS plugin
*
* @return {*} - videojs-http-streaming plugin.
*/
getHls() {
2022-04-11 18:20:50 +08:00
console.warn('hls-quality-selector: WARN: Using getHls options is deprecated. Use getVhs instead.');
return this.getVhs();
}
/**
* Returns VHS Plugin
*
* @return {*} - videojs-http-streaming plugin.
*/
getVhs() {
return this.player.tech({ IWillNotUseThisInPlugins: true }).vhs;
}
/**
* Binds listener for quality level changes.
*/
bindPlayerEvents() {
this.player.qualityLevels().on('addqualitylevel', this.onAddQualityLevel.bind(this));
}
/**
* Adds the quality menu button to the player control bar.
*/
createQualityButton() {
const player = this.player;
this._qualityButton = new ConcreteButton(player);
const placementIndex = player.controlBar.children().length - 2;
2021-02-18 13:45:39 -05:00
const concreteButtonInstance = player.controlBar.addChild(
this._qualityButton,
{ componentClass: 'qualitySelector' },
this.config.placementIndex || placementIndex
);
concreteButtonInstance.addClass('vjs-quality-selector');
if (!this.config.displayCurrentQuality) {
const icon = ` ${this.config.vjsIconClass || 'vjs-icon-hd'}`;
2021-02-18 13:45:39 -05:00
concreteButtonInstance.menuButton_.$('.vjs-icon-placeholder').className += icon;
} else {
this.setButtonInnerText(QUALITY_OPTIONS.AUTO);
}
concreteButtonInstance.removeClass('vjs-hidden');
}
2022-04-11 18:20:50 +08:00
resolveOriginalQualityLabel(abbreviatedForm, includeResolution) {
if (includeResolution && this.config.originalHeight) {
return abbreviatedForm
? __('Orig (%quality%) --[Video quality popup. Short form.]--', { quality: this.config.originalHeight + 'p' })
: __('Original (%quality%) --[Video quality popup. Long form.]--', {
quality: this.config.originalHeight + 'p',
});
} else {
// The allocated space for the button is fixed and happened to fit
// "Original", so we don't abbreviate for English. But it will most likely
// not fit for other languages, hence the 2 strings.
return abbreviatedForm
? __('Original --[Video quality button. Abbreviate to fit space.]--')
: __('Original --[Video quality button. Long form.]--');
}
}
/**
*Set inner button text.
*
* @param {string} text - the text to display in the button.
*/
setButtonInnerText(text) {
2022-04-11 18:20:50 +08:00
let str;
switch (text) {
case QUALITY_OPTIONS.AUTO:
str = QUALITY_OPTIONS.AUTO;
2022-04-11 18:20:50 +08:00
break;
case QUALITY_OPTIONS.ORIGINAL:
2022-04-11 18:20:50 +08:00
str = this.resolveOriginalQualityLabel(true, false);
break;
default:
str = text;
break;
}
this._qualityButton.menuButton_.$('.vjs-icon-placeholder').innerHTML = str;
}
/**
* Builds individual quality menu items.
*
* @param {Object} item - Individual quality menu item.
* @return {ConcreteMenuItem} - Menu item
*/
getQualityMenuItem(item) {
const player = this.player;
return new ConcreteMenuItem(player, item, this._qualityButton, this);
}
/**
* Executed when a quality level is added from HLS playlist.
*/
2022-04-20 15:38:18 -03:00
onAddQualityLevel(e, qualityOption) {
const player = this.player;
2022-04-20 15:38:18 -03:00
const defaultQuality = qualityOption || this.config.defaultQuality;
const qualityList = player.qualityLevels();
const levels = qualityList.levels_ || [];
let levelItems = [];
let nextLowestQualityItem;
let nextLowestQualityItemObj;
for (let i = 0; i < levels.length; ++i) {
const currentHeight = levels[i].height;
if (!levelItems.filter((_existingItem) => _existingItem.item?.value === currentHeight).length) {
const heightStr = currentHeight + 'p';
const levelItem = this.getQualityMenuItem.call(this, {
label: heightStr,
value: currentHeight,
selected: defaultQuality ? currentHeight === defaultQuality : undefined,
});
// Stop at index 0 since the list starts from max quality
const isLiveOriginal = defaultQuality === QUALITY_OPTIONS.ORIGINAL && player.isLivestream && i === 0;
const shouldCheckHeight =
defaultQuality && !nextLowestQualityItem && (currentHeight <= defaultQuality || isLiveOriginal);
if (shouldCheckHeight) {
nextLowestQualityItem = levelItem;
nextLowestQualityItemObj = {
label: heightStr,
value: currentHeight,
selected: true,
};
}
levelItems.push(levelItem);
}
}
if (nextLowestQualityItem) {
levelItems = levelItems.map((item) =>
item === nextLowestQualityItem ? this.getQualityMenuItem.call(this, nextLowestQualityItemObj) : item
);
}
levelItems.sort((current, next) => {
2021-02-18 13:45:39 -05:00
if (typeof current !== 'object' || typeof next !== 'object') {
return -1;
}
if (current.item.value < next.item.value) {
return -1;
}
if (current.item.value > next.item.value) {
return 1;
}
return 0;
});
if (!player.isLivestream) {
levelItems.push(
this.getQualityMenuItem.call(this, {
label: this.resolveOriginalQualityLabel(false, true),
value: QUALITY_OPTIONS.ORIGINAL,
selected: defaultQuality ? defaultQuality === QUALITY_OPTIONS.ORIGINAL : false,
})
);
2022-04-26 21:00:34 -03:00
if (defaultQuality === QUALITY_OPTIONS.ORIGINAL && !player.switchedFromDefaultQuality) {
this.swapSrcTo(QUALITY_OPTIONS.ORIGINAL);
}
}
2022-04-11 18:20:50 +08:00
levelItems.push(
this.getQualityMenuItem.call(this, {
label: QUALITY_OPTIONS.AUTO,
value: QUALITY_OPTIONS.AUTO,
selected: !defaultQuality ? true : defaultQuality === QUALITY_OPTIONS.AUTO,
2021-02-18 13:45:39 -05:00
})
);
2022-04-26 21:00:34 -03:00
// initial button inner text based on default quality setting, or next lowest
if (!this.player.switchedFromDefaultQuality) {
this.setButtonInnerText(
nextLowestQualityItemObj ? nextLowestQualityItemObj.label : defaultQuality || QUALITY_OPTIONS.AUTO
);
}
if (this._qualityButton) {
2021-02-18 13:45:39 -05:00
this._qualityButton.createItems = function () {
return levelItems;
};
this._qualityButton.update();
}
2022-04-26 21:00:34 -03:00
if (defaultQuality) {
this.player.qualityToSet =
nextLowestQualityItemObj?.value ||
(defaultQuality === QUALITY_OPTIONS.ORIGINAL && QUALITY_OPTIONS.ORIGINAL) ||
QUALITY_OPTIONS.AUTO;
}
}
swapSrcTo(mode = QUALITY_OPTIONS.ORIGINAL) {
const currentTime = this.player.currentTime();
this.player.src(mode === 'vhs' ? this.player.claimSrcVhs : this.player.claimSrcOriginal);
this.player.load();
this.player.currentTime(currentTime);
console.assert(mode === 'vhs' || mode === QUALITY_OPTIONS.ORIGINAL, 'Unexpected input');
}
/**
* Sets quality (based on media height)
*
* @param {number} height - A number representing HLS playlist.
*/
setQuality(height) {
const qualityList = this.player.qualityLevels();
2022-04-25 10:28:36 -03:00
const { initialQualityChange, setInitialQualityChange, doToast } = this.config;
if (!initialQualityChange) {
doToast({
message: __('You can also change your default quality on settings.'),
linkText: __('Settings'),
linkTarget: '/settings',
});
setInitialQualityChange(true);
}
// Set quality on plugin
this._currentQuality = height;
if (this.config.displayCurrentQuality) {
this.setButtonInnerText(
height === QUALITY_OPTIONS.AUTO
? QUALITY_OPTIONS.AUTO
: height === QUALITY_OPTIONS.ORIGINAL
? QUALITY_OPTIONS.ORIGINAL
: `${height}p`
);
}
for (let i = 0; i < qualityList.length; ++i) {
const quality = qualityList[i];
quality.enabled =
quality.height === height || height === QUALITY_OPTIONS.AUTO || height === QUALITY_OPTIONS.ORIGINAL;
}
2022-04-11 18:20:50 +08:00
if (height === QUALITY_OPTIONS.ORIGINAL) {
if (this.player.currentSrc() !== this.player.claimSrcOriginal.src) {
setTimeout(() => this.swapSrcTo(QUALITY_OPTIONS.ORIGINAL));
}
} else {
if (!this.player.isLivestream && this.player.currentSrc() !== this.player.claimSrcVhs.src) {
setTimeout(() => this.swapSrcTo('vhs'));
2022-04-20 15:38:18 -03:00
if (height !== 'auto') {
// -- Re-select quality --
// Until we have "persistent quality" implemented, we need to do this
// because the VHS internals default to "auto" when initialized,
// causing a GUI mismatch.
setTimeout(() => {
this.setQuality(height);
this.onAddQualityLevel(undefined, height);
}, 1000);
}
}
}
this._qualityButton.unpressButton();
}
/**
* Return the current set quality or 'auto'
*
* @return {string} the currently set quality
*/
getCurrentQuality() {
return this._currentQuality || QUALITY_OPTIONS.AUTO;
}
}
/**
* Function to invoke when the player is ready.
*
* This is a great place for your plugin to initialize itself. When this
* function is called, the player will have its DOM and child components
* in place.
*
* @function onPlayerReady
* @param {Player} player
* A Video.js player object.
*
* @param {Object} [options={}]
* A plain object containing options for the plugin.
*/
const onPlayerReady = (player, options) => {
player.addClass('vjs-hls-quality-selector');
player.hlsQualitySelector = new HlsQualitySelectorPlugin(player, options);
};
/**
* A video.js plugin.
*
* In the plugin function, the value of `this` is a video.js `Player`
* instance. You cannot rely on the player being in a "ready" state here,
* depending on how the plugin is invoked. This may or may not be important
* to you; if not, remove the wait for "ready"!
*
* @function hlsQualitySelector
* @param {Object} [options={}]
* An object of options left to the plugin author to define.
*/
2021-02-18 13:45:39 -05:00
const hlsQualitySelector = function (options) {
this.ready(() => {
onPlayerReady(this, videojs.mergeOptions(defaults, options));
});
};
// Register the plugin with video.js.
registerPlugin('hlsQualitySelector', hlsQualitySelector);
// Include the version number.
hlsQualitySelector.VERSION = VERSION;
export default hlsQualitySelector;
2021-02-18 13:45:39 -05:00
/* eslint-enable */