mirror of
https://github.com/shaka-project/shaka-player.git
synced 2026-06-24 17:35:10 +03:00
4cc4e96dbd
* Updates all Copyright years to 2015. * Adds licenses annotations to all JS. * Makes all licenses identical to avoid repeated appearance in the compiled output. * Drops fileoverview annotations, which do not affect docs output. * The linter still requires fileoverview on externs. This patch required a newer closure compiler, since the previous version we used had a bug regarding license annotations that caused the license comment block to appear in the output once per file regardless of uniqueness. Change-Id: I2e9272db680cba7ecc4613d97f1d3a94ac2244cc
288 lines
8.6 KiB
JavaScript
288 lines
8.6 KiB
JavaScript
/**
|
|
* @license
|
|
* Copyright 2015 Google Inc.
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
goog.provide('shaka.media.SegmentIndex');
|
|
|
|
goog.require('shaka.asserts');
|
|
goog.require('shaka.media.SegmentReference');
|
|
goog.require('shaka.util.ArrayUtils');
|
|
|
|
|
|
|
|
/**
|
|
* Creates a SegmentIndex, which maintains a set of SegmentReferences sorted by
|
|
* their start times.
|
|
*
|
|
* @param {!Array.<!shaka.media.SegmentReference>} references The set of
|
|
* SegmentReferences, which must be sorted by their start times.
|
|
*
|
|
* @constructor
|
|
* @struct
|
|
*/
|
|
shaka.media.SegmentIndex = function(references) {
|
|
/** @protected {!Array.<!shaka.media.SegmentReference>} */
|
|
this.references = references;
|
|
|
|
/** @protected {number} */
|
|
this.timestampCorrection = 0;
|
|
};
|
|
|
|
|
|
/**
|
|
* Destroys this SegmentIndex.
|
|
* @suppress {checkTypes} to set otherwise non-nullable types to null.
|
|
*/
|
|
shaka.media.SegmentIndex.prototype.destroy = function() {
|
|
this.references = null;
|
|
};
|
|
|
|
|
|
/**
|
|
* Gets the number of SegmentReferences.
|
|
*
|
|
* @return {number}
|
|
*/
|
|
shaka.media.SegmentIndex.prototype.length = function() {
|
|
return this.references.length;
|
|
};
|
|
|
|
|
|
/**
|
|
* Gets the first SegmentReference.
|
|
*
|
|
* @return {!shaka.media.SegmentReference} The first SegmentReference.
|
|
* @throws {RangeError} when there are no SegmentReferences.
|
|
*/
|
|
shaka.media.SegmentIndex.prototype.first = function() {
|
|
if (this.references.length == 0) {
|
|
throw new RangeError('SegmentIndex: There is no first SegmentReference.');
|
|
}
|
|
return this.references[0];
|
|
};
|
|
|
|
|
|
/**
|
|
* Gets the last SegmentReference.
|
|
*
|
|
* @return {!shaka.media.SegmentReference} The last SegmentReference.
|
|
* @throws {RangeError} when there are no SegmentReferences.
|
|
*/
|
|
shaka.media.SegmentIndex.prototype.last = function() {
|
|
if (this.references.length == 0) {
|
|
throw new RangeError('SegmentIndex: There is no last SegmentReference.');
|
|
}
|
|
return this.references[this.references.length - 1];
|
|
};
|
|
|
|
|
|
/**
|
|
* Gets the SegmentReference at the given index.
|
|
*
|
|
* @param {number} index
|
|
* @return {!shaka.media.SegmentReference}
|
|
* @throws {RangeError} when |index| is out of range.
|
|
*/
|
|
shaka.media.SegmentIndex.prototype.get = function(index) {
|
|
if (index < 0 || index >= this.references.length) {
|
|
throw new RangeError('SegmentIndex: The specified index is out of range.');
|
|
}
|
|
return this.references[index];
|
|
};
|
|
|
|
|
|
/**
|
|
* Finds a SegmentReference for the specified time.
|
|
*
|
|
* This function can trigger an update, which may add or remove
|
|
* SegmentReferences.
|
|
*
|
|
* @param {number} time The time in seconds.
|
|
* @return {shaka.media.SegmentReference} The SegmentReference for the
|
|
* specified time, or null if no such SegmentReference exists.
|
|
*/
|
|
shaka.media.SegmentIndex.prototype.find = function(time) {
|
|
var i = shaka.media.SegmentReference.find(this.references, time);
|
|
return i >= 0 ? this.references[i] : null;
|
|
};
|
|
|
|
|
|
/**
|
|
* Integrates |segmentIndex| into this SegmentIndex. "Integration" is
|
|
* implementation dependent, but can be assumed to combine the two
|
|
* SegmentIndexes somehow. Assumes that both SegmentIndexes correspond to the
|
|
* same stream (e.g., the same Representation).
|
|
*
|
|
* This function can trigger an update, which may add or remove
|
|
* SegmentReferences independent of integration.
|
|
*
|
|
* The default implementation merges |segmentIndex| into this SegmentIndex if
|
|
* it covers times greater than or equal to times that this SegmentIndex
|
|
* covers.
|
|
*
|
|
* @param {!shaka.media.SegmentIndex} segmentIndex
|
|
* @return {boolean} True on success; otherwise, return false.
|
|
*/
|
|
shaka.media.SegmentIndex.prototype.integrate = function(segmentIndex) {
|
|
this.merge(segmentIndex);
|
|
return true;
|
|
};
|
|
|
|
|
|
/**
|
|
* Merges |segmentIndex| into this SegmentIndex, but only if it covers times
|
|
* greater than or equal to times that this SegmentIndex covers.
|
|
*
|
|
* Takes into account timestamp corrections.
|
|
*
|
|
* @param {!shaka.media.SegmentIndex} segmentIndex
|
|
* @protected
|
|
*/
|
|
shaka.media.SegmentIndex.prototype.merge = function(segmentIndex) {
|
|
if (this.timestampCorrection != segmentIndex.timestampCorrection) {
|
|
var delta = this.timestampCorrection - segmentIndex.timestampCorrection;
|
|
shaka.log.v2(
|
|
'Shifting new SegmentReferences by', delta, 'seconds before merging.');
|
|
segmentIndex = new shaka.media.SegmentIndex(
|
|
shaka.media.SegmentReference.shift(segmentIndex.references, delta));
|
|
}
|
|
|
|
if (this.length() == 0) {
|
|
this.references = segmentIndex.references.slice(0);
|
|
this.assertCorrectReferences();
|
|
return;
|
|
}
|
|
|
|
if (segmentIndex.length() == 0) {
|
|
shaka.log.debug('Nothing to merge: new SegmentIndex is empty.');
|
|
return;
|
|
}
|
|
|
|
if (this.last().endTime == null) {
|
|
shaka.log.debug(
|
|
'Nothing to merge:',
|
|
'existing SegmentIndex ends at the end of the stream.');
|
|
return;
|
|
}
|
|
|
|
if ((segmentIndex.last().endTime != null) &&
|
|
(segmentIndex.last().endTime < this.last().endTime)) {
|
|
shaka.log.debug(
|
|
'Nothing to merge:',
|
|
'new SegmentIndex ends before the existing one ends.');
|
|
return;
|
|
}
|
|
|
|
// The new SegmentIndex starts after the existing SegmentIndex.
|
|
if (this.last().endTime <= segmentIndex.first().startTime) {
|
|
// Adjust the last existing segment so that it starts at the the start of
|
|
// the first new segment.
|
|
var adjustedReference = this.last().adjust(
|
|
this.last().startTime, segmentIndex.first().startTime);
|
|
var head = this.references.slice(0, -1).concat([adjustedReference]);
|
|
this.references = head.concat(segmentIndex.references);
|
|
this.assertCorrectReferences();
|
|
return;
|
|
}
|
|
|
|
// The new SegmentIndex starts before or in the middle of the existing
|
|
// SegmentIndex.
|
|
var i;
|
|
for (i = 0; i < this.references.length; ++i) {
|
|
if (this.references[i].endTime >= segmentIndex.first().startTime) {
|
|
break;
|
|
}
|
|
}
|
|
shaka.asserts.assert(i < this.references.length);
|
|
|
|
var head;
|
|
if (this.references[i].startTime < segmentIndex.first().startTime) {
|
|
// The first new segment starts in the middle of an existing segment, so
|
|
// compress the existing segment such that it ends at the start of the
|
|
// first new segment.
|
|
var adjustedReference = this.references[i].adjust(
|
|
this.references[i].startTime, segmentIndex.first().startTime);
|
|
head = this.references.slice(0, i).concat([adjustedReference]);
|
|
} else {
|
|
// The first new segment either starts before all existing segments or at
|
|
// the start of an existing segment.
|
|
shaka.asserts.assert(
|
|
(this.first().startTime > segmentIndex.first().startTime) ||
|
|
(this.references[i].startTime == segmentIndex.first().startTime));
|
|
head = this.references.slice(0, i);
|
|
}
|
|
this.references = head.concat(segmentIndex.references);
|
|
this.assertCorrectReferences();
|
|
};
|
|
|
|
|
|
/**
|
|
* Corrects each SegmentReference by the given timestamp correction. The
|
|
* previous timestamp correction, if it exists, is replaced.
|
|
*
|
|
* @param {number} timestampCorrection
|
|
* @return {number} The amount the SegmentReferences were shifted by.
|
|
*/
|
|
shaka.media.SegmentIndex.prototype.correct = function(timestampCorrection) {
|
|
var delta = timestampCorrection - this.timestampCorrection;
|
|
if (delta == 0) {
|
|
shaka.log.v2(
|
|
'Already applied timestamp correction of',
|
|
timestampCorrection,
|
|
'seconds to',
|
|
this);
|
|
return 0;
|
|
}
|
|
|
|
this.references = shaka.media.SegmentReference.shift(this.references, delta);
|
|
this.assertCorrectReferences();
|
|
this.timestampCorrection = timestampCorrection;
|
|
|
|
shaka.log.debug(
|
|
'Applied timestamp correction of',
|
|
timestampCorrection,
|
|
'seconds to SegmentIndex',
|
|
this);
|
|
return delta;
|
|
};
|
|
|
|
|
|
/**
|
|
* Gets the SegmentIndex's seek range. By default the SegmentIndex's entire
|
|
* span is seekable.
|
|
*
|
|
* @return {{start: number, end: ?number}} The seek range. If |end| is null
|
|
* then the seek end time continues to the end of the stream.
|
|
*/
|
|
shaka.media.SegmentIndex.prototype.getSeekRange = function() {
|
|
return this.length() > 0 ?
|
|
{ start: this.first().startTime, end: this.last().endTime } :
|
|
{ start: 0, end: 0 };
|
|
};
|
|
|
|
|
|
/**
|
|
* Asserts that the SegmentReferences meet all requirements.
|
|
*
|
|
* For debugging purposes.
|
|
*
|
|
* @protected
|
|
*/
|
|
shaka.media.SegmentIndex.prototype.assertCorrectReferences = function() {
|
|
shaka.media.SegmentReference.assertCorrectReferences(this.references);
|
|
};
|
|
|