Source: ui/seek_bar.js

  1. /**
  2. * @license
  3. * Copyright 2016 Google Inc.
  4. *
  5. * Licensed under the Apache License, Version 2.0 (the "License");
  6. * you may not use this file except in compliance with the License.
  7. * You may obtain a copy of the License at
  8. *
  9. * http://www.apache.org/licenses/LICENSE-2.0
  10. *
  11. * Unless required by applicable law or agreed to in writing, software
  12. * distributed under the License is distributed on an "AS IS" BASIS,
  13. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14. * See the License for the specific language governing permissions and
  15. * limitations under the License.
  16. */
  17. goog.provide('shaka.ui.SeekBar');
  18. goog.require('shaka.ui.Constants');
  19. goog.require('shaka.ui.Locales');
  20. goog.require('shaka.ui.Localization');
  21. goog.require('shaka.ui.RangeElement');
  22. goog.require('shaka.ui.Utils');
  23. goog.require('shaka.util.Timer');
  24. /**
  25. * @extends {shaka.ui.RangeElement}
  26. * @final
  27. * @export
  28. */
  29. shaka.ui.SeekBar = class extends shaka.ui.RangeElement {
  30. /**
  31. * @param {!HTMLElement} parent
  32. * @param {!shaka.ui.Controls} controls
  33. */
  34. constructor(parent, controls) {
  35. super(parent, controls,
  36. ['shaka-seek-bar-container'],
  37. [
  38. 'shaka-seek-bar',
  39. 'shaka-no-propagation',
  40. 'shaka-show-controls-on-mouse-over',
  41. ]);
  42. /** @private {!shaka.extern.UIConfiguration} */
  43. this.config_ = this.controls.getConfig();
  44. /**
  45. * This timer is used to introduce a delay between the user scrubbing across
  46. * the seek bar and the seek being sent to the player.
  47. *
  48. * @private {shaka.util.Timer}
  49. */
  50. this.seekTimer_ = new shaka.util.Timer(() => {
  51. this.video.currentTime = this.getValue();
  52. });
  53. /**
  54. * When user is scrubbing the seek bar - we should pause the video - see https://git.io/JUhHG
  55. * but will conditionally pause or play the video after scrubbing
  56. * depending on its previous state
  57. *
  58. * @private {boolean}
  59. */
  60. this.wasPlaying_ = false;
  61. this.eventManager.listen(this.localization,
  62. shaka.ui.Localization.LOCALE_UPDATED,
  63. () => this.updateAriaLabel_());
  64. this.eventManager.listen(this.localization,
  65. shaka.ui.Localization.LOCALE_CHANGED,
  66. () => this.updateAriaLabel_());
  67. // Initialize seek state and label.
  68. this.setValue(this.video.currentTime);
  69. this.update();
  70. this.updateAriaLabel_();
  71. }
  72. /** @override */
  73. async destroy() {
  74. if (this.seekTimer_) {
  75. this.seekTimer_.stop();
  76. this.seekTimer_ = null;
  77. }
  78. await super.destroy();
  79. }
  80. /**
  81. * Called by the base class when user interaction with the input element
  82. * begins.
  83. *
  84. * @override
  85. */
  86. onChangeStart() {
  87. this.wasPlaying_ = !this.video.paused;
  88. this.controls.setSeeking(true);
  89. this.video.pause();
  90. }
  91. /**
  92. * Update the video element's state to match the input element's state.
  93. * Called by the base class when the input element changes.
  94. *
  95. * @override
  96. */
  97. onChange() {
  98. if (!this.video.duration) {
  99. // Can't seek yet. Ignore.
  100. return;
  101. }
  102. // Update the UI right away.
  103. this.update();
  104. // We want to wait until the user has stopped moving the seek bar for a
  105. // little bit to reduce the number of times we ask the player to seek.
  106. //
  107. // To do this, we will start a timer that will fire in a little bit, but if
  108. // we see another seek bar change, we will cancel that timer and re-start
  109. // it.
  110. //
  111. // Calling |start| on an already pending timer will cancel the old request
  112. // and start the new one.
  113. this.seekTimer_.tickAfter(/* seconds= */ 0.125);
  114. }
  115. /**
  116. * Called by the base class when user interaction with the input element
  117. * ends.
  118. *
  119. * @override
  120. */
  121. onChangeEnd() {
  122. // They just let go of the seek bar, so cancel the timer and manually
  123. // call the event so that we can respond immediately.
  124. this.seekTimer_.tickNow();
  125. this.controls.setSeeking(false);
  126. if (this.wasPlaying_) {
  127. this.video.play();
  128. }
  129. }
  130. /** @return {boolean} */
  131. isShowing() {
  132. // It is showing by default, so it is hidden if shaka-hidden is in the list.
  133. return !this.container.classList.contains('shaka-hidden');
  134. }
  135. /**
  136. * Called by Controls on a timer to update the state of the seek bar.
  137. * Also called internally when the user interacts with the input element.
  138. */
  139. update() {
  140. const colors = this.config_.seekBarColors;
  141. const currentTime = this.getValue();
  142. const bufferedLength = this.video.buffered.length;
  143. const bufferedStart = bufferedLength ? this.video.buffered.start(0) : 0;
  144. const bufferedEnd =
  145. bufferedLength ? this.video.buffered.end(bufferedLength - 1) : 0;
  146. const seekRange = this.player.seekRange();
  147. const seekRangeSize = seekRange.end - seekRange.start;
  148. this.setRange(seekRange.start, seekRange.end);
  149. // Hide seekbar if the seek window is very small.
  150. if (this.player.isLive() &&
  151. seekRangeSize < shaka.ui.Constants.MIN_SEEK_WINDOW_TO_SHOW_SEEKBAR) {
  152. shaka.ui.Utils.setDisplay(this.container, false);
  153. } else {
  154. shaka.ui.Utils.setDisplay(this.container, true);
  155. if (bufferedLength == 0) {
  156. this.container.style.background = colors.base;
  157. } else {
  158. const clampedBufferStart = Math.max(bufferedStart, seekRange.start);
  159. const clampedBufferEnd = Math.min(bufferedEnd, seekRange.end);
  160. const clampedCurrentTime = Math.min(
  161. Math.max(currentTime, seekRange.start),
  162. seekRange.end);
  163. const bufferStartDistance = clampedBufferStart - seekRange.start;
  164. const bufferEndDistance = clampedBufferEnd - seekRange.start;
  165. const playheadDistance = clampedCurrentTime - seekRange.start;
  166. // NOTE: the fallback to zero eliminates NaN.
  167. const bufferStartFraction = (bufferStartDistance / seekRangeSize) || 0;
  168. const bufferEndFraction = (bufferEndDistance / seekRangeSize) || 0;
  169. const playheadFraction = (playheadDistance / seekRangeSize) || 0;
  170. const unbufferedColor =
  171. this.config_.showUnbufferedStart ? colors.base : colors.played;
  172. const makeColor = (color, fract) => color + ' ' + (fract * 100) + '%';
  173. const gradient = [
  174. 'to right',
  175. makeColor(unbufferedColor, bufferStartFraction),
  176. makeColor(colors.played, bufferStartFraction),
  177. makeColor(colors.played, playheadFraction),
  178. makeColor(colors.buffered, playheadFraction),
  179. makeColor(colors.buffered, bufferEndFraction),
  180. makeColor(colors.base, bufferEndFraction),
  181. ];
  182. this.container.style.background =
  183. 'linear-gradient(' + gradient.join(',') + ')';
  184. }
  185. }
  186. }
  187. /** @private */
  188. updateAriaLabel_() {
  189. this.bar.setAttribute(shaka.ui.Constants.ARIA_LABEL,
  190. this.localization.resolve(shaka.ui.Locales.Ids.SEEK));
  191. }
  192. };