Source: lib/media/time_ranges_utils.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.media.TimeRangesUtils');
  7. goog.require('shaka.util.Iterables');
  8. goog.require('shaka.util.Platform');
  9. /**
  10. * @summary A set of utility functions for dealing with TimeRanges objects.
  11. */
  12. shaka.media.TimeRangesUtils = class {
  13. /**
  14. * Gets the first timestamp in the buffer.
  15. *
  16. * @param {TimeRanges} b
  17. * @return {?number} The first buffered timestamp, in seconds, if |buffered|
  18. * is non-empty; otherwise, return null.
  19. */
  20. static bufferStart(b) {
  21. if (!b) {
  22. return null;
  23. }
  24. // Workaround Safari bug: https://bit.ly/2trx6O8
  25. if (b.length == 1 && b.end(0) - b.start(0) < 1e-6) {
  26. return null;
  27. }
  28. // Workaround Edge bug: https://bit.ly/2JYLPeB
  29. if (b.length == 1 && b.start(0) < 0) {
  30. return 0;
  31. }
  32. return b.length ? b.start(0) : null;
  33. }
  34. /**
  35. * Gets the last timestamp in the buffer.
  36. *
  37. * @param {TimeRanges} b
  38. * @return {?number} The last buffered timestamp, in seconds, if |buffered|
  39. * is non-empty; otherwise, return null.
  40. */
  41. static bufferEnd(b) {
  42. if (!b) {
  43. return null;
  44. }
  45. // Workaround Safari bug: https://bit.ly/2trx6O8
  46. if (b.length == 1 && b.end(0) - b.start(0) < 1e-6) {
  47. return null;
  48. }
  49. return b.length ? b.end(b.length - 1) : null;
  50. }
  51. /**
  52. * Determines if the given time is inside a buffered range. This includes
  53. * gaps, meaning that if the playhead is in a gap, it is considered buffered.
  54. * If there is a small gap between the playhead and buffer start, consider it
  55. * as buffered.
  56. *
  57. * @param {TimeRanges} b
  58. * @param {number} time Playhead time
  59. * @param {number=} smallGapLimit Set in configuration
  60. * @return {boolean}
  61. */
  62. static isBuffered(b, time, smallGapLimit = 0) {
  63. if (!b || !b.length) {
  64. return false;
  65. }
  66. // Workaround Safari bug: https://bit.ly/2trx6O8
  67. if (b.length == 1 && b.end(0) - b.start(0) < 1e-6) {
  68. return false;
  69. }
  70. if (time > b.end(b.length - 1)) {
  71. return false;
  72. }
  73. // Push the time forward by the gap limit so that it is more likely to be in
  74. // the range.
  75. return (time + smallGapLimit >= b.start(0));
  76. }
  77. /**
  78. * Computes how far ahead of the given timestamp is buffered. To provide
  79. * smooth playback while jumping gaps, we don't include the gaps when
  80. * calculating this.
  81. * This only includes the amount of content that is buffered.
  82. *
  83. * @param {TimeRanges} b
  84. * @param {number} time
  85. * @return {number} The number of seconds buffered, in seconds, ahead of the
  86. * given time.
  87. */
  88. static bufferedAheadOf(b, time) {
  89. if (!b || !b.length) {
  90. return 0;
  91. }
  92. // Workaround Safari bug: https://bit.ly/2trx6O8
  93. if (b.length == 1 && b.end(0) - b.start(0) < 1e-6) {
  94. return 0;
  95. }
  96. // NOTE: On IE11, buffered ranges may show appended data before the
  97. // associated append operation is complete.
  98. // We calculate the buffered amount by ONLY accounting for the content
  99. // buffered (i.e. we ignore the times of the gaps). We also buffer through
  100. // all gaps.
  101. // Therefore, we start at the end and add up all buffers until |time|.
  102. let result = 0;
  103. for (const {start, end} of shaka.media.TimeRangesUtils.getBufferedInfo(b)) {
  104. if (end > time) {
  105. result += end - Math.max(start, time);
  106. }
  107. }
  108. return result;
  109. }
  110. /**
  111. * Determines if the given time is inside a gap between buffered ranges. If
  112. * it is, this returns the index of the buffer that is *ahead* of the gap.
  113. *
  114. * @param {TimeRanges} b
  115. * @param {number} time
  116. * @return {?number} The index of the buffer after the gap, or null if not in
  117. * a gap.
  118. */
  119. static getGapIndex(b, time) {
  120. const Platform = shaka.util.Platform;
  121. const TimeRangesUtils = shaka.media.TimeRangesUtils;
  122. if (!b || !b.length) {
  123. return null;
  124. }
  125. // Workaround Safari bug: https://bit.ly/2trx6O8
  126. if (b.length == 1 && b.end(0) - b.start(0) < 1e-6) {
  127. return null;
  128. }
  129. // Some browsers will stop earlier than others before a gap (e.g. IE/Edge
  130. // stops 0.5 seconds before a gap). So for some browsers we need to use a
  131. // larger threshold. See: https://bit.ly/2K5xmJO
  132. const useLargeThreshold = Platform.isLegacyEdge() ||
  133. Platform.isIE() ||
  134. Platform.isTizen() ||
  135. Platform.isChromecast();
  136. const threshold = useLargeThreshold ? 0.5 : 0.1;
  137. const idx = TimeRangesUtils.getBufferedInfo(b).findIndex((item, i, arr) => {
  138. return item.start > time &&
  139. (i == 0 || arr[i - 1].end - time <= threshold);
  140. });
  141. return idx >= 0 ? idx : null;
  142. }
  143. /**
  144. * @param {TimeRanges} b
  145. * @return {!Array.<shaka.extern.BufferedRange>}
  146. */
  147. static getBufferedInfo(b) {
  148. if (!b) {
  149. return [];
  150. }
  151. const ret = [];
  152. for (const i of shaka.util.Iterables.range(b.length)) {
  153. ret.push({start: b.start(i), end: b.end(i)});
  154. }
  155. return ret;
  156. }
  157. };