2018-12-31 01:03:07 +01:00
import Debug from '../../conf/Debug' ;
import Scaler from './Scaler' ;
import Stretcher from './Stretcher' ;
import Zoom from './Zoom' ;
import PlayerData from '../video-data/PlayerData' ;
2019-01-20 23:01:45 +01:00
import ExtensionMode from '../../../common/enums/extension-mode.enum' ;
2019-02-19 21:10:49 +01:00
import Stretch from '../../../common/enums/stretch.enum' ;
2019-02-15 20:40:56 +01:00
import VideoAlignment from '../../../common/enums/video-alignment.enum' ;
2019-03-10 23:27:50 +01:00
import AspectRatio from '../../../common/enums/aspect-ratio.enum' ;
2018-12-31 01:03:07 +01:00
2019-02-21 21:21:25 +01:00
if ( Debug . debug ) {
2017-09-24 01:54:46 +02:00
console . log ( "Loading: Resizer.js" ) ;
2019-02-21 21:21:25 +01:00
}
2017-09-24 01:54:46 +02:00
2018-12-31 01:03:07 +01:00
class Resizer {
2017-09-24 01:54:46 +02:00
2018-11-02 21:19:34 +01:00
constructor ( videoData ) {
2018-05-12 02:51:58 +02:00
this . conf = videoData ;
2018-05-08 23:35:16 +02:00
this . video = videoData . video ;
2018-08-05 23:48:56 +02:00
this . settings = videoData . settings ;
2018-11-02 21:19:34 +01:00
this . extensionMode = videoData . extensionMode ;
2018-05-12 02:51:58 +02:00
2018-05-18 23:26:20 +02:00
this . scaler = new Scaler ( this . conf ) ;
this . stretcher = new Stretcher ( this . conf ) ;
this . zoom = new Zoom ( this . conf ) ;
2018-05-12 02:51:58 +02:00
2019-05-07 23:40:13 +02:00
this . cssCheckDisabled = false ;
2018-05-12 02:51:58 +02:00
// load up default values
2018-05-13 15:22:28 +02:00
this . correctedVideoDimensions = { } ;
this . currentCss = { } ;
2018-08-21 00:48:15 +02:00
this . currentStyleString = "" ;
2019-06-12 23:55:15 +02:00
this . currentPlayerStyleString = "" ;
2018-08-21 00:48:15 +02:00
this . currentCssValidFor = { } ;
2018-05-12 02:51:58 +02:00
// restore watchdog. While true, applyCss() tries to re-apply new css until this value becomes false again
// value becomes false when width and height of <video> tag match with what we want to set. Only necessary when
// calling _res_restore() for some weird reason.
this . restore _wd = false ;
2018-05-13 15:22:28 +02:00
2018-08-21 00:48:15 +02:00
// CSS watcher will trigger _very_ often for this many iterations
this . cssWatcherIncreasedFrequencyCounter = 0 ;
2018-09-17 00:39:32 +02:00
2019-02-16 01:54:41 +01:00
// this.lastAr = this.settings.getDefaultAr(); // this is the aspect ratio we start with
2019-03-10 23:27:50 +01:00
this . lastAr = { type : AspectRatio . Initial } ;
2019-02-19 21:10:49 +01:00
this . videoAlignment = this . settings . getDefaultVideoAlignment ( window . location . hostname ) ; // this is initial video alignment
2018-05-16 23:26:47 +02:00
this . destroyed = false ;
2018-08-30 00:56:15 +02:00
this . resizerId = ( Math . random ( 99 ) * 100 ) . toFixed ( 0 ) ;
2018-09-13 23:47:20 +02:00
if ( this . settings . active . pan ) {
2018-12-03 00:31:28 +01:00
console . log ( "can pan:" , this . settings . active . miscSettings . mousePan . enabled , "(default:" , this . settings . active . miscSettings . mousePan . enabled , ")" )
this . canPan = this . settings . active . miscSettings . mousePan . enabled ;
2018-09-13 23:47:20 +02:00
} else {
this . canPan = false ;
}
2019-08-23 02:25:48 +02:00
this . userCss = '' ;
this . userCssClassName = videoData . userCssClassName ;
2017-09-24 01:54:46 +02:00
}
2018-05-23 23:57:51 +02:00
start ( ) {
2018-09-23 19:46:40 +02:00
if ( ! this . destroyed ) {
this . startCssWatcher ( ) ;
}
2018-05-23 23:57:51 +02:00
}
stop ( ) {
this . stopCssWatcher ( ) ;
}
2019-08-23 02:25:48 +02:00
injectCss ( css ) {
this . conf . pageInfo . injectCss ( css ) ;
}
ejectCss ( css ) {
this . conf . pageInfo . ejectCss ( css ) ;
}
2019-08-24 00:28:08 +02:00
replaceCss ( oldCss , newCss ) {
this . conf . pageInfo . replaceCss ( oldCss , newCss ) ;
}
2019-08-23 02:25:48 +02:00
prepareCss ( css ) {
return ` . ${ this . userCssClassName } { ${ css } } ` ;
}
2018-05-16 23:26:47 +02:00
destroy ( ) {
2018-09-23 19:46:40 +02:00
if ( Debug . debug || Debug . init ) {
2018-08-30 00:56:15 +02:00
console . log ( ` [Resizer::destroy] <rid: ${ this . resizerId } > received destroy command. ` ) ;
}
2018-05-16 23:26:47 +02:00
this . destroyed = true ;
this . stopCssWatcher ( ) ;
}
2018-05-13 15:22:28 +02:00
2019-03-10 23:27:50 +01:00
calculateRatioForLegacyOptions ( ar ) {
// also present as modeToAr in Scaler.js
2019-05-07 23:40:13 +02:00
if ( ar . type !== AspectRatio . FitWidth && ar . type !== AspectRatio . FitHeight && ar . ratio ) {
2019-04-25 22:02:10 +02:00
return ar ;
2019-03-10 23:27:50 +01:00
}
// Skrbi za "stare" možnosti, kot na primer "na širino zaslona", "na višino zaslona" in "ponastavi".
// Približevanje opuščeno.
// handles "legacy" options, such as 'fit to widht', 'fit to height' and AspectRatio.Reset. No zoom tho
var ratioOut ;
if ( ! this . conf . video ) {
2019-04-25 22:02:10 +02:00
if ( Debug . debug ) {
2019-03-10 23:27:50 +01:00
console . log ( "[Scaler.js::modeToAr] No video??" , this . conf . video , "killing videoData" ) ;
}
this . conf . destroy ( ) ;
return null ;
}
2019-04-25 22:02:10 +02:00
if ( ! this . conf . player . dimensions ) {
2019-03-10 23:27:50 +01:00
ratioOut = screen . width / screen . height ;
2019-04-25 22:02:10 +02:00
} else {
2019-06-12 23:55:15 +02:00
if ( Debug . debug && Debug . resizer ) {
2019-05-07 23:40:13 +02:00
console . log ( ` [Resizer::calculateRatioForLegacyOptions] <rid: ${ this . resizerId } > Player dimensions: ` , this . conf . player . dimensions . width , 'x' , this . conf . player . dimensions . height , 'aspect ratio:' , this . conf . player . dimensions . width / this . conf . player . dimensions . height )
}
2019-03-10 23:27:50 +01:00
ratioOut = this . conf . player . dimensions . width / this . conf . player . dimensions . height ;
}
// POMEMBNO: lastAr je potrebno nastaviti šele po tem, ko kličemo _res_setAr(). _res_setAr() predvideva,
// da želimo nastaviti statično (type: 'static') razmerje stranic — tudi, če funkcijo kličemo tu oz. v ArDetect.
//
// IMPORTANT NOTE: lastAr needs to be set after _res_setAr() is called, as _res_setAr() assumes we're
// setting a static aspect ratio (even if the function is called from here or ArDetect).
var fileAr = this . conf . video . videoWidth / this . conf . video . videoHeight ;
if ( ar . type === AspectRatio . FitWidth ) {
2019-04-25 22:02:10 +02:00
ar . ratio = ratioOut > fileAr ? ratioOut : fileAr ;
2019-03-10 23:27:50 +01:00
}
else if ( ar . type === AspectRatio . FitHeight ) {
2019-04-25 22:02:10 +02:00
ar . ratio = ratioOut < fileAr ? ratioOut : fileAr ;
2019-03-10 23:27:50 +01:00
}
else if ( ar . type === AspectRatio . Reset ) {
if ( Debug . debug ) {
2019-04-25 22:02:10 +02:00
console . log ( "[Scaler.js::modeToAr] Using original aspect ratio -" , fileAr ) ;
2019-03-10 23:27:50 +01:00
}
2019-04-25 22:02:10 +02:00
ar . ratio = fileAr ;
} else {
return null ;
2019-03-10 23:27:50 +01:00
}
2019-04-25 22:02:10 +02:00
return ar ;
2019-03-10 23:27:50 +01:00
}
2018-05-24 23:29:30 +02:00
2018-05-12 02:51:58 +02:00
setAr ( ar , lastAr ) {
2018-08-30 00:56:15 +02:00
if ( this . destroyed ) {
return ;
}
2018-11-02 21:19:34 +01:00
2018-05-08 23:35:16 +02:00
if ( Debug . debug ) {
2018-08-30 00:56:15 +02:00
console . log ( '[Resizer::setAr] <rid:' + this . resizerId + '> trying to set ar. New ar:' , ar )
2018-05-08 23:35:16 +02:00
}
2017-09-24 01:54:46 +02:00
2019-03-10 23:27:50 +01:00
if ( ar == null ) {
2019-02-21 21:21:25 +01:00
return ;
}
2019-05-26 02:53:29 +02:00
if ( ar . type === AspectRatio . Automatic ||
ar . type === AspectRatio . Reset && this . lastAr . type === AspectRatio . Initial ) {
// some sites do things that interfere with our site (and aspect ratio setting in general)
// first, we check whether video contains anything we don't like
const siteSettings = this . settings . active . sites [ window . location . host ] ;
if ( siteSettings && siteSettings . autoarPreventConditions ) {
if ( siteSettings . autoarPreventConditions . videoStyleString ) {
2019-06-05 23:35:25 +02:00
const styleString = ( this . video . getAttribute ( 'style' ) || '' ) . split ( ';' ) ;
2019-05-26 02:53:29 +02:00
if ( siteSettings . autoarPreventConditions . videoStyleString . containsProperty ) {
const bannedProperties = siteSettings . autoarPreventConditions . videoStyleString . containsProperty ;
for ( const prop in bannedProperties ) {
for ( const s of styleString ) {
if ( s . trim ( ) . startsWith ( prop ) ) {
// check if css property has a list of allowed values:
if ( bannedProperties [ prop ] . allowedValues ) {
const styleValue = s . split ( ':' ) [ 1 ] . trim ( ) ;
// check if property value is on the list of allowed values
// if it's not, we aren't allowed to start aard
if ( bannedProperties [ prop ] . allowedValues . indexOf ( styleValue ) === - 1 ) {
if ( Debug . debug ) {
console . log ( "%c[Resizer::setAr] video style contains forbidden css property/value combo: " , "color: #900, background: #100" , prop , " — we aren't allowed to start autoar." )
}
return ;
}
} else {
// no allowed values, no problem. We have forbidden property
// and this means aard can't start.
if ( Debug . debug ) {
console . log ( "%c[Resizer::setAr] video style contains forbidden css property: " , "color: #900, background: #100" , prop , " — we aren't allowed to start autoar." )
}
return ;
}
}
}
}
}
}
}
}
2019-04-25 22:02:10 +02:00
if ( lastAr ) {
2019-05-07 23:40:13 +02:00
this . lastAr = this . calculateRatioForLegacyOptions ( lastAr ) ;
ar = this . calculateRatioForLegacyOptions ( ar ) ;
2018-05-12 02:51:58 +02:00
} else {
2019-04-25 22:02:10 +02:00
// NOTE: "fitw" "fith" and "reset" should ignore ar.ratio bit, but
// I'm not sure whether they do. Check that.
ar = this . calculateRatioForLegacyOptions ( ar ) ;
if ( ! ar ) {
2019-06-12 23:55:15 +02:00
if ( Debug . debug && Debug . resizer ) {
2019-04-25 22:02:10 +02:00
console . log ( ` [Resizer::setAr] < ${ this . resizerId } > Something wrong with ar or the player. Doing nothing. ` ) ;
}
return ;
2018-05-13 15:22:28 +02:00
}
2019-04-25 22:02:10 +02:00
this . lastAr = { type : ar . type , ratio : ar . ratio }
2018-05-12 02:51:58 +02:00
}
2018-05-01 23:09:58 +02:00
2019-03-10 23:27:50 +01:00
if ( this . extensionMode === ExtensionMode . Basic && ! PlayerData . isFullScreen ( ) && ar . type !== AspectRatio . Reset ) {
2018-11-02 21:51:51 +01:00
// don't actually apply or calculate css when using basic mode if not in fullscreen
// ... unless we're resetting the aspect ratio to original
return ;
2018-11-02 21:19:34 +01:00
}
2018-05-08 23:35:16 +02:00
if ( ! this . video ) {
2018-05-16 20:26:55 +02:00
// console.log("No video detected.")
2019-08-23 02:25:48 +02:00
this . conf . destroy ( ) ;
2018-05-08 23:35:16 +02:00
}
2018-05-01 23:09:58 +02:00
2019-02-15 20:40:56 +01:00
if ( this . extensionMode === ExtensionMode . Enabled || PlayerData . isFullScreen ( ) ) {
2018-11-02 21:19:34 +01:00
this . startCssWatcher ( ) ;
}
this . cssWatcherIncreasedFrequencyCounter = 20 ;
2018-09-23 19:46:40 +02:00
// // pause AR on basic stretch, unpause when using other mdoes
2018-07-15 16:22:32 +02:00
// fir sine reason unpause doesn't unpause. investigate that later
2019-02-16 01:54:41 +01:00
try {
2019-02-19 21:10:49 +01:00
if ( this . stretcher . mode === Stretch . Basic ) {
2019-02-16 01:54:41 +01:00
this . conf . arDetector . pause ( ) ;
} else {
2019-03-10 23:27:50 +01:00
if ( this . lastAr . type === AspectRatio . Automatic ) {
2019-02-16 01:54:41 +01:00
this . conf . arDetector . unpause ( ) ;
}
2019-02-15 20:40:56 +01:00
}
2019-02-16 01:54:41 +01:00
} catch ( e ) { // resizer starts before arDetector. this will do nothing but fail if arDetector isn't setup
2019-02-15 20:40:56 +01:00
}
2018-05-16 20:26:55 +02:00
2018-07-11 23:13:40 +02:00
// do stretch thingy
2019-02-19 21:10:49 +01:00
if ( this . stretcher . mode === Stretch . NoStretch || this . stretcher . mode === Stretch . Conditional ) {
2018-05-24 23:29:30 +02:00
var stretchFactors = this . scaler . calculateCrop ( ar ) ;
if ( ! stretchFactors || stretchFactors . error ) {
if ( Debug . debug ) {
2018-08-30 00:56:15 +02:00
console . log ( "[Resizer::setAr] <rid:" + this . resizerId + "> failed to set AR due to problem with calculating crop. Error:" , ( stretchFactors ? stretchFactors . error : stretchFactors ) ) ;
2018-05-24 23:29:30 +02:00
}
2019-05-07 23:40:13 +02:00
if ( stretchFactors . error === 'no_video' ) {
2018-05-24 23:29:30 +02:00
this . conf . destroy ( ) ;
}
2019-05-07 23:40:13 +02:00
if ( stretchFactors . error === 'illegal_video_dimensions' ) {
if ( Debug . debug ) {
console . log ( "[Resizer::setAr] <rid:" + this . resizerId + "> Illegal video dimensions found. We will pause everything." ) ;
}
// if we get illegal video dimensions, cssWatcher goes nuts. This is harmful,
// so we stop it until that sorts itself out
this . stopCssWatcher ( ) ;
}
2018-05-24 23:29:30 +02:00
return ;
2019-05-07 23:40:13 +02:00
} else {
this . startCssWatcher ( ) ;
2018-05-16 23:26:47 +02:00
}
2019-02-19 21:10:49 +01:00
if ( this . stretcher . mode === Stretch . Conditional ) {
2019-03-10 23:27:50 +01:00
this . stretcher . applyConditionalStretch ( stretchFactors , ar . ratio ) ;
2018-05-16 20:26:55 +02:00
}
2019-02-15 20:40:56 +01:00
if ( Debug . debug ) {
2019-02-19 21:10:49 +01:00
console . log ( "[Resizer::setAr] Processed stretch factors for " , this . stretcher . mode === Stretch . NoStretch ? 'stretch-free crop.' : 'crop with conditional stretch.' , 'Stretch factors are:' , stretchFactors ) ;
2019-02-15 20:40:56 +01:00
}
2019-02-19 21:10:49 +01:00
} else if ( this . stretcher . mode === Stretch . Hybrid ) {
2019-03-10 23:27:50 +01:00
var stretchFactors = this . stretcher . calculateStretch ( ar . ratio ) ;
2019-02-15 20:40:56 +01:00
if ( Debug . debug ) {
console . log ( '[Resizer::setAr] Processed stretch factors for hybrid stretch/crop. Stretch factors are:' , stretchFactors ) ;
}
2019-02-19 21:10:49 +01:00
} else if ( this . stretcher . mode === Stretch . Basic ) {
2018-06-15 00:33:10 +02:00
var stretchFactors = this . stretcher . calculateBasicStretch ( ) ;
2019-02-15 20:40:56 +01:00
if ( Debug . debug ) {
console . log ( '[Resizer::setAr] Processed stretch factors for basic stretch. Stretch factors are:' , stretchFactors ) ;
}
} else {
var stretchFactors = { xFactor : 1 , yFactor : 1 }
if ( Debug . debug ) {
console . log ( '[Resizer::setAr] Okay wtf happened? If you see this, something has gone wrong' , stretchFactors , "\n------[ i n f o d u m p ]------\nstretcher:" , this . stretcher ) ;
}
2018-05-16 20:26:55 +02:00
}
2018-05-24 23:29:30 +02:00
this . zoom . applyZoom ( stretchFactors ) ;
2018-05-02 17:52:25 +02:00
2018-05-24 23:29:30 +02:00
//TODO: correct these two
2018-05-25 21:37:09 +02:00
var translate = this . computeOffsets ( stretchFactors ) ;
this . applyCss ( stretchFactors , translate ) ;
2018-05-16 23:26:47 +02:00
2018-07-10 20:36:12 +02:00
}
2018-06-15 00:33:10 +02:00
2018-07-10 20:36:12 +02:00
resetLastAr ( ) {
2019-03-10 23:27:50 +01:00
this . lastAr = { type : AspectRatio . Initial } ;
2018-05-12 02:51:58 +02:00
}
2018-05-02 17:52:25 +02:00
2018-05-14 20:39:15 +02:00
setLastAr ( override ) {
this . lastAr = override ;
}
2018-05-15 21:40:53 +02:00
getLastAr ( ) {
return this . lastAr ;
}
2018-05-13 15:22:28 +02:00
setStretchMode ( stretchMode ) {
2019-02-19 21:10:49 +01:00
this . stretcher . setStretchMode ( stretchMode ) ;
2018-05-27 21:41:08 +02:00
this . restore ( ) ;
2018-05-13 15:22:28 +02:00
}
2018-05-02 17:52:25 +02:00
2018-12-02 23:51:34 +01:00
panHandler ( event , forcePan ) {
2018-09-13 23:47:20 +02:00
// console.log("this.conf.canPan:", this.conf.canPan)
2018-12-02 23:51:34 +01:00
if ( this . canPan || forcePan ) {
2018-09-13 23:47:20 +02:00
if ( ! this . conf . player || ! this . conf . player . element ) {
return ;
}
2018-12-02 23:51:34 +01:00
// dont allow weird floats
2019-02-19 21:10:49 +01:00
this . videoAlignment = VideoAlignment . Center ;
2018-12-02 23:51:34 +01:00
2018-09-13 23:47:20 +02:00
const player = this . conf . player . element ;
const relativeX = ( event . pageX - player . offsetLeft ) / player . offsetWidth ;
const relativeY = ( event . pageY - player . offsetTop ) / player . offsetHeight ;
2018-12-02 23:51:34 +01:00
if ( Debug . debug && Debug . mousemove ) {
console . log ( "[Resizer::panHandler] mousemove.pageX, pageY:" , event . pageX , event . pageY ,
"\nrelativeX/Y:" , relativeX , relativeY )
}
2018-09-13 23:47:20 +02:00
this . setPan ( relativeX , relativeY ) ;
}
}
2018-05-13 15:22:28 +02:00
setPan ( relativeMousePosX , relativeMousePosY ) {
// relativeMousePos[X|Y] - on scale from 0 to 1, how close is the mouse to player edges.
// use these values: top, left: 0, bottom, right: 1
if ( ! this . pan ) {
this . pan = { } ;
}
2018-02-04 17:39:26 +01:00
2018-12-03 00:31:28 +01:00
if ( this . settings . active . miscSettings . mousePanReverseMouse ) {
this . pan . relativeOffsetX = ( relativeMousePosX * 1.1 ) - 0.55 ;
this . pan . relativeOffsetY = ( relativeMousePosY * 1.1 ) - 0.55 ;
} else {
this . pan . relativeOffsetX = - ( relativeMousePosX * 1.1 ) + 0.55 ;
this . pan . relativeOffsetY = - ( relativeMousePosY * 1.1 ) + 0.55 ;
}
2018-09-13 23:47:20 +02:00
// if(Debug.debug){
// console.log("[Resizer::setPan] relative cursor pos:", relativeMousePosX, ",",relativeMousePosY, " | new pan obj:", this.pan)
// }
this . restore ( ) ;
2018-05-13 15:22:28 +02:00
}
2018-05-02 17:52:25 +02:00
2018-12-31 03:34:26 +01:00
setvideoAlignment ( videoAlignment ) {
this . videoAlignment = videoAlignment ;
2018-09-23 02:39:27 +02:00
this . restore ( ) ;
}
2018-05-13 15:22:28 +02:00
startCssWatcher ( ) {
2018-09-23 19:46:40 +02:00
if ( Debug . debug ) {
console . log ( "[Resizer.js::startCssWatcher] starting css watcher. Is resizer destroyed?" , this . destroyed ) ;
}
2018-08-30 00:56:15 +02:00
if ( this . destroyed ) {
return ;
}
2018-09-23 19:46:40 +02:00
2019-05-07 23:40:13 +02:00
this . cssCheckDisabled = false ;
2018-08-21 00:48:15 +02:00
if ( ! this . cssWatcherTimer ) {
this . scheduleCssWatcher ( 1 ) ;
} else {
clearTimeout ( this . cssWatcherTimer ) ;
this . scheduleCssWatcher ( 1 ) ;
}
}
2018-05-16 23:26:47 +02:00
2018-08-21 00:48:15 +02:00
scheduleCssWatcher ( timeout , force _reset ) {
2019-01-20 22:59:06 +01:00
if ( this . destroyed || ( this . extensionMode !== ExtensionMode . Enabled && ! PlayerData . isFullScreen ( ) ) ) {
2018-08-30 00:56:15 +02:00
return ;
}
2018-08-21 00:48:15 +02:00
if ( timeout === undefined ) {
this . cssCheck ( ) ; // no timeout = one-off
return ;
}
2019-05-07 23:40:13 +02:00
// one extra check to ensure we don't run css checks when we aren't supposed to
if ( this . cssCheckDisabled ) {
return ;
}
2018-08-21 00:48:15 +02:00
if ( this . cssWatcherTimeout ) {
clearTimeout ( this . cssWatcherTimer ) ;
2018-05-16 23:26:47 +02:00
}
2018-08-21 00:48:15 +02:00
var ths = this ;
this . cssWatcherTimer = setTimeout ( function ( ) {
ths . cssWatcherTimer = null ;
try {
2019-05-07 23:40:13 +02:00
if ( ! ths . cssCheckDisabled ) {
ths . cssCheck ( ) ;
}
2018-08-21 00:48:15 +02:00
} catch ( e ) {
if ( Debug . debug ) {
console . log ( "[Resizer.js::scheduleCssWatcher] Css check failed. Error:" , e ) ;
}
}
} , timeout ) ;
2018-05-13 15:22:28 +02:00
}
2017-09-24 01:54:46 +02:00
2018-06-15 00:33:10 +02:00
stopCssWatcher ( ) {
2019-03-10 23:27:50 +01:00
if ( Debug . debug ) {
2019-05-07 23:40:13 +02:00
console . log ( ` [Resizer.js] < ${ this . resizerId } > STOPPING CSS WATCHER! ` )
2019-03-10 23:27:50 +01:00
}
2019-05-07 23:40:13 +02:00
this . cssCheckDisabled = true ;
2018-05-13 15:22:28 +02:00
clearInterval ( this . cssWatcherTimeout ) ;
}
2018-05-02 17:52:25 +02:00
2018-06-15 00:33:10 +02:00
restore ( ) {
2018-05-13 15:22:28 +02:00
if ( Debug . debug ) {
2019-02-16 01:19:29 +01:00
console . log ( "[Resizer::restore] <rid:" + this . resizerId + "> attempting to restore aspect ratio. this & settings:" , { 'a_lastAr' : this . lastAr , 'this' : this , "settings" : this . settings } ) ;
2018-05-13 15:22:28 +02:00
}
// this is true until we verify that css has actually been applied
this . restore _wd = true ;
2019-03-10 23:27:50 +01:00
if ( this . lastAr . type === AspectRatio . Initial ) {
this . setAr ( { type : AspectRatio . Reset } ) ;
2018-05-13 15:22:28 +02:00
}
else {
2019-03-10 23:27:50 +01:00
if ( this . lastAr && this . lastAr . ratio === null ) {
2019-02-21 21:21:25 +01:00
throw "Last ar is null!"
}
2019-03-10 23:27:50 +01:00
this . setAr ( this . lastAr , this . lastAr )
2018-05-13 15:22:28 +02:00
}
}
2018-05-02 17:52:25 +02:00
2018-05-13 15:22:28 +02:00
reset ( ) {
2019-02-16 01:54:41 +01:00
this . setStretchMode ( this . settings . active . sites [ window . location . hostname ] ? this . settings . active . sites [ window . location . hostname ] . stretch : this . settings . active . sites [ '@global' ] . stretch ) ;
2018-05-13 15:22:28 +02:00
this . zoom . setZoom ( 1 ) ;
this . resetPan ( ) ;
2019-03-10 23:27:50 +01:00
this . setAr ( { type : AspectRatio . Reset } ) ;
2017-09-24 01:54:46 +02:00
}
2018-05-13 15:22:28 +02:00
2018-09-13 23:47:20 +02:00
setPanMode ( mode ) {
if ( mode === 'enable' ) {
this . canPan = true ;
} else if ( mode === 'disable' ) {
this . canPan = false ;
} else if ( mode === 'toggle' ) {
this . canPan = ! this . canPan ;
}
}
2018-05-13 15:22:28 +02:00
resetPan ( ) {
this . pan = undefined ;
2017-09-24 01:54:46 +02:00
}
2018-05-13 15:22:28 +02:00
2018-09-21 00:26:08 +02:00
setZoom ( zoomLevel , no _announce ) {
this . zoom . setZoom ( zoomLevel , no _announce ) ;
2018-09-18 23:37:33 +02:00
}
2018-05-24 20:50:37 +02:00
zoomStep ( step ) {
this . zoom . zoomStep ( step ) ;
}
2018-05-13 15:22:28 +02:00
resetZoom ( ) {
this . zoom . setZoom ( 1 ) ;
this . restore ( ) ;
2017-09-24 01:54:46 +02:00
}
2018-05-13 15:22:28 +02:00
resetCrop ( ) {
2019-03-10 23:27:50 +01:00
this . setAr ( { type : AspectRatio . Reset } ) ;
2017-09-24 01:54:46 +02:00
}
2018-05-13 15:22:28 +02:00
resetStretch ( ) {
2019-02-19 21:10:49 +01:00
this . stretcher . setStretchMode ( Stretch . NoStretch ) ;
2018-05-13 15:22:28 +02:00
this . restore ( ) ;
2017-09-24 01:54:46 +02:00
}
2018-01-02 03:36:29 +01:00
2018-05-13 15:22:28 +02:00
// mostly internal stuff
2018-05-25 21:37:09 +02:00
computeOffsets ( stretchFactors ) {
2018-05-13 15:22:28 +02:00
2018-09-13 23:47:20 +02:00
if ( Debug . debug ) {
2019-02-15 20:40:56 +01:00
console . log ( "[Resizer::computeOffsets] <rid:" + this . resizerId + "> video will be aligned to " , this . settings . active . sites [ '@global' ] . videoAlignment ) ;
2018-09-13 23:47:20 +02:00
}
2018-11-17 00:11:07 +01:00
const wdiff = this . conf . player . dimensions . width - this . conf . video . offsetWidth ;
const hdiff = this . conf . player . dimensions . height - this . conf . video . offsetHeight ;
2018-12-03 00:31:28 +01:00
const wdiffAfterZoom = this . conf . video . offsetWidth * stretchFactors . xFactor - this . conf . player . dimensions . width ;
const hdiffAfterZoom = this . conf . video . offsetHeight * stretchFactors . yFactor - this . conf . player . dimensions . height ;
2018-11-16 23:02:56 +01:00
var translate = {
x : wdiff * 0.5 ,
y : hdiff * 0.5 ,
} ;
2018-04-22 14:35:40 +02:00
2018-07-15 16:22:32 +02:00
if ( this . pan ) {
2018-09-13 23:47:20 +02:00
// don't offset when video is smaller than player
2018-12-03 00:31:28 +01:00
if ( wdiffAfterZoom < 0 && hdiffAfterZoom < 0 ) {
2018-09-13 23:47:20 +02:00
return translate ;
}
2018-12-03 00:31:28 +01:00
translate . x += wdiffAfterZoom * this . pan . relativeOffsetX * this . zoom . scale ;
translate . y += hdiffAfterZoom * this . pan . relativeOffsetY * this . zoom . scale ;
2018-05-13 15:22:28 +02:00
} else {
2019-02-15 20:40:56 +01:00
if ( this . videoAlignment == VideoAlignment . Left ) {
2018-12-03 00:31:28 +01:00
translate . x += wdiffAfterZoom * 0.5 ;
2018-05-13 15:22:28 +02:00
}
2019-02-15 20:40:56 +01:00
else if ( this . videoAlignment == VideoAlignment . Right ) {
2018-12-03 00:31:28 +01:00
translate . x -= wdiffAfterZoom * 0.5 ;
2018-05-13 15:22:28 +02:00
}
}
2019-06-11 01:34:02 +02:00
if ( Debug . debug ) {
2019-05-26 02:53:29 +02:00
console . log ( "[Resizer::_res_computeOffsets] <rid:" + this . resizerId + "> calculated offsets:\n\n" ,
'---- data in ----\n' ,
'player dimensions:' , { w : this . conf . player . dimensions . width , h : this . conf . player . dimensions . height } ,
'video dimensions: ' , { w : this . conf . video . offsetWidth , h : this . conf . video . offsetHeight } ,
'stretch factors: ' , stretchFactors ,
'pan & zoom: ' , this . pan , this . zoom ,
'\n\n---- data out ----\n' ,
'translate:' , translate ) ;
2019-06-11 01:34:02 +02:00
}
2018-09-13 23:47:20 +02:00
2018-05-25 21:37:09 +02:00
return translate ;
2018-05-13 15:22:28 +02:00
}
2019-06-12 23:55:15 +02:00
buildStyleArray ( existingStyleString , extraStyleString ) {
if ( existingStyleString ) {
const styleArray = existingStyleString . split ( ";" ) ;
2018-05-16 23:26:47 +02:00
2019-06-12 23:55:15 +02:00
if ( extraStyleString ) {
const extraCss = extraStyleString . split ( ';' ) ;
2019-06-11 01:34:02 +02:00
let dup = false ;
for ( const ecss of extraCss ) {
2019-06-12 23:55:15 +02:00
for ( let i in styleArray ) {
if ( ecss . split ( ':' ) [ 0 ] . trim ( ) === styleArray [ i ] . split ( ':' ) [ 0 ] . trim ( ) ) {
2019-06-11 01:34:02 +02:00
dup = true ;
2019-06-12 23:55:15 +02:00
styleArray [ i ] = ecss ;
2019-06-11 01:34:02 +02:00
}
if ( dup ) {
dup = false ;
continue ;
}
2019-06-12 23:55:15 +02:00
styleArray . push ( ecss ) ;
2019-06-11 01:34:02 +02:00
}
}
}
for ( var i in styleArray ) {
2019-06-12 23:55:15 +02:00
styleArray [ i ] = styleArray [ i ] . trim ( ) ;
2018-11-02 23:10:42 +01:00
// some sites do 'top: 50%; left: 50%; transform: <transform>' to center videos.
// we dont wanna, because we already center videos on our own
if ( styleArray [ i ] . startsWith ( "transform:" ) ||
styleArray [ i ] . startsWith ( "top:" ) ||
styleArray [ i ] . startsWith ( "left:" ) ||
styleArray [ i ] . startsWith ( "right:" ) ||
2019-02-16 01:19:29 +01:00
styleArray [ i ] . startsWith ( "bottom:" ) ) {
2018-05-25 21:37:09 +02:00
delete styleArray [ i ] ;
2018-05-13 15:22:28 +02:00
}
2018-01-02 03:36:29 +01:00
}
2019-06-12 23:55:15 +02:00
return styleArray ;
}
return [ ] ;
}
buildStyleString ( styleArray ) {
let styleString = '' ;
for ( var i in styleArray ) {
if ( styleArray [ i ] ) {
2019-08-23 02:25:48 +02:00
styleString += styleArray [ i ] + " !important; " ;
2019-06-12 23:55:15 +02:00
}
}
return styleString ;
}
applyCss ( stretchFactors , translate ) {
// apply extra CSS here. In case of duplicated properties, extraCss overrides
// default styleString
if ( ! this . video ) {
if ( Debug . debug ) {
console . log ( "[Resizer::applyCss] <rid:" + this . resizerId + "> Video went missing, doing nothing." ) ;
}
this . conf . destroy ( ) ;
return ;
}
if ( Debug . debug && Debug . resizer ) {
2019-08-23 02:25:48 +02:00
console . log ( "[Resizer::applyCss] <rid:" + this . resizerId + "> will apply css." , { stretchFactors , translate , video : this . video } ) ;
2018-05-13 15:22:28 +02:00
}
2019-06-12 23:55:15 +02:00
// save stuff for quick tests (before we turn numbers into css values):
this . currentVideoSettings = {
validFor : this . conf . player . dimensions ,
// videoWidth: dimensions.width,
// videoHeight: dimensions.height
}
let extraStyleString ;
try {
extraStyleString = this . settings . active . sites [ window . location . host ] . DOM . video . additionalCss ;
} catch ( e ) {
// do nothing. It's ok if no special settings are defined for this site, we'll just do defaults
}
2019-08-23 02:25:48 +02:00
const styleArray = this . buildStyleArray ( '' , extraStyleString )
2019-06-12 23:55:15 +02:00
2018-05-13 15:22:28 +02:00
// add remaining elements
2019-06-11 01:34:02 +02:00
if ( stretchFactors ) {
2018-11-16 23:02:56 +01:00
styleArray . push ( ` transform: translate( ${ translate . x } px, ${ translate . y } px) scale( ${ stretchFactors . xFactor } , ${ stretchFactors . yFactor } ) ` ) ;
2019-08-23 02:25:48 +02:00
styleArray . push ( "top: 0px !important; left: 0px !important; bottom: 0px !important; right: 0px" ) ;
2018-01-02 03:36:29 +01:00
}
2019-08-23 02:25:48 +02:00
const styleString = ` ${ this . buildStyleString ( styleArray ) } ${ extraStyleString || '' } ` ; // string returned by buildStyleString() should end with ; anyway
2018-04-22 14:35:40 +02:00
2019-06-12 23:55:15 +02:00
// build style string back
2019-06-14 02:15:24 +02:00
this . setStyleString ( styleString ) ;
2018-04-22 14:35:40 +02:00
}
2018-05-13 15:22:28 +02:00
2019-06-14 02:15:24 +02:00
setStyleString ( styleString ) {
2019-08-24 00:28:08 +02:00
this . currentCssValidFor = this . conf . player . dimensions ;
const newCssString = this . prepareCss ( styleString ) ;
2019-08-23 02:25:48 +02:00
2019-08-24 00:28:08 +02:00
// inject new CSS or replace existing one
if ( ! this . userCss ) {
this . injectCss ( newCssString ) ;
this . userCss = newCssString ;
} else {
this . replaceCss ( this . userCss , newCssString ) ;
this . userCss = newCssString ;
2019-08-23 02:25:48 +02:00
}
2019-06-12 23:55:15 +02:00
2019-08-24 00:28:08 +02:00
// browser checks and ignores duplicate classes, so no point in checking whether we've already
// added the extra class ourselves on top of that. Twice the work, but not much benefit
this . video . classList . add ( this . userCssClassName ) ;
2019-08-23 02:25:48 +02:00
2019-06-12 23:55:15 +02:00
if ( this . restore _wd ) {
if ( ! this . video ) {
2018-05-13 15:22:28 +02:00
if ( Debug . debug )
2018-08-30 00:56:15 +02:00
console . log ( "[Resizer::_res_setStyleString] <rid:" + this . resizerId + "> Video element went missing, nothing to do here." )
2018-05-13 15:22:28 +02:00
return ;
2017-12-02 21:09:08 +01:00
}
2019-08-23 02:25:48 +02:00
this . restore _wd = false ;
2017-11-13 22:35:04 +01:00
}
2018-05-13 15:22:28 +02:00
else {
if ( Debug . debug )
2018-08-30 00:56:15 +02:00
console . log ( "[Resizer::_res_setStyleString] <rid:" + this . resizerId + "> css applied. Style string:" , styleString ) ;
2018-05-13 15:22:28 +02:00
}
2017-11-13 22:35:04 +01:00
}
2018-04-25 20:39:52 +02:00
2018-08-21 00:48:15 +02:00
cssCheck ( ) {
2019-05-07 23:40:13 +02:00
if ( this . cssCheckDisabled ) {
throw "fucking dont"
return ;
}
2018-05-13 15:22:28 +02:00
// this means we haven't set our CSS yet, or that we changed video.
2018-08-21 00:48:15 +02:00
// if(! this.currentCss.tranform) {
// this.scheduleCssWatcher(200);
// return;
// }
2018-05-13 15:22:28 +02:00
2018-05-16 23:26:47 +02:00
// this means video went missing. videoData will be re-initialized when the next video is found
2019-06-12 23:55:15 +02:00
if ( ! this . video ) {
if ( Debug . debug && Debug . resizer ) {
2018-08-30 00:56:15 +02:00
console . log ( "[Resizer::cssCheck] <rid:" + this . resizerId + "> no video detecting, issuing destroy command" ) ;
}
2018-08-21 00:48:15 +02:00
this . conf . destroy ( ) ;
2018-05-13 15:22:28 +02:00
return ;
2018-05-16 23:26:47 +02:00
}
2018-05-13 15:22:28 +02:00
2018-08-30 00:56:15 +02:00
if ( this . destroyed ) {
2019-06-12 23:55:15 +02:00
if ( Debug . debug && Debug . resizer ) {
2018-08-30 00:56:15 +02:00
console . log ( "[Resizer::cssCheck] <rid:" + this . resizerId + "> destroyed flag is set, we shouldnt be running" ) ;
}
2018-09-23 19:46:40 +02:00
this . stopCssWatcher ( ) ;
2018-08-30 00:56:15 +02:00
return ;
}
2018-05-16 23:26:47 +02:00
2019-06-12 23:55:15 +02:00
let cssValid = true ;
2018-08-21 00:48:15 +02:00
2018-05-16 23:26:47 +02:00
// first, a quick test:
2019-06-12 23:55:15 +02:00
cssValid &= this . currentVideoSettings . validFor . width === this . conf . player . dimensions . width ;
if ( cssValid ) {
2019-08-23 02:25:48 +02:00
cssValid &= this . video . classList . contains ( this . userCssClassName ) ;
2019-06-12 23:55:15 +02:00
}
if ( cssValid && this . currentPlayerStyleString ) { // only check for changes to player element if we applied them before
const playerStyleString = this . player . element . getAttribute ( 'style' ) ;
cssValid &= this . currentPlayerStyleString === playerStyleString ;
}
if ( ! cssValid ) {
if ( Debug . debug && Debug . resizer ) {
2019-02-16 01:19:29 +01:00
console . log ( ` %c[Resizer::cssCheck] <rid: ${ this . resizerId } > something touched our style string. We need to re-apply the style. ` , { background : '#ddf' , color : '#007' } ) ;
}
2018-08-30 00:56:15 +02:00
this . restore ( ) ;
this . scheduleCssWatcher ( 10 ) ;
2019-06-12 23:55:15 +02:00
return ;
2018-08-30 00:56:15 +02:00
}
2018-08-21 00:48:15 +02:00
if ( this . cssWatcherIncreasedFrequencyCounter > 0 ) {
-- this . cssWatcherIncreasedFrequencyCounter ;
2019-05-07 23:40:13 +02:00
this . scheduleCssWatcher ( 20 ) ;
2018-08-21 00:48:15 +02:00
} else {
this . scheduleCssWatcher ( 1000 ) ;
2018-02-05 22:46:38 +01:00
}
}
}
2018-12-31 01:03:07 +01:00
export default Resizer ;