Source: VideoInputPreview.js

/**
 * Video input preview handler
 */
class VideoInputPreview{
	constructor(target, parameters){
		/**
		 * Base input
		 * @type {Element}
		 */
		this.input = target instanceof Element ? target : document.querySelector(target);

		/**
		 * Has the input changed ?
		 * @type {Boolean}
		 */
		this.changed = false;

		/**
		 * Parameters holder
		 * @private
		 */
		this._parameters = {
			preview: null,
			uploadIcon: null,
			lang: 'en',
			dictionary: null,
			...(parameters || {})
		};

        //Errors checking
        if(!this.input){
			throw console.warn('VideoInputPreview: '+(typeof target == 'string' ? 'The selector `'+target+'` didn\'t match any element.' : 'The element you provided was undefined'));
		}else if(this.input.classList.contains('vip-initialized')){
			throw console.warn('VideoInputPreview: The element has already been initialized.');
		}else{
			/**
			 * Upload icon
			 * @private
			 */
			this._uploadIcon = this._parameters.uploadIcon || '';

			this._loadDictionary();
			this._build();
			this._listen();
		}
	}

	/**
     * Loads the dictionary
     * @private
     */
    _loadDictionary(){
		/** @private */
		this._dictionary = {
			...this._parameters.dictionary,
			en: {
				upload: 'Upload video',
				...(this._parameters.dictionary || {}).en
			},
			fr: {
				upload: 'Télécharger une vidéo',
				...(this._parameters.dictionary || {}).fr
			}
		};
	}

	/**
	 * Builds the VideoInputPreview module in the DOM
	 * @private
	 */
	_build(){
		this.input.classList.add('vip-input', 'vip-initialized');

		/**
		 * Input preview
		 * @type {Element}
		 */
		this.preview = document.createElement('video');

		this.preview.classList.add('vip-preview');
		if(!this._parameters.preview) this.preview.poster = this._uploadIcon;
		this.preview.controls = 'true';
		this.preview.width = 200;

		this.preview.insertAdjacentHTML('beforeend', /*html*/`
			<source ${this._parameters.preview ? `src="${this._parameters.preview}"` : ''}>
		`);

		this.input.insertAdjacentElement('beforebegin', this.preview);
	}

	/**
	 * Listens to necessary events
	 * @private
	 */
	_listen(){
		this.input.addEventListener('change', () => {
			if(this.input.files && this.input.files[0]){
				if(this.input.files[0].type.startsWith('video/')){
					this.preview.firstElementChild.src = URL.createObjectURL(this.input.files[0]);
					this.preview.load();
					this.preview.poster = '';
				}

				this.changed = true;
			}
		});
	}

	/**
	 * Changes the preview image
	 * @param {String} src The preview URL
	 * @returns {ImageInputPreview} The current VideoInputPreview
	 */
	setPreview(src){
		this.preview.poster = src;

		return this;
	}

	/**
	 * Resets the input and the preview
	 * @returns {VideoInputPreview} The current VideoInputPreview
	 */
	reset(){
		this.input.value = '';
		this.preview.firstChild.src = '';

		this.changed = true;

		return this;
	}
}

// Initialize inputs the data- way
window.addEventListener('load', () => {
	let inputs = document.querySelectorAll('input[type="file"][data-video-preview]');

	inputs.forEach(input => {
		new VideoInputPreview(input, {
			preview: input.getAttribute('data-video-preview')
		});
	});
});