ultrawidify/src/ext/lib/video-data/VideoData.js

295 lines
7.6 KiB
JavaScript
Raw Normal View History

import Debug from '../../conf/Debug';
import PlayerData from './PlayerData';
import Resizer from '../video-transform/Resizer';
import ArDetector from '../ar-detect/ArDetector';
2018-05-09 00:03:22 +02:00
class VideoData {
constructor(video, settings, pageInfo, logger){
this.arSetupComplete = false;
2018-05-09 00:03:22 +02:00
this.video = video;
this.destroyed = false;
this.settings = settings;
2018-11-02 02:52:01 +01:00
this.pageInfo = pageInfo;
this.extensionMode = pageInfo.extensionMode;
this.logger = logger;
this.vdid = (Math.random()*100).toFixed();
this.userCssClassName = `uw-fuck-you-and-do-what-i-tell-you_${this.vdid}`;
// We'll replace cssWatcher (in resizer) with mutationObserver
const observerConf = {
attributes: true,
// attributeFilter: ['style', 'class'],
attributeOldValue: true,
};
this.observer = new MutationObserver(this.onVideoDimensionsChanged);
this.observer.observe(video, observerConf);
// POZOR: VRSTNI RED JE POMEMBEN (arDetect mora bit zadnji)
// NOTE: ORDERING OF OBJ INITIALIZATIONS IS IMPORTANT (arDetect needs to go last)
this.player = new PlayerData(this, logger);
this.resizer = new Resizer(this, logger);
this.arDetector = new ArDetector(this, logger); // this starts Ar detection. needs optional parameter that prevets ardetdctor from starting
2018-05-12 02:51:58 +02:00
// player dimensions need to be in:
// this.player.dimensions
// apply default align and stretch
this.logger.log('info', 'debug', "%c[VideoData::ctor] Initial resizer reset!", {background: '#afd', color: '#132'});
this.resizer.reset();
2018-11-02 02:52:01 +01:00
if (Debug.init) {
console.log("[VideoData::ctor] Created videoData with vdid", this.vdid,"\nextension mode:", this.extensionMode);
}
this.pageInfo.initMouseActionHandler(this);
this.video.classList.add(this.userCssClassName); // this also needs to be applied BEFORE we initialize resizer!
2019-08-31 18:21:49 +02:00
}
onVideoDimensionsChanged(mutationList, observer) {
2019-08-25 01:52:04 +02:00
if (!mutationList || this.video === undefined) { // something's wrong
if (observer && this.video) {
observer.disconnect();
}
return;
}
for (let mutation of mutationList) {
if (mutation.type === 'attributes') {
if (mutation.attributeName === 'class') {
if (!this.video.classList.contains(this.userCssClassName)) {
// force the page to include our class in classlist, if the classlist has been removed
this.video.classList.add(this.userCssClassName);
// } else if () {
// this bug should really get
} else {
this.restoreAr();
}
} else if (mutation.attributeName === 'style' && mutation.attributeOldValue !== this.video.getAttribute('style')) {
// if size of the video has changed, this may mean we need to recalculate/reapply
// last calculated aspect ratio
this.player.forceRefreshPlayerElement();
this.restoreAr();
2019-08-25 01:52:04 +02:00
} else if (mutation.attribute = 'src' && mutation.attributeOldValue !== this.video.getAttribute('src')) {
// try fixing alignment issue on video change
try {
this.player.forceRefreshPlayerElement();
2019-08-25 01:52:04 +02:00
this.restoreAr();
} catch (e) {
console.error("[VideoData::onVideoDimensionsChanged] There was an error when handling src change.", e);
}
}
}
}
2018-05-12 02:51:58 +02:00
}
firstTimeArdInit(){
if(this.destroyed) {
2019-08-25 01:52:04 +02:00
// throw {error: 'VIDEO_DATA_DESTROYED', data: {videoData: this}};
return;
}
if(! this.arSetupComplete){
this.arDetector = new ArDetector(this);
}
}
initArDetection() {
if(this.destroyed) {
2019-08-25 01:52:04 +02:00
// throw {error: 'VIDEO_DATA_DESTROYED', data: {videoData: this}};
return;
}
if(this.arDetector){
this.arDetector.init();
}
else{
this.arDetector = new ArDetector(this);
this.arDetector.init();
}
2018-05-12 02:51:58 +02:00
}
startArDetection() {
this.logger.log('info', 'debug', "[VideoData::startArDetection] starting AR detection")
if(this.destroyed) {
2019-08-25 01:52:04 +02:00
// throw {error: 'VIDEO_DATA_DESTROYED', data: {videoData: this}};
return;
}
2018-08-31 00:35:52 +02:00
if(!this.arDetector) {
this.arDetector.init();
}
2018-05-12 02:51:58 +02:00
this.arDetector.start();
2018-05-09 00:03:22 +02:00
}
rebootArDetection() {
if(this.destroyed) {
2019-08-25 01:52:04 +02:00
// throw {error: 'VIDEO_DATA_DESTROYED', data: {videoData: this}};
return;
}
this.arDetector.init();
}
stopArDetection() {
2018-08-31 00:35:52 +02:00
if (this.arDetector) {
this.arDetector.stop();
}
}
2018-05-09 00:03:22 +02:00
destroy() {
this.logger.log('info', ['debug', 'init'], `[VideoData::destroy] <vdid:${this.vdid}> received destroy command`);
2019-08-31 18:21:49 +02:00
this.video.classList.remove(this.userCssClassName);
this.pause();
this.destroyed = true;
2019-08-25 01:52:04 +02:00
if (this.arDetector){
try {
this.arDetector.stop();
this.arDetector.destroy();
} catch (e) {}
}
2019-08-25 01:52:04 +02:00
this.arDetector = undefined;
if (this.resizer){
try {
this.resizer.destroy();
} catch (e) {}
}
2019-08-25 01:52:04 +02:00
this.resizer = undefined;
if (this.player){
try {
this.player.destroy();
} catch (e) {}
}
if (this.observer) {
try {
this.observer.disconnect();
} catch (e) {}
}
2019-08-25 01:52:04 +02:00
this.player = undefined;
this.video = undefined;
2018-05-09 00:03:22 +02:00
}
2018-05-23 23:58:34 +02:00
pause(){
this.paused = true;
if(this.arDetector){
this.arDetector.stop();
}
if(this.player){
this.player.stop();
}
}
resume(){
if(this.destroyed) {
2019-08-25 01:52:04 +02:00
// throw {error: 'VIDEO_DATA_DESTROYED', data: {videoData: this}};
return;
}
2018-05-23 23:58:34 +02:00
this.paused = false;
try {
this.resizer.start();
2018-11-02 02:52:01 +01:00
if (this.player) {
this.player.start();
}
2018-05-23 23:58:34 +02:00
} catch (e) {
this.logger.log('error', 'debug', "[VideoData.js::resume] cannot resume for reasons. Will destroy videoData. Error here:", e);
2018-05-23 23:58:34 +02:00
this.destroy();
}
}
resumeAutoAr(){
if(this.arDetector){
this.startArDetection();
}
}
2019-02-16 01:19:29 +01:00
setManualTick(manualTick) {
if(this.arDetector){
this.arDetector.setManualTick(manualTick);
}
}
2019-02-16 01:19:29 +01:00
tick() {
if(this.arDetector){
this.arDetector.tick();
}
}
setLastAr(lastAr){
this.resizer.setLastAr(lastAr);
}
setAr(ar, lastAr){
this.resizer.setAr(ar, lastAr);
}
2018-09-14 00:10:57 +02:00
resetAr() {
this.resizer.reset();
}
resetLastAr() {
this.resizer.setLastAr('original');
}
panHandler(event, forcePan) {
if(this.destroyed) {
2019-08-25 01:52:04 +02:00
// throw {error: 'VIDEO_DATA_DESTROYED', data: {videoData: this}};
return;
}
if(!this.resizer) {
this.destroy();
return;
}
this.resizer.panHandler(event, forcePan);
2018-09-13 23:47:20 +02:00
}
setPanMode(mode) {
this.resizer.setPanMode(mode);
}
setvideoAlignment(videoAlignment) {
this.resizer.setvideoAlignment(videoAlignment);
}
restoreAr(){
this.resizer.restore();
}
setStretchMode(stretchMode){
this.resizer.setStretchMode(stretchMode);
}
2018-09-21 00:26:08 +02:00
setZoom(zoomLevel, no_announce){
this.resizer.setZoom(zoomLevel, no_announce);
}
zoomStep(step){
2018-09-13 23:47:20 +02:00
this.resizer.zoomStep(step);
}
2018-09-21 00:26:08 +02:00
announceZoom(scale){
this.pageInfo.announceZoom(scale);
}
markPlayer(name, color) {
if (this.player) {
this.player.markPlayer(name, color)
}
}
unmarkPlayer() {
this.player.unmarkPlayer();
}
isPlaying() {
// console.log("is playing? video:", this.video, "ctime:", this.video.currentTime,
// "paused/ended:", this.video.paused, this.video.ended,
// "is playing?", this.video && this.video.currentTime > 0 && !this.video.paused && !this.video.ended);
return this.video && this.video.currentTime > 0 && !this.video.paused && !this.video.ended;
}
}
export default VideoData;