diff --git a/src/ext/lib/video-data/VideoData.js b/src/ext/lib/video-data/VideoData.js index 25eb991..47e78a6 100644 --- a/src/ext/lib/video-data/VideoData.js +++ b/src/ext/lib/video-data/VideoData.js @@ -277,6 +277,54 @@ class VideoData { return a < b + diff && a > b - diff } + /** + * Gets the contents of the style attribute of the video element + * in a form of an object. + */ + getVideoStyle() { + // This will _always_ give us an array. Empty string gives an array + // that contains one element. That element is an empty string. + const styleArray = (this.video.getAttribute('style') || '').split(';'); + + const styleObject = {}; + + for (const style of styleArray) { + // not a valid CSS, so we skip those + if (style.indexOf(':') === -1) { + continue; + } + + // let's play _very_ safe + let {property, value} = style.split('!important')[0].split(':'); + value = value.trim(); + styleObject[property] = value; + } + + return styleObject; + } + + /** + * Some sites try to accomodate ultrawide users by "cropping" videos + * by setting 'style' attribute of the video element to 'height: X%', + * where 'X' is something greater than 100. + * + * This function gets that percentage and converts it into a factor. + */ + getHeightCompensationFactor() { + const heightStyle = this.getVideoStyle()?.height; + + if (!heightStyle || !heightStyle.endsWith('%')) { + return 1; + } + + const heightCompensationFactor = heightStyle.split('%')[0] / 100; + if (isNaN(heightCompensationFactor)) { + return 1; + } + return heightCompensationFactor; + } + + firstTimeArdInit(){ if(this.destroyed || this.invalid) { // throw {error: 'VIDEO_DATA_DESTROYED', data: {videoData: this}}; diff --git a/src/ext/lib/video-transform/Resizer.js b/src/ext/lib/video-transform/Resizer.js index af4f8cb..eec7daa 100644 --- a/src/ext/lib/video-transform/Resizer.js +++ b/src/ext/lib/video-transform/Resizer.js @@ -177,43 +177,41 @@ class Resizer { this.conf.pageInfo.updateCurrentCrop(ar); } - 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 - if (siteSettings?.autoarPreventConditions?.videoStyleString) { - const styleString = (this.video.getAttribute('style') || '').split(';'); + // 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 + // if (siteSettings?.autoarPreventConditions?.videoStyleString) { + // const styleString = (this.video.getAttribute('style') || '').split(';'); - 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)) { + // 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 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) { - this.logger.log('error', 'debug', "%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. - this.logger.log('info', 'debug', "%c[Resizer::setAr] video style contains forbidden css property: ", "color: #900, background: #100", prop, " — we aren't allowed to start autoar.") - return; - } - } - } - } - } - } - } - - + // // 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) { + // this.logger.log('error', 'debug', "%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. + // this.logger.log('info', 'debug', "%c[Resizer::setAr] video style contains forbidden css property: ", "color: #900, background: #100", prop, " — we aren't allowed to start autoar.") + // return; + // } + // } + // } + // } + // } + // } + // } if (lastAr) { this.lastAr = this.calculateRatioForLegacyOptions(lastAr); @@ -229,11 +227,11 @@ class Resizer { this.lastAr = {type: ar.type, ratio: ar.ratio} } - if (this.extensionMode === ExtensionMode.Basic && !PlayerData.isFullScreen() && ar.type !== AspectRatio.Reset) { - // 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; - } + // if (this.extensionMode === ExtensionMode.Basic && !PlayerData.isFullScreen() && ar.type !== AspectRatio.Reset) { + // // 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; + // } if (! this.video) { this.conf.destroy(); @@ -259,7 +257,6 @@ class Resizer { var stretchFactors = this.scaler.calculateCrop(ar); - if(! stretchFactors || stretchFactors.error){ this.logger.log('error', 'debug', `[Resizer::setAr] failed to set AR due to problem with calculating crop. Error:`, stretchFactors?.error); if (stretchFactors?.error === 'no_video'){ @@ -297,7 +294,7 @@ class Resizer { var stretchFactors = this.stretcher.calculateBasicStretch(); this.logger.log('info', 'debug', '[Resizer::setAr] Processed stretch factors for basic stretch. Stretch factors are:', stretchFactors); } else { - var stretchFactors = {xFactor: 1, yFactor: 1} + var stretchFactors = {xFactor: 1, yFactor: 1}; this.logger.log('error', 'debug', '[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); } diff --git a/src/ext/lib/video-transform/Scaler.js b/src/ext/lib/video-transform/Scaler.js index d846274..552a0e0 100644 --- a/src/ext/lib/video-transform/Scaler.js +++ b/src/ext/lib/video-transform/Scaler.js @@ -66,6 +66,28 @@ class Scaler { } calculateCrop(ar) { + /** + * STEP 1: NORMALIZE ASPECT RATIO + * + * Video width is normalized based on 100% of the parent. That means if the player AR + * is narrower than video ar, we need to pre-downscale the video. This scaling already + * undoes any zoom that style="height:123%" on the video element adds. + * + * Quick notes: + * * when I say 'video AR', I actually mean aspect ratio after we've compensated for + * any possible 'height:' stuffs in the style attribute of the video tag + * * because video width is normalized on 100% of the parent, we don't need to correct + * anything when the player is wider than the video. + */ + const streamAr = this.conf.video.videoWidth / this.conf.video.videoHeight; + const playerAr = this.conf.player.dimensions.width / this.conf.player.dimensions.height; + const compensatedStreamAr = streamAr * this.conf.videoData.getHeightCompensationFactor(); + + let arCorrectionFactor = 1; + if (playerAr < compensatedStreamAr) { + arCorrectionFactor = this.conf.player.dimensions.width / this.conf.video.offsetWidth; + } + if(!this.conf.video){ this.logger.log('info', 'debug', "[Scaler::calculateCrop] ERROR — no video detected. Conf:", this.conf, "video:", this.conf.video, "video dimensions:", this.conf.video && this.conf.video.videoWidth, '×', this.conf.video && this.conf.video.videoHeight); @@ -81,7 +103,7 @@ class Scaler { } if (ar.type === AspectRatio.Reset){ - return {xFactor: 1, yFactor: 1} + return {xFactor: arCorrectionFactor, yFactor: arCorrectionFactor} } // handle fuckie-wuckies @@ -102,15 +124,13 @@ class Scaler { // Dejansko razmerje stranic datoteke/