This commit is contained in:
Tamius Han 2021-09-19 21:22:12 +02:00
parent 5b1f73a69e
commit 502ce707e1

View File

@ -114,9 +114,9 @@ class ArDetector {
* We get one animation frame per this many ms. This means that our autodetection * We get one animation frame per this many ms. This means that our autodetection
* stuff must run in less than this many ms. This valuz is averaged out over multiple * stuff must run in less than this many ms. This valuz is averaged out over multiple
* samples for better accuracy. * samples for better accuracy.
* *
* Returns value in ms. * Returns value in ms.
* *
* A very important caveat: if autodetection takes up too much time, it WILL artificially * A very important caveat: if autodetection takes up too much time, it WILL artificially
* increase time budget. Therefore, you should use (and firstly even implement) getTimeBudget() * increase time budget. Therefore, you should use (and firstly even implement) getTimeBudget()
* that turns off autodetection for a second or so to gather accurate timing info. * that turns off autodetection for a second or so to gather accurate timing info.
@ -146,7 +146,7 @@ class ArDetector {
this.conf = videoData; this.conf = videoData;
this.video = videoData.video; this.video = videoData.video;
this.settings = videoData.settings; this.settings = videoData.settings;
this.sampleCols = []; this.sampleCols = [];
this.blackLevel = this.settings.active.arDetect.blackbar.blackLevel; this.blackLevel = this.settings.active.arDetect.blackbar.blackLevel;
@ -193,7 +193,7 @@ class ArDetector {
this.edgeDetector = new EdgeDetect(this); this.edgeDetector = new EdgeDetect(this);
// this.debugCanvas = new DebugCanvas(this); // this.debugCanvas = new DebugCanvas(this);
// //
// [1] initiate canvases // [1] initiate canvases
// //
@ -210,7 +210,7 @@ class ArDetector {
this.blackframeCanvas.remove(); this.blackframeCanvas.remove();
} }
// things to note: we'll be keeping canvas in memory only. // things to note: we'll be keeping canvas in memory only.
this.canvas = document.createElement("canvas"); this.canvas = document.createElement("canvas");
this.canvas.width = cwidth; this.canvas.width = cwidth;
this.canvas.height = cheight; this.canvas.height = cheight;
@ -233,10 +233,10 @@ class ArDetector {
let ncol = this.settings.active.arDetect.sampling.staticCols; let ncol = this.settings.active.arDetect.sampling.staticCols;
let nrow = this.settings.active.arDetect.sampling.staticRows; let nrow = this.settings.active.arDetect.sampling.staticRows;
let colSpacing = this.canvas.width / ncol; let colSpacing = this.canvas.width / ncol;
let rowSpacing = (this.canvas.height << 2) / nrow; let rowSpacing = (this.canvas.height << 2) / nrow;
this.sampleLines = []; this.sampleLines = [];
this.sampleCols = []; this.sampleCols = [];
@ -276,7 +276,7 @@ class ArDetector {
} catch (e) { } catch (e) {
this.canDoFallbackMode = false; this.canDoFallbackMode = false;
} }
// //
// [5] do other things setup needs to do // [5] do other things setup needs to do
// //
@ -289,12 +289,12 @@ class ArDetector {
this.canvasImageDataRowLength = cwidth << 2; this.canvasImageDataRowLength = cwidth << 2;
this.noLetterboxCanvasReset = false; this.noLetterboxCanvasReset = false;
if (this.settings.canStartAutoAr() ) { if (this.settings.canStartAutoAr() ) {
// this.main(); // this.main();
this.start(); this.start();
} }
if(Debug.debugCanvas.enabled){ if(Debug.debugCanvas.enabled){
// this.debugCanvas.init({width: cwidth, height: cheight}); // this.debugCanvas.init({width: cwidth, height: cheight});
// DebugCanvas.draw("test marker","test","rect", {x:5, y:5}, {width: 5, height: 5}); // DebugCanvas.draw("test marker","test","rect", {x:5, y:5}, {width: 5, height: 5});
@ -324,7 +324,7 @@ class ArDetector {
this.conf.resizer.setLastAr({type: AspectRatioType.Automatic, ratio: this.defaultAr}); this.conf.resizer.setLastAr({type: AspectRatioType.Automatic, ratio: this.defaultAr});
} }
this._paused = false; this._paused = false;
this._halted = false; this._halted = false;
this._exited = false; this._exited = false;
@ -342,13 +342,16 @@ class ArDetector {
} }
this.animationFrameHandle = window.requestAnimationFrame( (ts) => this.animationFrameBootstrap(ts)); this.animationFrameHandle = window.requestAnimationFrame( (ts) => this.animationFrameBootstrap(ts));
this.logger.log('info', 'debug', `"%c[ArDetect::startLoop] <@${this.arid}> AARD loop started.`, _ard_console_start);
} }
stop() { stop() {
if (this.animationFrameHandle) { if (this.animationFrameHandle) {
this.logger.log('info', 'debug', `"%c[ArDetect::stop] <@${this.arid}> Stopping AnimationFrame loop.`, _ard_console_stop);
window.cancelAnimationFrame(this.animationFrameHandle); window.cancelAnimationFrame(this.animationFrameHandle);
} else {
this.logger.log('info', 'debug', `"%c[ArDetect::stop] <@${this.arid}> AnimationFrame loop is already paused (due to an earlier call of this function).`);
} }
this.logger.log('info', 'debug', `"%c[ArDetect::stop] <@${this.arid}> Stopping AnimationFrame loop.`, _ard_console_stop);
} }
unpause() { unpause() {
@ -405,7 +408,7 @@ class ArDetector {
/** /**
* Checks whether conditions for granting a frame check are fulfilled * Checks whether conditions for granting a frame check are fulfilled
* @returns * @returns
*/ */
private canTriggerFrameCheck() { private canTriggerFrameCheck() {
if (this._paused || this._halted || this._exited) { if (this._paused || this._halted || this._exited) {
@ -415,6 +418,7 @@ class ArDetector {
// if video was paused & we know that we already checked that frame, // if video was paused & we know that we already checked that frame,
// we will not check it again. // we will not check it again.
const videoState = this.getVideoPlaybackState(); const videoState = this.getVideoPlaybackState();
if (videoState !== VideoPlaybackState.Playing) { if (videoState !== VideoPlaybackState.Playing) {
if (this.status.lastVideoStatus === videoState) { if (this.status.lastVideoStatus === videoState) {
return false; return false;
@ -435,10 +439,10 @@ class ArDetector {
timeout = 100; timeout = 100;
} }
// don't allow more than 1 instance // don't allow more than 1 instance
if(this.setupTimer){ if(this.setupTimer){
clearTimeout(this.setupTimer); clearTimeout(this.setupTimer);
} }
let ths = this; let ths = this;
this.setupTimer = setTimeout(function(){ this.setupTimer = setTimeout(function(){
ths.setupTimer = null; ths.setupTimer = null;
@ -475,8 +479,8 @@ class ArDetector {
/** /**
* Adds execution time sample for performance metrics * Adds execution time sample for performance metrics
* @param performanceObject * @param performanceObject
* @param executionTime * @param executionTime
*/ */
private addPerformanceTimeMeasure(performanceObject, executionTime) { private addPerformanceTimeMeasure(performanceObject, executionTime) {
performanceObject.sampleTime[performanceObject.currentIndex] = executionTime; performanceObject.sampleTime[performanceObject.currentIndex] = executionTime;
@ -489,19 +493,19 @@ class ArDetector {
/** /**
* Returns time ultrawidify spends on certain aspects of autodetection. * Returns time ultrawidify spends on certain aspects of autodetection.
* *
* The returned object contains the following: * The returned object contains the following:
* *
* eyeballedTimeBudget a very inaccurate time budget * eyeballedTimeBudget a very inaccurate time budget
* fps framerate at which we run * fps framerate at which we run
* aardTime time spent on average frameCheck loop. * aardTime time spent on average frameCheck loop.
* It's a nearly useless metric, because * It's a nearly useless metric, because
* frameCheck can exit very early. * frameCheck can exit very early.
* drawWindowTime how much time browser spends on executing * drawWindowTime how much time browser spends on executing
* drawWindow() calls. * drawWindow() calls.
* getImageData how much time browser spends on executing * getImageData how much time browser spends on executing
* getImageData() calls. * getImageData() calls.
* *
* Most of these are on "per frame" basis and averaged. * Most of these are on "per frame" basis and averaged.
*/ */
getTimings() { getTimings() {
@ -528,12 +532,16 @@ class ArDetector {
* This is the "main loop" for aspect ratio autodetection * This is the "main loop" for aspect ratio autodetection
*/ */
private async animationFrameBootstrap(timestamp: number) { private async animationFrameBootstrap(timestamp: number) {
// this.logger.log('info', 'arDetect_verbose', `[ArDetect::animationFrameBootstrap] <@${this.arid}> New animation frame.\nmanualTickEnabled: ${!this.manualTickEnabled}\ncan trigger frame check? ${this.canTriggerFrameCheck()}\nnext tick? ${this._nextTick}\n => (a&b | c) => Can we do tick? ${ (!this.manualTickEnabled && this.canTriggerFrameCheck()) || this._nextTick}\n\ncan we continue running? ${this && !this._halted && !this._paused}`);
// do timekeeping first // do timekeeping first
this.addPerformanceTimeMeasure(this.performance.animationFrame, timestamp - this.performance.animationFrame.lastTime); this.addPerformanceTimeMeasure(this.performance.animationFrame, timestamp - this.performance.animationFrame.lastTime);
this.performance.animationFrame.lastTime = timestamp; this.performance.animationFrame.lastTime = timestamp;
// trigger frame check, if we're allowed to // trigger frame check, if we're allowed to
if ( (!this.manualTickEnabled && this.canTriggerFrameCheck()) || this._nextTick) { if ( (!this.manualTickEnabled && this.canTriggerFrameCheck()) || this._nextTick) {
this.logger.log('info', 'arDetect_verbose', `[ArDetect::animationFrameBootstrap] <@${this.arid}> Processing next tick.`);
this._nextTick = false; this._nextTick = false;
try { try {
@ -541,17 +549,17 @@ class ArDetector {
await this.frameCheck(); await this.frameCheck();
this.addPerformanceTimeMeasure(this.performance.aard, performance.now() - startTime); this.addPerformanceTimeMeasure(this.performance.aard, performance.now() - startTime);
} catch (e) { } catch (e) {
this.logger.log('error', 'debug', `%c[ArDetect::main] <@${this.arid}> Frame check failed:`, "color: #000, background: #f00", e); this.logger.log('error', 'debug', `%c[ArDetect::animationFrameBootstrap] <@${this.arid}> Frame check failed:`, "color: #000, background: #f00", e);
} }
} }
if (this && !this._halted && !this._paused) { if (this && !this._halted && !this._paused) {
this.animationFrameHandle = window.requestAnimationFrame( (ts) => this.animationFrameBootstrap(ts)); this.animationFrameHandle = window.requestAnimationFrame( (ts) => this.animationFrameBootstrap(ts));
} else if (this._halted) { } else if (this._halted) {
this.logger.log('info', 'debug', `%c[ArDetect::main] <@${this.arid}> Main autodetection loop exited. Halted? ${this._halted}`, _ard_console_stop); this.logger.log('info', 'debug', `%c[ArDetect::animationFrameBootstrap] <@${this.arid}> Main autodetection loop exited. Halted? ${this._halted}`, _ard_console_stop);
this._exited = true; this._exited = true;
} else { } else {
this.logger.log('info', 'debug', `[ArDetect::main] <@${this.arid}> Not renewing animation frame for some reason. Paused? ${this._paused}; Halted?: ${this._halted}, Exited?: ${this._exited}`); this.logger.log('info', 'debug', `[ArDetect::animationFrameBootstrap] <@${this.arid}> Not renewing animation frame for some reason. Paused? ${this._paused}; Halted?: ${this._halted}, Exited?: ${this._exited}`);
} }
} }
@ -565,11 +573,11 @@ class ArDetector {
} }
let letterbox = edges.top + edges.bottom; let letterbox = edges.top + edges.bottom;
if (! this.fallbackMode) { if (! this.fallbackMode) {
// Since video is stretched to fit the canvas, we need to take that into account when calculating target // Since video is stretched to fit the canvas, we need to take that into account when calculating target
// aspect ratio and correct our calculations to account for that // aspect ratio and correct our calculations to account for that
const fileAr = this.video.videoWidth / this.video.videoHeight; const fileAr = this.video.videoWidth / this.video.videoHeight;
const canvasAr = this.canvas.width / this.canvas.height; const canvasAr = this.canvas.width / this.canvas.height;
@ -590,7 +598,7 @@ class ArDetector {
// fallback mode behaves a wee bit differently // fallback mode behaves a wee bit differently
let zoomFactor = 1; let zoomFactor = 1;
// there's stuff missing from the canvas. We need to assume canvas' actual height is bigger by a factor x, where // there's stuff missing from the canvas. We need to assume canvas' actual height is bigger by a factor x, where
// x = [video.zoomedHeight] / [video.unzoomedHeight] // x = [video.zoomedHeight] / [video.unzoomedHeight]
// //
@ -598,7 +606,7 @@ class ArDetector {
// letterbox += [video.zoomedHeight] - [video.unzoomedHeight] // letterbox += [video.zoomedHeight] - [video.unzoomedHeight]
let vbr = this.video.getBoundingClientRect(); let vbr = this.video.getBoundingClientRect();
zoomFactor = vbr.height / this.video.clientHeight; zoomFactor = vbr.height / this.video.clientHeight;
letterbox += vbr.height - this.video.clientHeight; letterbox += vbr.height - this.video.clientHeight;
@ -608,7 +616,7 @@ class ArDetector {
this.logger.log('info', 'arDetect', `%c[ArDetect::calculateArFromEdges] <@${this.arid}> Edge is in the no-trigger zone. Aspect ratio change is not triggered.`) this.logger.log('info', 'arDetect', `%c[ArDetect::calculateArFromEdges] <@${this.arid}> Edge is in the no-trigger zone. Aspect ratio change is not triggered.`)
return; return;
} }
// varnostno območje, ki naj ostane črno (da lahko v fallback načinu odkrijemo ožanje razmerja stranic). // varnostno območje, ki naj ostane črno (da lahko v fallback načinu odkrijemo ožanje razmerja stranic).
// x2, ker je safetyBorderPx definiran za eno stran. // x2, ker je safetyBorderPx definiran za eno stran.
// safety border so we can detect aspect ratio narrowing (21:9 -> 16:9). // safety border so we can detect aspect ratio narrowing (21:9 -> 16:9).
@ -625,7 +633,7 @@ class ArDetector {
return; return;
} }
this.detectedAr = trueAr; this.detectedAr = trueAr;
// poglejmo, če se je razmerje stranic spremenilo // poglejmo, če se je razmerje stranic spremenilo
// check if aspect ratio is changed: // check if aspect ratio is changed:
let lastAr = this.conf.resizer.getLastAr(); let lastAr = this.conf.resizer.getLastAr();
@ -634,14 +642,14 @@ class ArDetector {
// že nastavili. // že nastavili.
// //
// we can only deny aspect ratio changes if we use automatic mode and if aspect ratio was set from here. // we can only deny aspect ratio changes if we use automatic mode and if aspect ratio was set from here.
let arDiff = trueAr - lastAr.ratio; let arDiff = trueAr - lastAr.ratio;
if (arDiff < 0) if (arDiff < 0)
arDiff = -arDiff; arDiff = -arDiff;
const arDiff_percent = arDiff / trueAr; const arDiff_percent = arDiff / trueAr;
// ali je sprememba v mejah dovoljenega? Če da -> fertik // ali je sprememba v mejah dovoljenega? Če da -> fertik
// is ar variance within acceptable levels? If yes -> we done // is ar variance within acceptable levels? If yes -> we done
this.logger.log('info', 'arDetect', `%c[ArDetect::processAr] <@${this.arid}> New aspect ratio varies from the old one by this much:\n`,"color: #aaf","old Ar", lastAr.ratio, "current ar", trueAr, "arDiff (absolute):",arDiff,"ar diff (relative to new ar)", arDiff_percent); this.logger.log('info', 'arDetect', `%c[ArDetect::processAr] <@${this.arid}> New aspect ratio varies from the old one by this much:\n`,"color: #aaf","old Ar", lastAr.ratio, "current ar", trueAr, "arDiff (absolute):",arDiff,"ar diff (relative to new ar)", arDiff_percent);
@ -653,7 +661,7 @@ class ArDetector {
this.logger.log('info', 'arDetect', `%c[ArDetect::processAr] <@${this.arid}> aspect ratio change accepted — diff %: ${arDiff_percent}`, "background: #153; color: #4f9"); this.logger.log('info', 'arDetect', `%c[ArDetect::processAr] <@${this.arid}> aspect ratio change accepted — diff %: ${arDiff_percent}`, "background: #153; color: #4f9");
} }
this.logger.log('info', 'debug', `%c[ArDetect::processAr] <@${this.arid}> Triggering aspect ratio change. New aspect ratio: ${trueAr}`, _ard_console_change); this.logger.log('info', 'debug', `%c[ArDetect::processAr] <@${this.arid}> Triggering aspect ratio change. New aspect ratio: ${trueAr}`, _ard_console_change);
this.conf.resizer.updateAr({type: AspectRatioType.Automatic, ratio: trueAr}); this.conf.resizer.updateAr({type: AspectRatioType.Automatic, ratio: trueAr});
} }
@ -676,7 +684,7 @@ class ArDetector {
if (!this.blackframeContext) { if (!this.blackframeContext) {
this.init(); this.init();
} }
let startTime; let startTime;
let partialDrawImageTime = 0; let partialDrawImageTime = 0;
let sampleCols = this.sampleCols.slice(0); let sampleCols = this.sampleCols.slice(0);
@ -687,7 +695,7 @@ class ArDetector {
try { try {
startTime = performance.now(); startTime = performance.now();
// do it in ghetto async. This way, other javascript function should be able to // do it in ghetto async. This way, other javascript function should be able to
// get a chance to do something _before_ we process our data // get a chance to do something _before_ we process our data
await new Promise<void>( await new Promise<void>(
resolve => { resolve => {
@ -712,10 +720,10 @@ class ArDetector {
if (! this.canvasReadyForDrawWindow()) { if (! this.canvasReadyForDrawWindow()) {
// this means canvas needs to be resized, so we'll just re-run setup with all those new parameters // this means canvas needs to be resized, so we'll just re-run setup with all those new parameters
this.halt(); this.halt();
let newCanvasWidth = window.innerHeight * (this.video.videoWidth / this.video.videoHeight); let newCanvasWidth = window.innerHeight * (this.video.videoWidth / this.video.videoHeight);
let newCanvasHeight = window.innerHeight; let newCanvasHeight = window.innerHeight;
if (this.conf.resizer.videoAlignment === VideoAlignmentType.Center) { if (this.conf.resizer.videoAlignment === VideoAlignmentType.Center) {
this.canvasDrawWindowHOffset = Math.round((window.innerWidth - newCanvasWidth) * 0.5); this.canvasDrawWindowHOffset = Math.round((window.innerWidth - newCanvasWidth) * 0.5);
} else if (this.conf.resizer.videoAlignment === VideoAlignmentType.Left) { } else if (this.conf.resizer.videoAlignment === VideoAlignmentType.Left) {
@ -725,9 +733,9 @@ class ArDetector {
} }
this.setup(newCanvasWidth, newCanvasHeight); this.setup(newCanvasWidth, newCanvasHeight);
return; return;
} }
// if this is the case, we'll first draw on canvas, as we'll need intermediate canvas if we want to get a // if this is the case, we'll first draw on canvas, as we'll need intermediate canvas if we want to get a
// smaller sample for blackframe check // smaller sample for blackframe check
this.fallbackMode = true; this.fallbackMode = true;
@ -740,10 +748,12 @@ class ArDetector {
return; // it's prolly just a fluke, so we do nothing special here return; // it's prolly just a fluke, so we do nothing special here
} }
// draw blackframe sample from our main sample: // draw blackframe sample from our main sample:
await new Promise<void>(resolve => { await new Promise<void>(
this.blackframeContext.drawImage(this.canvas, this.blackframeCanvas.width, this.blackframeCanvas.height); resolve => {
resolve(); this.blackframeContext.drawImage(this.canvas, this.blackframeCanvas.width, this.blackframeCanvas.height);
}); resolve();
}
);
partialDrawImageTime += performance.now() - startTime; partialDrawImageTime += performance.now() - startTime;
this.logger.log('info', 'arDetect_verbose', `%c[ArDetect::frameCheck] canvas.drawImage seems to have worked`, "color:#000; backgroud:#2f5;"); this.logger.log('info', 'arDetect_verbose', `%c[ArDetect::frameCheck] canvas.drawImage seems to have worked`, "color:#000; backgroud:#2f5;");
@ -760,10 +770,12 @@ class ArDetector {
// if we are in normal mode though, the frame has yet to be drawn // if we are in normal mode though, the frame has yet to be drawn
if (!this.fallbackMode) { if (!this.fallbackMode) {
startTime = performance.now(); startTime = performance.now();
await new Promise<void>(resolve => { await new Promise<void>(
this.context.drawImage(this.video, 0, 0, this.canvas.width, this.canvas.height); resolve => {
resolve(); this.context.drawImage(this.video, 0, 0, this.canvas.width, this.canvas.height);
}) resolve();
}
)
partialDrawImageTime += performance.now() - startTime; partialDrawImageTime += performance.now() - startTime;
} }
@ -793,20 +805,20 @@ class ArDetector {
// if we look further we need to reset this value back to false. Otherwise we'll only get CSS reset once // if we look further we need to reset this value back to false. Otherwise we'll only get CSS reset once
// per video/pageload instead of every time letterbox goes away (this can happen more than once per vid) // per video/pageload instead of every time letterbox goes away (this can happen more than once per vid)
this.noLetterboxCanvasReset = false; this.noLetterboxCanvasReset = false;
// poglejmo, če obrežemo preveč. // poglejmo, če obrežemo preveč.
// let's check if we're cropping too much // let's check if we're cropping too much
const guardLineOut = this.guardLine.check(imageData, this.fallbackMode); const guardLineOut = this.guardLine.check(imageData, this.fallbackMode);
// če ni padla nobena izmed funkcij, potem se razmerje stranic ni spremenilo // če ni padla nobena izmed funkcij, potem se razmerje stranic ni spremenilo
// if both succeed, then aspect ratio hasn't changed. // if both succeed, then aspect ratio hasn't changed.
if (!guardLineOut.imageFail && !guardLineOut.blackbarFail) { if (!guardLineOut.imageFail && !guardLineOut.blackbarFail) {
this.logger.log('info', 'arDetect_verbose', `%c[ArDetect::frameCheck] guardLine tests were successful. (no imagefail and no blackbarfail)\n`, "color: #afa", guardLineOut); this.logger.log('info', 'arDetect_verbose', `%c[ArDetect::frameCheck] guardLine tests were successful. (no imagefail and no blackbarfail)\n`, "color: #afa", guardLineOut);
this.clearImageData(imageData); this.clearImageData(imageData);
return; return;
} }
// drugače nadaljujemo, našemu vzorcu stolpcev pa dodamo tiste stolpce, ki so // drugače nadaljujemo, našemu vzorcu stolpcev pa dodamo tiste stolpce, ki so
// kršili blackbar (če obstajajo) ter jih razvrstimo // kršili blackbar (če obstajajo) ter jih razvrstimo
// otherwise we continue. We add blackbar violations to the list of the cols // otherwise we continue. We add blackbar violations to the list of the cols
// we'll sample and sort them // we'll sample and sort them
@ -815,7 +827,7 @@ class ArDetector {
(a: number, b: number) => a - b (a: number, b: number) => a - b
); );
} }
// if we're in fallback mode and blackbar test failed, we restore CSS and quit // if we're in fallback mode and blackbar test failed, we restore CSS and quit
// (since the new letterbox edge isn't present in our sample due to technical // (since the new letterbox edge isn't present in our sample due to technical
// limitations) // limitations)
@ -828,8 +840,8 @@ class ArDetector {
this.clearImageData(imageData); this.clearImageData(imageData);
return; return;
} }
// If aspect ratio changes from narrower to wider, we first check for presence of pillarbox. Presence of pillarbox indicates // If aspect ratio changes from narrower to wider, we first check for presence of pillarbox. Presence of pillarbox indicates
// a chance of a logo on black background. We could cut easily cut too much. Because there's a somewhat significant chance // a chance of a logo on black background. We could cut easily cut too much. Because there's a somewhat significant chance
// that we will cut too much, we rather avoid doing anything at all. There's gonna be a next chance. // that we will cut too much, we rather avoid doing anything at all. There's gonna be a next chance.
@ -842,26 +854,26 @@ class ArDetector {
this.guardLine.reset(); this.guardLine.reset();
} else { } else {
this.logger.log('info', 'arDetect_verbose', `[ArDetect::frameCheck] Guardline failed, blackbar didn't, and we got pillarbox. Doing nothing.`); this.logger.log('info', 'arDetect_verbose', `[ArDetect::frameCheck] Guardline failed, blackbar didn't, and we got pillarbox. Doing nothing.`);
} }
this.clearImageData(imageData); this.clearImageData(imageData);
return; return;
} }
} }
} catch(e) { } catch(e) {
this.logger.log('info', 'arDetect', `[ArDetect::frameCheck] something went wrong while checking for pillarbox. Error:\n`, e); this.logger.log('info', 'arDetect', `[ArDetect::frameCheck] something went wrong while checking for pillarbox. Error:\n`, e);
} }
// pa poglejmo, kje se končajo črne letvice na vrhu in na dnu videa. // pa poglejmo, kje se končajo črne letvice na vrhu in na dnu videa.
// let's see where black bars end. // let's see where black bars end.
this.sampleCols_current = sampleCols.length; this.sampleCols_current = sampleCols.length;
// blackSamples -> {res_top, res_bottom} // blackSamples -> {res_top, res_bottom}
let edgePost = this.edgeDetector.findBars(imageData, sampleCols, EdgeDetectPrimaryDirection.Vertical, EdgeDetectQuality.Improved, guardLineOut, bfanalysis); let edgePost = this.edgeDetector.findBars(imageData, sampleCols, EdgeDetectPrimaryDirection.Vertical, EdgeDetectQuality.Improved, guardLineOut, bfanalysis);
this.logger.log('info', 'arDetect_verbose', `%c[ArDetect::frameCheck] edgeDetector returned this\n`, "color: #aaf", edgePost); this.logger.log('info', 'arDetect_verbose', `%c[ArDetect::frameCheck] edgeDetector returned this\n`, "color: #aaf", edgePost);
if (edgePost.status !== EdgeStatus.ARKnown){ if (edgePost.status !== EdgeStatus.ARKnown){
// rob ni bil zaznan, zato ne naredimo ničesar. // rob ni bil zaznan, zato ne naredimo ničesar.
// no edge was detected. Let's leave things as they were // no edge was detected. Let's leave things as they were
@ -872,12 +884,12 @@ class ArDetector {
} }
let newAr = this.calculateArFromEdges(edgePost); let newAr = this.calculateArFromEdges(edgePost);
this.logger.log('info', 'arDetect_verbose', `%c[ArDetect::frameCheck] Triggering aspect ration change! new ar: ${newAr}`, "color: #aaf"); this.logger.log('info', 'arDetect_verbose', `%c[ArDetect::frameCheck] Triggering aspect ration change! new ar: ${newAr}`, "color: #aaf");
// we also know edges for guardline, so set them. // we also know edges for guardline, so set them.
// we need to be mindful of fallbackMode though // we need to be mindful of fallbackMode though
// if edges are okay and not invalid, we also // if edges are okay and not invalid, we also
// allow automatic aspect ratio correction. If edges // allow automatic aspect ratio correction. If edges
// are bogus, we remain aspect ratio unchanged. // are bogus, we remain aspect ratio unchanged.
try { try {
@ -898,16 +910,16 @@ class ArDetector {
this.processAr(newAr); this.processAr(newAr);
} catch (e) { } catch (e) {
// edges weren't gucci, so we'll just reset // edges weren't gucci, so we'll just reset
// the aspect ratio to defaults // the aspect ratio to defaults
this.logger.log('error', 'arDetect', `%c[ArDetect::frameCheck] There was a problem setting blackbar. Doing nothing. Error:`, e); this.logger.log('error', 'arDetect', `%c[ArDetect::frameCheck] There was a problem setting blackbar. Doing nothing. Error:`, e);
try { try {
this.guardLine.reset(); this.guardLine.reset();
} catch (e) { } catch (e) {
// no guardline, no bigge // no guardline, no bigge
} }
// WE DO NOT RESET ASPECT RATIO HERE IN CASE OF PROBLEMS, CAUSES UNWARRANTED RESETS: // WE DO NOT RESET ASPECT RATIO HERE IN CASE OF PROBLEMS, CAUSES UNWARRANTED RESETS:
// (eg. here: https://www.youtube.com/watch?v=nw5Z93Yt-UQ&t=410) // (eg. here: https://www.youtube.com/watch?v=nw5Z93Yt-UQ&t=410)
// //
// this.conf.resizer.setAr({type: AspectRatioType.Automatic, ratio: this.defaultAr}); // this.conf.resizer.setAr({type: AspectRatioType.Automatic, ratio: this.defaultAr});
@ -917,7 +929,7 @@ class ArDetector {
} }
resetBlackLevel(){ resetBlackLevel(){
this.blackLevel = this.settings.active.arDetect.blackbar.blackLevel; this.blackLevel = this.settings.active.arDetect.blackbar.blackLevel;
} }
blackLevelTest_full() { blackLevelTest_full() {
@ -943,7 +955,7 @@ class ArDetector {
let blackPixelCount = 0; let blackPixelCount = 0;
const bfImageData = this.blackframeContext.getImageData(0, 0, cols, rows).data; const bfImageData = this.blackframeContext.getImageData(0, 0, cols, rows).data;
const blackTreshold = this.blackLevel + this.settings.active.arDetect.blackbar.frameThreshold; const blackTreshold = this.blackLevel + this.settings.active.arDetect.blackbar.frameThreshold;
// we do some recon for letterbox and pillarbox. While this can't determine whether letterbox/pillarbox exists // we do some recon for letterbox and pillarbox. While this can't determine whether letterbox/pillarbox exists
// with sufficient level of certainty due to small sample resolution, it can still give us some hints for later // with sufficient level of certainty due to small sample resolution, it can still give us some hints for later
@ -972,7 +984,7 @@ class ArDetector {
max_g = max_g > bfImageData[i+1] ? max_g : bfImageData[i+1]; max_g = max_g > bfImageData[i+1] ? max_g : bfImageData[i+1];
max_b = max_b > bfImageData[i+2] ? max_b : bfImageData[i+2]; max_b = max_b > bfImageData[i+2] ? max_b : bfImageData[i+2];
} }
r = ~~(i/rows); r = ~~(i/rows);
c = i % cols; c = i % cols;
@ -1056,19 +1068,19 @@ class ArDetector {
/** /**
* Does a quick test to see if the aspect ratio is correct * Does a quick test to see if the aspect ratio is correct
* Returns 'true' if there's a chance of letterbox existing, false if not. * Returns 'true' if there's a chance of letterbox existing, false if not.
* @param imageData * @param imageData
* @param sampleCols * @param sampleCols
* @returns * @returns
*/ */
fastLetterboxPresenceTest(imageData, sampleCols) { fastLetterboxPresenceTest(imageData, sampleCols) {
// fast test to see if aspect ratio is correct. // fast test to see if aspect ratio is correct.
// returns 'true' if presence of letterbox is possible. // returns 'true' if presence of letterbox is possible.
// returns 'false' if we found a non-black edge pixel. // returns 'false' if we found a non-black edge pixel.
// If we detect anything darker than blackLevel, we modify blackLevel to the new lowest value // If we detect anything darker than blackLevel, we modify blackLevel to the new lowest value
const rowOffset = this.canvas.width * (this.canvas.height - 1); const rowOffset = this.canvas.width * (this.canvas.height - 1);
let currentMin = 255, currentMax = 0, colOffset_r, colOffset_g, colOffset_b, colOffset_rb, colOffset_gb, colOffset_bb, blthreshold = this.settings.active.arDetect.blackbar.threshold; let currentMin = 255, currentMax = 0, colOffset_r, colOffset_g, colOffset_b, colOffset_rb, colOffset_gb, colOffset_bb, blthreshold = this.settings.active.arDetect.blackbar.threshold;
// detect black level. if currentMax comes above blackbar + blackbar threshold, we know we aren't letterboxed // detect black level. if currentMax comes above blackbar + blackbar threshold, we know we aren't letterboxed
for (let i = 0; i < sampleCols.length; ++i){ for (let i = 0; i < sampleCols.length; ++i){