ultrawidify/src/ext/lib/video-transform/Scaler.js

156 lines
6.4 KiB
JavaScript
Raw Normal View History

import Debug from '../../conf/Debug';
2019-03-10 23:27:50 +01:00
import AspectRatio from '../../../common/enums/aspect-ratio.enum';
// računa velikost videa za približevanje/oddaljevanje
// does video size calculations for zooming/cropping
class Scaler {
// internal variables
// functions
2019-09-03 22:55:10 +02:00
constructor(videoData) {
2018-05-18 23:26:20 +02:00
this.conf = videoData;
2019-09-03 22:55:10 +02:00
this.logger = videoData.logger;
}
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
modeToAr (ar) {
if (ar.type !== AspectRatio.FitWidth && ar.type !== AspectRatio.FitHeight && ar.ratio) {
2019-03-10 23:27:50 +01:00
return ar.ratio;
}
2019-03-10 23:27:50 +01:00
var ratioOut;
if (!this.conf.video) {
this.logger.log('error', 'debug', "[Scaler.js::modeToAr] No video??",this.conf.video, "killing videoData");
2018-05-18 23:26:20 +02:00
this.conf.destroy();
return null;
}
2020-09-20 12:26:03 +02:00
if (!this.conf.player.dimensions) {
2019-03-10 23:27:50 +01:00
ratioOut = screen.width / screen.height;
2020-09-20 12:26:03 +02:00
} else {
2019-03-10 23:27:50 +01:00
ratioOut = this.conf.player.dimensions.width / this.conf.player.dimensions.height;
}
2018-08-20 22:45:43 +02:00
// 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.
//
2018-08-20 22:45:43 +02:00
// 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;
2020-09-20 12:26:03 +02:00
if (ar.type === AspectRatio.FitWidth) {
2019-03-10 23:27:50 +01:00
ratioOut > fileAr ? ratioOut : fileAr
ar.ratio = ratioOut;
return ratioOut;
}
2020-09-20 12:26:03 +02:00
else if (ar.type === AspectRatio.FitHeight) {
2019-03-10 23:27:50 +01:00
ratioOut < fileAr ? ratioOut : fileAr
ar.ratio = ratioOut;
return ratioOut;
}
2020-09-20 12:26:03 +02:00
else if (ar.type === AspectRatio.Reset) {
this.logger.log('info', 'debug', "[Scaler.js::modeToAr] Using original aspect ratio -", fileAr)
2019-03-10 23:27:50 +01:00
ar.ar = fileAr;
return fileAr;
}
return null;
}
2019-03-10 23:27:50 +01:00
calculateCrop(ar) {
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);
2018-05-18 23:26:20 +02:00
this.conf.destroy();
return {error: "no_video"};
}
if (this.conf.video.videoWidth == 0 || this.conf.video.videoHeight == 0) {
// that's illegal, but not illegal enough to just blast our shit to high hell
// mr officer will let you go with a warning this time around
this.logger.log('error', 'debug', "[Scaler::calculateCrop] Video has illegal dimensions. Video dimensions:", this.conf.video && this.conf.video.videoWidth, '×', this.conf.video && this.conf.video.videoHeight);
return {error: "illegal_video_dimensions"};
}
2019-03-10 23:27:50 +01:00
if (ar.type === AspectRatio.Reset){
return {xFactor: 1, yFactor: 1}
}
// handle fuckie-wuckies
if (!ar.ratio){
this.logger.log('error', 'scaler', "[Scaler::calculateCrop] no ar?", ar.ratio, " -- we were given this mode:", ar);
2019-04-25 22:02:10 +02:00
return {error: "no_ar", ratio: ar.ratio};
}
this.logger.log('info', 'scaler', "[Scaler::calculateCrop] trying to set ar. args are: ar->",ar.ratio,"; this.conf.player.dimensions->",this.conf.player.dimensions.width, "×", this.conf.player.dimensions.height, "| obj:", this.conf.player.dimensions);
if( (! this.conf.player.dimensions) || this.conf.player.dimensions.width === 0 || this.conf.player.dimensions.height === 0 ){
this.logger.log('error', 'scaler', "[Scaler::calculateCrop] ERROR — no (or invalid) this.conf.player.dimensions:",this.conf.player.dimensions);
return {error: "this.conf.player.dimensions_error"};
}
// zdaj lahko končno začnemo računati novo velikost videa
// we can finally start computing required video dimensions now:
// Dejansko razmerje stranic datoteke/<video> značke
// Actual aspect ratio of the file/<video> tag
var fileAr = this.conf.video.videoWidth / this.conf.video.videoHeight;
var playerAr = this.conf.player.dimensions.width / this.conf.player.dimensions.height;
if (ar.type === AspectRatio.Initial || !ar.ratio) {
2019-04-25 22:02:10 +02:00
ar.ratio = fileAr;
}
this.logger.log('info', 'scaler', "[Scaler::calculateCrop] ar is " ,ar.ratio, ", file ar is", fileAr, ", this.conf.player.dimensions are ", this.conf.player.dimensions.width, "×", this.conf.player.dimensions.height, "| obj:", this.conf.player.dimensions);
var videoDimensions = {
xFactor: 1,
yFactor: 1,
actualWidth: 0, // width of the video (excluding pillarbox) when <video> tag height is equal to width
actualHeight: 0, // height of the video (excluding letterbox) when <video> tag height is equal to height
}
2019-10-27 22:11:07 +01:00
if (fileAr < playerAr) {
if (fileAr < ar.ratio){
// in this situation we have to crop letterbox on top/bottom of the player
// we cut it, but never more than the player
2019-04-25 22:02:10 +02:00
videoDimensions.xFactor = Math.min(ar.ratio, playerAr) / fileAr;
videoDimensions.yFactor = videoDimensions.xFactor;
2019-10-27 22:11:07 +01:00
} else {
// in this situation, we would be cutting pillarbox. Inside horizontal player.
// I don't think so. Except exceptions, we'll wait for bug reports.
videoDimensions.xFactor = 1;
videoDimensions.yFactor = 1;
}
} else {
if (fileAr < ar.ratio || playerAr < ar.ratio){
2019-10-27 22:11:07 +01:00
// in this situation, we need to add extra letterbox on top of our letterbox
// this means we simply don't crop anything _at all_
videoDimensions.xFactor = 1;
videoDimensions.yFactor = 1;
2019-10-27 22:11:07 +01:00
} else {
// meant for handling pillarbox crop. not quite implemented.
videoDimensions.xFactor = fileAr / Math.min(ar.ratio, playerAr);
videoDimensions.yFactor = videoDimensions.xFactor;
// videoDimensions.xFactor = Math.max(ar.ratio, playerAr) * fileAr;
// videoDimensions.yFactor = videoDimensions.xFactor;
2019-10-27 22:11:07 +01:00
}
}
this.logger.log('info', 'scaler', "[Scaler::calculateCrop] Crop factor calculated — ", videoDimensions.xFactor);
return videoDimensions;
}
}
export default Scaler;