mirror of
https://github.com/shaka-project/shaka-player.git
synced 2026-06-14 15:56:38 +03:00
1c8539698c
Changes: - The look has been changed to make it more similar to YouTube: - The main background color is now black, and the font is white. - Presentation time has been moved to the bottom. - Cast and airplay buttons are now more accessible. - Tooltips have been enabled except on mobile platforms. - The ad information has been moved to appear in the same position as the presentation time when the ad is present. - A mark indicating the current quality has been added (e.g.: HD, 2K, 4K, 8K) - The spinner has been replaced with one that works well on Smart TVs and is very similar to the current one. The animation is included in the SVG element itself rather than through CSS. - More LESS variables have been added to make customization easier in forks. - The maximum size of the menus is dynamically calculated so that they never extend outside the video container. - The size of the subtitle container when the UI appears is now calculated dynamically. - The Demo has been updated to show the seekbar when trickplays are enabled. - UI performance on Smart TVs has been improved (Tested on Tizen 5.0) - Many offsets that were hardcoded have been removed, but not all (in CSS). Close https://github.com/shaka-project/shaka-player/issues/8406
250 lines
6.9 KiB
JavaScript
250 lines
6.9 KiB
JavaScript
/*! @license
|
|
* Shaka Player
|
|
* Copyright 2016 Google LLC
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
|
|
goog.provide('shaka.ui.OverflowMenu');
|
|
|
|
goog.require('goog.asserts');
|
|
goog.require('shaka.ads.Utils');
|
|
goog.require('shaka.log');
|
|
goog.require('shaka.ui.Controls');
|
|
goog.require('shaka.ui.Element');
|
|
goog.require('shaka.ui.Enums');
|
|
goog.require('shaka.ui.Locales');
|
|
goog.require('shaka.ui.Localization');
|
|
goog.require('shaka.ui.Utils');
|
|
goog.require('shaka.util.Dom');
|
|
goog.require('shaka.util.Iterables');
|
|
|
|
|
|
/**
|
|
* @extends {shaka.ui.Element}
|
|
* @final
|
|
* @export
|
|
*/
|
|
shaka.ui.OverflowMenu = class extends shaka.ui.Element {
|
|
/**
|
|
* @param {!HTMLElement} parent
|
|
* @param {!shaka.ui.Controls} controls
|
|
*/
|
|
constructor(parent, controls) {
|
|
super(parent, controls);
|
|
|
|
/** @private {!shaka.extern.UIConfiguration} */
|
|
this.config_ = this.controls.getConfig();
|
|
|
|
/** @private {HTMLElement} */
|
|
this.controlsContainer_ = this.controls.getControlsContainer();
|
|
|
|
/** @private {HTMLElement } */
|
|
this.videoContainer_ = this.controls.getVideoContainer();
|
|
|
|
/** @private {!Array<shaka.extern.IUIElement>} */
|
|
this.children_ = [];
|
|
|
|
this.addOverflowMenuButton_();
|
|
|
|
this.addOverflowMenu_();
|
|
|
|
this.createChildren_();
|
|
|
|
this.eventManager.listen(
|
|
this.localization, shaka.ui.Localization.LOCALE_UPDATED, () => {
|
|
this.updateAriaLabel_();
|
|
});
|
|
|
|
this.eventManager.listen(
|
|
this.localization, shaka.ui.Localization.LOCALE_CHANGED, () => {
|
|
this.updateAriaLabel_();
|
|
});
|
|
|
|
this.eventManager.listen(
|
|
this.adManager, shaka.ads.Utils.AD_STARTED, () => {
|
|
if (this.ad && this.ad.isLinear()) {
|
|
shaka.ui.Utils.setDisplay(this.overflowMenuButton_, false);
|
|
}
|
|
});
|
|
|
|
this.eventManager.listen(
|
|
this.adManager, shaka.ads.Utils.AD_STOPPED, () => {
|
|
shaka.ui.Utils.setDisplay(this.overflowMenuButton_, true);
|
|
});
|
|
|
|
|
|
this.eventManager.listen(
|
|
this.controls, 'submenuopen', () => {
|
|
// Hide the main overflow menu if one of the sub menus has
|
|
// been opened.
|
|
shaka.ui.Utils.setDisplay(this.overflowMenu_, false);
|
|
});
|
|
|
|
|
|
this.eventManager.listen(
|
|
this.overflowMenu_, 'touchstart', (event) => {
|
|
this.controls.setLastTouchEventTime(Date.now());
|
|
event.stopPropagation();
|
|
});
|
|
|
|
this.eventManager.listen(this.overflowMenuButton_, 'click', () => {
|
|
if (!this.controls.isOpaque()) {
|
|
return;
|
|
}
|
|
this.onOverflowMenuButtonClick_();
|
|
});
|
|
|
|
this.updateAriaLabel_();
|
|
|
|
if (this.ad && this.ad.isLinear()) {
|
|
// There was already an ad.
|
|
shaka.ui.Utils.setDisplay(this.overflowMenuButton_, false);
|
|
}
|
|
}
|
|
|
|
/** @override */
|
|
release() {
|
|
this.controlsContainer_ = null;
|
|
|
|
for (const element of this.children_) {
|
|
element.release();
|
|
}
|
|
|
|
this.children_ = [];
|
|
super.release();
|
|
}
|
|
|
|
/**
|
|
* @param {string} name
|
|
* @param {!shaka.extern.IUIElement.Factory} factory
|
|
* @export
|
|
*/
|
|
static registerElement(name, factory) {
|
|
shaka.ui.OverflowMenu.elementNamesToFactories_.set(name, factory);
|
|
}
|
|
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
addOverflowMenu_() {
|
|
/** @private {!HTMLElement} */
|
|
this.overflowMenu_ = shaka.util.Dom.createHTMLElement('div');
|
|
this.overflowMenu_.classList.add('shaka-overflow-menu');
|
|
this.overflowMenu_.classList.add('shaka-no-propagation');
|
|
this.overflowMenu_.classList.add('shaka-show-controls-on-mouse-over');
|
|
this.overflowMenu_.classList.add('shaka-hidden');
|
|
this.controlsContainer_.appendChild(this.overflowMenu_);
|
|
}
|
|
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
addOverflowMenuButton_() {
|
|
/** @private {!HTMLButtonElement} */
|
|
this.overflowMenuButton_ = shaka.util.Dom.createButton();
|
|
this.overflowMenuButton_.classList.add('shaka-overflow-menu-button');
|
|
this.overflowMenuButton_.classList.add('shaka-no-propagation');
|
|
this.overflowMenuButton_.classList.add('material-icons-round');
|
|
this.overflowMenuButton_.classList.add('shaka-tooltip');
|
|
this.overflowMenuButton_.textContent =
|
|
shaka.ui.Enums.MaterialDesignIcons.OPEN_OVERFLOW;
|
|
const markEl = shaka.util.Dom.createHTMLElement('span');
|
|
markEl.classList.add('shaka-overflow-quality-mark');
|
|
markEl.style.display = 'none';
|
|
this.overflowMenuButton_.appendChild(markEl);
|
|
this.parent.appendChild(this.overflowMenuButton_);
|
|
}
|
|
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
createChildren_() {
|
|
for (const name of this.config_.overflowMenuButtons) {
|
|
if (shaka.ui.OverflowMenu.elementNamesToFactories_.get(name)) {
|
|
const factory =
|
|
shaka.ui.OverflowMenu.elementNamesToFactories_.get(name);
|
|
goog.asserts.assert(this.controls, 'Controls should not be null!');
|
|
this.children_.push(factory.create(this.overflowMenu_, this.controls));
|
|
} else {
|
|
shaka.log.alwaysWarn('Unrecognized overflow menu element requested:',
|
|
name);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/** @private */
|
|
onOverflowMenuButtonClick_() {
|
|
if (this.controls.anySettingsMenusAreOpen()) {
|
|
this.controls.hideSettingsMenus();
|
|
} else {
|
|
shaka.ui.Utils.setDisplay(this.overflowMenu_, true);
|
|
this.controls.computeOpacity();
|
|
|
|
// If overflow menu has currently visible buttons, focus on the
|
|
// first one, when the menu opens.
|
|
const isDisplayed =
|
|
(element) => element.classList.contains('shaka-hidden') == false;
|
|
|
|
const Iterables = shaka.util.Iterables;
|
|
if (Iterables.some(this.overflowMenu_.childNodes, isDisplayed)) {
|
|
// Focus on the first visible child of the overflow menu
|
|
const visibleElements =
|
|
Iterables.filter(this.overflowMenu_.childNodes, isDisplayed);
|
|
/** @type {!HTMLElement} */ (visibleElements[0]).focus();
|
|
}
|
|
this.computeMaxHeight_();
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
updateAriaLabel_() {
|
|
const LocIds = shaka.ui.Locales.Ids;
|
|
this.overflowMenuButton_.ariaLabel =
|
|
this.localization.resolve(LocIds.MORE_SETTINGS);
|
|
}
|
|
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
computeMaxHeight_() {
|
|
const rectMenu = this.overflowMenu_.getBoundingClientRect();
|
|
const styleMenu = window.getComputedStyle(this.overflowMenu_);
|
|
const paddingTop = parseFloat(styleMenu.paddingTop);
|
|
const paddingBottom = parseFloat(styleMenu.paddingBottom);
|
|
const rectContainer = this.videoContainer_.getBoundingClientRect();
|
|
const heightIntersection =
|
|
rectMenu.bottom - rectContainer.top - paddingTop - paddingBottom;
|
|
|
|
this.overflowMenu_.style.maxHeight = heightIntersection + 'px';
|
|
}
|
|
};
|
|
|
|
|
|
/**
|
|
* @implements {shaka.extern.IUIElement.Factory}
|
|
* @final
|
|
*/
|
|
shaka.ui.OverflowMenu.Factory = class {
|
|
/** @override */
|
|
create(rootElement, controls) {
|
|
return new shaka.ui.OverflowMenu(rootElement, controls);
|
|
}
|
|
};
|
|
|
|
shaka.ui.Controls.registerElement(
|
|
'overflow_menu', new shaka.ui.OverflowMenu.Factory());
|
|
|
|
|
|
/** @private {!Map<string, !shaka.extern.IUIElement.Factory>} */
|
|
shaka.ui.OverflowMenu.elementNamesToFactories_ = new Map();
|
|
|