chromium/media/test/data/player.html

<html>
<body onload="RunTest();">
<div id="player_container"></div>
<div id="logs"></div>
</body>

<script type="text/javascript">
// <audio> or <video> player element.
var player;
var tag = '';
var logs = document.getElementById('logs');
var heartbeatCount = 0;
var isSizeTest = false;

function Log(message) {
  logs.innerHTML = message + '<br>' + logs.innerHTML;
  console.log(message);
}

function onLogEvent(e) {
  Log('Event: ' + e.type + ', State: {' + getMediaStatus(e.target) + '}');
}

function getTimeRanges(range) {
  const result = [];
  for (let i = 0; i < range.length; i++) {
    result.push(`[${range.start(i)},${range.end(i)}]`);
  }
  return '[' + result.join(', ') + ']';
}

function getMediaStatus(video) {
  if (video == null) {
    return 'null';
  }
  const result = [];
  result.push(`readyState: ${video.readyState}`);
  result.push(`networkState: ${video.networkState}`);
  result.push(`paused: ${video.paused}`);
  result.push(`ended: ${video.ended}`);
  result.push(`currentTime: ${video.currentTime}`);
  result.push(`duration: ${video.duration}`);
  result.push(`buffered: ${getTimeRanges(video.buffered)}`);
  result.push(`played: ${getTimeRanges(video.played)}`);
  if (tag == 'video') {
    result.push(`size: ${video.videoWidth}x${video.videoHeight}`);
  }
  if (video.error) {
    result.push(`error: {${video.error.code},${video.error.message}}`);
  }
  if (window.internals) {
    result.push(`suspended: ${internals.isMediaElementSuspended(video) ?
      'true' : 'false'}`);
  }
  return result.join(', ');
}

// Listen for |event| from |element|, set document.title = |event| upon event.
function InstallTitleEventHandler(element, event) {
  element.addEventListener(event, function(e) {
    onLogEvent(e);

    if (!isSizeTest) {
      document.title = event.toUpperCase();
      return;
    }

    var video = e.target;
    document.title =
        event.toUpperCase() + ` ${video.videoWidth} ${video.videoHeight}`;
  }, false);
}

function InstallTitleErrorEventHandler(element, error_substr) {
  element.addEventListener('error', function(e) {
    onLogEvent(e);

    // Element's error attribute must be populated.
    var mediaError = element.error;
    if (mediaError == null) {
      Log('ERROR: Null element error attribute while handling error event.');
      Failed();
      return;
    }

    Log('INFO: Element error message is ' + mediaError.message);

    if (error_substr != null && !mediaError.message.includes(error_substr)) {
      Log('ERROR: Element error message (' + mediaError.message +
          ') does not contain expected substring (' + error_substr + ')');
      Failed();
      return;
    }
    document.title = 'ERROR';
  }, false);
}

function Failed() {
  document.title = 'FAILED';
  return false;
}

function SeekTestStep(e) {
  if (e) onLogEvent(e);

  // Remove the previous triggers for this step.
  player.removeEventListener('ended', SeekTestStep, false);
  player.removeEventListener('timeupdate', SeekTestTimeoutSetup, false);

  // Test completes on the next ended event.
  InstallTitleEventHandler(player, 'ended');

  player.currentTime = 0.9 * player.duration;
  Log('Seeking back to ' + 0.9 * player.duration);
  player.play().catch(function(error) {
    Log(error);
  });
}

function SeekTestTimeoutSetup() {
  if (player.currentTime < 2)
    return;

  Log('Triggering seek due to timeout @ ' + player.currentTime);
  SeekTestStep();
}

// Uses URL query parameters to create an audio or video element using a given
// source.  URL must be of the form:
// "player.html?[tag]=[media_url][&loop=[true/false]][&error_substr=substr]".
//
// Plays the media and waits for X seconds of playback or the ended event, at
// which point the test seeks near the end of the file and resumes playback.
// Test completes when the second ended event occurs or an error event occurs at
// any time.
// There is an optional loop query parameter, which when set to true will cause
// the created media element to loop.
// There is an optional error_substr query parameter, which when set will cause
// any error event handling to fail if the MediaError.message does not contain
// the specified substring resulting from decodeURIComponent().
function RunTest() {
  var url_parts = window.location.href.split('?');
  if (url_parts.length != 2)
    return Failed();

  var media_url = '';
  var loop = false;
  var error_substr = null;

  var query_params = url_parts[1].split('&');
  for (var query_param in query_params) {
    var query_parts = query_params[query_param].split('=');
    if (query_parts.length != 2) {
      return Failed();
    }

    if (query_parts[0] == 'audio' || query_parts[0] == 'video') {
      tag = query_parts[0];
      media_url = query_parts[1];
      continue;
    }

    if (query_parts[0] == 'loop') {
      loop = (query_parts[1] == 'true');
      continue;
    }

    if (query_parts[0] == 'error_substr') {
      error_substr = decodeURIComponent(query_parts[1]);
      continue;
    }

    if (query_parts[0] == 'sizetest') {
      isSizeTest = true;
      continue;
    }
  }

  if (tag != 'audio' && tag != 'video') {
    return Failed();
  }

  // Create player and insert into DOM.
  player = document.createElement(tag);
  player.controls = true;
  document.getElementById('player_container').appendChild(player);

  // Transition to the seek test after X seconds of playback or when the ended
  // event occurs, whichever happens first.
  player.addEventListener('ended', SeekTestStep, false);
  player.addEventListener('timeupdate', SeekTestTimeoutSetup, false);

  // Log events.
  player.addEventListener('abort', onLogEvent);
  player.addEventListener('canplay', onLogEvent);
  player.addEventListener('canplaythrough', onLogEvent);
  player.addEventListener('durationchange', onLogEvent);
  player.addEventListener('emptied', onLogEvent);
  player.addEventListener('loadeddata', onLogEvent);
  player.addEventListener('loadedmetadata', onLogEvent);
  player.addEventListener('loadstart', onLogEvent);
  player.addEventListener('play', onLogEvent);
  player.addEventListener('playing', onLogEvent);
  player.addEventListener('progress', onLogEvent);
  player.addEventListener('resize', onLogEvent);
  player.addEventListener('stalled', onLogEvent);
  player.addEventListener('suspend', onLogEvent);
  player.addEventListener('timeupdate', onLogEvent);
  player.addEventListener('waiting', onLogEvent);

  setInterval(function () {
    onLogEvent({'type': 'heartbeat #' + ++heartbeatCount, 'target': player});
  }, 1000);

  // Ensure we percolate up any error events, and optionally verify they contain
  // the expected error substring in their message.
  InstallTitleErrorEventHandler(player, error_substr);

  // Starts the player.
  player.loop = loop;
  player.preload = 'auto';
  player.src = media_url;
  Log('Starting test by calling ' + tag + '.play()');
  player.play().catch(function(error) {
    Log(error);
  });
}
</script>
</html>