Source: ui/hidden_seek_button.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.ui.HiddenSeekButton');
  7. goog.require('shaka.ui.Element');
  8. goog.require('shaka.util.Timer');
  9. goog.require('shaka.util.Dom');
  10. goog.requireType('shaka.ui.Controls');
  11. /**
  12. * @extends {shaka.ui.Element}
  13. * @export
  14. */
  15. shaka.ui.HiddenSeekButton = class extends shaka.ui.Element {
  16. /**
  17. * @param {!HTMLElement} parent
  18. * @param {!shaka.ui.Controls} controls
  19. */
  20. constructor(parent, controls) {
  21. super(parent, controls);
  22. /** @private {?number} */
  23. this.lastTouchEventTimeSet_ = null;
  24. /** @private {boolean} */
  25. this.triggeredTouchValid_ = false;
  26. /**
  27. * Keeps track of whether the user has moved enough
  28. * to be considered scrolling.
  29. * @private {boolean}
  30. */
  31. this.hasMoved_ = false;
  32. /**
  33. * Touch-start coordinates for detecting scroll distance.
  34. * @private {?number}
  35. */
  36. this.touchStartX_ = null;
  37. /** @private {?number} */
  38. this.touchStartY_ = null;
  39. /**
  40. * Timer used to hide the seek button container. In the timer’s callback,
  41. * if the seek value is still 0s, we interpret it as a single tap
  42. * (play/pause). If not, we perform the seek.
  43. * @private {shaka.util.Timer}
  44. */
  45. this.hideSeekButtonContainerTimer_ = new shaka.util.Timer(() => {
  46. const seekSeconds = parseInt(this.seekValue_.textContent, 10);
  47. if (seekSeconds === 0) {
  48. this.controls.onContainerClick();
  49. }
  50. this.hideSeekButtonContainer_();
  51. });
  52. /** @protected {!HTMLElement} */
  53. this.seekContainer = shaka.util.Dom.createHTMLElement('div');
  54. this.parent.appendChild(this.seekContainer);
  55. /** @private {!HTMLElement} */
  56. this.seekValue_ = shaka.util.Dom.createHTMLElement('span');
  57. this.seekValue_.textContent = '0s';
  58. this.seekContainer.appendChild(this.seekValue_);
  59. /** @protected {!HTMLElement} */
  60. this.seekIcon = shaka.util.Dom.createHTMLElement('span');
  61. this.seekIcon.classList.add(
  62. 'shaka-forward-rewind-container-icon');
  63. this.seekContainer.appendChild(this.seekIcon);
  64. /** @protected {boolean} */
  65. this.isRewind = false;
  66. // ---------------------------------------------------------------
  67. // TOUCH EVENT LISTENERS for SCROLL vs. TAP DETECTION
  68. // ---------------------------------------------------------------
  69. this.eventManager.listen(this.seekContainer, 'touchstart', (e) => {
  70. const event = /** @type {!TouchEvent} */(e);
  71. this.onTouchStart_(event);
  72. });
  73. this.eventManager.listen(this.seekContainer, 'touchmove', (e) => {
  74. const event = /** @type {!TouchEvent} */(e);
  75. this.onTouchMove_(event);
  76. });
  77. this.eventManager.listen(this.seekContainer, 'touchend', (e) => {
  78. const event = /** @type {!TouchEvent} */(e);
  79. this.onTouchEnd_(event);
  80. });
  81. }
  82. /**
  83. * Called when the user starts touching the screen.
  84. * We record the initial touch coordinates for scroll detection.
  85. * @param {!TouchEvent} event
  86. * @private
  87. */
  88. onTouchStart_(event) {
  89. // Only proceed if controls are visible.
  90. if (!this.controls.isOpaque()) {
  91. return;
  92. }
  93. // If multiple touches, handle or ignore as needed. Here, we assume
  94. // single-touch.
  95. if (event.touches.length > 0) {
  96. this.touchStartX_ = event.touches[0].clientX;
  97. this.touchStartY_ = event.touches[0].clientY;
  98. }
  99. this.hasMoved_ = false;
  100. }
  101. /**
  102. * Called when the user moves the finger on the screen.
  103. * If the movement exceeds the scroll threshold, we mark this as scrolling.
  104. * @param {!TouchEvent} event
  105. * @private
  106. */
  107. onTouchMove_(event) {
  108. if (event.touches.length > 0 &&
  109. this.touchStartX_ != null &&
  110. this.touchStartY_ != null) {
  111. const dx = event.touches[0].clientX - this.touchStartX_;
  112. const dy = event.touches[0].clientY - this.touchStartY_;
  113. const distance = Math.sqrt(dx * dx + dy * dy);
  114. if (distance > shaka.ui.HiddenSeekButton.SCROLL_THRESHOLD_) {
  115. this.hasMoved_ = true;
  116. }
  117. }
  118. }
  119. /**
  120. * Called when the user lifts the finger from the screen.
  121. * If we haven't moved beyond the threshold, treat it as a tap.
  122. * @param {!TouchEvent} event
  123. * @private
  124. */
  125. onTouchEnd_(event) {
  126. // Only proceed if controls are visible.
  127. if (!this.controls.isOpaque()) {
  128. return;
  129. }
  130. // If user scrolled, don't handle as a tap.
  131. if (this.hasMoved_) {
  132. return;
  133. }
  134. // If any settings menus are open, this tap closes them instead of toggling
  135. // play/seek.
  136. if (this.controls.anySettingsMenusAreOpen()) {
  137. event.preventDefault();
  138. this.controls.hideSettingsMenus();
  139. return;
  140. }
  141. // Normal tap logic (single vs double tap).
  142. if (this.controls.getConfig().tapSeekDistance > 0) {
  143. event.preventDefault();
  144. this.onSeekButtonClick_();
  145. }
  146. }
  147. /**
  148. * Determines whether this tap is a single tap (leading to play/pause)
  149. * or a double tap (leading to a seek). We use a 500 ms window.
  150. * @private
  151. */
  152. onSeekButtonClick_() {
  153. const tapSeekDistance = this.controls.getConfig().tapSeekDistance;
  154. const doubleTapWindow = shaka.ui.HiddenSeekButton.DOUBLE_TAP_WINDOW_;
  155. if (!this.triggeredTouchValid_) {
  156. // First tap: start our 500 ms "double-tap" timer.
  157. this.triggeredTouchValid_ = true;
  158. this.lastTouchEventTimeSet_ = Date.now();
  159. this.hideSeekButtonContainerTimer_.tickAfter(doubleTapWindow);
  160. } else if ((this.lastTouchEventTimeSet_ +
  161. doubleTapWindow * 1000) > Date.now()) {
  162. // Second tap arrived in time — interpret as a double tap to seek.
  163. this.hideSeekButtonContainerTimer_.stop();
  164. this.lastTouchEventTimeSet_ = Date.now();
  165. let position = parseInt(this.seekValue_.textContent, 10);
  166. if (this.isRewind) {
  167. position -= tapSeekDistance;
  168. } else {
  169. position += tapSeekDistance;
  170. }
  171. this.seekValue_.textContent = position.toString() + 's';
  172. this.seekContainer.style.opacity = '1';
  173. // Restart timer if user might tap again (triple tap).
  174. this.hideSeekButtonContainerTimer_.tickAfter(doubleTapWindow);
  175. }
  176. }
  177. /**
  178. * If the seek value is zero, interpret it as a single tap (play/pause).
  179. * Otherwise, apply the seek and reset.
  180. * @private
  181. */
  182. hideSeekButtonContainer_() {
  183. const seekSeconds = parseInt(this.seekValue_.textContent, 10);
  184. if (seekSeconds !== 0) {
  185. // Perform the seek.
  186. this.video.currentTime = this.controls.getDisplayTime() + seekSeconds;
  187. }
  188. // Hide and reset.
  189. this.seekContainer.style.opacity = '0';
  190. this.triggeredTouchValid_ = false;
  191. this.seekValue_.textContent = '0s';
  192. }
  193. };
  194. /**
  195. * The amount of time, in seconds, to double-tap detection.
  196. *
  197. * @const {number}
  198. */
  199. shaka.ui.HiddenSeekButton.DOUBLE_TAP_WINDOW_ = 0.5;
  200. /**
  201. * Minimum distance (px) the finger must move during touch to consider it a
  202. * scroll rather than a tap.
  203. *
  204. * @const {number}
  205. */
  206. shaka.ui.HiddenSeekButton.SCROLL_THRESHOLD_ = 10;