/**
* videojs-record
* @version 2.0.6
* @see https://github.com/collab-project/videojs-record
* @copyright 2014-2018 Collab
* @license MIT
*/
(function(f){if(typeof exports==="object"&&typeof module!=="undefined"){module.exports=f()}else if(typeof define==="function"&&define.amd){define([],f)}else{var g;if(typeof window!=="undefined"){g=window}else if(typeof global!=="undefined"){g=global}else if(typeof self!=="undefined"){g=self}else{g=this}(g.videojs || (g.videojs = {})).record = f()}})(function(){var define,module,exports;return (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o'
});
}
}]);
return AnimationDisplay;
}(Component);
Component.registerComponent('AnimationDisplay', AnimationDisplay);
exports.default = AnimationDisplay;
},{}],2:[function(require,module,exports){
'use strict';
Object.defineProperty(exports, "__esModule", {
value: true
});
var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
var _get = function get(object, property, receiver) { if (object === null) object = Function.prototype; var desc = Object.getOwnPropertyDescriptor(object, property); if (desc === undefined) { var parent = Object.getPrototypeOf(object); if (parent === null) { return undefined; } else { return get(parent, property, receiver); } } else if ("value" in desc) { return desc.value; } else { var getter = desc.get; if (getter === undefined) { return undefined; } return getter.call(receiver); } };
function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; }
function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; }
/**
* @file camera-button.js
* @since 2.0.0
*/
var Button = videojs.getComponent('Button');
var Component = videojs.getComponent('Component');
/**
* Button to toggle between create and retry snapshot image.
*
* @class
* @augments videojs.Button
*/
var CameraButton = function (_Button) {
_inherits(CameraButton, _Button);
function CameraButton() {
_classCallCheck(this, CameraButton);
return _possibleConstructorReturn(this, (CameraButton.__proto__ || Object.getPrototypeOf(CameraButton)).apply(this, arguments));
}
_createClass(CameraButton, [{
key: 'buildCSSClass',
/**
* Builds the default DOM `className`.
*
* @return {string}
* The DOM `className` for this object.
*/
value: function buildCSSClass() {
return 'vjs-camera-button vjs-control vjs-button vjs-icon-photo-camera';
}
/**
* Enable the `CameraButton` element so that it can be activated or clicked.
*/
}, {
key: 'enable',
value: function enable() {
_get(CameraButton.prototype.__proto__ || Object.getPrototypeOf(CameraButton.prototype), 'enable', this).call(this);
this.on(this.player_, 'startRecord', this.onStart);
this.on(this.player_, 'stopRecord', this.onStop);
}
/**
* Disable the `CameraButton` element so that it cannot be activated or clicked.
*/
}, {
key: 'disable',
value: function disable() {
_get(CameraButton.prototype.__proto__ || Object.getPrototypeOf(CameraButton.prototype), 'disable', this).call(this);
this.off(this.player_, 'startRecord', this.onStart);
this.off(this.player_, 'stopRecord', this.onStop);
}
/**
* This gets called when the button is clicked.
*
* @param {EventTarget~Event} event
* The `tap` or `click` event that caused this function to be
* called.
*
* @listens tap
* @listens click
*/
}, {
key: 'handleClick',
value: function handleClick(event) {
var recorder = this.player_.record();
if (!recorder.isProcessing()) {
// create snapshot
recorder.start();
} else {
// retry
recorder.retrySnapshot();
// reset camera button
this.onStop();
}
}
/**
* Add the vjs-icon-replay class to the element so it can change appearance.
*
* @param {EventTarget~Event} [event]
* The event that caused this function to run.
*
* @listens Player#startRecord
*/
}, {
key: 'onStart',
value: function onStart(event) {
// replace element class so it can change appearance
this.removeClass('vjs-icon-photo-camera');
this.addClass('vjs-icon-replay');
// change the button text
this.controlText('Retry');
}
/**
* Add the vjs-icon-photo-camera class to the element so it can change appearance.
*
* @param {EventTarget~Event} [event]
* The event that caused this function to run.
*
* @listens Player#stopRecord
*/
}, {
key: 'onStop',
value: function onStop(event) {
// replace element class so it can change appearance
this.removeClass('vjs-icon-replay');
this.addClass('vjs-icon-photo-camera');
// change the button text
this.controlText('Image');
}
}]);
return CameraButton;
}(Button);
/**
* The text that should display over the `CameraButton`s controls. Added for localization.
*
* @type {string}
* @private
*/
CameraButton.prototype.controlText_ = 'Image';
Component.registerComponent('CameraButton', CameraButton);
exports.default = CameraButton;
},{}],3:[function(require,module,exports){
'use strict';
Object.defineProperty(exports, "__esModule", {
value: true
});
var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; }
function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; }
/**
* @file device-button.js
* @since 2.0.0
*/
var Button = videojs.getComponent('Button');
var Component = videojs.getComponent('Component');
/**
* Button to select recording device.
*
* @class
* @augments videojs.Button
*/
var DeviceButton = function (_Button) {
_inherits(DeviceButton, _Button);
function DeviceButton() {
_classCallCheck(this, DeviceButton);
return _possibleConstructorReturn(this, (DeviceButton.__proto__ || Object.getPrototypeOf(DeviceButton)).apply(this, arguments));
}
_createClass(DeviceButton, [{
key: 'handleClick',
/**
* This gets called when this button gets:
*
* - Clicked (via the `click` event, listening starts in the constructor)
* - Tapped (via the `tap` event, listening starts in the constructor)
*
* @param {EventTarget~Event} event
* The `keydown`, `tap`, or `click` event that caused this function to be
* called.
*
* @listens tap
* @listens click
*/
value: function handleClick(event) {
// open device dialog
this.player_.record().getDevice();
}
}]);
return DeviceButton;
}(Button);
/**
* The text that should display over the `DeviceButton`s controls. Added for localization.
*
* @type {string}
* @private
*/
DeviceButton.prototype.controlText_ = 'Device';
Component.registerComponent('DeviceButton', DeviceButton);
exports.default = DeviceButton;
},{}],4:[function(require,module,exports){
'use strict';
Object.defineProperty(exports, "__esModule", {
value: true
});
var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
var _get = function get(object, property, receiver) { if (object === null) object = Function.prototype; var desc = Object.getOwnPropertyDescriptor(object, property); if (desc === undefined) { var parent = Object.getPrototypeOf(object); if (parent === null) { return undefined; } else { return get(parent, property, receiver); } } else if ("value" in desc) { return desc.value; } else { var getter = desc.get; if (getter === undefined) { return undefined; } return getter.call(receiver); } };
function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; }
function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; }
/**
* @file record-canvas
* @since 2.0.0
*/
var Component = videojs.getComponent('Component');
/**
* Canvas for displaying snapshot image.
*
* @class
* @augments videojs.Component
*/
var RecordCanvas = function (_Component) {
_inherits(RecordCanvas, _Component);
function RecordCanvas() {
_classCallCheck(this, RecordCanvas);
return _possibleConstructorReturn(this, (RecordCanvas.__proto__ || Object.getPrototypeOf(RecordCanvas)).apply(this, arguments));
}
_createClass(RecordCanvas, [{
key: 'createEl',
/**
* Create the `RecordCanvas`s DOM element.
*
* @return {Element}
* The dom element that gets created.
*/
value: function createEl() {
return _get(RecordCanvas.prototype.__proto__ || Object.getPrototypeOf(RecordCanvas.prototype), 'createEl', this).call(this, 'div', {
className: 'vjs-record-canvas',
innerHTML: ''
});
}
}]);
return RecordCanvas;
}(Component);
Component.registerComponent('RecordCanvas', RecordCanvas);
exports.default = RecordCanvas;
},{}],5:[function(require,module,exports){
'use strict';
Object.defineProperty(exports, "__esModule", {
value: true
});
var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
var _get = function get(object, property, receiver) { if (object === null) object = Function.prototype; var desc = Object.getOwnPropertyDescriptor(object, property); if (desc === undefined) { var parent = Object.getPrototypeOf(object); if (parent === null) { return undefined; } else { return get(parent, property, receiver); } } else if ("value" in desc) { return desc.value; } else { var getter = desc.get; if (getter === undefined) { return undefined; } return getter.call(receiver); } };
function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; }
function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; }
/**
* @file record-indicator.js
* @since 2.0.0
*/
var Component = videojs.getComponent('Component');
/**
* Icon indicating recording is active.
*
* @class
* @augments videojs.Component
*/
var RecordIndicator = function (_Component) {
_inherits(RecordIndicator, _Component);
/**
* The constructor function for the class.
*
* @private
* @param {(videojs.Player|Object)} player - Video.js player instance.
* @param {Object} options - Player options.
*/
function RecordIndicator(player, options) {
_classCallCheck(this, RecordIndicator);
var _this = _possibleConstructorReturn(this, (RecordIndicator.__proto__ || Object.getPrototypeOf(RecordIndicator)).call(this, player, options));
_this.enable();
return _this;
}
/**
* Create the `RecordIndicator`s DOM element.
*
* @return {Element}
* The dom element that gets created.
*/
_createClass(RecordIndicator, [{
key: 'createEl',
value: function createEl() {
return _get(RecordIndicator.prototype.__proto__ || Object.getPrototypeOf(RecordIndicator.prototype), 'createEl', this).call(this, 'div', {
className: 'vjs-record-indicator vjs-control',
dir: 'ltr'
});
}
/**
* Enable event handlers.
*/
}, {
key: 'enable',
value: function enable() {
this.on(this.player_, 'startRecord', this.show);
this.on(this.player_, 'stopRecord', this.hide);
}
/**
* Disable event handlers.
*/
}, {
key: 'disable',
value: function disable() {
this.off(this.player_, 'startRecord', this.show);
this.off(this.player_, 'stopRecord', this.hide);
}
}]);
return RecordIndicator;
}(Component);
Component.registerComponent('RecordIndicator', RecordIndicator);
exports.default = RecordIndicator;
},{}],6:[function(require,module,exports){
'use strict';
Object.defineProperty(exports, "__esModule", {
value: true
});
var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
var _get = function get(object, property, receiver) { if (object === null) object = Function.prototype; var desc = Object.getOwnPropertyDescriptor(object, property); if (desc === undefined) { var parent = Object.getPrototypeOf(object); if (parent === null) { return undefined; } else { return get(parent, property, receiver); } } else if ("value" in desc) { return desc.value; } else { var getter = desc.get; if (getter === undefined) { return undefined; } return getter.call(receiver); } };
function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; }
function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; }
/**
* @file record-toggle.js
* @since 2.0.0
*/
var Button = videojs.getComponent('Button');
var Component = videojs.getComponent('Component');
/**
* Button to toggle between start and stop recording.
*
* @class
* @augments videojs.Button
*/
var RecordToggle = function (_Button) {
_inherits(RecordToggle, _Button);
function RecordToggle() {
_classCallCheck(this, RecordToggle);
return _possibleConstructorReturn(this, (RecordToggle.__proto__ || Object.getPrototypeOf(RecordToggle)).apply(this, arguments));
}
_createClass(RecordToggle, [{
key: 'buildCSSClass',
/**
* Builds the default DOM `className`.
*
* @return {string}
* The DOM `className` for this object.
*/
value: function buildCSSClass() {
return 'vjs-record-button vjs-control vjs-button vjs-icon-record-start';
}
/**
* Enable the `RecordToggle` element so that it can be activated or clicked.
*/
}, {
key: 'enable',
value: function enable() {
_get(RecordToggle.prototype.__proto__ || Object.getPrototypeOf(RecordToggle.prototype), 'enable', this).call(this);
this.on(this.player_, 'startRecord', this.onStart);
this.on(this.player_, 'stopRecord', this.onStop);
}
/**
* Disable the `RecordToggle` element so that it cannot be activated or clicked.
*/
}, {
key: 'disable',
value: function disable() {
_get(RecordToggle.prototype.__proto__ || Object.getPrototypeOf(RecordToggle.prototype), 'disable', this).call(this);
this.off(this.player_, 'startRecord', this.onStart);
this.off(this.player_, 'stopRecord', this.onStop);
}
/**
* This gets called when the button is clicked.
*
* @param {EventTarget~Event} event
* The `tap` or `click` event that caused this function to be
* called.
*
* @listens tap
* @listens click
*/
}, {
key: 'handleClick',
value: function handleClick(event) {
var recorder = this.player_.record();
if (!recorder.isRecording()) {
recorder.start();
} else {
recorder.stop();
}
}
/**
* Add the vjs-icon-record-stop class to the element so it can change appearance.
*
* @param {EventTarget~Event} [event]
* The event that caused this function to run.
*
* @listens Player#startRecord
*/
}, {
key: 'onStart',
value: function onStart(event) {
// replace element class so it can change appearance
this.removeClass('vjs-icon-record-start');
this.addClass('vjs-icon-record-stop');
// change the button text
this.controlText('Stop');
}
/**
* Add the vjs-icon-record-start class to the element so it can change appearance.
*
* @param {EventTarget~Event} [event]
* The event that caused this function to run.
*
* @listens Player#stopRecord
*/
}, {
key: 'onStop',
value: function onStop(event) {
// replace element class so it can change appearance
this.removeClass('vjs-icon-record-stop');
this.addClass('vjs-icon-record-start');
// change the button text
this.controlText('Record');
}
}]);
return RecordToggle;
}(Button);
/**
* The text that should display over the `RecordToggle`s controls. Added for localization.
*
* @type {string}
* @private
*/
RecordToggle.prototype.controlText_ = 'Record';
Component.registerComponent('RecordToggle', RecordToggle);
exports.default = RecordToggle;
},{}],7:[function(require,module,exports){
'use strict';
Object.defineProperty(exports, "__esModule", {
value: true
});
/**
* @file defaults.js
* @since 2.0.0
*/
//plugin defaults
var pluginDefaultOptions = {
// Single snapshot image.
image: false,
// Include audio in the recorded clip.
audio: false,
// Include video in the recorded clip.
video: false,
// Animated GIF.
animation: false,
// Maximum length of the recorded clip.
maxLength: 10,
// Width of the recorded video frames.
frameWidth: 320,
// Height of the recorded video frames.
frameHeight: 240,
// Enables console logging for debugging purposes.
debug: false,
// The mime type for the video recorder. Default to 'video/webm'.
// Use 'video/mp4' (Firefox) or 'video/webm;codecs=H264' (Chrome 52 and
// newer) for MP4.
videoMimeType: 'video/webm',
// Video recorder type to use. This allows you to specify an alternative
// recorder class, e.g. WhammyRecorder. Defaults to 'auto' which let's
// recordrtc specify the best available recorder type.
videoRecorderType: 'auto',
// Audio recording library to use. Legal values are 'recordrtc',
// 'libvorbis.js', 'opus-recorder', 'lamejs' and 'recorder.js'.
audioEngine: 'recordrtc',
// Audio recorder type to use. This allows you to specify an alternative
// recorder class, e.g. StereoAudioRecorder. Defaults to 'auto' which let's
// recordrtc specify the best available recorder type. Currently this
// setting is only used with the 'recordrtc' audioEngine.
audioRecorderType: 'auto',
// The mime type for the audio recorder. Defaults to 'auto' which will pick
// the best option available in the browser (e.g. either 'audio/wav',
// 'audio/ogg' or 'audio/webm').
audioMimeType: 'auto',
// The size of the audio buffer (in sample-frames) which needs to
// be processed each time onprocessaudio is called.
// From the spec: This value controls how frequently the audioprocess event is
// dispatched and how many sample-frames need to be processed each call.
// Lower values for buffer size will result in a lower (better) latency.
// Higher values will be necessary to avoid audio breakup and glitches.
// Legal values are 256, 512, 1024, 2048, 4096, 8192 or 16384.
audioBufferSize: 4096,
// The audio sample rate (in sample-frames per second) at which the
// AudioContext handles audio. It is assumed that all AudioNodes
// in the context run at this rate. In making this assumption,
// sample-rate converters or "varispeed" processors are not supported
// in real-time processing.
// The sampleRate parameter describes the sample-rate of the
// linear PCM audio data in the buffer in sample-frames per second.
// An implementation must support sample-rates in at least
// the range 22050 to 96000.
audioSampleRate: 44100,
// Allows you to record single-channel audio, which can reduce the
// filesize.
audioChannels: 2,
// URL for the audio worker.
audioWorkerURL: '',
// Frame rate in frames per second.
animationFrameRate: 200,
// Sets quality of color quantization (conversion of images to the
// maximum 256 colors allowed by the GIF specification).
// Lower values (minimum = 1) produce better colors,
// but slow processing significantly. 10 is the default,
// and produces good color mapping at reasonable speeds.
// Values greater than 20 do not yield significant improvements
// in speed.
animationQuality: 10,
// Accepts numbers in milliseconds; use this to force intervals-based blobs.
timeSlice: 0
};
exports.default = pluginDefaultOptions;
},{}],8:[function(require,module,exports){
'use strict';
Object.defineProperty(exports, "__esModule", {
value: true
});
var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; }
function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; }
/**
* @file record-engine.js
* @since 2.0.0
*/
var Component = videojs.getComponent('Component');
// supported recorder plugin engines
var RECORDRTC = 'recordrtc';
var LIBVORBISJS = 'libvorbis.js';
var RECORDERJS = 'recorder.js';
var LAMEJS = 'lamejs';
var OPUSRECORDER = 'opus-recorder';
/**
* Base class for recorder backends.
* @class
* @augments videojs.Component
*/
var RecordEngine = function (_Component) {
_inherits(RecordEngine, _Component);
/**
* Creates an instance of this class.
*
* @param {Player} player
* The `Player` that this class should be attached to.
*
* @param {Object} [options]
* The key/value store of player options.
*/
function RecordEngine(player, options) {
_classCallCheck(this, RecordEngine);
// auto mixin the evented mixin (required since video.js v6.6.0)
options.evented = true;
return _possibleConstructorReturn(this, (RecordEngine.__proto__ || Object.getPrototypeOf(RecordEngine)).call(this, player, options));
}
/**
* Remove any temporary data and references to streams.
* @private
*/
_createClass(RecordEngine, [{
key: 'dispose',
value: function dispose() {
// dispose previous recording
if (this.recordedData !== undefined) {
URL.revokeObjectURL(this.recordedData);
}
}
/**
* Add filename and timestamp to recorded file object.
*
* @param {(blob|file)} fileObj - Blob or File object.
*/
}, {
key: 'addFileInfo',
value: function addFileInfo(fileObj) {
var now = new Date();
fileObj.lastModifiedDate = now;
// guess extension name from mime type, e.g. audio/ogg, but
// any extension is valid here. Chrome also accepts extended
// mime types like video/webm;codecs=h264,vp9,opus
var fileExtension = '.' + fileObj.type.split('/')[1];
if (fileExtension.indexOf(';') > -1) {
fileExtension = fileExtension.split(';')[0];
}
// use timestamp in filename, e.g. 1451180941326.ogg
fileObj.name = now.getTime() + fileExtension;
}
/**
* Invoked when recording is stopped and resulting stream is available.
*
* @param {blob} data - Reference to the recorded Blob.
*/
}, {
key: 'onStopRecording',
value: function onStopRecording(data) {
this.recordedData = data;
// add filename and timestamp to recorded file object
this.addFileInfo(this.recordedData);
// remove reference to recorded stream
this.dispose();
// notify listeners
this.trigger('recordComplete');
}
/**
* Show save as dialog in browser so the user can store the recorded media
* locally.
*
* @param {object} name - Object with names for the particular blob(s)
* you want to save. File extensions are added automatically. For
* example: {'video': 'name-of-video-file'}. Supported keys are
* 'audio', 'video' and 'gif'.
*/
}, {
key: 'saveAs',
value: function saveAs(name) {
var fileName = name[Object.keys(name)[0]];
if (typeof navigator.msSaveOrOpenBlob !== 'undefined') {
return navigator.msSaveOrOpenBlob(this.recordedData, fileName);
} else if (typeof navigator.msSaveBlob !== 'undefined') {
return navigator.msSaveBlob(this.recordedData, fileName);
}
var hyperlink = document.createElement('a');
hyperlink.href = URL.createObjectURL(this.recordedData);
hyperlink.download = fileName;
hyperlink.style = 'display:none;opacity:0;color:transparent;';
(document.body || document.documentElement).appendChild(hyperlink);
if (typeof hyperlink.click === 'function') {
hyperlink.click();
} else {
hyperlink.target = '_blank';
hyperlink.dispatchEvent(new MouseEvent('click', {
view: window,
bubbles: true,
cancelable: true
}));
}
URL.revokeObjectURL(hyperlink.href);
}
}]);
return RecordEngine;
}(Component);
// expose component for external plugins
videojs.RecordEngine = RecordEngine;
Component.registerComponent('RecordEngine', RecordEngine);
exports.RecordEngine = RecordEngine;
exports.RECORDRTC = RECORDRTC;
exports.LIBVORBISJS = LIBVORBISJS;
exports.RECORDERJS = RECORDERJS;
exports.LAMEJS = LAMEJS;
exports.OPUSRECORDER = OPUSRECORDER;
},{}],9:[function(require,module,exports){
'use strict';
Object.defineProperty(exports, "__esModule", {
value: true
});
/**
* @file record-mode.js
* @since 2.0.0
*/
// recorder modes
var IMAGE_ONLY = 'image_only';
var AUDIO_ONLY = 'audio_only';
var VIDEO_ONLY = 'video_only';
var AUDIO_VIDEO = 'audio_video';
var ANIMATION = 'animation';
var getRecorderMode = function getRecorderMode(image, audio, video, animation) {
if (isModeEnabled(image)) {
return IMAGE_ONLY;
} else if (isModeEnabled(animation)) {
return ANIMATION;
} else if (isModeEnabled(audio) && !isModeEnabled(video)) {
return AUDIO_ONLY;
} else if (isModeEnabled(audio) && isModeEnabled(video)) {
return AUDIO_VIDEO;
} else if (!isModeEnabled(audio) && isModeEnabled(video)) {
return VIDEO_ONLY;
}
};
/**
* Return boolean indicating whether mode is enabled or not.
*/
var isModeEnabled = function isModeEnabled(mode) {
return mode === Object(mode) || mode === true;
};
exports.getRecorderMode = getRecorderMode;
exports.IMAGE_ONLY = IMAGE_ONLY;
exports.AUDIO_ONLY = AUDIO_ONLY;
exports.VIDEO_ONLY = VIDEO_ONLY;
exports.AUDIO_VIDEO = AUDIO_VIDEO;
exports.ANIMATION = ANIMATION;
},{}],10:[function(require,module,exports){
'use strict';
Object.defineProperty(exports, "__esModule", {
value: true
});
var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
var _get = function get(object, property, receiver) { if (object === null) object = Function.prototype; var desc = Object.getOwnPropertyDescriptor(object, property); if (desc === undefined) { var parent = Object.getPrototypeOf(object); if (parent === null) { return undefined; } else { return get(parent, property, receiver); } } else if ("value" in desc) { return desc.value; } else { var getter = desc.get; if (getter === undefined) { return undefined; } return getter.call(receiver); } };
var _recordEngine = require('./record-engine');
var _detectBrowser = require('../utils/detect-browser');
var _recordMode = require('./record-mode');
function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; }
function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; } /**
* @file record-rtc.js
* @since 2.0.0
*/
var Component = videojs.getComponent('Component');
/**
* Engine used with the MRecordRTC class in the RecordRTC library.
*
* @class
* @augments videojs.RecordEngine
*/
var RecordRTCEngine = function (_RecordEngine) {
_inherits(RecordRTCEngine, _RecordEngine);
function RecordRTCEngine() {
_classCallCheck(this, RecordRTCEngine);
return _possibleConstructorReturn(this, (RecordRTCEngine.__proto__ || Object.getPrototypeOf(RecordRTCEngine)).apply(this, arguments));
}
_createClass(RecordRTCEngine, [{
key: 'setup',
/**
* Setup recording engine.
*/
value: function setup(stream, mediaType, debug) {
this.inputStream = stream;
this.mediaType = mediaType;
this.debug = debug;
// setup RecordRTC
this.engine = new RecordRTC.MRecordRTC();
this.engine.mediaType = this.mediaType;
this.engine.disableLogs = !this.debug;
this.engine.mimeType = this.mimeType;
// audio settings
this.engine.bufferSize = this.bufferSize;
this.engine.sampleRate = this.sampleRate;
this.engine.numberOfAudioChannels = this.audioChannels;
// video/canvas settings
this.engine.video = this.video;
this.engine.canvas = this.canvas;
// animated gif settings
this.engine.quality = this.quality;
this.engine.frameRate = this.frameRate;
if (this.onTimeStamp !== undefined) {
this.engine.timeSlice = this.timeSlice;
this.engine.onTimeStamp = this.onTimeStamp;
}
// connect stream to recording engine
this.engine.addStream(this.inputStream);
}
/**
* Remove any temporary data and references to streams.
*/
}, {
key: 'dispose',
value: function dispose() {
_get(RecordRTCEngine.prototype.__proto__ || Object.getPrototypeOf(RecordRTCEngine.prototype), 'dispose', this).call(this);
if (typeof this.engine.destroy === 'function') {
this.engine.destroy();
}
}
/**
* Start recording.
*/
}, {
key: 'start',
value: function start() {
this.engine.startRecording();
}
/**
* Stop recording. Result will be available async when onStopRecording
* is called.
*/
}, {
key: 'stop',
value: function stop() {
this.engine.stopRecording(this.onStopRecording.bind(this));
}
/**
* Pause recording.
*/
}, {
key: 'pause',
value: function pause() {
this.engine.pauseRecording();
}
/**
* Resume recording.
*/
}, {
key: 'resume',
value: function resume() {
this.engine.resumeRecording();
}
/**
* Show save as dialog in browser so the user can store the recorded media
* locally.
*
* @param {object} name - Object with names for the particular blob(s)
* you want to save. File extensions are added automatically. For
* example: {'video': 'name-of-video-file'}. Supported keys are
* 'audio', 'video' and 'gif'.
*/
}, {
key: 'saveAs',
value: function saveAs(name) {
if (this.engine && name !== undefined) {
this.engine.save(name);
}
}
/**
* Invoked when recording is stopped and resulting stream is available.
*
* @private
* @param {string} audioVideoURL - Reference to the recorded Blob
* object, e.g. 'blob:http://localhost:8080/10100016-4248-9949-b0d6-0bb40db56eba'
* @param {string} type - Media type, eg. 'video' or 'audio'.
*/
}, {
key: 'onStopRecording',
value: function onStopRecording(audioVideoURL, type) {
var _this2 = this;
// store reference to recorded stream URL
this.mediaURL = audioVideoURL;
// store reference to recorded stream data
var recordType = this.player().record().getRecordType();
this.engine.getBlob(function (recording) {
switch (recordType) {
case _recordMode.AUDIO_ONLY:
_this2.recordedData = recording.audio;
_this2.addFileInfo(_this2.recordedData);
// notify listeners
_this2.trigger('recordComplete');
break;
case _recordMode.VIDEO_ONLY:
case _recordMode.AUDIO_VIDEO:
// when recording both audio and video, recordrtc
// calls this twice on chrome, first with audio data
// and then with video data.
// on firefox it's called once but with a single
// blob that includes both audio and video data.
if (recording.video !== undefined) {
// data is video-only but on firefox audio+video
_this2.recordedData = recording.video;
// on the chrome browser two blobs are created
// containing the separate audio/video streams.
if (recordType === _recordMode.AUDIO_VIDEO && (0, _detectBrowser.isChrome)()) {
// store both audio and video
_this2.recordedData = recording;
for (var mtype in _this2.recordedData) {
_this2.addFileInfo(_this2.recordedData[mtype]);
}
} else {
_this2.addFileInfo(_this2.recordedData);
}
// notify listeners
_this2.trigger('recordComplete');
}
break;
case _recordMode.ANIMATION:
_this2.recordedData = recording.gif;
_this2.addFileInfo(_this2.recordedData);
// notify listeners
_this2.trigger('recordComplete');
break;
}
});
}
}]);
return RecordRTCEngine;
}(_recordEngine.RecordEngine);
// expose plugin
videojs.RecordRTCEngine = RecordRTCEngine;
Component.registerComponent('RecordRTCEngine', RecordRTCEngine);
exports.default = RecordRTCEngine;
},{"../utils/detect-browser":12,"./record-engine":8,"./record-mode":9}],11:[function(require,module,exports){
'use strict';
Object.defineProperty(exports, "__esModule", {
value: true
});
/**
* @file browser-shim.js
* @since 2.0.0
*/
var setSrcObject = function setSrcObject(stream, element, ignoreCreateObjectURL) {
if ('createObjectURL' in URL && !ignoreCreateObjectURL) {
try {
element.src = URL.createObjectURL(stream);
} catch (e) {
setSrcObject(stream, element, true);
return;
}
} else if ('srcObject' in element) {
element.srcObject = stream;
} else if ('mozSrcObject' in element) {
element.mozSrcObject = stream;
} else {
console.log('createObjectURL/srcObject both are not supported.');
}
};
exports.default = setSrcObject;
},{}],12:[function(require,module,exports){
'use strict';
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.isSafari = exports.isChrome = exports.isOpera = exports.isEdge = exports.detectBrowser = undefined;
var _window = require('global/window');
var _window2 = _interopRequireDefault(_window);
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
/**
* Browser detector.
*
* @private
* @return {object} result containing browser, version and minVersion
* properties.
*/
var detectBrowser = function detectBrowser() {
// returned result object
var result = {};
result.browser = null;
result.version = null;
result.minVersion = null;
// fail early if it's not a browser
if (typeof _window2.default === 'undefined' || !_window2.default.navigator) {
result.browser = 'Not a supported browser.';
return result;
}
// Firefox
if (navigator.mozGetUserMedia) {
result.browser = 'firefox';
result.version = extractVersion(navigator.userAgent, /Firefox\/(\d+)\./, 1);
result.minVersion = 31;
} else if (navigator.webkitGetUserMedia) {
// Chrome, Chromium, Webview, Opera
if (_window2.default.webkitRTCPeerConnection) {
result.browser = 'chrome';
result.version = extractVersion(navigator.userAgent, /Chrom(e|ium)\/(\d+)\./, 2);
result.minVersion = 38;
} else {
// Safari (in an unpublished version) or unknown webkit-based.
if (navigator.userAgent.match(/Version\/(\d+).(\d+)/)) {
result.browser = 'safari';
result.version = extractVersion(navigator.userAgent, /AppleWebKit\/(\d+)\./, 1);
result.minVersion = 11;
} else {
// unknown webkit-based browser.
result.browser = 'Unsupported webkit-based browser ' + 'with GUM support but no WebRTC support.';
return result;
}
}
// Edge
} else if (navigator.mediaDevices && navigator.userAgent.match(/Edge\/(\d+).(\d+)$/)) {
result.browser = 'edge';
result.version = extractVersion(navigator.userAgent, /Edge\/(\d+).(\d+)$/, 2);
result.minVersion = 10547;
} else if (navigator.mediaDevices && navigator.userAgent.match(/AppleWebKit\/(\d+)\./)) {
// Safari, with webkitGetUserMedia removed.
result.browser = 'safari';
result.version = extractVersion(navigator.userAgent, /AppleWebKit\/(\d+)\./, 1);
} else {
// default fallthrough: not supported.
result.browser = 'Not a supported browser.';
return result;
}
return result;
};
/**
* Extract browser version out of the provided user agent string.
*
* @private
* @param {!string} uastring - userAgent string.
* @param {!string} expr - Regular expression used as match criteria.
* @param {!number} pos - position in the version string to be
* returned.
* @return {!number} browser version.
*/
/**
* @file detect-browser.js
* @since 2.0.0
*/
var extractVersion = function extractVersion(uastring, expr, pos) {
var match = uastring.match(expr);
return match && match.length >= pos && parseInt(match[pos], 10);
};
var isEdge = function isEdge() {
return detectBrowser().browser === 'edge';
};
var isSafari = function isSafari() {
return detectBrowser().browser === 'safari';
};
var isOpera = function isOpera() {
return !!_window2.default.opera || navigator.userAgent.indexOf('OPR/') !== -1;
};
var isChrome = function isChrome() {
return detectBrowser().browser === 'chrome';
};
exports.detectBrowser = detectBrowser;
exports.isEdge = isEdge;
exports.isOpera = isOpera;
exports.isChrome = isChrome;
exports.isSafari = isSafari;
},{"global/window":14}],13:[function(require,module,exports){
'use strict';
Object.defineProperty(exports, "__esModule", {
value: true
});
/**
* @file format-time.js
* @since 2.0.0
*/
/**
* Format seconds as a time string, H:MM:SS, M:SS or M:SS:MMM.
*
* Supplying a guide (in seconds) will force a number of leading zeros
* to cover the length of the guide.
*
* @param {number} seconds - Number of seconds to be turned into a
* string.
* @param {number} guide - Number (in seconds) to model the string
* after.
* @param {number} msDisplayMax - Number (in milliseconds) to model the string
* after.
* @return {string} Time formatted as H:MM:SS, M:SS or M:SS:MMM, e.g.
* 0:00:12.
* @private
*/
var formatTime = function formatTime(seconds, guide, msDisplayMax) {
// Default to using seconds as guide
seconds = seconds < 0 ? 0 : seconds;
guide = guide || seconds;
var s = Math.floor(seconds % 60),
m = Math.floor(seconds / 60 % 60),
h = Math.floor(seconds / 3600),
gm = Math.floor(guide / 60 % 60),
gh = Math.floor(guide / 3600),
ms = Math.floor((seconds - s) * 1000);
// handle invalid times
if (isNaN(seconds) || seconds === Infinity) {
// '-' is false for all relational operators (e.g. <, >=) so this
// setting will add the minimum number of fields specified by the
// guide
h = m = s = ms = '-';
}
// Check if we need to show milliseconds
if (guide > 0 && guide < msDisplayMax) {
if (ms < 100) {
if (ms < 10) {
ms = '00' + ms;
} else {
ms = '0' + ms;
}
}
ms = ':' + ms;
} else {
ms = '';
}
// Check if we need to show hours
h = h > 0 || gh > 0 ? h + ':' : '';
// If hours are showing, we may need to add a leading zero.
// Always show at least one digit of minutes.
m = ((h || gm >= 10) && m < 10 ? '0' + m : m) + ':';
// Check if leading zero is need for seconds
s = s < 10 ? '0' + s : s;
return h + m + s + ms;
};
exports.default = formatTime;
},{}],14:[function(require,module,exports){
(function (global){
var win;
if (typeof window !== "undefined") {
win = window;
} else if (typeof global !== "undefined") {
win = global;
} else if (typeof self !== "undefined"){
win = self;
} else {
win = {};
}
module.exports = win;
}).call(this,typeof global !== "undefined" ? global : typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {})
},{}],15:[function(require,module,exports){
(function (global){
'use strict';
var _typeof = typeof Symbol === "function" && typeof Symbol.iterator === "symbol" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; };
var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
var _get = function get(object, property, receiver) { if (object === null) object = Function.prototype; var desc = Object.getOwnPropertyDescriptor(object, property); if (desc === undefined) { var parent = Object.getPrototypeOf(object); if (parent === null) { return undefined; } else { return get(parent, property, receiver); } } else if ("value" in desc) { return desc.value; } else { var getter = desc.get; if (getter === undefined) { return undefined; } return getter.call(receiver); } };
var _animationDisplay = require('./controls/animation-display');
var _animationDisplay2 = _interopRequireDefault(_animationDisplay);
var _recordCanvas = require('./controls/record-canvas');
var _recordCanvas2 = _interopRequireDefault(_recordCanvas);
var _deviceButton = require('./controls/device-button');
var _deviceButton2 = _interopRequireDefault(_deviceButton);
var _cameraButton = require('./controls/camera-button');
var _cameraButton2 = _interopRequireDefault(_cameraButton);
var _recordToggle = require('./controls/record-toggle');
var _recordToggle2 = _interopRequireDefault(_recordToggle);
var _recordIndicator = require('./controls/record-indicator');
var _recordIndicator2 = _interopRequireDefault(_recordIndicator);
var _defaults = require('./defaults');
var _defaults2 = _interopRequireDefault(_defaults);
var _formatTime = require('./utils/format-time');
var _formatTime2 = _interopRequireDefault(_formatTime);
var _browserShim = require('./utils/browser-shim');
var _browserShim2 = _interopRequireDefault(_browserShim);
var _detectBrowser = require('./utils/detect-browser');
var _recordRtc = require('./engine/record-rtc');
var _recordRtc2 = _interopRequireDefault(_recordRtc);
var _recordEngine = require('./engine/record-engine');
var _recordMode = require('./engine/record-mode');
var _video = (typeof window !== "undefined" ? window['videojs'] : typeof global !== "undefined" ? global['videojs'] : null);
var _video2 = _interopRequireDefault(_video);
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; }
function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; } /**
* @file videojs.record.js
*
* The main file for the videojs-record project.
* MIT license: https://github.com/collab-project/videojs-record/blob/master/LICENSE
*/
var Plugin = _video2.default.getPlugin('plugin');
var Player = _video2.default.getComponent('Player');
var AUTO = 'auto';
// monkey-patch play (#152)
Player.prototype.play = function play() {
var retval = this.techGet_('play');
// silence errors (unhandled promise from play)
if (retval !== undefined && typeof retval.then === 'function') {
retval.then(null, function (e) {});
}
return retval;
};
/**
* Record audio/video/images using the Video.js player.
*
* @class
* @augments videojs.Plugin
*/
var Record = function (_Plugin) {
_inherits(Record, _Plugin);
/**
* The constructor function for the class.
*
* @param {(videojs.Player|Object)} player
* @param {Object} options - Player options.
*/
function Record(player, options) {
_classCallCheck(this, Record);
// setup plugin options
var _this = _possibleConstructorReturn(this, (Record.__proto__ || Object.getPrototypeOf(Record)).call(this, player, options));
_this.loadOptions();
// (re)set recorder state
_this.resetState();
// add device button with icon based on type
var deviceIcon = 'av-perm';
switch (_this.getRecordType()) {
case _recordMode.IMAGE_ONLY:
case _recordMode.VIDEO_ONLY:
case _recordMode.ANIMATION:
deviceIcon = 'video-perm';
break;
case _recordMode.AUDIO_ONLY:
deviceIcon = 'audio-perm';
break;
}
_deviceButton2.default.prototype.buildCSSClass = function () {
// use dynamic icon class
return 'vjs-device-button vjs-control vjs-icon-' + deviceIcon;
};
player.deviceButton = new _deviceButton2.default(player, options);
player.addChild(player.deviceButton);
// add blinking record indicator
player.recordIndicator = new _recordIndicator2.default(player, options);
player.recordIndicator.hide();
player.addChild(player.recordIndicator);
// add canvas for recording and displaying image
player.recordCanvas = new _recordCanvas2.default(player, options);
player.recordCanvas.hide();
player.addChild(player.recordCanvas);
// add image for animation display
player.animationDisplay = new _animationDisplay2.default(player, options);
player.animationDisplay.hide();
player.addChild(player.animationDisplay);
// add camera button
player.cameraButton = new _cameraButton2.default(player, options);
player.cameraButton.hide();
// add record toggle
player.recordToggle = new _recordToggle2.default(player, options);
player.recordToggle.hide();
// wait until player ui is ready
_this.player.one('ready', _this.setupUI.bind(_this));
return _this;
}
/**
* Setup plugin options.
*/
_createClass(Record, [{
key: 'loadOptions',
value: function loadOptions() {
var recordOptions = _video2.default.mergeOptions(_defaults2.default, this.player.options_.plugins.record);
// record settings
this.recordImage = recordOptions.image;
this.recordAudio = recordOptions.audio;
this.recordVideo = recordOptions.video;
this.recordAnimation = recordOptions.animation;
this.maxLength = recordOptions.maxLength;
this.debug = recordOptions.debug;
this.recordTimeSlice = recordOptions.timeSlice;
// video/canvas settings
this.videoFrameWidth = recordOptions.frameWidth;
this.videoFrameHeight = recordOptions.frameHeight;
this.videoRecorderType = recordOptions.videoRecorderType;
this.videoMimeType = recordOptions.videoMimeType;
// audio settings
this.audioEngine = recordOptions.audioEngine;
this.audioRecorderType = recordOptions.audioRecorderType;
this.audioWorkerURL = recordOptions.audioWorkerURL;
this.audioBufferSize = recordOptions.audioBufferSize;
this.audioSampleRate = recordOptions.audioSampleRate;
this.audioChannels = recordOptions.audioChannels;
this.audioMimeType = recordOptions.audioMimeType;
// animation settings
this.animationFrameRate = recordOptions.animationFrameRate;
this.animationQuality = recordOptions.animationQuality;
}
/**
* Player UI is ready.
* @private
*/
}, {
key: 'setupUI',
value: function setupUI() {
var _this2 = this;
// insert custom controls on left-side of controlbar
this.player.controlBar.addChild(this.player.cameraButton);
this.player.controlBar.el().insertBefore(this.player.cameraButton.el(), this.player.controlBar.el().firstChild);
this.player.controlBar.el().insertBefore(this.player.recordToggle.el(), this.player.controlBar.el().firstChild);
// get rid of unused controls
if (this.player.controlBar.remainingTimeDisplay !== undefined) {
this.player.controlBar.remainingTimeDisplay.el().style.display = 'none';
}
if (this.player.controlBar.liveDisplay !== undefined) {
this.player.controlBar.liveDisplay.el().style.display = 'none';
}
// loop feature is never used in this plugin
this.player.loop(false);
// tweak player UI based on type
switch (this.getRecordType()) {
case _recordMode.AUDIO_ONLY:
// reference to videojs-wavesurfer plugin
this.surfer = this.player.wavesurfer();
break;
case _recordMode.IMAGE_ONLY:
case _recordMode.VIDEO_ONLY:
case _recordMode.AUDIO_VIDEO:
case _recordMode.ANIMATION:
// customize controls
this.player.bigPlayButton.hide();
// loadedmetadata resets the durationDisplay for the
// first time
this.player.one('loadedmetadata', function () {
// display max record time
_this2.setDuration(_this2.maxLength);
});
// the native controls don't work for this UI so disable
// them no matter what
if (this.player.usingNativeControls_ === true) {
if (this.player.tech_.el_ !== undefined) {
this.player.tech_.el_.controls = false;
}
}
// clicking or tapping the player video element should not try
// to start playback
this.player.removeTechControlsListeners_();
if (this.player.options_.controls) {
// progress control isn't used by this plugin
this.player.controlBar.progressControl.hide();
// prevent controlbar fadeout
this.player.on('userinactive', function (event) {
_this2.player.userActive(true);
});
// videojs automatically hides the controls when no valid 'source'
// element is included in the video or audio tag. Don't. Ever again.
this.player.controlBar.show();
this.player.controlBar.el().style.display = 'flex';
}
break;
}
// disable time display events that constantly try to reset the current time
// and duration values
this.player.off('timeupdate');
this.player.off('durationchange');
this.player.off('loadedmetadata');
// display max record time
this.setDuration(this.maxLength);
// hide play control
this.player.controlBar.playToggle.hide();
}
/**
* Indicates whether the plugin is currently recording or not.
*
* @return {boolean} Plugin currently recording or not.
*/
}, {
key: 'isRecording',
value: function isRecording() {
return this._recording;
}
/**
* Indicates whether the plugin is currently processing recorded data
* or not.
*
* @return {boolean} Plugin processing or not.
*/
}, {
key: 'isProcessing',
value: function isProcessing() {
return this._processing;
}
/**
* Indicates whether the plugin is destroyed or not.
*
* @return {boolean} Plugin destroyed or not.
*/
}, {
key: 'isDestroyed',
value: function isDestroyed() {
return this.player && this.player.children() === null;
}
/**
* Open the browser's recording device selection dialog.
*/
}, {
key: 'getDevice',
value: function getDevice() {
// define device callbacks once
if (this.deviceReadyCallback === undefined) {
this.deviceReadyCallback = this.onDeviceReady.bind(this);
}
if (this.deviceErrorCallback === undefined) {
this.deviceErrorCallback = this.onDeviceError.bind(this);
}
if (this.engineStopCallback === undefined) {
this.engineStopCallback = this.onRecordComplete.bind(this);
}
// ask the browser to give the user access to the media device
// and get a stream reference in the callback function
switch (this.getRecordType()) {
case _recordMode.AUDIO_ONLY:
// setup microphone
this.mediaType = {
audio: this.audioRecorderType === AUTO ? true : this.audioRecorderType,
video: false
};
// remove existing microphone listeners
this.surfer.surfer.microphone.un('deviceReady', this.deviceReadyCallback);
this.surfer.surfer.microphone.un('deviceError', this.deviceErrorCallback);
// setup new microphone listeners
this.surfer.surfer.microphone.on('deviceReady', this.deviceReadyCallback);
this.surfer.surfer.microphone.on('deviceError', this.deviceErrorCallback);
// disable existing playback events
this.surfer.setupPlaybackEvents(false);
// (re)set surfer liveMode
this.surfer.liveMode = true;
this.surfer.surfer.microphone.paused = false;
// open browser device selection dialog
this.surfer.surfer.microphone.start();
break;
case _recordMode.IMAGE_ONLY:
case _recordMode.VIDEO_ONLY:
// setup camera
this.mediaType = {
audio: false,
video: this.videoRecorderType === AUTO ? true : this.videoRecorderType
};
navigator.mediaDevices.getUserMedia({
audio: false,
video: this.getRecordType() === _recordMode.IMAGE_ONLY ? this.recordImage : this.recordVideo
}).then(this.onDeviceReady.bind(this)).catch(this.onDeviceError.bind(this));
break;
case _recordMode.AUDIO_VIDEO:
// setup camera and microphone
this.mediaType = {
audio: this.audioRecorderType === AUTO ? true : this.audioRecorderType,
video: this.videoRecorderType === AUTO ? true : this.videoRecorderType
};
navigator.mediaDevices.getUserMedia({
audio: this.recordAudio,
video: this.recordVideo
}).then(this.onDeviceReady.bind(this)).catch(this.onDeviceError.bind(this));
break;
case _recordMode.ANIMATION:
// setup camera
this.mediaType = {
// animated GIF
audio: false,
video: false,
gif: true
};
navigator.mediaDevices.getUserMedia({
audio: false,
video: this.recordAnimation
}).then(this.onDeviceReady.bind(this)).catch(this.onDeviceError.bind(this));
break;
}
}
/**
* Invoked when the device is ready.
* @private
* @param stream: LocalMediaStream instance.
*/
}, {
key: 'onDeviceReady',
value: function onDeviceReady(stream) {
var _this3 = this;
this._deviceActive = true;
// store reference to stream for stopping etc.
this.stream = stream;
// hide device selection button
this.player.deviceButton.hide();
// reset time (e.g. when stopDevice was used)
this.setDuration(this.maxLength);
this.setCurrentTime(0);
// hide play/pause control (e.g. when stopDevice was used)
this.player.controlBar.playToggle.hide();
// reset playback listeners
this.off(this.player, 'timeupdate', this.playbackTimeUpdate);
this.off(this.player, 'ended', this.playbackTimeUpdate);
// setup recording engine
if (this.getRecordType() !== _recordMode.IMAGE_ONLY) {
// currently libvorbis.js, recorder.js, opus-recorder and lamejs
// are only supported in audio-only mode
if (this.getRecordType() !== _recordMode.AUDIO_ONLY && (this.audioEngine === _recordEngine.LIBVORBISJS || this.audioEngine === _recordEngine.RECORDERJS || this.audioEngine === _recordEngine.LAMEJS || this.audioEngine === _recordEngine.OPUSRECORDER)) {
throw new Error('Currently ' + this.audioEngine + ' is only supported in audio-only mode.');
}
// get recorder class
var EngineClass;
switch (this.audioEngine) {
case _recordEngine.RECORDRTC:
// RecordRTC.js (default)
EngineClass = _video2.default.RecordRTCEngine;
break;
case _recordEngine.LIBVORBISJS:
// libvorbis.js
EngineClass = _video2.default.LibVorbisEngine;
break;
case _recordEngine.RECORDERJS:
// recorder.js
EngineClass = _video2.default.RecorderjsEngine;
break;
case _recordEngine.LAMEJS:
// lamejs
EngineClass = _video2.default.LamejsEngine;
break;
case _recordEngine.OPUSRECORDER:
// opus-recorder
EngineClass = _video2.default.OpusRecorderEngine;
break;
default:
// unknown engine
throw new Error('Unknown audioEngine: ' + this.audioEngine);
}
try {
// connect stream to recording engine
this.engine = new EngineClass(this.player, this.player.options_);
} catch (err) {
console.error(err);
throw new Error('Could not load ' + this.audioEngine + ' plugin');
}
// listen for events
this.engine.on('recordComplete', this.engineStopCallback);
// audio settings
this.engine.bufferSize = this.audioBufferSize;
this.engine.sampleRate = this.audioSampleRate;
this.engine.audioChannels = this.audioChannels;
this.engine.audioWorkerURL = this.audioWorkerURL;
// mime type
this.engine.mimeType = {
video: this.videoMimeType,
gif: 'image/gif'
};
if (this.audioMimeType !== null && this.audioMimeType !== AUTO) {
this.engine.mimeType.audio = this.audioMimeType;
}
// video/canvas settings
this.engine.video = {
width: this.videoFrameWidth,
height: this.videoFrameHeight
};
this.engine.canvas = {
width: this.videoFrameWidth,
height: this.videoFrameHeight
};
// animated GIF settings
this.engine.quality = this.animationQuality;
this.engine.frameRate = this.animationFrameRate;
// timeSlice
if (this.recordTimeSlice && this.recordTimeSlice > 0) {
this.engine.timeSlice = this.recordTimeSlice;
this.engine.onTimeStamp = this.onTimeStamp.bind(this);
}
// initialize recorder
this.engine.setup(this.stream, this.mediaType, this.debug);
// show elements that should never be hidden in animation,
// audio and/or video modus
var uiElements = [this.player.controlBar.currentTimeDisplay, this.player.controlBar.timeDivider, this.player.controlBar.durationDisplay];
uiElements.forEach(function (element) {
if (element !== undefined) {
element.el().style.display = 'block';
element.show();
}
});
// show record button
this.player.recordToggle.show();
} else {
// disable record indicator
this.player.recordIndicator.disable();
// setup UI for retrying snapshot (e.g. when stopDevice was
// used)
this.retrySnapshot();
// reset and show camera button
this.player.cameraButton.onStop();
this.player.cameraButton.show();
}
// setup preview
if (this.getRecordType() !== _recordMode.AUDIO_ONLY) {
// show live preview
this.mediaElement = this.player.el().firstChild;
this.mediaElement.controls = false;
// mute incoming audio for feedback loops
this.mediaElement.muted = true;
// hide the volume bar while it's muted
this.displayVolumeControl(false);
// load stream
this.load(this.stream);
// stream loading is async, so we wait until it's ready to play
// the stream
this.player.one('loadedmetadata', function () {
// start stream
_this3.mediaElement.play();
// forward to listeners
_this3.player.trigger('deviceReady');
});
} else {
// forward to listeners
this.player.trigger('deviceReady');
}
}
/**
* Invoked when an device error occurred.
* @private
*/
}, {
key: 'onDeviceError',
value: function onDeviceError(code) {
this._deviceActive = false;
// store code
this.player.deviceErrorCode = code;
// forward error to player
this.player.trigger('deviceError');
}
/**
* Start recording.
*/
}, {
key: 'start',
value: function start() {
var _this4 = this;
if (!this.isProcessing()) {
this._recording = true;
// hide play/pause control
this.player.controlBar.playToggle.hide();
// reset playback listeners
this.off(this.player, 'timeupdate', this.playbackTimeUpdate);
this.off(this.player, 'ended', this.playbackTimeUpdate);
// start preview
switch (this.getRecordType()) {
case _recordMode.AUDIO_ONLY:
// disable playback events
this.surfer.setupPlaybackEvents(false);
// start/resume live audio visualization
this.surfer.surfer.microphone.paused = false;
this.surfer.liveMode = true;
this.surfer.surfer.microphone.play();
break;
case _recordMode.VIDEO_ONLY:
case _recordMode.AUDIO_VIDEO:
// preview video stream in video element
this.startVideoPreview();
break;
case _recordMode.ANIMATION:
// hide the first frame
this.player.recordCanvas.hide();
// hide the animation
this.player.animationDisplay.hide();
// show preview video
this.mediaElement.style.display = 'block';
// for animations, capture the first frame
// that can be displayed as soon as recording
// is complete
this.captureFrame().then(function (result) {
// start video preview **after** capturing first frame
_this4.startVideoPreview();
});
break;
}
// start recording
switch (this.getRecordType()) {
case _recordMode.IMAGE_ONLY:
// create snapshot
this.createSnapshot();
// notify UI
this.player.trigger('startRecord');
break;
case _recordMode.VIDEO_ONLY:
case _recordMode.AUDIO_VIDEO:
case _recordMode.ANIMATION:
// wait for media stream on video element to actually load
this.player.one('loadedmetadata', function () {
// start actually recording process
_this4.startRecording();
});
break;
default:
// all resources have already loaded, so we can start
// recording right away
this.startRecording();
}
}
}
/**
* Start recording.
* @private
*/
}, {
key: 'startRecording',
value: function startRecording() {
// register starting point
this.paused = false;
this.pauseTime = this.pausedTime = 0;
this.startTime = new Date().getTime();
// start countdown
this.countDown = this.player.setInterval(this.onCountDown.bind(this), 100);
// cleanup previous recording
if (this.engine !== undefined) {
this.engine.dispose();
}
// start recording stream
this.engine.start();
// notify UI
this.player.trigger('startRecord');
}
/**
* Stop recording.
*/
}, {
key: 'stop',
value: function stop() {
if (!this.isProcessing()) {
this._recording = false;
this._processing = true;
if (this.getRecordType() !== _recordMode.IMAGE_ONLY) {
// notify UI
this.player.trigger('stopRecord');
// stop countdown
this.player.clearInterval(this.countDown);
// stop recording stream (result will be available async)
if (this.engine) {
this.engine.stop();
}
} else {
if (this.player.recordedData) {
// notify listeners that image data is (already) available
this.player.trigger('finishRecord');
}
}
}
}
/**
* Stop device(s) and recording if active.
*/
}, {
key: 'stopDevice',
value: function stopDevice() {
if (this.isRecording()) {
// stop stream once recorded data is available,
// otherwise it'll break recording
this.player.one('finishRecord', this.stopStream.bind(this));
// stop recording
this.stop();
} else {
// stop stream now, since there's no recorded data available
this.stopStream();
}
}
/**
* Stop stream and device.
*/
}, {
key: 'stopStream',
value: function stopStream() {
// stop stream and device
if (this.stream) {
this._deviceActive = false;
if (this.getRecordType() === _recordMode.AUDIO_ONLY) {
// make the microphone plugin stop it's device
this.surfer.surfer.microphone.stopDevice();
return;
}
this.stream.getTracks().forEach(function (stream) {
stream.stop();
});
}
}
/**
* Pause recording.
*/
}, {
key: 'pause',
value: function pause() {
if (!this.paused) {
this.pauseTime = new Date().getTime();
this.paused = true;
this.engine.pause();
}
}
/**
* Resume recording.
*/
}, {
key: 'resume',
value: function resume() {
if (this.paused) {
this.pausedTime += new Date().getTime() - this.pauseTime;
this.engine.resume();
this.paused = false;
}
}
/**
* Invoked when recording completed and the resulting stream is
* available.
* @private
*/
}, {
key: 'onRecordComplete',
value: function onRecordComplete() {
var _this5 = this;
// store reference to recorded stream data
this.player.recordedData = this.engine.recordedData;
// change the replay button back to a play button
this.player.controlBar.playToggle.removeClass('vjs-ended');
this.player.controlBar.playToggle.show();
// notify listeners that data is available
this.player.trigger('finishRecord');
switch (this.getRecordType()) {
case _recordMode.AUDIO_ONLY:
// pause player so user can start playback
this.surfer.pause();
// setup events for playback
this.surfer.setupPlaybackEvents(true);
// display loader
this.player.loadingSpinner.show();
// restore interaction with controls after waveform
// rendering is complete
this.surfer.surfer.once('ready', function () {
_this5._processing = false;
});
// visualize recorded stream
this.load(this.player.recordedData);
break;
case _recordMode.VIDEO_ONLY:
case _recordMode.AUDIO_VIDEO:
// pausing the player so we can visualize the recorded data
// will trigger an async video.js 'pause' event that we
// have to wait for.
this.player.one('pause', function () {
// video data is ready
_this5._processing = false;
// hide loader
_this5.player.loadingSpinner.hide();
// show stream total duration
_this5.setDuration(_this5.streamDuration);
// update time during playback and at end
_this5.on(_this5.player, 'timeupdate', _this5.playbackTimeUpdate);
_this5.on(_this5.player, 'ended', _this5.playbackTimeUpdate);
// unmute local audio during playback
if (_this5.getRecordType() === _recordMode.AUDIO_VIDEO) {
_this5.mediaElement.muted = false;
// show the volume bar when it's unmuted
_this5.displayVolumeControl(true);
}
// load recorded media
if ((0, _detectBrowser.isChrome)() && _this5.getRecordType() === _recordMode.AUDIO_VIDEO) {
// use video property on Chrome
_this5.load(_this5.player.recordedData.video);
} else {
_this5.load(_this5.player.recordedData);
}
});
// pause player so user can start playback
this.player.pause();
break;
case _recordMode.ANIMATION:
// animation data is ready
this._processing = false;
// hide loader
this.player.loadingSpinner.hide();
// show animation total duration
this.setDuration(this.streamDuration);
// hide preview video
this.mediaElement.style.display = 'none';
// show the first frame
this.player.recordCanvas.show();
// pause player so user can start playback
this.player.pause();
// show animation on play
this.on(this.player, 'play', this.showAnimation);
// hide animation on pause
this.on(this.player, 'pause', this.hideAnimation);
break;
}
}
/**
* Invoked during recording and displays the remaining time.
* @private
*/
}, {
key: 'onCountDown',
value: function onCountDown() {
if (!this.paused) {
var now = new Date().getTime();
var duration = this.maxLength;
var currentTime = (now - (this.startTime + this.pausedTime)) / 1000;
this.streamDuration = currentTime;
if (currentTime >= duration) {
// at the end
currentTime = duration;
// stop recording
this.stop();
}
// update duration
this.setDuration(duration);
// update current time
this.setCurrentTime(currentTime, duration);
// notify listeners
this.player.trigger('progressRecord');
}
}
/**
* Get the current time of the recorded stream during playback.
*
* Returns 0 if no recording is available (yet).
*/
}, {
key: 'getCurrentTime',
value: function getCurrentTime() {
var currentTime = isNaN(this.streamCurrentTime) ? 0 : this.streamCurrentTime;
if (this.getRecordType() === _recordMode.AUDIO_ONLY) {
currentTime = this.surfer.getCurrentTime();
}
return currentTime;
}
/**
* Updates the player's element displaying the current time.
*
* @private
* @param {number} [currentTime=0] - Current position of the
* playhead (in seconds).
* @param {number} [duration=0] - Duration in seconds.
*/
}, {
key: 'setCurrentTime',
value: function setCurrentTime(currentTime, duration) {
currentTime = isNaN(currentTime) ? 0 : currentTime;
duration = isNaN(duration) ? 0 : duration;
switch (this.getRecordType()) {
case _recordMode.AUDIO_ONLY:
this.surfer.setCurrentTime(currentTime, duration);
break;
case _recordMode.VIDEO_ONLY:
case _recordMode.AUDIO_VIDEO:
case _recordMode.ANIMATION:
this.streamCurrentTime = Math.min(currentTime, duration);
// update current time display component
this.player.controlBar.currentTimeDisplay.formattedTime_ = this.player.controlBar.currentTimeDisplay.contentEl().lastChild.textContent = (0, _formatTime2.default)(this.streamCurrentTime, duration, this.msDisplayMax);
break;
}
}
/**
* Get the length of the recorded stream in seconds.
*
* Returns 0 if no recording is available (yet).
*/
}, {
key: 'getDuration',
value: function getDuration() {
var duration = isNaN(this.streamDuration) ? 0 : this.streamDuration;
return duration;
}
/**
* Updates the player's element displaying the duration time.
*
* @param {number} [duration=0] - Duration in seconds.
* @private
*/
}, {
key: 'setDuration',
value: function setDuration(duration) {
duration = isNaN(duration) ? 0 : duration;
switch (this.getRecordType()) {
case _recordMode.AUDIO_ONLY:
this.surfer.setDuration(duration);
break;
case _recordMode.VIDEO_ONLY:
case _recordMode.AUDIO_VIDEO:
case _recordMode.ANIMATION:
// update duration display component
this.player.controlBar.durationDisplay.formattedTime_ = this.player.controlBar.durationDisplay.contentEl().lastChild.textContent = (0, _formatTime2.default)(duration, duration, this.msDisplayMax);
break;
}
}
/**
* Start loading data.
*
* @param {(string|blob|file)} url - Either the URL of the media file,
* a Blob, a File object or MediaStream.
*/
}, {
key: 'load',
value: function load(url) {
switch (this.getRecordType()) {
case _recordMode.AUDIO_ONLY:
// visualize recorded Blob stream
this.surfer.load(url);
break;
case _recordMode.IMAGE_ONLY:
case _recordMode.VIDEO_ONLY:
case _recordMode.AUDIO_VIDEO:
case _recordMode.ANIMATION:
if (url instanceof Blob || url instanceof File) {
// assign blob using createObjectURL
(0, _browserShim2.default)(url, this.mediaElement, false);
} else {
// assign stream without createObjectURL
(0, _browserShim2.default)(url, this.mediaElement, true);
}
break;
}
}
/**
* Show save as dialog in browser so the user can store the recorded media
* locally.
*
* @param {object} name - Object with one or more names for the particular
* blob(s) you want to save. File extensions are added automatically.
* For example: {'video': 'name-of-video-file'}. Supported keys are
* 'audio', 'video' and 'gif'.
*/
}, {
key: 'saveAs',
value: function saveAs(name) {
if (this.engine && name !== undefined) {
this.engine.saveAs(name);
}
}
/**
* Destroy plugin only.
*
* Use `destroy` to remove the plugin and the player.
*/
}, {
key: 'dispose',
value: function dispose() {
// disable common event listeners
this.player.off('ready');
this.player.off('userinactive');
this.player.off('loadedmetadata');
// prevent callbacks if recording is in progress
if (this.engine) {
this.engine.dispose();
this.engine.off('recordComplete', this.engineStopCallback);
}
// stop recording and device
this.stop();
this.stopDevice();
// stop countdown
this.player.clearInterval(this.countDown);
// dispose wavesurfer.js
if (this.getRecordType() == _recordMode.AUDIO_ONLY) {
if (this.surfer) {
// also disposes player
this.surfer.destroy();
}
}
this.resetState();
_get(Record.prototype.__proto__ || Object.getPrototypeOf(Record.prototype), 'dispose', this).call(this);
}
/**
* Destroy plugin and players and cleanup resources.
*/
}, {
key: 'destroy',
value: function destroy() {
this.player.dispose();
}
/**
* Reset the plugin.
*/
}, {
key: 'reset',
value: function reset() {
var _this6 = this;
// prevent callbacks if recording is in progress
if (this.engine) {
this.engine.dispose();
this.engine.off('recordComplete', this.engineStopCallback);
}
// stop recording and device
this.stop();
this.stopDevice();
// stop countdown
this.player.clearInterval(this.countDown);
// reset options
this.loadOptions();
// reset recorder state
this.resetState();
// reset record time
this.setDuration(this.maxLength);
this.setCurrentTime(0);
// reset player
this.player.reset();
switch (this.getRecordType()) {
case _recordMode.AUDIO_ONLY:
if (this.surfer && this.surfer.surfer) {
// empty last frame
this.surfer.surfer.empty();
}
break;
case _recordMode.IMAGE_ONLY:
case _recordMode.ANIMATION:
// reset UI
this.player.recordCanvas.hide();
this.player.cameraButton.hide();
break;
}
// hide play control
this.player.controlBar.playToggle.hide();
// show device selection button
this.player.deviceButton.show();
// hide record button
this.player.recordToggle.hide();
// loadedmetadata resets the durationDisplay for the
// first time
this.player.one('loadedmetadata', function () {
// display max record time
_this6.setDuration(_this6.maxLength);
});
}
/**
* Reset the plugin recorder state.
* @private
*/
}, {
key: 'resetState',
value: function resetState() {
this._recording = false;
this._processing = false;
this._deviceActive = false;
this.devices = [];
}
/**
* Get recorder type.
*/
}, {
key: 'getRecordType',
value: function getRecordType() {
return (0, _recordMode.getRecorderMode)(this.recordImage, this.recordAudio, this.recordVideo, this.recordAnimation);
}
/**
* Create and display snapshot image.
* @private
*/
}, {
key: 'createSnapshot',
value: function createSnapshot() {
var _this7 = this;
this.captureFrame().then(function (result) {
// turn the canvas data into base64 data with a PNG header
_this7.player.recordedData = result.toDataURL('image/png');
// hide preview video
_this7.mediaElement.style.display = 'none';
// show the snapshot
_this7.player.recordCanvas.show();
// stop recording
_this7.stop();
});
}
/**
* Reset UI for retrying a snapshot image.
* @private
*/
}, {
key: 'retrySnapshot',
value: function retrySnapshot() {
this._processing = false;
// retry: hide the snapshot
this.player.recordCanvas.hide();
// show preview video
this.player.el().firstChild.style.display = 'block';
}
/**
* Capture frame from camera and copy data to canvas.
* @private
*/
}, {
key: 'captureFrame',
value: function captureFrame() {
var _this8 = this;
var detected = (0, _detectBrowser.detectBrowser)();
var recordCanvas = this.player.recordCanvas.el().firstChild;
// set the canvas size to the dimensions of the camera,
// which also wipes the content of the canvas
recordCanvas.width = this.player.width();
recordCanvas.height = this.player.height();
return new Promise(function (resolve, reject) {
// MediaCapture is only supported on:
// - Chrome 60 and newer (see
// https://github.com/w3c/mediacapture-image/blob/gh-pages/implementation-status.md)
// - Firefox behind flag (https://bugzilla.mozilla.org/show_bug.cgi?id=888177)
//
// importing ImageCapture can fail when enabling chrome flag is still required.
// if so; ignore and continue
if (detected.browser === 'chrome' && detected.version >= 60 && (typeof ImageCapture === 'undefined' ? 'undefined' : _typeof(ImageCapture)) === (typeof Function === 'undefined' ? 'undefined' : _typeof(Function))) {
try {
var track = _this8.stream.getVideoTracks()[0];
var imageCapture = new ImageCapture(track);
imageCapture.takePhoto().then(function (blob) {
return createImageBitmap(blob);
}).then(function (imageBitmap) {
// get a frame and copy it onto the canvas
_this8.drawCanvas(recordCanvas, imageBitmap);
// notify others
resolve(recordCanvas);
});
return;
} catch (err) {}
}
// no ImageCapture available: do it the oldskool way
// get a frame and copy it onto the canvas
_this8.drawCanvas(recordCanvas, _this8.mediaElement);
// notify others
resolve(recordCanvas);
});
}
/**
* Draw image frame on canvas element.
* @private
*/
}, {
key: 'drawCanvas',
value: function drawCanvas(canvas, element) {
canvas.getContext('2d').drawImage(element, 0, 0, canvas.width, canvas.height);
}
/**
* Start preview of video stream.
* @private
*/
}, {
key: 'startVideoPreview',
value: function startVideoPreview() {
// disable playback events
this.off('timeupdate');
this.off('durationchange');
this.off('loadedmetadata');
this.off('play');
// mute local audio
this.mediaElement.muted = true;
// hide volume control to prevent feedback
this.displayVolumeControl(false);
// start or resume live preview
this.load(this.stream);
this.mediaElement.play();
}
/**
* Show animated GIF.
* @private
*/
}, {
key: 'showAnimation',
value: function showAnimation() {
var animationDisplay = this.player.animationDisplay.el().firstChild;
// set the image size to the dimensions of the recorded animation
animationDisplay.width = this.player.width();
animationDisplay.height = this.player.height();
// hide the first frame
this.player.recordCanvas.hide();
// show the animation
(0, _browserShim2.default)(this.player.recordedData, animationDisplay, false);
this.player.animationDisplay.show();
}
/**
* Hide animated GIF.
* @private
*/
}, {
key: 'hideAnimation',
value: function hideAnimation() {
// show the first frame
this.player.recordCanvas.show();
// hide the animation
this.player.animationDisplay.hide();
}
/**
* Update time during playback.
* @private
*/
}, {
key: 'playbackTimeUpdate',
value: function playbackTimeUpdate() {
this.setCurrentTime(this.player.currentTime(), this.streamDuration);
}
/**
* Received new timestamp (when timeSlice option is enabled).
* @private
*/
}, {
key: 'onTimeStamp',
value: function onTimeStamp(current, all) {
this.player.currentTimestamp = current;
this.player.allTimestamps = all;
// get blob (only for MediaStreamRecorder)
var internal;
switch (this.getRecordType()) {
case _recordMode.AUDIO_ONLY:
internal = this.engine.engine.audioRecorder;
break;
case _recordMode.ANIMATION:
internal = this.engine.engine.gifRecorder;
break;
default:
internal = this.engine.engine.videoRecorder;
}
internal = internal.getInternalRecorder();
if (internal instanceof MediaStreamRecorder === true) {
this.player.recordedData = internal.getArrayOfBlobs();
}
// notify others
this.player.trigger('timestamp');
}
/**
* Collects information about the media input and output devices
* available on the system.
*
* Returns an array.
*/
}, {
key: 'enumerateDevices',
value: function enumerateDevices() {
var _this9 = this;
if (!navigator.mediaDevices || !navigator.mediaDevices.enumerateDevices) {
this.player.enumerateErrorCode = 'enumerateDevices() not supported.';
this.player.trigger('enumerateError');
return;
}
// List cameras and microphones.
navigator.mediaDevices.enumerateDevices(this).then(function (devices) {
_this9.devices = [];
devices.forEach(function (device) {
_this9.devices.push(device);
});
// notify listeners
_this9.player.trigger('enumerateReady');
}).catch(function (err) {
_this9.player.enumerateErrorCode = err;
_this9.player.trigger('enumerateError');
});
}
/**
* Show or hide the volume menu.
*
* @private
* @param {boolean} display - Hide/show volume control.
*/
}, {
key: 'displayVolumeControl',
value: function displayVolumeControl(display) {
if (this.player.controlBar.volumePanel !== undefined) {
if (display === true) {
display = 'flex';
} else {
display = 'none';
}
this.player.controlBar.volumePanel.el().style.display = display;
}
}
}]);
return Record;
}(Plugin);
// version nr gets replaced during build
Record.VERSION = '2.0.6';
// register plugin
_video2.default.Record = Record;
_video2.default.registerPlugin('record', Record);
// export plugin
module.exports = {
Record: Record
};
}).call(this,typeof global !== "undefined" ? global : typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {})
},{"./controls/animation-display":1,"./controls/camera-button":2,"./controls/device-button":3,"./controls/record-canvas":4,"./controls/record-indicator":5,"./controls/record-toggle":6,"./defaults":7,"./engine/record-engine":8,"./engine/record-mode":9,"./engine/record-rtc":10,"./utils/browser-shim":11,"./utils/detect-browser":12,"./utils/format-time":13}]},{},[15])(15)
});
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["node_modules/browser-pack/_prelude.js","es5/controls/animation-display.js","es5/controls/camera-button.js","es5/controls/device-button.js","es5/controls/record-canvas.js","es5/controls/record-indicator.js","es5/controls/record-toggle.js","es5/defaults.js","es5/engine/record-engine.js","es5/engine/record-mode.js","es5/engine/record-rtc.js","es5/utils/browser-shim.js","es5/utils/detect-browser.js","es5/utils/format-time.js","node_modules/global/window.js"],"names":[],"mappings":";;;;;;;AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7JA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5KA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7OA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACxEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","'use strict';\n\nObject.defineProperty(exports, \"__esModule\", {\n  value: true\n});\n\nvar _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if (\"value\" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();\n\nvar _get = function get(object, property, receiver) { if (object === null) object = Function.prototype; var desc = Object.getOwnPropertyDescriptor(object, property); if (desc === undefined) { var parent = Object.getPrototypeOf(object); if (parent === null) { return undefined; } else { return get(parent, property, receiver); } } else if (\"value\" in desc) { return desc.value; } else { var getter = desc.get; if (getter === undefined) { return undefined; } return getter.call(receiver); } };\n\nfunction _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError(\"Cannot call a class as a function\"); } }\n\nfunction _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError(\"this hasn't been initialised - super() hasn't been called\"); } return call && (typeof call === \"object\" || typeof call === \"function\") ? call : self; }\n\nfunction _inherits(subClass, superClass) { if (typeof superClass !== \"function\" && superClass !== null) { throw new TypeError(\"Super expression must either be null or a function, not \" + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; }\n\n/**\n * @file animation-display.js\n * @since 2.0.0\n */\n\nvar Component = videojs.getComponent('Component');\n\n/**\n * Image for displaying animated GIF image.\n *\n * @class\n * @augments videojs.Component\n*/\n\nvar AnimationDisplay = function (_Component) {\n  _inherits(AnimationDisplay, _Component);\n\n  function AnimationDisplay() {\n    _classCallCheck(this, AnimationDisplay);\n\n    return _possibleConstructorReturn(this, (AnimationDisplay.__proto__ || Object.getPrototypeOf(AnimationDisplay)).apply(this, arguments));\n  }\n\n  _createClass(AnimationDisplay, [{\n    key: 'createEl',\n\n\n    /**\n     * Create the `AnimationDisplay`s DOM element.\n     *\n     * @return {Element}\n     *         The dom element that gets created.\n     */\n    value: function createEl() {\n      return _get(AnimationDisplay.prototype.__proto__ || Object.getPrototypeOf(AnimationDisplay.prototype), 'createEl', this).call(this, 'div', {\n        className: 'vjs-animation-display',\n        innerHTML: '<img />'\n      });\n    }\n  }]);\n\n  return AnimationDisplay;\n}(Component);\n\nComponent.registerComponent('AnimationDisplay', AnimationDisplay);\n\nexports.default = AnimationDisplay;","'use strict';\n\nObject.defineProperty(exports, \"__esModule\", {\n  value: true\n});\n\nvar _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if (\"value\" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();\n\nvar _get = function get(object, property, receiver) { if (object === null) object = Function.prototype; var desc = Object.getOwnPropertyDescriptor(object, property); if (desc === undefined) { var parent = Object.getPrototypeOf(object); if (parent === null) { return undefined; } else { return get(parent, property, receiver); } } else if (\"value\" in desc) { return desc.value; } else { var getter = desc.get; if (getter === undefined) { return undefined; } return getter.call(receiver); } };\n\nfunction _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError(\"Cannot call a class as a function\"); } }\n\nfunction _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError(\"this hasn't been initialised - super() hasn't been called\"); } return call && (typeof call === \"object\" || typeof call === \"function\") ? call : self; }\n\nfunction _inherits(subClass, superClass) { if (typeof superClass !== \"function\" && superClass !== null) { throw new TypeError(\"Super expression must either be null or a function, not \" + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; }\n\n/**\n * @file camera-button.js\n * @since 2.0.0\n */\n\nvar Button = videojs.getComponent('Button');\nvar Component = videojs.getComponent('Component');\n\n/**\n * Button to toggle between create and retry snapshot image.\n *\n * @class\n * @augments videojs.Button\n*/\n\nvar CameraButton = function (_Button) {\n  _inherits(CameraButton, _Button);\n\n  function CameraButton() {\n    _classCallCheck(this, CameraButton);\n\n    return _possibleConstructorReturn(this, (CameraButton.__proto__ || Object.getPrototypeOf(CameraButton)).apply(this, arguments));\n  }\n\n  _createClass(CameraButton, [{\n    key: 'buildCSSClass',\n\n    /**\n     * Builds the default DOM `className`.\n     *\n     * @return {string}\n     *         The DOM `className` for this object.\n     */\n    value: function buildCSSClass() {\n      return 'vjs-camera-button vjs-control vjs-button vjs-icon-photo-camera';\n    }\n\n    /**\n     * Enable the `CameraButton` element so that it can be activated or clicked.\n     */\n\n  }, {\n    key: 'enable',\n    value: function enable() {\n      _get(CameraButton.prototype.__proto__ || Object.getPrototypeOf(CameraButton.prototype), 'enable', this).call(this);\n\n      this.on(this.player_, 'startRecord', this.onStart);\n      this.on(this.player_, 'stopRecord', this.onStop);\n    }\n\n    /**\n     * Disable the `CameraButton` element so that it cannot be activated or clicked.\n     */\n\n  }, {\n    key: 'disable',\n    value: function disable() {\n      _get(CameraButton.prototype.__proto__ || Object.getPrototypeOf(CameraButton.prototype), 'disable', this).call(this);\n\n      this.off(this.player_, 'startRecord', this.onStart);\n      this.off(this.player_, 'stopRecord', this.onStop);\n    }\n\n    /**\n     * This gets called when the button is clicked.\n     *\n     * @param {EventTarget~Event} event\n     *        The `tap` or `click` event that caused this function to be\n     *        called.\n     *\n     * @listens tap\n     * @listens click\n     */\n\n  }, {\n    key: 'handleClick',\n    value: function handleClick(event) {\n      var recorder = this.player_.record();\n\n      if (!recorder.isProcessing()) {\n        // create snapshot\n        recorder.start();\n      } else {\n        // retry\n        recorder.retrySnapshot();\n\n        // reset camera button\n        this.onStop();\n      }\n    }\n\n    /**\n     * Add the vjs-icon-replay class to the element so it can change appearance.\n     *\n     * @param {EventTarget~Event} [event]\n     *        The event that caused this function to run.\n     *\n     * @listens Player#startRecord\n     */\n\n  }, {\n    key: 'onStart',\n    value: function onStart(event) {\n      // replace element class so it can change appearance\n      this.removeClass('vjs-icon-photo-camera');\n      this.addClass('vjs-icon-replay');\n\n      // change the button text\n      this.controlText('Retry');\n    }\n\n    /**\n     * Add the vjs-icon-photo-camera class to the element so it can change appearance.\n     *\n     * @param {EventTarget~Event} [event]\n     *        The event that caused this function to run.\n     *\n     * @listens Player#stopRecord\n     */\n\n  }, {\n    key: 'onStop',\n    value: function onStop(event) {\n      // replace element class so it can change appearance\n      this.removeClass('vjs-icon-replay');\n      this.addClass('vjs-icon-photo-camera');\n\n      // change the button text\n      this.controlText('Image');\n    }\n  }]);\n\n  return CameraButton;\n}(Button);\n\n/**\n * The text that should display over the `CameraButton`s controls. Added for localization.\n *\n * @type {string}\n * @private\n */\n\n\nCameraButton.prototype.controlText_ = 'Image';\n\nComponent.registerComponent('CameraButton', CameraButton);\n\nexports.default = CameraButton;","'use strict';\n\nObject.defineProperty(exports, \"__esModule\", {\n  value: true\n});\n\nvar _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if (\"value\" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();\n\nfunction _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError(\"Cannot call a class as a function\"); } }\n\nfunction _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError(\"this hasn't been initialised - super() hasn't been called\"); } return call && (typeof call === \"object\" || typeof call === \"function\") ? call : self; }\n\nfunction _inherits(subClass, superClass) { if (typeof superClass !== \"function\" && superClass !== null) { throw new TypeError(\"Super expression must either be null or a function, not \" + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; }\n\n/**\n * @file device-button.js\n * @since 2.0.0\n */\n\nvar Button = videojs.getComponent('Button');\nvar Component = videojs.getComponent('Component');\n\n/**\n * Button to select recording device.\n *\n * @class\n * @augments videojs.Button\n*/\n\nvar DeviceButton = function (_Button) {\n  _inherits(DeviceButton, _Button);\n\n  function DeviceButton() {\n    _classCallCheck(this, DeviceButton);\n\n    return _possibleConstructorReturn(this, (DeviceButton.__proto__ || Object.getPrototypeOf(DeviceButton)).apply(this, arguments));\n  }\n\n  _createClass(DeviceButton, [{\n    key: 'handleClick',\n\n    /**\n     * This gets called when this button gets:\n     *\n     * - Clicked (via the `click` event, listening starts in the constructor)\n     * - Tapped (via the `tap` event, listening starts in the constructor)\n     *\n     * @param {EventTarget~Event} event\n     *        The `keydown`, `tap`, or `click` event that caused this function to be\n     *        called.\n     *\n     * @listens tap\n     * @listens click\n     */\n    value: function handleClick(event) {\n      // open device dialog\n      this.player_.record().getDevice();\n    }\n  }]);\n\n  return DeviceButton;\n}(Button);\n\n/**\n * The text that should display over the `DeviceButton`s controls. Added for localization.\n *\n * @type {string}\n * @private\n */\n\n\nDeviceButton.prototype.controlText_ = 'Device';\n\nComponent.registerComponent('DeviceButton', DeviceButton);\n\nexports.default = DeviceButton;","'use strict';\n\nObject.defineProperty(exports, \"__esModule\", {\n  value: true\n});\n\nvar _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if (\"value\" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();\n\nvar _get = function get(object, property, receiver) { if (object === null) object = Function.prototype; var desc = Object.getOwnPropertyDescriptor(object, property); if (desc === undefined) { var parent = Object.getPrototypeOf(object); if (parent === null) { return undefined; } else { return get(parent, property, receiver); } } else if (\"value\" in desc) { return desc.value; } else { var getter = desc.get; if (getter === undefined) { return undefined; } return getter.call(receiver); } };\n\nfunction _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError(\"Cannot call a class as a function\"); } }\n\nfunction _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError(\"this hasn't been initialised - super() hasn't been called\"); } return call && (typeof call === \"object\" || typeof call === \"function\") ? call : self; }\n\nfunction _inherits(subClass, superClass) { if (typeof superClass !== \"function\" && superClass !== null) { throw new TypeError(\"Super expression must either be null or a function, not \" + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; }\n\n/**\n * @file record-canvas\n * @since 2.0.0\n */\n\nvar Component = videojs.getComponent('Component');\n\n/**\n * Canvas for displaying snapshot image.\n *\n * @class\n * @augments videojs.Component\n*/\n\nvar RecordCanvas = function (_Component) {\n  _inherits(RecordCanvas, _Component);\n\n  function RecordCanvas() {\n    _classCallCheck(this, RecordCanvas);\n\n    return _possibleConstructorReturn(this, (RecordCanvas.__proto__ || Object.getPrototypeOf(RecordCanvas)).apply(this, arguments));\n  }\n\n  _createClass(RecordCanvas, [{\n    key: 'createEl',\n\n\n    /**\n     * Create the `RecordCanvas`s DOM element.\n     *\n     * @return {Element}\n     *         The dom element that gets created.\n     */\n    value: function createEl() {\n      return _get(RecordCanvas.prototype.__proto__ || Object.getPrototypeOf(RecordCanvas.prototype), 'createEl', this).call(this, 'div', {\n        className: 'vjs-record-canvas',\n        innerHTML: '<canvas></canvas>'\n      });\n    }\n  }]);\n\n  return RecordCanvas;\n}(Component);\n\nComponent.registerComponent('RecordCanvas', RecordCanvas);\n\nexports.default = RecordCanvas;","'use strict';\n\nObject.defineProperty(exports, \"__esModule\", {\n  value: true\n});\n\nvar _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if (\"value\" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();\n\nvar _get = function get(object, property, receiver) { if (object === null) object = Function.prototype; var desc = Object.getOwnPropertyDescriptor(object, property); if (desc === undefined) { var parent = Object.getPrototypeOf(object); if (parent === null) { return undefined; } else { return get(parent, property, receiver); } } else if (\"value\" in desc) { return desc.value; } else { var getter = desc.get; if (getter === undefined) { return undefined; } return getter.call(receiver); } };\n\nfunction _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError(\"Cannot call a class as a function\"); } }\n\nfunction _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError(\"this hasn't been initialised - super() hasn't been called\"); } return call && (typeof call === \"object\" || typeof call === \"function\") ? call : self; }\n\nfunction _inherits(subClass, superClass) { if (typeof superClass !== \"function\" && superClass !== null) { throw new TypeError(\"Super expression must either be null or a function, not \" + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; }\n\n/**\n * @file record-indicator.js\n * @since 2.0.0\n */\n\nvar Component = videojs.getComponent('Component');\n\n/**\n * Icon indicating recording is active.\n *\n * @class\n * @augments videojs.Component\n*/\n\nvar RecordIndicator = function (_Component) {\n  _inherits(RecordIndicator, _Component);\n\n  /**\n   * The constructor function for the class.\n   *\n   * @private\n   * @param {(videojs.Player|Object)} player - Video.js player instance.\n   * @param {Object} options - Player options.\n   */\n  function RecordIndicator(player, options) {\n    _classCallCheck(this, RecordIndicator);\n\n    var _this = _possibleConstructorReturn(this, (RecordIndicator.__proto__ || Object.getPrototypeOf(RecordIndicator)).call(this, player, options));\n\n    _this.enable();\n    return _this;\n  }\n\n  /**\n   * Create the `RecordIndicator`s DOM element.\n   *\n   * @return {Element}\n   *         The dom element that gets created.\n   */\n\n\n  _createClass(RecordIndicator, [{\n    key: 'createEl',\n    value: function createEl() {\n      return _get(RecordIndicator.prototype.__proto__ || Object.getPrototypeOf(RecordIndicator.prototype), 'createEl', this).call(this, 'div', {\n        className: 'vjs-record-indicator vjs-control',\n        dir: 'ltr'\n      });\n    }\n\n    /**\n     * Enable event handlers.\n     */\n\n  }, {\n    key: 'enable',\n    value: function enable() {\n      this.on(this.player_, 'startRecord', this.show);\n      this.on(this.player_, 'stopRecord', this.hide);\n    }\n\n    /**\n     * Disable event handlers.\n     */\n\n  }, {\n    key: 'disable',\n    value: function disable() {\n      this.off(this.player_, 'startRecord', this.show);\n      this.off(this.player_, 'stopRecord', this.hide);\n    }\n  }]);\n\n  return RecordIndicator;\n}(Component);\n\nComponent.registerComponent('RecordIndicator', RecordIndicator);\n\nexports.default = RecordIndicator;","'use strict';\n\nObject.defineProperty(exports, \"__esModule\", {\n  value: true\n});\n\nvar _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if (\"value\" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();\n\nvar _get = function get(object, property, receiver) { if (object === null) object = Function.prototype; var desc = Object.getOwnPropertyDescriptor(object, property); if (desc === undefined) { var parent = Object.getPrototypeOf(object); if (parent === null) { return undefined; } else { return get(parent, property, receiver); } } else if (\"value\" in desc) { return desc.value; } else { var getter = desc.get; if (getter === undefined) { return undefined; } return getter.call(receiver); } };\n\nfunction _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError(\"Cannot call a class as a function\"); } }\n\nfunction _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError(\"this hasn't been initialised - super() hasn't been called\"); } return call && (typeof call === \"object\" || typeof call === \"function\") ? call : self; }\n\nfunction _inherits(subClass, superClass) { if (typeof superClass !== \"function\" && superClass !== null) { throw new TypeError(\"Super expression must either be null or a function, not \" + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; }\n\n/**\n * @file record-toggle.js\n * @since 2.0.0\n */\n\nvar Button = videojs.getComponent('Button');\nvar Component = videojs.getComponent('Component');\n\n/**\n * Button to toggle between start and stop recording.\n *\n * @class\n * @augments videojs.Button\n*/\n\nvar RecordToggle = function (_Button) {\n  _inherits(RecordToggle, _Button);\n\n  function RecordToggle() {\n    _classCallCheck(this, RecordToggle);\n\n    return _possibleConstructorReturn(this, (RecordToggle.__proto__ || Object.getPrototypeOf(RecordToggle)).apply(this, arguments));\n  }\n\n  _createClass(RecordToggle, [{\n    key: 'buildCSSClass',\n\n    /**\n     * Builds the default DOM `className`.\n     *\n     * @return {string}\n     *         The DOM `className` for this object.\n     */\n    value: function buildCSSClass() {\n      return 'vjs-record-button vjs-control vjs-button vjs-icon-record-start';\n    }\n\n    /**\n     * Enable the `RecordToggle` element so that it can be activated or clicked.\n     */\n\n  }, {\n    key: 'enable',\n    value: function enable() {\n      _get(RecordToggle.prototype.__proto__ || Object.getPrototypeOf(RecordToggle.prototype), 'enable', this).call(this);\n\n      this.on(this.player_, 'startRecord', this.onStart);\n      this.on(this.player_, 'stopRecord', this.onStop);\n    }\n\n    /**\n     * Disable the `RecordToggle` element so that it cannot be activated or clicked.\n     */\n\n  }, {\n    key: 'disable',\n    value: function disable() {\n      _get(RecordToggle.prototype.__proto__ || Object.getPrototypeOf(RecordToggle.prototype), 'disable', this).call(this);\n\n      this.off(this.player_, 'startRecord', this.onStart);\n      this.off(this.player_, 'stopRecord', this.onStop);\n    }\n\n    /**\n     * This gets called when the button is clicked.\n     *\n     * @param {EventTarget~Event} event\n     *        The `tap` or `click` event that caused this function to be\n     *        called.\n     *\n     * @listens tap\n     * @listens click\n     */\n\n  }, {\n    key: 'handleClick',\n    value: function handleClick(event) {\n      var recorder = this.player_.record();\n      if (!recorder.isRecording()) {\n        recorder.start();\n      } else {\n        recorder.stop();\n      }\n    }\n\n    /**\n     * Add the vjs-icon-record-stop class to the element so it can change appearance.\n     *\n     * @param {EventTarget~Event} [event]\n     *        The event that caused this function to run.\n     *\n     * @listens Player#startRecord\n     */\n\n  }, {\n    key: 'onStart',\n    value: function onStart(event) {\n      // replace element class so it can change appearance\n      this.removeClass('vjs-icon-record-start');\n      this.addClass('vjs-icon-record-stop');\n\n      // change the button text\n      this.controlText('Stop');\n    }\n\n    /**\n     * Add the vjs-icon-record-start class to the element so it can change appearance.\n     *\n     * @param {EventTarget~Event} [event]\n     *        The event that caused this function to run.\n     *\n     * @listens Player#stopRecord\n     */\n\n  }, {\n    key: 'onStop',\n    value: function onStop(event) {\n      // replace element class so it can change appearance\n      this.removeClass('vjs-icon-record-stop');\n      this.addClass('vjs-icon-record-start');\n\n      // change the button text\n      this.controlText('Record');\n    }\n  }]);\n\n  return RecordToggle;\n}(Button);\n\n/**\n * The text that should display over the `RecordToggle`s controls. Added for localization.\n *\n * @type {string}\n * @private\n */\n\n\nRecordToggle.prototype.controlText_ = 'Record';\n\nComponent.registerComponent('RecordToggle', RecordToggle);\n\nexports.default = RecordToggle;","'use strict';\n\nObject.defineProperty(exports, \"__esModule\", {\n    value: true\n});\n/**\n * @file defaults.js\n * @since 2.0.0\n */\n\n//plugin defaults\nvar pluginDefaultOptions = {\n    // Single snapshot image.\n    image: false,\n    // Include audio in the recorded clip.\n    audio: false,\n    // Include video in the recorded clip.\n    video: false,\n    // Animated GIF.\n    animation: false,\n    // Maximum length of the recorded clip.\n    maxLength: 10,\n    // Width of the recorded video frames.\n    frameWidth: 320,\n    // Height of the recorded video frames.\n    frameHeight: 240,\n    // Enables console logging for debugging purposes.\n    debug: false,\n    // The mime type for the video recorder. Default to 'video/webm'.\n    // Use 'video/mp4' (Firefox) or 'video/webm;codecs=H264' (Chrome 52 and\n    // newer) for MP4.\n    videoMimeType: 'video/webm',\n    // Video recorder type to use. This allows you to specify an alternative\n    // recorder class, e.g. WhammyRecorder. Defaults to 'auto' which let's\n    // recordrtc specify the best available recorder type.\n    videoRecorderType: 'auto',\n    // Audio recording library to use. Legal values are 'recordrtc',\n    // 'libvorbis.js', 'opus-recorder', 'lamejs' and 'recorder.js'.\n    audioEngine: 'recordrtc',\n    // Audio recorder type to use. This allows you to specify an alternative\n    // recorder class, e.g. StereoAudioRecorder. Defaults to 'auto' which let's\n    // recordrtc specify the best available recorder type. Currently this\n    // setting is only used with the 'recordrtc' audioEngine.\n    audioRecorderType: 'auto',\n    // The mime type for the audio recorder. Defaults to 'auto' which will pick\n    // the best option available in the browser (e.g. either 'audio/wav',\n    // 'audio/ogg' or 'audio/webm').\n    audioMimeType: 'auto',\n    // The size of the audio buffer (in sample-frames) which needs to\n    // be processed each time onprocessaudio is called.\n    // From the spec: This value controls how frequently the audioprocess event is\n    // dispatched and how many sample-frames need to be processed each call.\n    // Lower values for buffer size will result in a lower (better) latency.\n    // Higher values will be necessary to avoid audio breakup and glitches.\n    // Legal values are 256, 512, 1024, 2048, 4096, 8192 or 16384.\n    audioBufferSize: 4096,\n    // The audio sample rate (in sample-frames per second) at which the\n    // AudioContext handles audio. It is assumed that all AudioNodes\n    // in the context run at this rate. In making this assumption,\n    // sample-rate converters or \"varispeed\" processors are not supported\n    // in real-time processing.\n    // The sampleRate parameter describes the sample-rate of the\n    // linear PCM audio data in the buffer in sample-frames per second.\n    // An implementation must support sample-rates in at least\n    // the range 22050 to 96000.\n    audioSampleRate: 44100,\n    // Allows you to record single-channel audio, which can reduce the\n    // filesize.\n    audioChannels: 2,\n    // URL for the audio worker.\n    audioWorkerURL: '',\n    // Frame rate in frames per second.\n    animationFrameRate: 200,\n    // Sets quality of color quantization (conversion of images to the\n    // maximum 256 colors allowed by the GIF specification).\n    // Lower values (minimum = 1) produce better colors,\n    // but slow processing significantly. 10 is the default,\n    // and produces good color mapping at reasonable speeds.\n    // Values greater than 20 do not yield significant improvements\n    // in speed.\n    animationQuality: 10,\n    // Accepts numbers in milliseconds; use this to force intervals-based blobs.\n    timeSlice: 0\n};\n\nexports.default = pluginDefaultOptions;","'use strict';\n\nObject.defineProperty(exports, \"__esModule\", {\n    value: true\n});\n\nvar _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if (\"value\" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();\n\nfunction _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError(\"Cannot call a class as a function\"); } }\n\nfunction _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError(\"this hasn't been initialised - super() hasn't been called\"); } return call && (typeof call === \"object\" || typeof call === \"function\") ? call : self; }\n\nfunction _inherits(subClass, superClass) { if (typeof superClass !== \"function\" && superClass !== null) { throw new TypeError(\"Super expression must either be null or a function, not \" + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; }\n\n/**\n * @file record-engine.js\n * @since 2.0.0\n */\n\nvar Component = videojs.getComponent('Component');\n\n// supported recorder plugin engines\nvar RECORDRTC = 'recordrtc';\nvar LIBVORBISJS = 'libvorbis.js';\nvar RECORDERJS = 'recorder.js';\nvar LAMEJS = 'lamejs';\nvar OPUSRECORDER = 'opus-recorder';\n\n/**\n * Base class for recorder backends.\n * @class\n * @augments videojs.Component\n */\n\nvar RecordEngine = function (_Component) {\n    _inherits(RecordEngine, _Component);\n\n    /**\n     * Creates an instance of this class.\n     *\n     * @param  {Player} player\n     *         The `Player` that this class should be attached to.\n     *\n     * @param  {Object} [options]\n     *         The key/value store of player options.\n     */\n    function RecordEngine(player, options) {\n        _classCallCheck(this, RecordEngine);\n\n        // auto mixin the evented mixin (required since video.js v6.6.0)\n        options.evented = true;\n\n        return _possibleConstructorReturn(this, (RecordEngine.__proto__ || Object.getPrototypeOf(RecordEngine)).call(this, player, options));\n    }\n\n    /**\n     * Remove any temporary data and references to streams.\n     * @private\n     */\n\n\n    _createClass(RecordEngine, [{\n        key: 'dispose',\n        value: function dispose() {\n            // dispose previous recording\n            if (this.recordedData !== undefined) {\n                URL.revokeObjectURL(this.recordedData);\n            }\n        }\n\n        /**\n         * Add filename and timestamp to recorded file object.\n         *\n         * @param {(blob|file)} fileObj - Blob or File object.\n         */\n\n    }, {\n        key: 'addFileInfo',\n        value: function addFileInfo(fileObj) {\n            var now = new Date();\n            fileObj.lastModifiedDate = now;\n\n            // guess extension name from mime type, e.g. audio/ogg, but\n            // any extension is valid here. Chrome also accepts extended\n            // mime types like video/webm;codecs=h264,vp9,opus\n            var fileExtension = '.' + fileObj.type.split('/')[1];\n            if (fileExtension.indexOf(';') > -1) {\n                fileExtension = fileExtension.split(';')[0];\n            }\n\n            // use timestamp in filename, e.g. 1451180941326.ogg\n            fileObj.name = now.getTime() + fileExtension;\n        }\n\n        /**\n         * Invoked when recording is stopped and resulting stream is available.\n         *\n         * @param {blob} data - Reference to the recorded Blob.\n         */\n\n    }, {\n        key: 'onStopRecording',\n        value: function onStopRecording(data) {\n            this.recordedData = data;\n\n            // add filename and timestamp to recorded file object\n            this.addFileInfo(this.recordedData);\n\n            // remove reference to recorded stream\n            this.dispose();\n\n            // notify listeners\n            this.trigger('recordComplete');\n        }\n\n        /**\n         * Show save as dialog in browser so the user can store the recorded media\n         * locally.\n         *\n         * @param {object} name - Object with names for the particular blob(s)\n         *     you want to save. File extensions are added automatically. For\n         *     example: {'video': 'name-of-video-file'}. Supported keys are\n         *     'audio', 'video' and 'gif'.\n         */\n\n    }, {\n        key: 'saveAs',\n        value: function saveAs(name) {\n            var fileName = name[Object.keys(name)[0]];\n\n            if (typeof navigator.msSaveOrOpenBlob !== 'undefined') {\n                return navigator.msSaveOrOpenBlob(this.recordedData, fileName);\n            } else if (typeof navigator.msSaveBlob !== 'undefined') {\n                return navigator.msSaveBlob(this.recordedData, fileName);\n            }\n\n            var hyperlink = document.createElement('a');\n            hyperlink.href = URL.createObjectURL(this.recordedData);\n            hyperlink.download = fileName;\n\n            hyperlink.style = 'display:none;opacity:0;color:transparent;';\n            (document.body || document.documentElement).appendChild(hyperlink);\n\n            if (typeof hyperlink.click === 'function') {\n                hyperlink.click();\n            } else {\n                hyperlink.target = '_blank';\n                hyperlink.dispatchEvent(new MouseEvent('click', {\n                    view: window,\n                    bubbles: true,\n                    cancelable: true\n                }));\n            }\n\n            URL.revokeObjectURL(hyperlink.href);\n        }\n    }]);\n\n    return RecordEngine;\n}(Component);\n\n// expose component for external plugins\n\n\nvideojs.RecordEngine = RecordEngine;\nComponent.registerComponent('RecordEngine', RecordEngine);\n\nexports.RecordEngine = RecordEngine;\nexports.RECORDRTC = RECORDRTC;\nexports.LIBVORBISJS = LIBVORBISJS;\nexports.RECORDERJS = RECORDERJS;\nexports.LAMEJS = LAMEJS;\nexports.OPUSRECORDER = OPUSRECORDER;","'use strict';\n\nObject.defineProperty(exports, \"__esModule\", {\n    value: true\n});\n/**\n * @file record-mode.js\n * @since 2.0.0\n */\n\n// recorder modes\nvar IMAGE_ONLY = 'image_only';\nvar AUDIO_ONLY = 'audio_only';\nvar VIDEO_ONLY = 'video_only';\nvar AUDIO_VIDEO = 'audio_video';\nvar ANIMATION = 'animation';\n\nvar getRecorderMode = function getRecorderMode(image, audio, video, animation) {\n    if (isModeEnabled(image)) {\n        return IMAGE_ONLY;\n    } else if (isModeEnabled(animation)) {\n        return ANIMATION;\n    } else if (isModeEnabled(audio) && !isModeEnabled(video)) {\n        return AUDIO_ONLY;\n    } else if (isModeEnabled(audio) && isModeEnabled(video)) {\n        return AUDIO_VIDEO;\n    } else if (!isModeEnabled(audio) && isModeEnabled(video)) {\n        return VIDEO_ONLY;\n    }\n};\n\n/**\n * Return boolean indicating whether mode is enabled or not.\n */\nvar isModeEnabled = function isModeEnabled(mode) {\n    return mode === Object(mode) || mode === true;\n};\n\nexports.getRecorderMode = getRecorderMode;\nexports.IMAGE_ONLY = IMAGE_ONLY;\nexports.AUDIO_ONLY = AUDIO_ONLY;\nexports.VIDEO_ONLY = VIDEO_ONLY;\nexports.AUDIO_VIDEO = AUDIO_VIDEO;\nexports.ANIMATION = ANIMATION;","'use strict';\n\nObject.defineProperty(exports, \"__esModule\", {\n    value: true\n});\n\nvar _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if (\"value\" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();\n\nvar _get = function get(object, property, receiver) { if (object === null) object = Function.prototype; var desc = Object.getOwnPropertyDescriptor(object, property); if (desc === undefined) { var parent = Object.getPrototypeOf(object); if (parent === null) { return undefined; } else { return get(parent, property, receiver); } } else if (\"value\" in desc) { return desc.value; } else { var getter = desc.get; if (getter === undefined) { return undefined; } return getter.call(receiver); } };\n\nvar _recordEngine = require('./record-engine');\n\nvar _detectBrowser = require('../utils/detect-browser');\n\nvar _recordMode = require('./record-mode');\n\nfunction _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError(\"Cannot call a class as a function\"); } }\n\nfunction _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError(\"this hasn't been initialised - super() hasn't been called\"); } return call && (typeof call === \"object\" || typeof call === \"function\") ? call : self; }\n\nfunction _inherits(subClass, superClass) { if (typeof superClass !== \"function\" && superClass !== null) { throw new TypeError(\"Super expression must either be null or a function, not \" + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; } /**\n                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                * @file record-rtc.js\n                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                * @since 2.0.0\n                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                */\n\nvar Component = videojs.getComponent('Component');\n\n/**\n * Engine used with the MRecordRTC class in the RecordRTC library.\n *\n * @class\n * @augments videojs.RecordEngine\n */\n\nvar RecordRTCEngine = function (_RecordEngine) {\n    _inherits(RecordRTCEngine, _RecordEngine);\n\n    function RecordRTCEngine() {\n        _classCallCheck(this, RecordRTCEngine);\n\n        return _possibleConstructorReturn(this, (RecordRTCEngine.__proto__ || Object.getPrototypeOf(RecordRTCEngine)).apply(this, arguments));\n    }\n\n    _createClass(RecordRTCEngine, [{\n        key: 'setup',\n\n\n        /**\n         * Setup recording engine.\n         */\n        value: function setup(stream, mediaType, debug) {\n            this.inputStream = stream;\n            this.mediaType = mediaType;\n            this.debug = debug;\n\n            // setup RecordRTC\n            this.engine = new RecordRTC.MRecordRTC();\n            this.engine.mediaType = this.mediaType;\n            this.engine.disableLogs = !this.debug;\n            this.engine.mimeType = this.mimeType;\n\n            // audio settings\n            this.engine.bufferSize = this.bufferSize;\n            this.engine.sampleRate = this.sampleRate;\n            this.engine.numberOfAudioChannels = this.audioChannels;\n\n            // video/canvas settings\n            this.engine.video = this.video;\n            this.engine.canvas = this.canvas;\n\n            // animated gif settings\n            this.engine.quality = this.quality;\n            this.engine.frameRate = this.frameRate;\n            if (this.onTimeStamp !== undefined) {\n                this.engine.timeSlice = this.timeSlice;\n                this.engine.onTimeStamp = this.onTimeStamp;\n            }\n\n            // connect stream to recording engine\n            this.engine.addStream(this.inputStream);\n        }\n\n        /**\n         * Remove any temporary data and references to streams.\n         */\n\n    }, {\n        key: 'dispose',\n        value: function dispose() {\n            _get(RecordRTCEngine.prototype.__proto__ || Object.getPrototypeOf(RecordRTCEngine.prototype), 'dispose', this).call(this);\n\n            if (typeof this.engine.destroy === 'function') {\n                this.engine.destroy();\n            }\n        }\n\n        /**\n         * Start recording.\n         */\n\n    }, {\n        key: 'start',\n        value: function start() {\n            this.engine.startRecording();\n        }\n\n        /**\n         * Stop recording. Result will be available async when onStopRecording\n         * is called.\n         */\n\n    }, {\n        key: 'stop',\n        value: function stop() {\n            this.engine.stopRecording(this.onStopRecording.bind(this));\n        }\n\n        /**\n         * Pause recording.\n         */\n\n    }, {\n        key: 'pause',\n        value: function pause() {\n            this.engine.pauseRecording();\n        }\n\n        /**\n         * Resume recording.\n         */\n\n    }, {\n        key: 'resume',\n        value: function resume() {\n            this.engine.resumeRecording();\n        }\n\n        /**\n         * Show save as dialog in browser so the user can store the recorded media\n         * locally.\n         *\n         * @param {object} name - Object with names for the particular blob(s)\n         *     you want to save. File extensions are added automatically. For\n         *     example: {'video': 'name-of-video-file'}. Supported keys are\n         *     'audio', 'video' and 'gif'.\n         */\n\n    }, {\n        key: 'saveAs',\n        value: function saveAs(name) {\n            if (this.engine && name !== undefined) {\n                this.engine.save(name);\n            }\n        }\n\n        /**\n         * Invoked when recording is stopped and resulting stream is available.\n         *\n         * @private\n         * @param {string} audioVideoURL - Reference to the recorded Blob\n         *     object, e.g. 'blob:http://localhost:8080/10100016-4248-9949-b0d6-0bb40db56eba'\n         * @param {string} type - Media type, eg. 'video' or 'audio'.\n         */\n\n    }, {\n        key: 'onStopRecording',\n        value: function onStopRecording(audioVideoURL, type) {\n            var _this2 = this;\n\n            // store reference to recorded stream URL\n            this.mediaURL = audioVideoURL;\n\n            // store reference to recorded stream data\n            var recordType = this.player().record().getRecordType();\n            this.engine.getBlob(function (recording) {\n                switch (recordType) {\n                    case _recordMode.AUDIO_ONLY:\n                        _this2.recordedData = recording.audio;\n\n                        _this2.addFileInfo(_this2.recordedData);\n\n                        // notify listeners\n                        _this2.trigger('recordComplete');\n                        break;\n\n                    case _recordMode.VIDEO_ONLY:\n                    case _recordMode.AUDIO_VIDEO:\n                        // when recording both audio and video, recordrtc\n                        // calls this twice on chrome, first with audio data\n                        // and then with video data.\n                        // on firefox it's called once but with a single\n                        // blob that includes both audio and video data.\n                        if (recording.video !== undefined) {\n                            // data is video-only but on firefox audio+video\n                            _this2.recordedData = recording.video;\n\n                            // on the chrome browser two blobs are created\n                            // containing the separate audio/video streams.\n                            if (recordType === _recordMode.AUDIO_VIDEO && (0, _detectBrowser.isChrome)()) {\n                                // store both audio and video\n                                _this2.recordedData = recording;\n\n                                for (var mtype in _this2.recordedData) {\n                                    _this2.addFileInfo(_this2.recordedData[mtype]);\n                                }\n                            } else {\n                                _this2.addFileInfo(_this2.recordedData);\n                            }\n\n                            // notify listeners\n                            _this2.trigger('recordComplete');\n                        }\n                        break;\n\n                    case _recordMode.ANIMATION:\n                        _this2.recordedData = recording.gif;\n\n                        _this2.addFileInfo(_this2.recordedData);\n\n                        // notify listeners\n                        _this2.trigger('recordComplete');\n                        break;\n                }\n            });\n        }\n    }]);\n\n    return RecordRTCEngine;\n}(_recordEngine.RecordEngine);\n\n// expose plugin\n\n\nvideojs.RecordRTCEngine = RecordRTCEngine;\n\nComponent.registerComponent('RecordRTCEngine', RecordRTCEngine);\n\nexports.default = RecordRTCEngine;","'use strict';\n\nObject.defineProperty(exports, \"__esModule\", {\n    value: true\n});\n/**\n * @file browser-shim.js\n * @since 2.0.0\n */\n\nvar setSrcObject = function setSrcObject(stream, element, ignoreCreateObjectURL) {\n    if ('createObjectURL' in URL && !ignoreCreateObjectURL) {\n        try {\n            element.src = URL.createObjectURL(stream);\n        } catch (e) {\n            setSrcObject(stream, element, true);\n            return;\n        }\n    } else if ('srcObject' in element) {\n        element.srcObject = stream;\n    } else if ('mozSrcObject' in element) {\n        element.mozSrcObject = stream;\n    } else {\n        console.log('createObjectURL/srcObject both are not supported.');\n    }\n};\n\nexports.default = setSrcObject;","'use strict';\n\nObject.defineProperty(exports, \"__esModule\", {\n    value: true\n});\nexports.isSafari = exports.isChrome = exports.isOpera = exports.isEdge = exports.detectBrowser = undefined;\n\nvar _window = require('global/window');\n\nvar _window2 = _interopRequireDefault(_window);\n\nfunction _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }\n\n/**\n * Browser detector.\n *\n * @private\n * @return {object} result containing browser, version and minVersion\n *     properties.\n */\nvar detectBrowser = function detectBrowser() {\n    // returned result object\n    var result = {};\n    result.browser = null;\n    result.version = null;\n    result.minVersion = null;\n\n    // fail early if it's not a browser\n    if (typeof _window2.default === 'undefined' || !_window2.default.navigator) {\n        result.browser = 'Not a supported browser.';\n        return result;\n    }\n\n    // Firefox\n    if (navigator.mozGetUserMedia) {\n        result.browser = 'firefox';\n        result.version = extractVersion(navigator.userAgent, /Firefox\\/(\\d+)\\./, 1);\n        result.minVersion = 31;\n    } else if (navigator.webkitGetUserMedia) {\n        // Chrome, Chromium, Webview, Opera\n        if (_window2.default.webkitRTCPeerConnection) {\n            result.browser = 'chrome';\n            result.version = extractVersion(navigator.userAgent, /Chrom(e|ium)\\/(\\d+)\\./, 2);\n            result.minVersion = 38;\n        } else {\n            // Safari (in an unpublished version) or unknown webkit-based.\n            if (navigator.userAgent.match(/Version\\/(\\d+).(\\d+)/)) {\n                result.browser = 'safari';\n                result.version = extractVersion(navigator.userAgent, /AppleWebKit\\/(\\d+)\\./, 1);\n                result.minVersion = 11;\n            } else {\n                // unknown webkit-based browser.\n                result.browser = 'Unsupported webkit-based browser ' + 'with GUM support but no WebRTC support.';\n                return result;\n            }\n        }\n        // Edge\n    } else if (navigator.mediaDevices && navigator.userAgent.match(/Edge\\/(\\d+).(\\d+)$/)) {\n        result.browser = 'edge';\n        result.version = extractVersion(navigator.userAgent, /Edge\\/(\\d+).(\\d+)$/, 2);\n        result.minVersion = 10547;\n    } else if (navigator.mediaDevices && navigator.userAgent.match(/AppleWebKit\\/(\\d+)\\./)) {\n        // Safari, with webkitGetUserMedia removed.\n        result.browser = 'safari';\n        result.version = extractVersion(navigator.userAgent, /AppleWebKit\\/(\\d+)\\./, 1);\n    } else {\n        // default fallthrough: not supported.\n        result.browser = 'Not a supported browser.';\n        return result;\n    }\n\n    return result;\n};\n\n/**\n * Extract browser version out of the provided user agent string.\n *\n * @private\n * @param {!string} uastring - userAgent string.\n * @param {!string} expr - Regular expression used as match criteria.\n * @param {!number} pos - position in the version string to be\n *     returned.\n * @return {!number} browser version.\n */\n/**\n * @file detect-browser.js\n * @since 2.0.0\n */\n\nvar extractVersion = function extractVersion(uastring, expr, pos) {\n    var match = uastring.match(expr);\n    return match && match.length >= pos && parseInt(match[pos], 10);\n};\n\nvar isEdge = function isEdge() {\n    return detectBrowser().browser === 'edge';\n};\n\nvar isSafari = function isSafari() {\n    return detectBrowser().browser === 'safari';\n};\n\nvar isOpera = function isOpera() {\n    return !!_window2.default.opera || navigator.userAgent.indexOf('OPR/') !== -1;\n};\n\nvar isChrome = function isChrome() {\n    return detectBrowser().browser === 'chrome';\n};\n\nexports.detectBrowser = detectBrowser;\nexports.isEdge = isEdge;\nexports.isOpera = isOpera;\nexports.isChrome = isChrome;\nexports.isSafari = isSafari;","'use strict';\n\nObject.defineProperty(exports, \"__esModule\", {\n    value: true\n});\n/**\n * @file format-time.js\n * @since 2.0.0\n */\n\n/**\n * Format seconds as a time string, H:MM:SS, M:SS or M:SS:MMM.\n *\n * Supplying a guide (in seconds) will force a number of leading zeros\n * to cover the length of the guide.\n *\n * @param {number} seconds - Number of seconds to be turned into a\n *     string.\n * @param {number} guide - Number (in seconds) to model the string\n *     after.\n * @param {number} msDisplayMax - Number (in milliseconds) to model the string\n *     after.\n * @return {string} Time formatted as H:MM:SS, M:SS or M:SS:MMM, e.g.\n *     0:00:12.\n * @private\n */\nvar formatTime = function formatTime(seconds, guide, msDisplayMax) {\n    // Default to using seconds as guide\n    seconds = seconds < 0 ? 0 : seconds;\n    guide = guide || seconds;\n    var s = Math.floor(seconds % 60),\n        m = Math.floor(seconds / 60 % 60),\n        h = Math.floor(seconds / 3600),\n        gm = Math.floor(guide / 60 % 60),\n        gh = Math.floor(guide / 3600),\n        ms = Math.floor((seconds - s) * 1000);\n\n    // handle invalid times\n    if (isNaN(seconds) || seconds === Infinity) {\n        // '-' is false for all relational operators (e.g. <, >=) so this\n        // setting will add the minimum number of fields specified by the\n        // guide\n        h = m = s = ms = '-';\n    }\n\n    // Check if we need to show milliseconds\n    if (guide > 0 && guide < msDisplayMax) {\n        if (ms < 100) {\n            if (ms < 10) {\n                ms = '00' + ms;\n            } else {\n                ms = '0' + ms;\n            }\n        }\n        ms = ':' + ms;\n    } else {\n        ms = '';\n    }\n\n    // Check if we need to show hours\n    h = h > 0 || gh > 0 ? h + ':' : '';\n\n    // If hours are showing, we may need to add a leading zero.\n    // Always show at least one digit of minutes.\n    m = ((h || gm >= 10) && m < 10 ? '0' + m : m) + ':';\n\n    // Check if leading zero is need for seconds\n    s = s < 10 ? '0' + s : s;\n\n    return h + m + s + ms;\n};\n\nexports.default = formatTime;","var win;\n\nif (typeof window !== \"undefined\") {\n    win = window;\n} else if (typeof global !== \"undefined\") {\n    win = global;\n} else if (typeof self !== \"undefined\"){\n    win = self;\n} else {\n    win = {};\n}\n\nmodule.exports = win;\n"]}