Source: lib/media/playhead_observer.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.media.IPlayheadObserver');
  7. goog.provide('shaka.media.PlayheadObserverManager');
  8. goog.require('shaka.util.IReleasable');
  9. goog.require('shaka.util.Timer');
  10. /**
  11. * A playhead observer is a system that watches for meaningful changes in state
  12. * that are dependent on playhead information. The observer is responsible for
  13. * managing its own listeners.
  14. *
  15. * @extends {shaka.util.IReleasable}
  16. * @interface
  17. */
  18. shaka.media.IPlayheadObserver = class {
  19. /**
  20. * Check again (using an update playhead summary) if an event should be fired.
  21. * If an event should be fired, fire it.
  22. *
  23. * @param {number} positionInSeconds
  24. * @param {boolean} wasSeeking
  25. */
  26. poll(positionInSeconds, wasSeeking) {}
  27. };
  28. /**
  29. * The playhead observer manager is responsible for owning playhead observer
  30. * instances and polling them when needed. Destroying the manager will destroy
  31. * all observers managed by the manager.
  32. *
  33. * @implements {shaka.util.IReleasable}
  34. * @final
  35. */
  36. shaka.media.PlayheadObserverManager = class {
  37. /**
  38. * @param {!HTMLMediaElement} mediaElement
  39. */
  40. constructor(mediaElement) {
  41. /** @private {HTMLMediaElement} */
  42. this.mediaElement_ = mediaElement;
  43. /**
  44. * The set of all observers that this manager is responsible for updating.
  45. * We are using a set to ensure that we don't double update an observer if
  46. * it is accidentally added twice.
  47. *
  48. * @private {!Set.<shaka.media.IPlayheadObserver>}
  49. */
  50. this.observers_ = new Set();
  51. /**
  52. * To fire events semi-accurately, poll the observers 4 times a second. This
  53. * should be frequent enough to trigger an event close enough to its actual
  54. * occurrence without the user noticing a delay.
  55. *
  56. * @private {shaka.util.Timer}
  57. */
  58. this.pollingLoop_ = new shaka.util.Timer(() => {
  59. this.pollAllObservers_(/* seeking= */ false);
  60. }).tickEvery(/* seconds= */ 0.25);
  61. }
  62. /** @override */
  63. release() {
  64. // We need to stop the loop or else we may try to use a released resource.
  65. this.pollingLoop_.stop();
  66. for (const observer of this.observers_) {
  67. observer.release();
  68. }
  69. this.observers_.clear();
  70. }
  71. /**
  72. * Have the playhead observer manager manage a new observer. This will ensure
  73. * that observers are only tracked once within the manager. After this call,
  74. * the manager will be responsible for the life cycle of |observer|.
  75. *
  76. * @param {!shaka.media.IPlayheadObserver} observer
  77. */
  78. manage(observer) {
  79. this.observers_.add(observer);
  80. }
  81. /**
  82. * Notify all the observers that we just seeked.
  83. */
  84. notifyOfSeek() {
  85. this.pollAllObservers_(/* seeking= */ true);
  86. }
  87. /**
  88. * @param {boolean} seeking
  89. * @private
  90. */
  91. pollAllObservers_(seeking) {
  92. for (const observer of this.observers_) {
  93. observer.poll(
  94. this.mediaElement_.currentTime,
  95. seeking);
  96. }
  97. }
  98. };