2020-01-21 16:14:45 +01:00
|
|
|
/**
|
|
|
|
* EGroupware eTemplate2 - JS Description object
|
|
|
|
*
|
|
|
|
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
|
|
|
|
* @package etemplate
|
|
|
|
* @subpackage api
|
|
|
|
* @link http://www.egroupware.org
|
|
|
|
* @author Hadi Nategh <hn[at]stylite.de>
|
|
|
|
* @copyright Stylite AG
|
|
|
|
* @version $Id$
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*egw:uses
|
|
|
|
/vendor/bower-asset/jquery/dist/jquery.js;
|
|
|
|
et2_core_interfaces;
|
|
|
|
et2_core_baseWidget;
|
|
|
|
*/
|
|
|
|
|
|
|
|
import { et2_baseWidget } from './et2_core_baseWidget'
|
|
|
|
import {ClassWithAttributes} from "./et2_core_inheritance";
|
2020-05-07 16:14:07 +02:00
|
|
|
import {WidgetConfig, et2_register_widget} from "./et2_core_widget";
|
2020-01-21 16:14:45 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* This widget represents the HTML5 video tag with all its optional attributes
|
|
|
|
*
|
|
|
|
* The widget can be created in the following ways:
|
|
|
|
* <code>
|
|
|
|
* var videoTag = et2_createWidget("video", {
|
|
|
|
* video_src: "../../test.mp4",
|
|
|
|
* src_type: "video/mp4",
|
|
|
|
* muted: true,
|
|
|
|
* autoplay: true,
|
|
|
|
* controls: true,
|
|
|
|
* poster: "../../poster.jpg",
|
|
|
|
* loop: true,
|
|
|
|
* height: 100,
|
|
|
|
* width: 200,
|
|
|
|
* });
|
|
|
|
* </code>
|
|
|
|
* Or by adding XET-tag in your template (.xet) file:
|
|
|
|
* <code>
|
|
|
|
* <video [attributes...]/>
|
|
|
|
* </code>
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Class which implements the "video" XET-Tag
|
|
|
|
*
|
|
|
|
* @augments et2_baseWidget
|
|
|
|
*/
|
2020-05-06 14:53:02 +02:00
|
|
|
export class et2_video extends et2_baseWidget implements et2_IDOMNode
|
2020-01-21 16:14:45 +01:00
|
|
|
{
|
|
|
|
static readonly _attributes: any = {
|
|
|
|
"video_src": {
|
|
|
|
"name": "Video",
|
|
|
|
"type": "string",
|
|
|
|
"description": "Source of video to display"
|
|
|
|
},
|
|
|
|
"src_type": {
|
|
|
|
"name": "Source type",
|
|
|
|
"type": "string",
|
|
|
|
"description": "Defines the type the stream source provided"
|
|
|
|
},
|
|
|
|
"muted": {
|
|
|
|
"name": "Audio control",
|
|
|
|
"type": "boolean",
|
|
|
|
"default": false,
|
|
|
|
"description": "Defines that the audio output of the video should be muted"
|
|
|
|
},
|
|
|
|
"autoplay": {
|
2021-02-02 11:02:26 +01:00
|
|
|
"name": "Autoplay",
|
2020-01-21 16:14:45 +01:00
|
|
|
"type": "boolean",
|
|
|
|
"default": false,
|
|
|
|
"description": "Defines if Video will start playing as soon as it is ready"
|
|
|
|
},
|
2021-02-02 11:02:26 +01:00
|
|
|
starttime: {
|
|
|
|
"name": "Inital position of video",
|
|
|
|
"type": "float",
|
|
|
|
"default": 0,
|
|
|
|
"description": "Set initial position of video"
|
|
|
|
},
|
2020-01-21 16:14:45 +01:00
|
|
|
"controls": {
|
|
|
|
"name": "Control buttons",
|
|
|
|
"type": "boolean",
|
|
|
|
"default": false,
|
|
|
|
"description": "Defines if Video controls, play/pause buttons should be displayed"
|
|
|
|
},
|
|
|
|
"poster": {
|
|
|
|
"name": "Video Poster",
|
|
|
|
"type": "string",
|
|
|
|
"default": "",
|
|
|
|
"description": "Specifies an image to be shown while video is loading or before user play it"
|
|
|
|
},
|
|
|
|
"loop": {
|
|
|
|
"name": "Video loop",
|
|
|
|
"type": "boolean",
|
|
|
|
"default": false,
|
|
|
|
"description": "Defines if the video should be played repeatedly"
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2021-02-15 16:58:04 +01:00
|
|
|
video: JQuery<HTMLVideoElement|HTMLIFrameElement|HTMLElement> = null;
|
|
|
|
|
|
|
|
youtube: any;
|
|
|
|
private static youtube_player_states = {
|
|
|
|
unstarted: 0,
|
|
|
|
ended: 1,
|
|
|
|
playing: 2,
|
|
|
|
paused: 3,
|
|
|
|
buffering: 4,
|
|
|
|
video_cued: 5
|
|
|
|
};
|
2020-01-21 16:14:45 +01:00
|
|
|
|
2021-02-15 12:24:33 +01:00
|
|
|
/**
|
|
|
|
* keeps internal state of previousTime video played
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
private _previousTime: number = 0;
|
|
|
|
|
2020-01-21 16:14:45 +01:00
|
|
|
constructor(_parent, _attrs? : WidgetConfig, _child? : object)
|
|
|
|
{
|
2020-01-22 18:24:34 +01:00
|
|
|
super(_parent, _attrs, ClassWithAttributes.extendAttributes(et2_video._attributes, _child || {}));
|
2020-01-21 16:14:45 +01:00
|
|
|
|
|
|
|
//Create Video tag
|
2021-02-15 16:58:04 +01:00
|
|
|
this.video = jQuery(document.createElement(this._isYoutube()?"div":"video"));
|
|
|
|
if (this._isYoutube())
|
|
|
|
{
|
|
|
|
this.video.attr('id', this.id);
|
|
|
|
// 2. This code loads the IFrame Player API code asynchronously.
|
|
|
|
let tag = document.createElement('script');
|
|
|
|
tag.src = "https://www.youtube.com/iframe_api";
|
|
|
|
let firstScriptTag = document.getElementsByTagName('script')[0];
|
|
|
|
firstScriptTag.parentNode.insertBefore(tag, firstScriptTag);
|
|
|
|
}
|
|
|
|
if (!this._isYoutube() && this.options.controls)
|
2020-01-21 16:14:45 +01:00
|
|
|
{
|
|
|
|
this.video.attr("controls", 1);
|
|
|
|
}
|
2021-02-15 16:58:04 +01:00
|
|
|
if (!this._isYoutube() && this.options.autoplay)
|
2020-01-21 16:14:45 +01:00
|
|
|
{
|
|
|
|
this.video.attr("autoplay", 1);
|
|
|
|
}
|
|
|
|
if (this.options.muted)
|
|
|
|
{
|
|
|
|
this.video.attr("muted", 1);
|
|
|
|
}
|
|
|
|
if (this.options.video_src)
|
|
|
|
{
|
|
|
|
this.set_src(this.options.video_src);
|
|
|
|
}
|
|
|
|
if (this.options.loop)
|
|
|
|
{
|
|
|
|
this.video.attr("loop", 1);
|
|
|
|
}
|
|
|
|
this.setDOMNode(this.video[0]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set video src
|
|
|
|
*
|
|
|
|
* @param {string} _value url
|
|
|
|
*/
|
|
|
|
set_src(_value: string) {
|
2021-02-15 16:58:04 +01:00
|
|
|
var self = this;
|
|
|
|
if (_value && !this._isYoutube())
|
2020-01-21 16:14:45 +01:00
|
|
|
{
|
|
|
|
let source = jQuery(document.createElement('source'))
|
|
|
|
.attr('src',_value)
|
|
|
|
.appendTo(this.video);
|
|
|
|
|
|
|
|
if (this.options.src_type)
|
|
|
|
{
|
|
|
|
source.attr('type', this.options.src_type);
|
|
|
|
}
|
|
|
|
}
|
2021-02-15 16:58:04 +01:00
|
|
|
else if(_value)
|
|
|
|
{
|
|
|
|
//initiate youtube Api object, it gets called automatically by iframe_api script from the api
|
|
|
|
window.onYouTubeIframeAPIReady = function() {
|
|
|
|
self.youtube = new YT.Player( self.id, {
|
|
|
|
height: '100%',
|
|
|
|
width: 'auto',
|
|
|
|
videoId: '', //TODO get youtube video id
|
|
|
|
events: {
|
|
|
|
'onReady': jQuery.proxy(self._onReady, self)
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
2020-01-21 16:14:45 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set autoplay option for video
|
|
|
|
* -If autoplay is set, video would be played automatically after the page is loaded
|
|
|
|
*
|
|
|
|
* @param {string} _value true set the autoplay and false not to set
|
|
|
|
*/
|
|
|
|
set_autoplay(_value: string)
|
|
|
|
{
|
2021-02-15 16:58:04 +01:00
|
|
|
if (_value && !this._isYoutube())
|
2020-01-21 16:14:45 +01:00
|
|
|
{
|
|
|
|
this.video.attr("autoplay", _value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set controls option for video
|
|
|
|
*
|
|
|
|
* @param {string} _value true set the autoplay and false not to set
|
|
|
|
*/
|
|
|
|
set_controls(_value: string)
|
|
|
|
{
|
2021-02-15 16:58:04 +01:00
|
|
|
if (_value && !this._isYoutube())
|
2020-01-21 16:14:45 +01:00
|
|
|
{
|
|
|
|
this.video.attr("controls", _value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set poster attribute in order to specify
|
|
|
|
* an image to be shown while video is loading or before user play it
|
|
|
|
*
|
2020-05-08 12:20:37 +02:00
|
|
|
* @param {string} _url url or image spec like "api/mime128_video"
|
2020-01-21 16:14:45 +01:00
|
|
|
*/
|
|
|
|
set_poster(_url: string)
|
|
|
|
{
|
|
|
|
if (_url)
|
|
|
|
{
|
2020-05-08 12:20:37 +02:00
|
|
|
if (_url[0] !== '/' && !_url.match(/^https?:\/\//))
|
|
|
|
{
|
|
|
|
_url = this.egw().image(_url);
|
|
|
|
}
|
2020-01-21 16:14:45 +01:00
|
|
|
this.video.attr("poster", _url);
|
|
|
|
}
|
|
|
|
}
|
2020-05-10 14:52:47 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Seek to a time / position
|
|
|
|
*
|
|
|
|
* @param _vtime in seconds
|
|
|
|
*/
|
|
|
|
public seek_video(_vtime : number)
|
|
|
|
{
|
2021-02-15 16:58:04 +01:00
|
|
|
if (this._isYoutube())
|
|
|
|
{
|
|
|
|
this.youtube.seekTo(_vtime);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
(<HTMLVideoElement>this.video[0]).currentTime = _vtime;
|
|
|
|
}
|
2020-05-10 14:52:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Play video
|
|
|
|
*/
|
|
|
|
public play_video() : Promise<void>
|
|
|
|
{
|
2021-02-15 16:58:04 +01:00
|
|
|
return this._isYoutube()?this.youtube.playVideo():(<HTMLVideoElement>this.video[0]).play();
|
2020-05-10 14:52:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Pause video
|
|
|
|
*/
|
|
|
|
public pause_video()
|
|
|
|
{
|
2021-02-15 16:58:04 +01:00
|
|
|
if (this._isYoutube())
|
|
|
|
{
|
|
|
|
this.youtube.pauseVideo();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
(<HTMLVideoElement>this.video[0]).pause();
|
|
|
|
}
|
2020-05-10 14:52:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-02-15 12:24:33 +01:00
|
|
|
* play video
|
|
|
|
* ***Internal use and should not be overriden in its extended class***
|
2020-05-10 14:52:47 +02:00
|
|
|
*/
|
2021-02-15 12:24:33 +01:00
|
|
|
public play() : Promise<void>
|
2020-05-10 14:52:47 +02:00
|
|
|
{
|
2021-02-15 16:58:04 +01:00
|
|
|
return this._isYoutube() && this.youtube?.playVideo ?this.youtube.playVideo():(<HTMLVideoElement>this.video[0]).play();
|
2021-02-15 12:24:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get/set current video time / position in seconds
|
|
|
|
* @return returns currentTime
|
|
|
|
*/
|
|
|
|
public currentTime(_time?) : number
|
|
|
|
{
|
2021-02-15 16:58:04 +01:00
|
|
|
if (_time)
|
|
|
|
{
|
|
|
|
if (this._isYoutube())
|
|
|
|
{
|
|
|
|
this.youtube.seekTo(_time);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
(<HTMLVideoElement>this.video[0]).currentTime = _time;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (this._isYoutube())
|
|
|
|
{
|
|
|
|
return this.youtube?.getCurrentTime ? this.youtube.getCurrentTime() : 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return (<HTMLVideoElement>this.video[0]).currentTime;
|
|
|
|
}
|
2020-05-10 14:52:47 +02:00
|
|
|
}
|
2021-02-02 12:35:28 +01:00
|
|
|
|
2021-02-15 12:24:33 +01:00
|
|
|
/**
|
|
|
|
* get duration time
|
|
|
|
* @return returns duration time
|
|
|
|
*/
|
|
|
|
public duration() : number
|
|
|
|
{
|
2021-02-15 16:58:04 +01:00
|
|
|
if (this._isYoutube())
|
|
|
|
{
|
|
|
|
return this.youtube?.getDuration ? this.youtube.getDuration() : 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return (<HTMLVideoElement>this.video[0]).duration;
|
|
|
|
}
|
2021-02-15 12:24:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* get pasued
|
|
|
|
* @return returns paused flag
|
|
|
|
*/
|
|
|
|
public paused() : boolean
|
|
|
|
{
|
2021-02-15 16:58:04 +01:00
|
|
|
if (this._isYoutube())
|
|
|
|
{
|
|
|
|
return this.youtube.getPlayerState() == et2_video.youtube_player_states.paused;
|
|
|
|
}
|
|
|
|
return (<HTMLVideoElement>this.video[0]).paused;
|
2021-02-15 12:24:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* get ended
|
|
|
|
* @return returns ended flag
|
|
|
|
*/
|
|
|
|
public ended() : boolean
|
|
|
|
{
|
2021-02-15 16:58:04 +01:00
|
|
|
if (this._isYoutube())
|
|
|
|
{
|
|
|
|
return this.youtube.getPlayerState() == et2_video.youtube_player_states.ended;
|
|
|
|
}
|
|
|
|
return (<HTMLVideoElement>this.video[0]).ended;
|
2021-02-15 12:24:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* get/set priviousTime
|
|
|
|
* @param _time
|
|
|
|
* @return returns time
|
|
|
|
*/
|
|
|
|
public previousTime(_time?) : number
|
|
|
|
{
|
|
|
|
if (_time) this._previousTime = _time;
|
|
|
|
return this._previousTime;
|
|
|
|
}
|
|
|
|
|
2021-02-02 12:35:28 +01:00
|
|
|
doLoadingFinished(): boolean
|
|
|
|
{
|
|
|
|
super.doLoadingFinished();
|
|
|
|
let self = this;
|
2021-02-15 16:58:04 +01:00
|
|
|
if (!this._isYoutube())
|
|
|
|
{
|
|
|
|
this.video[0].addEventListener("loadedmetadata", function(){
|
|
|
|
self._onReady();
|
|
|
|
});
|
|
|
|
}
|
2021-02-02 12:35:28 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
public videoLoadnigIsFinished()
|
|
|
|
{
|
|
|
|
if (this.options.starttime)
|
|
|
|
{
|
|
|
|
this.seek_video(this.options.starttime);
|
|
|
|
}
|
|
|
|
}
|
2021-02-15 16:58:04 +01:00
|
|
|
|
|
|
|
private _onReady()
|
|
|
|
{
|
|
|
|
let event = document.createEvent("Event");
|
|
|
|
event.initEvent('et2_video.onReady.'+this.id, true, true);
|
|
|
|
this.video[0].dispatchEvent(event);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* check if the video is a youtube type
|
|
|
|
* @return return true if it's a youtube type video
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
private _isYoutube() : boolean
|
|
|
|
{
|
|
|
|
return !!this.options.src_type.match('youtube');
|
|
|
|
}
|
2020-05-07 16:14:07 +02:00
|
|
|
}
|
|
|
|
et2_register_widget(et2_video, ["video"]);
|