Source: ui/playback_rate_selection.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.ui.PlaybackRateSelection');
  7. goog.require('shaka.ui.Controls');
  8. goog.require('shaka.ui.Enums');
  9. goog.require('shaka.ui.Locales');
  10. goog.require('shaka.ui.Localization');
  11. goog.require('shaka.ui.OverflowMenu');
  12. goog.require('shaka.ui.SettingsMenu');
  13. goog.require('shaka.ui.Utils');
  14. goog.require('shaka.util.Dom');
  15. goog.requireType('shaka.ui.Controls');
  16. /**
  17. * @extends {shaka.ui.SettingsMenu}
  18. * @final
  19. * @export
  20. */
  21. shaka.ui.PlaybackRateSelection = class extends shaka.ui.SettingsMenu {
  22. /**
  23. * @param {!HTMLElement} parent
  24. * @param {!shaka.ui.Controls} controls
  25. */
  26. constructor(parent, controls) {
  27. super(parent, controls, shaka.ui.Enums.MaterialDesignIcons.PLAYBACK_RATE);
  28. this.button.classList.add('shaka-playbackrate-button');
  29. this.menu.classList.add('shaka-playback-rates');
  30. this.button.classList.add('shaka-tooltip-status');
  31. this.eventManager.listen(
  32. this.localization, shaka.ui.Localization.LOCALE_UPDATED, () => {
  33. this.updateLocalizedStrings_();
  34. });
  35. this.eventManager.listen(
  36. this.localization, shaka.ui.Localization.LOCALE_CHANGED, () => {
  37. this.updateLocalizedStrings_();
  38. });
  39. this.eventManager.listen(this.player, 'loaded', () => {
  40. this.updatePlaybackRateSelection_(this.player.getPlaybackRate());
  41. });
  42. this.eventManager.listen(this.player, 'ratechange', () => {
  43. this.updatePlaybackRateSelection_(this.player.getPlaybackRate());
  44. });
  45. /** @type {!Map.<string, number>} */
  46. this.playbackRates_ = new Map(this.controls.getConfig().playbackRates
  47. .map((rate) => [rate + 'x', rate]));
  48. // Set up all the strings in the user's preferred language.
  49. this.updateLocalizedStrings_();
  50. this.addPlaybackRates_();
  51. this.updatePlaybackRateSelection_(this.player.getPlaybackRate());
  52. }
  53. /**
  54. * @private
  55. */
  56. updateLocalizedStrings_() {
  57. const LocIds = shaka.ui.Locales.Ids;
  58. this.backButton.ariaLabel = this.localization.resolve(LocIds.BACK);
  59. this.button.ariaLabel = this.localization.resolve(LocIds.PLAYBACK_RATE);
  60. this.nameSpan.textContent = this.localization.resolve(LocIds.PLAYBACK_RATE);
  61. this.backSpan.textContent = this.localization.resolve(LocIds.PLAYBACK_RATE);
  62. }
  63. /**
  64. * Update checkmark icon and related class and attribute for the chosen rate
  65. * button.
  66. * @param {number} rate The video playback rate.
  67. * @private
  68. */
  69. updatePlaybackRateSelection_(rate) {
  70. // Remove the old checkmark icon and related tags and classes if it exists.
  71. const checkmarkIcon = shaka.ui.Utils.getDescendantIfExists(
  72. this.menu, 'material-icons-round shaka-chosen-item');
  73. if (checkmarkIcon) {
  74. const previouslySelectedButton = checkmarkIcon.parentElement;
  75. previouslySelectedButton.removeAttribute('aria-selected');
  76. const previouslySelectedSpan =
  77. previouslySelectedButton.getElementsByTagName('span')[0];
  78. previouslySelectedSpan.classList.remove('shaka-chosen-item');
  79. previouslySelectedButton.removeChild(checkmarkIcon);
  80. }
  81. // Find the button that represents the newly selected playback rate.
  82. // Add the checkmark icon, related tags and classes to the newly selected
  83. // button.
  84. const span = Array.from(this.menu.querySelectorAll('span')).find((el) => {
  85. return this.playbackRates_.get(el.textContent) == rate;
  86. });
  87. if (span) {
  88. const button = span.parentElement;
  89. button.appendChild(shaka.ui.Utils.checkmarkIcon());
  90. button.ariaSelected = 'true';
  91. span.classList.add('shaka-chosen-item');
  92. }
  93. // Set the label to display the current playback rate in the overflow menu,
  94. // in the format of '1x', '1.5x', etc.
  95. this.currentSelection.textContent = rate + 'x';
  96. this.button.setAttribute('shaka-status', rate + 'x');
  97. }
  98. /** @private */
  99. addPlaybackRates_() {
  100. for (const rateStr of this.playbackRates_.keys()) {
  101. const button = shaka.util.Dom.createButton();
  102. const span = shaka.util.Dom.createHTMLElement('span');
  103. span.textContent = rateStr;
  104. button.appendChild(span);
  105. this.eventManager.listen(button, 'click', () => {
  106. this.video.playbackRate = this.playbackRates_.get(rateStr);
  107. this.video.defaultPlaybackRate = this.playbackRates_.get(rateStr);
  108. });
  109. this.menu.appendChild(button);
  110. }
  111. shaka.ui.Utils.focusOnTheChosenItem(this.menu);
  112. }
  113. };
  114. /**
  115. * @implements {shaka.extern.IUIElement.Factory}
  116. * @final
  117. */
  118. shaka.ui.PlaybackRateSelection.Factory = class {
  119. /** @override */
  120. create(rootElement, controls) {
  121. return new shaka.ui.PlaybackRateSelection(rootElement, controls);
  122. }
  123. };
  124. shaka.ui.OverflowMenu.registerElement(
  125. 'playback_rate', new shaka.ui.PlaybackRateSelection.Factory());
  126. shaka.ui.Controls.registerElement(
  127. 'playback_rate', new shaka.ui.PlaybackRateSelection.Factory());