1
0
Fork 0
mirror of https://github.com/jellyfin/jellyfin-web synced 2025-03-30 19:56:21 +00:00
jellyfin-web/dashboard-ui/bower_components/hls.js/src/controller/stream-controller.js

1193 lines
46 KiB
JavaScript
Raw Normal View History

2015-12-16 00:30:14 -05:00
/*
2016-02-24 22:15:07 -05:00
* Stream Controller
2015-12-16 00:30:14 -05:00
*/
import Demuxer from '../demux/demuxer';
import Event from '../events';
2016-01-18 14:07:26 -05:00
import EventHandler from '../event-handler';
2015-12-16 00:30:14 -05:00
import {logger} from '../utils/logger';
import BinarySearch from '../utils/binary-search';
import LevelHelper from '../helper/level-helper';
import {ErrorTypes, ErrorDetails} from '../errors';
const State = {
2016-02-24 22:15:07 -05:00
ERROR : 'ERROR',
STARTING : 'STARTING',
IDLE : 'IDLE',
PAUSED : 'PAUSED',
KEY_LOADING : 'KEY_LOADING',
FRAG_LOADING : 'FRAG_LOADING',
FRAG_LOADING_WAITING_RETRY : 'FRAG_LOADING_WAITING_RETRY',
WAITING_LEVEL : 'WAITING_LEVEL',
PARSING : 'PARSING',
PARSED : 'PARSED',
ENDED : 'ENDED'
2015-12-16 00:30:14 -05:00
};
2016-02-24 22:15:07 -05:00
class StreamController extends EventHandler {
2015-12-16 00:30:14 -05:00
constructor(hls) {
2016-02-24 22:15:07 -05:00
super(hls,
Event.MEDIA_ATTACHED,
2016-01-18 14:07:26 -05:00
Event.MEDIA_DETACHING,
Event.MANIFEST_PARSED,
Event.LEVEL_LOADED,
Event.KEY_LOADED,
Event.FRAG_LOADED,
Event.FRAG_PARSING_INIT_SEGMENT,
Event.FRAG_PARSING_DATA,
Event.FRAG_PARSED,
2016-02-24 22:15:07 -05:00
Event.ERROR,
Event.BUFFER_APPENDED,
Event.BUFFER_FLUSHED);
2015-12-16 00:30:14 -05:00
this.config = hls.config;
this.audioCodecSwap = false;
2016-01-13 15:58:12 -05:00
this.ticks = 0;
2015-12-16 00:30:14 -05:00
this.ontick = this.tick.bind(this);
}
destroy() {
this.stop();
2016-01-18 14:07:26 -05:00
EventHandler.prototype.destroy.call(this);
2015-12-16 00:30:14 -05:00
this.state = State.IDLE;
}
startLoad() {
2016-02-03 18:00:01 -05:00
if (this.levels) {
var media = this.media, lastCurrentTime = this.lastCurrentTime;
2016-02-24 22:15:07 -05:00
this.stop();
this.demuxer = new Demuxer(this.hls);
this.timer = setInterval(this.ontick, 100);
this.level = -1;
this.fragLoadError = 0;
2016-02-03 18:00:01 -05:00
if (media && lastCurrentTime) {
2016-02-24 22:15:07 -05:00
logger.log(`configure startPosition @${lastCurrentTime}`);
2015-12-16 00:30:14 -05:00
if (!this.lastPaused) {
logger.log('resuming video');
2016-02-03 18:00:01 -05:00
media.play();
2015-12-16 00:30:14 -05:00
}
this.state = State.IDLE;
} else {
2016-02-24 22:15:07 -05:00
this.lastCurrentTime = this.startPosition ? this.startPosition : 0;
2015-12-16 00:30:14 -05:00
this.state = State.STARTING;
}
this.nextLoadPosition = this.startPosition = this.lastCurrentTime;
this.tick();
} else {
2016-02-03 18:00:01 -05:00
logger.warn('cannot start loading as manifest not parsed yet');
2015-12-16 00:30:14 -05:00
}
}
stop() {
this.bufferRange = [];
2016-01-25 15:28:29 -05:00
this.stalled = false;
2015-12-16 00:30:14 -05:00
var frag = this.fragCurrent;
if (frag) {
if (frag.loader) {
frag.loader.abort();
}
this.fragCurrent = null;
}
this.fragPrevious = null;
2016-02-24 22:15:07 -05:00
logger.log('trigger BUFFER_RESET');
this.hls.trigger(Event.BUFFER_RESET);
2015-12-16 00:30:14 -05:00
if (this.timer) {
clearInterval(this.timer);
this.timer = null;
}
if (this.demuxer) {
this.demuxer.destroy();
this.demuxer = null;
}
}
tick() {
2016-01-13 15:58:12 -05:00
this.ticks++;
if (this.ticks === 1) {
this.doTick();
if (this.ticks > 1) {
setTimeout(this.tick, 1);
}
this.ticks = 0;
}
}
doTick() {
2016-02-24 22:15:07 -05:00
var pos, level, levelDetails, hls = this.hls, config = hls.config;
//logger.log(this.state);
2015-12-16 00:30:14 -05:00
switch(this.state) {
case State.ERROR:
//don't do anything in error state to avoid breaking further ...
2016-02-24 22:15:07 -05:00
case State.PAUSED:
//don't do anything in paused state either ...
2015-12-16 00:30:14 -05:00
break;
case State.STARTING:
// determine load level
this.startLevel = hls.startLevel;
if (this.startLevel === -1) {
// -1 : guess start Level by doing a bitrate test by loading first fragment of lowest quality level
this.startLevel = 0;
this.fragBitrateTest = true;
}
// set new level to playlist loader : this will trigger start level load
this.level = hls.nextLoadLevel = this.startLevel;
this.state = State.WAITING_LEVEL;
this.loadedmetadata = false;
break;
case State.IDLE:
2016-02-24 22:15:07 -05:00
// if video not attached AND
// start fragment already requested OR start frag prefetch disable
// exit loop
// => if media not attached but start frag prefetch is enabled and start frag not requested yet, we will not exit loop
if (!this.media &&
(this.startFragRequested || !config.startFragPrefetch)) {
2015-12-16 00:30:14 -05:00
break;
}
// determine next candidate fragment to be loaded, based on current position and
// end of buffer position
// ensure 60s of buffer upfront
// if we have not yet loaded any fragment, start loading from start position
if (this.loadedmetadata) {
pos = this.media.currentTime;
} else {
pos = this.nextLoadPosition;
}
// determine next load level
2016-02-24 22:15:07 -05:00
if (this.startFragRequested === false) {
2015-12-16 00:30:14 -05:00
level = this.startLevel;
} else {
// we are not at playback start, get next load level from level Controller
level = hls.nextLoadLevel;
}
2016-02-24 22:15:07 -05:00
var bufferInfo = this.bufferInfo(pos,config.maxBufferHole),
2015-12-16 00:30:14 -05:00
bufferLen = bufferInfo.len,
bufferEnd = bufferInfo.end,
fragPrevious = this.fragPrevious,
maxBufLen;
// compute max Buffer Length that we could get from this load level, based on level bitrate. don't buffer more than 60 MB and more than 30s
if ((this.levels[level]).hasOwnProperty('bitrate')) {
2016-02-24 22:15:07 -05:00
maxBufLen = Math.max(8 * config.maxBufferSize / this.levels[level].bitrate, config.maxBufferLength);
maxBufLen = Math.min(maxBufLen, config.maxMaxBufferLength);
2015-12-16 00:30:14 -05:00
} else {
2016-02-24 22:15:07 -05:00
maxBufLen = config.maxBufferLength;
2015-12-16 00:30:14 -05:00
}
// if buffer length is less than maxBufLen try to load a new fragment
if (bufferLen < maxBufLen) {
// set next load level : this will trigger a playlist load if needed
hls.nextLoadLevel = level;
this.level = level;
levelDetails = this.levels[level].details;
// if level info not retrieved yet, switch state and wait for level retrieval
2016-01-18 14:07:26 -05:00
// if live playlist, ensure that new playlist has been refreshed to avoid loading/try to load
// a useless and outdated fragment (that might even introduce load error if it is already out of the live playlist)
if (typeof levelDetails === 'undefined' || levelDetails.live && this.levelLastLoaded !== level) {
2015-12-16 00:30:14 -05:00
this.state = State.WAITING_LEVEL;
break;
}
// find fragment index, contiguous with end of buffer position
let fragments = levelDetails.fragments,
fragLen = fragments.length,
start = fragments[0].start,
end = fragments[fragLen-1].start + fragments[fragLen-1].duration,
frag;
// in case of live playlist we need to ensure that requested position is not located before playlist start
if (levelDetails.live) {
// check if requested position is within seekable boundaries :
//logger.log(`start/pos/bufEnd/seeking:${start.toFixed(3)}/${pos.toFixed(3)}/${bufferEnd.toFixed(3)}/${this.media.seeking}`);
2016-03-09 12:40:22 -05:00
let maxLatency = config.liveMaxLatencyDuration !== undefined ? config.liveMaxLatencyDuration : config.liveMaxLatencyDurationCount*levelDetails.targetduration;
if (bufferEnd < Math.max(start, end - maxLatency)) {
let targetLatency = config.liveSyncDuration !== undefined ? config.liveSyncDuration : config.liveSyncDurationCount * levelDetails.targetduration;
this.seekAfterBuffered = start + Math.max(0, levelDetails.totalduration - targetLatency);
2015-12-16 00:30:14 -05:00
logger.log(`buffer end: ${bufferEnd} is located too far from the end of live sliding playlist, media position will be reseted to: ${this.seekAfterBuffered.toFixed(3)}`);
bufferEnd = this.seekAfterBuffered;
}
2016-02-24 22:15:07 -05:00
if (this.startFragRequested && !levelDetails.PTSKnown) {
2015-12-16 00:30:14 -05:00
/* we are switching level on live playlist, but we don't have any PTS info for that quality level ...
try to load frag matching with next SN.
even if SN are not synchronized between playlists, loading this frag will help us
compute playlist sliding and find the right one after in case it was not the right consecutive one */
if (fragPrevious) {
var targetSN = fragPrevious.sn + 1;
if (targetSN >= levelDetails.startSN && targetSN <= levelDetails.endSN) {
frag = fragments[targetSN - levelDetails.startSN];
logger.log(`live playlist, switching playlist, load frag with next SN: ${frag.sn}`);
}
}
if (!frag) {
/* we have no idea about which fragment should be loaded.
so let's load mid fragment. it will help computing playlist sliding and find the right one
*/
frag = fragments[Math.min(fragLen - 1, Math.round(fragLen / 2))];
logger.log(`live playlist, switching playlist, unknown, load middle frag : ${frag.sn}`);
}
}
} else {
// VoD playlist: if bufferEnd before start of playlist, load first fragment
if (bufferEnd < start) {
frag = fragments[0];
}
}
if (!frag) {
2016-03-16 13:43:01 -04:00
let foundFrag;
let maxFragLookUpTolerance = config.maxFragLookUpTolerance;
2015-12-16 00:30:14 -05:00
if (bufferEnd < end) {
2016-03-16 13:43:01 -04:00
if (bufferEnd > end - maxFragLookUpTolerance) {
maxFragLookUpTolerance = 0;
}
2015-12-16 00:30:14 -05:00
foundFrag = BinarySearch.search(fragments, (candidate) => {
2016-03-16 13:43:01 -04:00
// offset should be within fragment boundary - config.maxFragLookUpTolerance
// this is to cope with situations like
// bufferEnd = 9.991
// frag[Ø] : [0,10]
// frag[1] : [10,20]
// bufferEnd is within frag[0] range ... although what we are expecting is to return frag[1] here
// frag start frag start+duration
// |-----------------------------|
// <---> <--->
// ...--------><-----------------------------><---------....
// previous frag matching fragment next frag
// return -1 return 0 return 1
2015-12-16 00:30:14 -05:00
//logger.log(`level/sn/start/end/bufEnd:${level}/${candidate.sn}/${candidate.start}/${(candidate.start+candidate.duration)}/${bufferEnd}`);
2016-03-16 13:43:01 -04:00
if ((candidate.start + candidate.duration - maxFragLookUpTolerance) <= bufferEnd) {
2015-12-16 00:30:14 -05:00
return 1;
}
2016-03-16 13:43:01 -04:00
else if (candidate.start - maxFragLookUpTolerance > bufferEnd) {
2015-12-16 00:30:14 -05:00
return -1;
}
return 0;
});
} else {
// reach end of playlist
foundFrag = fragments[fragLen-1];
}
if (foundFrag) {
frag = foundFrag;
start = foundFrag.start;
//logger.log('find SN matching with pos:' + bufferEnd + ':' + frag.sn);
if (fragPrevious && frag.level === fragPrevious.level && frag.sn === fragPrevious.sn) {
if (frag.sn < levelDetails.endSN) {
frag = fragments[frag.sn + 1 - levelDetails.startSN];
logger.log(`SN just loaded, load next one: ${frag.sn}`);
} else {
// have we reached end of VOD playlist ?
if (!levelDetails.live) {
2016-02-24 22:15:07 -05:00
this.hls.trigger(Event.BUFFER_EOS);
this.state = State.ENDED;
2015-12-16 00:30:14 -05:00
}
frag = null;
}
}
}
}
if(frag) {
//logger.log(' loading frag ' + i +',pos/bufEnd:' + pos.toFixed(3) + '/' + bufferEnd.toFixed(3));
if ((frag.decryptdata.uri != null) && (frag.decryptdata.key == null)) {
logger.log(`Loading key for ${frag.sn} of [${levelDetails.startSN} ,${levelDetails.endSN}],level ${level}`);
this.state = State.KEY_LOADING;
hls.trigger(Event.KEY_LOADING, {frag: frag});
} else {
logger.log(`Loading ${frag.sn} of [${levelDetails.startSN} ,${levelDetails.endSN}],level ${level}, currentTime:${pos},bufferEnd:${bufferEnd.toFixed(3)}`);
frag.autoLevel = hls.autoLevelEnabled;
if (this.levels.length > 1) {
frag.expectedLen = Math.round(frag.duration * this.levels[level].bitrate / 8);
frag.trequest = performance.now();
}
// ensure that we are not reloading the same fragments in loop ...
if (this.fragLoadIdx !== undefined) {
this.fragLoadIdx++;
} else {
this.fragLoadIdx = 0;
}
if (frag.loadCounter) {
frag.loadCounter++;
2016-02-24 22:15:07 -05:00
let maxThreshold = config.fragLoadingLoopThreshold;
2015-12-16 00:30:14 -05:00
// if this frag has already been loaded 3 times, and if it has been reloaded recently
if (frag.loadCounter > maxThreshold && (Math.abs(this.fragLoadIdx - frag.loadIdx) < maxThreshold)) {
hls.trigger(Event.ERROR, {type: ErrorTypes.MEDIA_ERROR, details: ErrorDetails.FRAG_LOOP_LOADING_ERROR, fatal: false, frag: frag});
return;
}
} else {
frag.loadCounter = 1;
}
frag.loadIdx = this.fragLoadIdx;
this.fragCurrent = frag;
2016-02-24 22:15:07 -05:00
this.startFragRequested = true;
2015-12-16 00:30:14 -05:00
hls.trigger(Event.FRAG_LOADING, {frag: frag});
this.state = State.FRAG_LOADING;
}
}
}
break;
case State.WAITING_LEVEL:
level = this.levels[this.level];
// check if playlist is already loaded
if (level && level.details) {
this.state = State.IDLE;
}
break;
case State.FRAG_LOADING:
/*
monitor fragment retrieval time...
we compute expected time of arrival of the complete fragment.
we compare it to expected time of buffer starvation
*/
let v = this.media,frag = this.fragCurrent;
/* only monitor frag retrieval time if
(video not paused OR first fragment being loaded) AND autoswitching enabled AND not lowest level AND multiple levels */
if (v && (!v.paused || this.loadedmetadata === false) && frag.autoLevel && this.level && this.levels.length > 1) {
2016-03-16 13:43:01 -04:00
let requestDelay = performance.now() - frag.trequest;
2015-12-16 00:30:14 -05:00
// monitor fragment load progress after half of expected fragment duration,to stabilize bitrate
if (requestDelay > (500 * frag.duration)) {
2016-03-16 13:43:01 -04:00
let loadRate = Math.max(1,frag.loaded * 1000 / requestDelay); // byte/s; at least 1 byte/s to avoid division by zero
2015-12-16 00:30:14 -05:00
if (frag.expectedLen < frag.loaded) {
frag.expectedLen = frag.loaded;
}
pos = v.currentTime;
2016-03-16 13:43:01 -04:00
let fragLoadedDelay = (frag.expectedLen - frag.loaded) / loadRate;
let bufferStarvationDelay = this.bufferInfo(pos,config.maxBufferHole).end - pos;
// consider emergency switch down only if we have less than 2 frag buffered AND
// time to finish loading current fragment is bigger than buffer starvation delay
// ie if we risk buffer starvation if bw does not increase quickly
if (bufferStarvationDelay < 2*frag.duration && fragLoadedDelay > bufferStarvationDelay) {
let fragLevelNextLoadedDelay, nextLoadLevel;
// lets iterate through lower level and try to find the biggest one that could avoid rebuffering
// we start from current level - 1 and we step down , until we find a matching level
for (nextLoadLevel = this.level - 1 ; nextLoadLevel >=0 ; nextLoadLevel--) {
// compute time to load next fragment at lower level
// 0.8 : consider only 80% of current bw to be conservative
// 8 = bits per byte (bps/Bps)
fragLevelNextLoadedDelay = frag.duration * this.levels[nextLoadLevel].bitrate / (8 * 0.8 * loadRate);
logger.log(`fragLoadedDelay/bufferStarvationDelay/fragLevelNextLoadedDelay[${nextLoadLevel}] :${fragLoadedDelay.toFixed(1)}/${bufferStarvationDelay.toFixed(1)}/${fragLevelNextLoadedDelay.toFixed(1)}`);
if (fragLevelNextLoadedDelay < bufferStarvationDelay) {
// we found a lower level that be rebuffering free with current estimated bw !
break;
}
}
// only emergency switch down if it takes less time to load new fragment at lowest level instead
// of finishing loading current one ...
if (fragLevelNextLoadedDelay < fragLoadedDelay) {
// ensure nextLoadLevel is not negative
nextLoadLevel = Math.max(0,nextLoadLevel);
// force next load level in auto mode
hls.nextLoadLevel = nextLoadLevel;
// abort fragment loading ...
logger.warn(`loading too slow, abort fragment loading and switch to level ${nextLoadLevel}`);
//abort fragment loading
frag.loader.abort();
hls.trigger(Event.FRAG_LOAD_EMERGENCY_ABORTED, {frag: frag});
// switch back to IDLE state to request new fragment at lower level
this.state = State.IDLE;
}
2015-12-16 00:30:14 -05:00
}
}
}
break;
2016-01-13 15:58:12 -05:00
case State.FRAG_LOADING_WAITING_RETRY:
var now = performance.now();
var retryDate = this.retryDate;
var media = this.media;
var isSeeking = media && media.seeking;
// if current time is gt than retryDate, or if media seeking let's switch to IDLE state to retry loading
if(!retryDate || (now >= retryDate) || isSeeking) {
logger.log(`mediaController: retryDate reached, switch back to IDLE state`);
this.state = State.IDLE;
}
break;
2015-12-16 00:30:14 -05:00
case State.PARSING:
// nothing to do, wait for fragment being parsed
break;
case State.PARSED:
2016-02-24 22:15:07 -05:00
// nothing to do, wait for all buffers to be appended
2015-12-16 00:30:14 -05:00
break;
2016-01-25 15:28:29 -05:00
case State.ENDED:
break;
2015-12-16 00:30:14 -05:00
default:
break;
}
// check buffer
this._checkBuffer();
2016-01-13 15:58:12 -05:00
// check/update current fragment
this._checkFragmentChanged();
2015-12-16 00:30:14 -05:00
}
bufferInfo(pos,maxHoleDuration) {
2016-02-24 22:15:07 -05:00
var media = this.media;
if (media) {
var vbuffered = media.buffered, buffered = [],i;
for (i = 0; i < vbuffered.length; i++) {
buffered.push({start: vbuffered.start(i), end: vbuffered.end(i)});
}
return this.bufferedInfo(buffered,pos,maxHoleDuration);
} else {
return {len: 0, start: 0, end: 0, nextStart : undefined} ;
2015-12-16 00:30:14 -05:00
}
}
bufferedInfo(buffered,pos,maxHoleDuration) {
var buffered2 = [],
// bufferStart and bufferEnd are buffer boundaries around current video position
bufferLen,bufferStart, bufferEnd,bufferStartNext,i;
// sort on buffer.start/smaller end (IE does not always return sorted buffered range)
buffered.sort(function (a, b) {
var diff = a.start - b.start;
if (diff) {
return diff;
} else {
return b.end - a.end;
}
});
// there might be some small holes between buffer time range
// consider that holes smaller than maxHoleDuration are irrelevant and build another
// buffer time range representations that discards those holes
for (i = 0; i < buffered.length; i++) {
var buf2len = buffered2.length;
if(buf2len) {
var buf2end = buffered2[buf2len - 1].end;
// if small hole (value between 0 or maxHoleDuration ) or overlapping (negative)
if((buffered[i].start - buf2end) < maxHoleDuration) {
// merge overlapping time ranges
// update lastRange.end only if smaller than item.end
// e.g. [ 1, 15] with [ 2,8] => [ 1,15] (no need to modify lastRange.end)
// whereas [ 1, 8] with [ 2,15] => [ 1,15] ( lastRange should switch from [1,8] to [1,15])
if(buffered[i].end > buf2end) {
buffered2[buf2len - 1].end = buffered[i].end;
}
} else {
// big hole
buffered2.push(buffered[i]);
}
} else {
// first value
buffered2.push(buffered[i]);
}
}
for (i = 0, bufferLen = 0, bufferStart = bufferEnd = pos; i < buffered2.length; i++) {
var start = buffered2[i].start,
end = buffered2[i].end;
//logger.log('buf start/end:' + buffered.start(i) + '/' + buffered.end(i));
if ((pos + maxHoleDuration) >= start && pos < end) {
// play position is inside this buffer TimeRange, retrieve end of buffer position and buffer length
bufferStart = start;
2016-03-09 12:40:22 -05:00
bufferEnd = end;
2015-12-16 00:30:14 -05:00
bufferLen = bufferEnd - pos;
} else if ((pos + maxHoleDuration) < start) {
bufferStartNext = start;
2016-01-18 14:07:26 -05:00
break;
2015-12-16 00:30:14 -05:00
}
}
return {len: bufferLen, start: bufferStart, end: bufferEnd, nextStart : bufferStartNext};
}
getBufferRange(position) {
2016-03-09 12:40:22 -05:00
var i, range,
bufferRange = this.bufferRange;
if (bufferRange) {
for (i = bufferRange.length - 1; i >=0; i--) {
range = bufferRange[i];
if (position >= range.start && position <= range.end) {
return range;
}
2015-12-16 00:30:14 -05:00
}
}
return null;
}
get currentLevel() {
if (this.media) {
var range = this.getBufferRange(this.media.currentTime);
if (range) {
return range.frag.level;
}
}
return -1;
}
get nextBufferRange() {
if (this.media) {
// first get end range of current fragment
return this.followingBufferRange(this.getBufferRange(this.media.currentTime));
} else {
return null;
}
}
followingBufferRange(range) {
if (range) {
// try to get range of next fragment (500ms after this range)
return this.getBufferRange(range.end + 0.5);
}
return null;
}
get nextLevel() {
var range = this.nextBufferRange;
if (range) {
return range.frag.level;
} else {
return -1;
}
}
isBuffered(position) {
var v = this.media, buffered = v.buffered;
for (var i = 0; i < buffered.length; i++) {
if (position >= buffered.start(i) && position <= buffered.end(i)) {
return true;
}
}
return false;
}
_checkFragmentChanged() {
var rangeCurrent, currentTime, video = this.media;
if (video && video.seeking === false) {
currentTime = video.currentTime;
/* if video element is in seeked state, currentTime can only increase.
(assuming that playback rate is positive ...)
As sometimes currentTime jumps back to zero after a
media decode error, check this, to avoid seeking back to
wrong position after a media decode error
*/
if(currentTime > video.playbackRate*this.lastCurrentTime) {
this.lastCurrentTime = currentTime;
}
if (this.isBuffered(currentTime)) {
rangeCurrent = this.getBufferRange(currentTime);
} else if (this.isBuffered(currentTime + 0.1)) {
/* ensure that FRAG_CHANGED event is triggered at startup,
when first video frame is displayed and playback is paused.
add a tolerance of 100ms, in case current position is not buffered,
check if current pos+100ms is buffered and use that buffer range
for FRAG_CHANGED event reporting */
rangeCurrent = this.getBufferRange(currentTime + 0.1);
}
if (rangeCurrent) {
var fragPlaying = rangeCurrent.frag;
if (fragPlaying !== this.fragPlaying) {
this.fragPlaying = fragPlaying;
this.hls.trigger(Event.FRAG_CHANGED, {frag: fragPlaying});
}
}
}
}
/*
on immediate level switch :
- pause playback if playing
- cancel any pending load request
- and trigger a buffer flush
*/
immediateLevelSwitch() {
logger.log('immediateLevelSwitch');
if (!this.immediateSwitch) {
this.immediateSwitch = true;
this.previouslyPaused = this.media.paused;
this.media.pause();
}
var fragCurrent = this.fragCurrent;
if (fragCurrent && fragCurrent.loader) {
fragCurrent.loader.abort();
}
this.fragCurrent = null;
// flush everything
2016-02-24 22:15:07 -05:00
this.hls.trigger(Event.BUFFER_FLUSHING, {startOffset: 0, endOffset: Number.POSITIVE_INFINITY});
this.state = State.PAUSED;
2015-12-16 00:30:14 -05:00
// increase fragment load Index to avoid frag loop loading error after buffer flush
this.fragLoadIdx += 2 * this.config.fragLoadingLoopThreshold;
// speed up switching, trigger timer function
this.tick();
}
/*
on immediate level switch end, after new fragment has been buffered :
- nudge video decoder by slightly adjusting video currentTime
- resume the playback if needed
*/
immediateLevelSwitchEnd() {
this.immediateSwitch = false;
this.media.currentTime -= 0.0001;
if (!this.previouslyPaused) {
this.media.play();
}
}
nextLevelSwitch() {
/* try to switch ASAP without breaking video playback :
in order to ensure smooth but quick level switching,
we need to find the next flushable buffer range
we should take into account new segment fetch time
*/
var fetchdelay, currentRange, nextRange;
currentRange = this.getBufferRange(this.media.currentTime);
2016-02-24 22:15:07 -05:00
if (currentRange && currentRange.start > 1) {
2015-12-16 00:30:14 -05:00
// flush buffer preceding current fragment (flush until current fragment start offset)
// minus 1s to avoid video freezing, that could happen if we flush keyframe of current video ...
2016-02-24 22:15:07 -05:00
this.hls.trigger(Event.BUFFER_FLUSHING, {startOffset: 0, endOffset: currentRange.start - 1});
this.state = State.PAUSED;
2015-12-16 00:30:14 -05:00
}
if (!this.media.paused) {
// add a safety delay of 1s
var nextLevelId = this.hls.nextLoadLevel,nextLevel = this.levels[nextLevelId], fragLastKbps = this.fragLastKbps;
if (fragLastKbps && this.fragCurrent) {
fetchdelay = this.fragCurrent.duration * nextLevel.bitrate / (1000 * fragLastKbps) + 1;
} else {
fetchdelay = 0;
}
} else {
fetchdelay = 0;
}
//logger.log('fetchdelay:'+fetchdelay);
// find buffer range that will be reached once new fragment will be fetched
nextRange = this.getBufferRange(this.media.currentTime + fetchdelay);
if (nextRange) {
// we can flush buffer range following this one without stalling playback
nextRange = this.followingBufferRange(nextRange);
if (nextRange) {
// flush position is the start position of this new buffer
2016-02-24 22:15:07 -05:00
this.hls.trigger(Event.BUFFER_FLUSHING, {startOffset: nextRange.start, endOffset: Number.POSITIVE_INFINITY});
this.state = State.PAUSED;
2015-12-16 00:30:14 -05:00
// if we are here, we can also cancel any loading/demuxing in progress, as they are useless
var fragCurrent = this.fragCurrent;
if (fragCurrent && fragCurrent.loader) {
fragCurrent.loader.abort();
}
this.fragCurrent = null;
2016-02-24 22:15:07 -05:00
// increase fragment load Index to avoid frag loop loading error after buffer flush
this.fragLoadIdx += 2 * this.config.fragLoadingLoopThreshold;
2015-12-16 00:30:14 -05:00
}
}
}
2016-02-24 22:15:07 -05:00
onMediaAttached(data) {
2015-12-16 00:30:14 -05:00
var media = this.media = data.media;
2016-02-24 22:15:07 -05:00
this.onvseeking = this.onMediaSeeking.bind(this);
this.onvseeked = this.onMediaSeeked.bind(this);
this.onvended = this.onMediaEnded.bind(this);
media.addEventListener('seeking', this.onvseeking);
media.addEventListener('seeked', this.onvseeked);
media.addEventListener('ended', this.onvended);
if(this.levels && this.config.autoStartLoad) {
this.startLoad();
}
2015-12-16 00:30:14 -05:00
}
onMediaDetaching() {
var media = this.media;
if (media && media.ended) {
logger.log('MSE detaching and video ended, reset startPosition');
this.startPosition = this.lastCurrentTime = 0;
}
// reset fragment loading counter on MSE detaching to avoid reporting FRAG_LOOP_LOADING_ERROR after error recovery
var levels = this.levels;
if (levels) {
// reset fragment load counter
levels.forEach(level => {
if(level.details) {
level.details.fragments.forEach(fragment => {
fragment.loadCounter = undefined;
});
}
});
}
2016-02-24 22:15:07 -05:00
// remove video listeners
if (media) {
media.removeEventListener('seeking', this.onvseeking);
media.removeEventListener('seeked', this.onvseeked);
media.removeEventListener('ended', this.onvended);
this.onvseeking = this.onvseeked = this.onvended = null;
2015-12-16 00:30:14 -05:00
}
2016-02-24 22:15:07 -05:00
this.media = null;
this.loadedmetadata = false;
this.stop();
2015-12-16 00:30:14 -05:00
}
onMediaSeeking() {
if (this.state === State.FRAG_LOADING) {
// check if currently loaded fragment is inside buffer.
//if outside, cancel fragment loading, otherwise do nothing
2016-01-18 14:07:26 -05:00
if (this.bufferInfo(this.media.currentTime,this.config.maxBufferHole).len === 0) {
2015-12-16 00:30:14 -05:00
logger.log('seeking outside of buffer while fragment load in progress, cancel fragment load');
var fragCurrent = this.fragCurrent;
if (fragCurrent) {
if (fragCurrent.loader) {
fragCurrent.loader.abort();
}
this.fragCurrent = null;
}
this.fragPrevious = null;
// switch to IDLE state to load new fragment
this.state = State.IDLE;
}
2016-01-25 15:28:29 -05:00
} else if (this.state === State.ENDED) {
// switch to IDLE state to check for potential new fragment
this.state = State.IDLE;
2015-12-16 00:30:14 -05:00
}
if (this.media) {
this.lastCurrentTime = this.media.currentTime;
}
// avoid reporting fragment loop loading error in case user is seeking several times on same position
if (this.fragLoadIdx !== undefined) {
this.fragLoadIdx += 2 * this.config.fragLoadingLoopThreshold;
}
// tick to speed up processing
this.tick();
}
onMediaSeeked() {
// tick to speed up FRAGMENT_PLAYING triggering
this.tick();
}
onMediaEnded() {
logger.log('media ended');
// reset startPosition and lastCurrentTime to restart playback @ stream beginning
this.startPosition = this.lastCurrentTime = 0;
}
2016-01-18 14:07:26 -05:00
onManifestParsed(data) {
2016-02-24 22:15:07 -05:00
var aac = false, heaac = false, codec;
2015-12-16 00:30:14 -05:00
data.levels.forEach(level => {
// detect if we have different kind of audio codecs used amongst playlists
2016-02-24 22:15:07 -05:00
codec = level.audioCodec;
if (codec) {
if (codec.indexOf('mp4a.40.2') !== -1) {
2015-12-16 00:30:14 -05:00
aac = true;
}
2016-02-24 22:15:07 -05:00
if (codec.indexOf('mp4a.40.5') !== -1) {
2015-12-16 00:30:14 -05:00
heaac = true;
}
}
});
2016-02-24 22:15:07 -05:00
this.audioCodecSwitch = (aac && heaac);
if (this.audioCodecSwitch) {
logger.log('both AAC/HE-AAC audio found in levels; declaring level codec as HE-AAC');
2015-12-16 00:30:14 -05:00
}
this.levels = data.levels;
this.startLevelLoaded = false;
2016-02-24 22:15:07 -05:00
this.startFragRequested = false;
if (this.config.autoStartLoad) {
2015-12-16 00:30:14 -05:00
this.startLoad();
}
}
2016-01-18 14:07:26 -05:00
onLevelLoaded(data) {
2015-12-16 00:30:14 -05:00
var newDetails = data.details,
newLevelId = data.level,
curLevel = this.levels[newLevelId],
2016-02-24 22:15:07 -05:00
duration = newDetails.totalduration,
sliding = 0;
2015-12-16 00:30:14 -05:00
logger.log(`level ${newLevelId} loaded [${newDetails.startSN},${newDetails.endSN}],duration:${duration}`);
2016-01-18 14:07:26 -05:00
this.levelLastLoaded = newLevelId;
2015-12-16 00:30:14 -05:00
if (newDetails.live) {
var curDetails = curLevel.details;
if (curDetails) {
// we already have details for that level, merge them
LevelHelper.mergeDetails(curDetails,newDetails);
2016-02-24 22:15:07 -05:00
sliding = newDetails.fragments[0].start;
2015-12-16 00:30:14 -05:00
if (newDetails.PTSKnown) {
2016-02-24 22:15:07 -05:00
logger.log(`live playlist sliding:${sliding.toFixed(3)}`);
2015-12-16 00:30:14 -05:00
} else {
logger.log('live playlist - outdated PTS, unknown sliding');
}
} else {
newDetails.PTSKnown = false;
logger.log('live playlist - first load, unknown sliding');
}
} else {
newDetails.PTSKnown = false;
}
// override level info
curLevel.details = newDetails;
this.hls.trigger(Event.LEVEL_UPDATED, { details: newDetails, level: newLevelId });
// compute start position
2016-02-24 22:15:07 -05:00
if (this.startFragRequested === false) {
2015-12-16 00:30:14 -05:00
// if live playlist, set start position to be fragment N-this.config.liveSyncDurationCount (usually 3)
if (newDetails.live) {
2016-03-09 12:40:22 -05:00
let targetLatency = this.config.liveSyncDuration !== undefined ? this.config.liveSyncDuration : this.config.liveSyncDurationCount * newDetails.targetduration;
this.startPosition = Math.max(0, sliding + duration - targetLatency);
2015-12-16 00:30:14 -05:00
}
this.nextLoadPosition = this.startPosition;
}
// only switch batck to IDLE state if we were waiting for level to start downloading a new fragment
if (this.state === State.WAITING_LEVEL) {
this.state = State.IDLE;
}
//trigger handler right now
this.tick();
}
onKeyLoaded() {
if (this.state === State.KEY_LOADING) {
this.state = State.IDLE;
this.tick();
}
}
2016-01-18 14:07:26 -05:00
onFragLoaded(data) {
2015-12-16 00:30:14 -05:00
var fragCurrent = this.fragCurrent;
if (this.state === State.FRAG_LOADING &&
fragCurrent &&
data.frag.level === fragCurrent.level &&
data.frag.sn === fragCurrent.sn) {
if (this.fragBitrateTest === true) {
// switch back to IDLE state ... we just loaded a fragment to determine adequate start bitrate and initialize autoswitch algo
this.state = State.IDLE;
this.fragBitrateTest = false;
data.stats.tparsed = data.stats.tbuffered = performance.now();
this.hls.trigger(Event.FRAG_BUFFERED, {stats: data.stats, frag: fragCurrent});
} else {
this.state = State.PARSING;
// transmux the MPEG-TS data to ISO-BMFF segments
this.stats = data.stats;
var currentLevel = this.levels[this.level],
details = currentLevel.details,
duration = details.totalduration,
start = fragCurrent.start,
level = fragCurrent.level,
sn = fragCurrent.sn,
2016-02-24 22:15:07 -05:00
audioCodec = currentLevel.audioCodec || this.config.defaultAudioCodec;
2016-01-13 15:58:12 -05:00
if(this.audioCodecSwap) {
2015-12-16 00:30:14 -05:00
logger.log('swapping playlist audio codec');
2016-01-13 15:58:12 -05:00
if(audioCodec === undefined) {
audioCodec = this.lastAudioCodec;
}
2016-02-24 22:15:07 -05:00
if(audioCodec) {
if(audioCodec.indexOf('mp4a.40.5') !==-1) {
audioCodec = 'mp4a.40.2';
} else {
audioCodec = 'mp4a.40.5';
}
2015-12-16 00:30:14 -05:00
}
}
2016-02-24 22:15:07 -05:00
this.pendingAppending = 0;
2015-12-16 00:30:14 -05:00
logger.log(`Demuxing ${sn} of [${details.startSN} ,${details.endSN}],level ${level}`);
this.demuxer.push(data.payload, audioCodec, currentLevel.videoCodec, start, fragCurrent.cc, level, sn, duration, fragCurrent.decryptdata);
}
}
2015-12-23 12:46:01 -05:00
this.fragLoadError = 0;
2015-12-16 00:30:14 -05:00
}
2016-01-18 14:07:26 -05:00
onFragParsingInitSegment(data) {
2015-12-16 00:30:14 -05:00
if (this.state === State.PARSING) {
2016-02-24 22:15:07 -05:00
var tracks = data.tracks, trackName, track;
// include levelCodec in audio and video tracks
track = tracks.audio;
if(track) {
2016-03-10 12:59:32 -05:00
var audioCodec = this.levels[this.level].audioCodec,
ua = navigator.userAgent.toLowerCase();
2016-02-24 22:15:07 -05:00
if(audioCodec && this.audioCodecSwap) {
logger.log('swapping playlist audio codec');
if(audioCodec.indexOf('mp4a.40.5') !==-1) {
audioCodec = 'mp4a.40.2';
} else {
audioCodec = 'mp4a.40.5';
}
}
// in case AAC and HE-AAC audio codecs are signalled in manifest
// force HE-AAC , as it seems that most browsers prefers that way,
2016-03-10 12:59:32 -05:00
// except for mono streams OR on FF
2016-02-24 22:15:07 -05:00
// these conditions might need to be reviewed ...
if (this.audioCodecSwitch) {
// don't force HE-AAC if mono stream
if(track.metadata.channelCount !== 1 &&
// don't force HE-AAC if firefox
ua.indexOf('firefox') === -1) {
audioCodec = 'mp4a.40.5';
}
2015-12-16 00:30:14 -05:00
}
2016-03-10 12:59:32 -05:00
// HE-AAC is broken on Android, always signal audio codec as AAC even if variant manifest states otherwise
if(ua.indexOf('android') !== -1) {
audioCodec = 'mp4a.40.2';
logger.log(`Android: force audio codec to` + audioCodec);
}
2016-02-24 22:15:07 -05:00
track.levelCodec = audioCodec;
2015-12-16 00:30:14 -05:00
}
2016-02-24 22:15:07 -05:00
track = tracks.video;
if(track) {
track.levelCodec = this.levels[this.level].videoCodec;
2015-12-16 00:30:14 -05:00
}
2016-02-24 22:15:07 -05:00
// if remuxer specify that a unique track needs to generated,
// let's merge all tracks together
if (data.unique) {
var mergedTrack = {
codec : '',
levelCodec : ''
};
for (trackName in data.tracks) {
track = tracks[trackName];
mergedTrack.container = track.container;
if (mergedTrack.codec) {
mergedTrack.codec += ',';
mergedTrack.levelCodec += ',';
}
if(track.codec) {
mergedTrack.codec += track.codec;
}
if (track.levelCodec) {
mergedTrack.levelCodec += track.levelCodec;
}
2015-12-16 00:30:14 -05:00
}
2016-02-24 22:15:07 -05:00
tracks = { audiovideo : mergedTrack };
2015-12-16 00:30:14 -05:00
}
2016-02-24 22:15:07 -05:00
this.hls.trigger(Event.BUFFER_CODECS,tracks);
// loop through tracks that are going to be provided to bufferController
for (trackName in tracks) {
track = tracks[trackName];
logger.log(`track:${trackName},container:${track.container},codecs[level/parsed]=[${track.levelCodec}/${track.codec}]`);
var initSegment = track.initSegment;
if (initSegment) {
this.pendingAppending++;
this.hls.trigger(Event.BUFFER_APPENDING, {type: trackName, data: initSegment});
}
2015-12-16 00:30:14 -05:00
}
//trigger handler right now
this.tick();
}
}
2016-01-18 14:07:26 -05:00
onFragParsingData(data) {
2015-12-16 00:30:14 -05:00
if (this.state === State.PARSING) {
this.tparse2 = Date.now();
var level = this.levels[this.level],
frag = this.fragCurrent;
2016-02-24 22:15:07 -05:00
2016-01-13 15:58:12 -05:00
logger.log(`parsed ${data.type},PTS:[${data.startPTS.toFixed(3)},${data.endPTS.toFixed(3)}],DTS:[${data.startDTS.toFixed(3)}/${data.endDTS.toFixed(3)}],nb:${data.nb}`);
2015-12-16 00:30:14 -05:00
2016-02-24 22:15:07 -05:00
var drift = LevelHelper.updateFragPTS(level.details,frag.sn,data.startPTS,data.endPTS),
hls = this.hls;
hls.trigger(Event.LEVEL_PTS_UPDATED, {details: level.details, level: this.level, drift: drift});
[data.data1, data.data2].forEach(buffer => {
if (buffer) {
this.pendingAppending++;
hls.trigger(Event.BUFFER_APPENDING, {type: data.type, data: buffer});
}
});
2015-12-16 00:30:14 -05:00
this.nextLoadPosition = data.endPTS;
this.bufferRange.push({type: data.type, start: data.startPTS, end: data.endPTS, frag: frag});
//trigger handler right now
this.tick();
} else {
2016-02-24 22:15:07 -05:00
logger.warn(`not in PARSING state but ${this.state}, ignoring FRAG_PARSING_DATA event`);
2015-12-16 00:30:14 -05:00
}
}
onFragParsed() {
if (this.state === State.PARSING) {
this.stats.tparsed = performance.now();
2016-02-24 22:15:07 -05:00
this.state = State.PARSED;
this._checkAppendedParsed();
}
}
2016-02-04 13:19:10 -05:00
2016-02-24 22:15:07 -05:00
onBufferAppended() {
switch (this.state) {
case State.PARSING:
case State.PARSED:
this.pendingAppending--;
this._checkAppendedParsed();
break;
default:
break;
}
}
2016-02-04 13:19:10 -05:00
2016-02-24 22:15:07 -05:00
_checkAppendedParsed() {
//trigger handler right now
if (this.state === State.PARSED && this.pendingAppending === 0) {
var frag = this.fragCurrent, stats = this.stats;
if (frag) {
this.fragPrevious = frag;
stats.tbuffered = performance.now();
this.fragLastKbps = Math.round(8 * stats.length / (stats.tbuffered - stats.tfirst));
this.hls.trigger(Event.FRAG_BUFFERED, {stats: stats, frag: frag});
logger.log(`media buffered : ${this.timeRangesToString(this.media.buffered)}`);
2016-02-04 13:19:10 -05:00
this.state = State.IDLE;
}
2015-12-16 00:30:14 -05:00
this.tick();
}
}
2016-01-18 14:07:26 -05:00
onError(data) {
2015-12-16 00:30:14 -05:00
switch(data.details) {
case ErrorDetails.FRAG_LOAD_ERROR:
case ErrorDetails.FRAG_LOAD_TIMEOUT:
2016-01-13 15:58:12 -05:00
if(!data.fatal) {
var loadError = this.fragLoadError;
if(loadError) {
loadError++;
} else {
loadError=1;
}
if (loadError <= this.config.fragLoadingMaxRetry) {
this.fragLoadError = loadError;
// reset load counter to avoid frag loop loading error
data.frag.loadCounter = 0;
// exponential backoff capped to 64s
var delay = Math.min(Math.pow(2,loadError-1)*this.config.fragLoadingRetryDelay,64000);
logger.warn(`mediaController: frag loading failed, retry in ${delay} ms`);
this.retryDate = performance.now() + delay;
// retry loading state
this.state = State.FRAG_LOADING_WAITING_RETRY;
} else {
logger.error(`mediaController: ${data.details} reaches max retry, redispatch as fatal ...`);
// redispatch same error but with fatal set to true
data.fatal = true;
2016-01-18 14:07:26 -05:00
this.hls.trigger(Event.ERROR, data);
2016-01-13 15:58:12 -05:00
this.state = State.ERROR;
}
2015-12-23 12:46:01 -05:00
}
break;
2015-12-16 00:30:14 -05:00
case ErrorDetails.FRAG_LOOP_LOADING_ERROR:
case ErrorDetails.LEVEL_LOAD_ERROR:
case ErrorDetails.LEVEL_LOAD_TIMEOUT:
case ErrorDetails.KEY_LOAD_ERROR:
case ErrorDetails.KEY_LOAD_TIMEOUT:
// if fatal error, stop processing, otherwise move to IDLE to retry loading
logger.warn(`mediaController: ${data.details} while loading frag,switch to ${data.fatal ? 'ERROR' : 'IDLE'} state ...`);
this.state = data.fatal ? State.ERROR : State.IDLE;
break;
2016-03-09 12:40:22 -05:00
case ErrorDetails.BUFFER_FULL_ERROR:
2016-02-24 22:15:07 -05:00
// trigger a smooth level switch to empty buffers
// also reduce max buffer length as it might be too high. we do this to avoid loop flushing ...
this.config.maxMaxBufferLength/=2;
logger.warn(`reduce max buffer length to ${this.config.maxMaxBufferLength}s and trigger a nextLevelSwitch to flush old buffer and fix QuotaExceededError`);
this.nextLevelSwitch();
break;
2015-12-16 00:30:14 -05:00
default:
break;
}
}
_checkBuffer() {
var media = this.media;
if(media) {
// compare readyState
var readyState = media.readyState;
// if ready state different from HAVE_NOTHING (numeric value 0), we are allowed to seek
if(readyState) {
2016-02-24 22:15:07 -05:00
var targetSeekPosition, currentTime;
2015-12-16 00:30:14 -05:00
// if seek after buffered defined, let's seek if within acceptable range
var seekAfterBuffered = this.seekAfterBuffered;
if(seekAfterBuffered) {
if(media.duration >= seekAfterBuffered) {
2016-02-24 22:15:07 -05:00
targetSeekPosition = seekAfterBuffered;
2015-12-16 00:30:14 -05:00
this.seekAfterBuffered = undefined;
}
2016-01-13 15:58:12 -05:00
} else {
2016-02-24 22:15:07 -05:00
currentTime = media.currentTime;
var loadedmetadata = this.loadedmetadata;
2016-01-25 15:28:29 -05:00
2016-02-24 22:15:07 -05:00
// adjust currentTime to start position on loaded metadata
if(!loadedmetadata && media.buffered.length) {
this.loadedmetadata = true;
// only adjust currentTime if not equal to 0
if (!currentTime && currentTime !== this.startPosition) {
targetSeekPosition = this.startPosition;
}
2016-01-25 15:28:29 -05:00
}
2016-02-24 22:15:07 -05:00
}
if (targetSeekPosition) {
currentTime = targetSeekPosition;
logger.log(`target seek position:${targetSeekPosition}`);
}
var bufferInfo = this.bufferInfo(currentTime,0),
expectedPlaying = !(media.paused || media.ended || media.seeking || readyState < 2),
2016-03-09 12:40:22 -05:00
jumpThreshold = 0.4, // tolerance needed as some browsers stalls playback before reaching buffered range end
2016-02-24 22:15:07 -05:00
playheadMoving = currentTime > media.playbackRate*this.lastCurrentTime;
2016-01-13 15:58:12 -05:00
2016-02-24 22:15:07 -05:00
if (this.stalled && playheadMoving) {
this.stalled = false;
}
// check buffer upfront
// if less than 200ms is buffered, and media is expected to play but playhead is not moving,
// and we have a new buffer range available upfront, let's seek to that one
if(bufferInfo.len <= jumpThreshold) {
if(playheadMoving || !expectedPlaying) {
// playhead moving or media not playing
jumpThreshold = 0;
} else {
// playhead not moving AND media expected to play
logger.log(`playback seems stuck @${currentTime}`);
if(!this.stalled) {
this.hls.trigger(Event.ERROR, {type: ErrorTypes.MEDIA_ERROR, details: ErrorDetails.BUFFER_STALLED_ERROR, fatal: false});
this.stalled = true;
2016-01-13 15:58:12 -05:00
}
2016-02-24 22:15:07 -05:00
}
// if we are below threshold, try to jump if next buffer range is close
if(bufferInfo.len <= jumpThreshold) {
2016-03-09 12:40:22 -05:00
// no buffer available @ currentTime, check if next buffer is close (within a config.maxSeekHole second range)
2016-02-24 22:15:07 -05:00
var nextBufferStart = bufferInfo.nextStart, delta = nextBufferStart-currentTime;
if(nextBufferStart &&
(delta < this.config.maxSeekHole) &&
2016-03-09 12:40:22 -05:00
(delta > 0) &&
2016-02-24 22:15:07 -05:00
!media.seeking) {
// next buffer is close ! adjust currentTime to nextBufferStart
// this will ensure effective video decoding
logger.log(`adjust currentTime from ${media.currentTime} to next buffered @ ${nextBufferStart}`);
media.currentTime = nextBufferStart;
2016-03-09 12:40:22 -05:00
this.hls.trigger(Event.ERROR, {type: ErrorTypes.MEDIA_ERROR, details: ErrorDetails.BUFFER_SEEK_OVER_HOLE, fatal: false});
2015-12-16 00:30:14 -05:00
}
}
2016-02-24 22:15:07 -05:00
} else {
if (targetSeekPosition && media.currentTime !== targetSeekPosition) {
logger.log(`adjust currentTime from ${media.currentTime} to ${targetSeekPosition}`);
media.currentTime = targetSeekPosition;
}
2015-12-16 00:30:14 -05:00
}
}
}
}
2016-02-24 22:15:07 -05:00
onBufferFlushed() {
/* after successful buffer flushing, rebuild buffer Range array
loop through existing buffer range and check if
corresponding range is still buffered. only push to new array already buffered range
*/
var newRange = [],range,i;
for (i = 0; i < this.bufferRange.length; i++) {
range = this.bufferRange[i];
if (this.isBuffered((range.start + range.end) / 2)) {
newRange.push(range);
}
}
this.bufferRange = newRange;
// handle end of immediate switching if needed
if (this.immediateSwitch) {
this.immediateLevelSwitchEnd();
}
// move to IDLE once flush complete. this should trigger new fragment loading
this.state = State.IDLE;
// reset reference to frag
this.fragPrevious = null;
2015-12-16 00:30:14 -05:00
}
2016-02-24 22:15:07 -05:00
swapAudioCodec() {
this.audioCodecSwap = !this.audioCodecSwap;
2015-12-16 00:30:14 -05:00
}
timeRangesToString(r) {
var log = '', len = r.length;
for (var i=0; i<len; i++) {
log += '[' + r.start(i) + ',' + r.end(i) + ']';
}
return log;
}
}
2016-02-24 22:15:07 -05:00
export default StreamController;
2015-12-16 00:30:14 -05:00