Autodetection persists over multiple videos

This commit is contained in:
Tamius Han 2018-05-16 23:26:47 +02:00
parent 82abee42bd
commit 4fe5ce6bcb
7 changed files with 169 additions and 60 deletions

View File

@ -41,7 +41,7 @@ var ExtensionConf = {
ignoreEdgeMargin: 0.20, // we ignore anything that pokes over the black line this close to the edge ignoreEdgeMargin: 0.20, // we ignore anything that pokes over the black line this close to the edge
// (relative to width of the sample) // (relative to width of the sample)
imageTestTreshold: 0.1, // when testing for image, this much pixels must be over blackbarTreshold imageTestTreshold: 0.1, // when testing for image, this much pixels must be over blackbarTreshold
edgeTolerancePx: 3, // black edge violation is performed this far from reported 'last black pixel' edgeTolerancePx: 2, // black edge violation is performed this far from reported 'last black pixel'
edgeTolerancePercent: null // unused. same as above, except use % of canvas height instead of pixels edgeTolerancePercent: null // unused. same as above, except use % of canvas height instead of pixels
}, },
fallbackMode: { fallbackMode: {

View File

@ -1,6 +1,7 @@
class VideoData { class VideoData {
constructor(video){ constructor(video){
this.arSetupComplete = false;
this.video = video; this.video = video;
// POZOR: VRSTNI RED JE POMEMBEN (arDetect mora bit zadnji) // POZOR: VRSTNI RED JE POMEMBEN (arDetect mora bit zadnji)
@ -14,8 +15,17 @@ class VideoData {
// this.player.dimensions // this.player.dimensions
} }
firstTimeArdInit(){
if(! this.arSetupComplete){
this.arDetector = new ArDetector(this);
}
}
initArDetection() { initArDetection() {
if(ths.arDetector)
this.arDetector.init(); this.arDetector.init();
else
this.arDetector = new ArDetector(this);
} }
startArDetection() { startArDetection() {
@ -23,7 +33,15 @@ class VideoData {
} }
destroy() { destroy() {
if(this.arDetector){
this.arDetector.stop(); this.arDetector.stop();
this.arDetector.destroy();
}
this.arDetector = null;
if(this.resizer){
this.resizer.destroy();
}
this.video = null;
} }
setLastAr(lastAr){ setLastAr(lastAr){

View File

@ -28,6 +28,10 @@ class ArDetector {
this.setup(ExtensionConf.arDetect.hSamples, ExtensionConf.arDetect.vSamples); this.setup(ExtensionConf.arDetect.hSamples, ExtensionConf.arDetect.vSamples);
} }
destroy(){
this.debugCanvas.destroy();
}
setup(cwidth, cheight){ setup(cwidth, cheight){
if(Debug.debug){ if(Debug.debug){
console.log("[ArDetect::setup] Starting autodetection setup"); console.log("[ArDetect::setup] Starting autodetection setup");
@ -73,7 +77,7 @@ class ArDetector {
if(this.video.videoWidth === 0 || this.video.videoHeight === 0 ){ if(this.video.videoWidth === 0 || this.video.videoHeight === 0 ){
if(! this.timer) if(! this.timer)
this.setupTimer = setTimeout(_arSetup, 100); this.setupTimer = setTimeout(this.init(), 100);
return; return;
} }
@ -145,6 +149,8 @@ class ArDetector {
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});
} }
this.conf.arSetupComplete = true;
} }
start(){ start(){
@ -332,8 +338,8 @@ class ArDetector {
if(! this.video){ if(! this.video){
if(Debug.debug || Debug.warnings_critical) if(Debug.debug || Debug.warnings_critical)
console.log("[ArDetect::_ard_vdraw] Video went missing. Stopping current instance of automatic detection.") console.log("[ArDetect::_ard_vdraw] Video went missing. Destroying current instance of videoData.")
this.stop(); this.conf.destroy();
return; return;
} }
@ -350,8 +356,8 @@ class ArDetector {
var how_far_treshold = 8; // how much can the edge pixel vary (*4) var how_far_treshold = 8; // how much can the edge pixel vary (*4)
if(this.video == null || this.video.ended ){ if(this.video.ended ){
// we slow down if ended or null. Detecting is pointless. // we slow down if ended. Detecting is pointless.
this.scheduleFrameCheck(ExtensionConf.arDetect.timer_paused); this.scheduleFrameCheck(ExtensionConf.arDetect.timer_paused);
return false; return false;
@ -502,7 +508,7 @@ class ArDetector {
// let's chec if we ever reset CSS. If we haven't, then we do so. If we did, then we don't. // let's chec if we ever reset CSS. If we haven't, then we do so. If we did, then we don't.
// while resetting the CSS, we also reset guardline top and bottom back to null. // while resetting the CSS, we also reset guardline top and bottom back to null.
if(! GlobalVars.arDetect.noLetterboxCanvasReset){ if(! this.noLetterboxCanvasReset){
this.conf.resizer.reset({type: "auto", ar: null}); this.conf.resizer.reset({type: "auto", ar: null});
this.guardLine.top = null; this.guardLine.top = null;
this.guardLine.bottom = null; this.guardLine.bottom = null;
@ -518,7 +524,7 @@ class ArDetector {
// css resetiral enkrat na video/pageload namesto vsakič, ko so za nekaj časa obrobe odstranejene // css resetiral enkrat na video/pageload namesto vsakič, ko so za nekaj časa obrobe odstranejene
// 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)
GlobalVars.arDetect.noLetterboxCanvasReset = false; this.noLetterboxCanvasReset = false;
// let's do a quick test to see if we're on a black frame // let's do a quick test to see if we're on a black frame
// TODO: reimplement but with less bullshit // TODO: reimplement but with less bullshit
@ -542,7 +548,7 @@ class ArDetector {
if (fallbackMode && guardLineOut.blackbarFail) { if (fallbackMode && guardLineOut.blackbarFail) {
this.conf.resizer.reset({type: "auto", ar: null}); this.conf.resizer.reset({type: "auto", ar: null});
this.guardLine.reset(); this.guardLine.reset();
this.arDetect.noLetterboxCanvasReset = true; this.noLetterboxCanvasReset = true;
triggerTimeout = this.getTimeout(baseTimeout, startTime); triggerTimeout = this.getTimeout(baseTimeout, startTime);
this.scheduleFrameCheck(triggerTimeout); //no letterbox, no problem this.scheduleFrameCheck(triggerTimeout); //no letterbox, no problem
@ -565,7 +571,7 @@ class ArDetector {
// 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.
try{
if(guardLineOut.blackbarFail || guardLineOut.imageFail){ if(guardLineOut.blackbarFail || guardLineOut.imageFail){
if(this.edgeDetector.findBars(image, null, EdgeDetectPrimaryDirection.HORIZONTAL).status === 'ar_known'){ if(this.edgeDetector.findBars(image, null, EdgeDetectPrimaryDirection.HORIZONTAL).status === 'ar_known'){
@ -573,7 +579,7 @@ class ArDetector {
console.log("[ArDetect::_ard_vdraw] Detected blackbar violation and pillarbox. Resetting to default aspect ratio."); console.log("[ArDetect::_ard_vdraw] Detected blackbar violation and pillarbox. Resetting to default aspect ratio.");
} }
if(! guardLineResult){ if(guardLineOut.blackbarFail){
this.conf.resizer.reset({type: "auto", ar: null}); this.conf.resizer.reset({type: "auto", ar: null});
this.guardLine.reset(); this.guardLine.reset();
} }
@ -584,6 +590,7 @@ class ArDetector {
return; return;
} }
} }
}catch(e) {console.log("deeee",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.

View File

@ -34,6 +34,10 @@ class DebugCanvas {
console.log("debug canvas is:", this.canvas, "context:", this.context) console.log("debug canvas is:", this.canvas, "context:", this.context)
} }
destroy(){
this.canvas.remove();
}
setBuffer(buffer) { setBuffer(buffer) {
// this.imageBuffer = buffer.splice(0); // this.imageBuffer = buffer.splice(0);
this.imageBuffer = new Uint8ClampedArray(buffer); this.imageBuffer = new Uint8ClampedArray(buffer);
@ -77,8 +81,8 @@ class DebugCanvas {
} }
DebugCanvasClasses = { DebugCanvasClasses = {
VIOLATION: {color: '#ff4400', colorRgb: [255, 68, 0], text: 'violation (general)'}, VIOLATION: {color: '#ff0000', colorRgb: [255, 00, 0], text: 'violation (general)'},
WARN: {color: '#d08539', colorRgb: [208, 133, 57], text: 'lesser violation (general)'}, WARN: {color: '#d0d039', colorRgb: [208, 208, 57], text: 'lesser violation (general)'},
GUARDLINE_BLACKBAR: {color: '#3333FF', colorRgb: [51, 51, 255], text: 'guardline/blackbar (expected value)'}, GUARDLINE_BLACKBAR: {color: '#3333FF', colorRgb: [51, 51, 255], text: 'guardline/blackbar (expected value)'},
GUARDLINE_IMAGE: {color: '#000088', colorRgb: [0, 0, 136], text: 'guardline/image (expected value)'}, GUARDLINE_IMAGE: {color: '#000088', colorRgb: [0, 0, 136], text: 'guardline/image (expected value)'},

View File

@ -12,33 +12,69 @@ class PageInfo {
} }
rescan(count){ rescan(count){
try{
var vids = document.getElementsByTagName('video'); var vids = document.getElementsByTagName('video');
if(!vids || vids.length == 0){ if(!vids || vids.length == 0){
this.hasVideos = false; this.hasVideos = false;
this.scheduleRescan();
return; return;
} }
// debugger; // debugger;
// add new videos // add new videos
for(var video of vids){ // for(var video of vids){
var existing = this.videos.find( (x) => { // var existing = this.videos.find( (x) => {
if (x == video.video) // if (video && x == video.video)
return x; // return x;
if (x.currentSrc == video.video.currentSrc){ // if (video && x.currentSrc == video.video.currentSrc){
return x; // return x;
} // }
}) // })
if(existing){ // if(existing){
video.video = existing; // video.video = existing;
// } else {
// this.videos.push(
// new VideoData(video)
// );
// }
// }
if(this.videos.length > 0){
if(vids[0] == this.videos[0].video){
// do nothing
} else { } else {
this.videos.push( this.videos[0].destroy();
new VideoData(video) this.videos[0] = new VideoData(vids[0]);
);
} }
} else {
this.videos.push(new VideoData(vids[0]));
} }
console.log("Rescan complete. Total videos?", this.videos.length)
}catch(e){
console.log("rescan error:",e)
}
this.scheduleRescan();
}
scheduleRescan(){
try{
if(this.rescanTimer){
clearTimeout(this.rescanTimer);
}
var ths = this;
this.rescanTimer = setTimeout(function(){
ths.rescanTimer = null;
ths.rescan();
ths = null;
}, 1000)
}catch(e){console.log("eee",e)}
} }
initArDetection(){ initArDetection(){

View File

@ -30,8 +30,13 @@ class Resizer {
this.restore_wd = false; this.restore_wd = false;
this.lastAr = {type: 'original'}; this.lastAr = {type: 'original'};
this.destroyed = false;
} }
destroy(){
this.destroyed = true;
this.stopCssWatcher();
}
setAr(ar, lastAr){ setAr(ar, lastAr){
if(Debug.debug){ if(Debug.debug){
@ -50,15 +55,17 @@ class Resizer {
if (! this.video) { if (! this.video) {
// console.log("No video detected.") // console.log("No video detected.")
// this.videoData.destroy(); this.videoData.destroy();
} }
var dimensions = Scaler.calculateCrop(ar, this.video, this.conf.player.dimensions); var dimensions = Scaler.calculateCrop(ar, this.video, this.conf.player.dimensions);
if(dimensions.error){ if(! dimensions || dimensions.error){
if(Debug.debug){ if(Debug.debug){
console.log("[Resizer::setAr] failed to set AR due to problem with calculating crop. Error:", dimensions.error) console.log("[Resizer::setAr] failed to set AR due to problem with calculating crop. Error:", (dimensions ? dimensions.error : dimensions));
}
if(dimensions.error === 'no_video'){
this.conf.destroy();
} }
return; return;
} }
@ -75,6 +82,9 @@ class Resizer {
var cssOffsets = this.computeOffsets(dimensions); var cssOffsets = this.computeOffsets(dimensions);
this.applyCss(cssOffsets, stretchFactors); this.applyCss(cssOffsets, stretchFactors);
if(! this.destroyed)
this.startCssWatcher();
} }
setLastAr(override){ setLastAr(override){
@ -101,10 +111,18 @@ class Resizer {
} }
startCssWatcher(){ startCssWatcher(){
this.cssWatcherTimeout = setInterval(this.cssWatcher, 200); // this.haltCssWatcher = false;
if(!this.cssWatcherTimeout){
if(Debug.debug)
console.log("[Resizer.js] STARTING CSS WATCHER")
this.cssWatcherTimeout = setInterval(this.cssWatcher, 200, this);
}
} }
stopCssWatcher(){ stopCssWatcher(){
if(Debug.debug) console.log("[Resizer.js] STOPPING CSS WATCHER!")
clearInterval(this.cssWatcherTimeout); clearInterval(this.cssWatcherTimeout);
} }
@ -190,12 +208,20 @@ class Resizer {
applyCss(dimensions, stretchFactors){ applyCss(dimensions, stretchFactors){
if(this.video == undefined || this.video == null){ if (! this.video ){
if(Debug.debug) if(Debug.debug)
console.log("[Resizer::_res_applyCss] Video went missing, doing nothing."); console.log("[Resizer::_res_applyCss] Video went missing, doing nothing.");
this.conf.destroy();
return; return;
} }
// 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
}
if(Debug.debug) if(Debug.debug)
console.log("[Resizer::_res_applyCss] Starting to apply css. this is what we're getting in:", dimensions); console.log("[Resizer::_res_applyCss] Starting to apply css. this is what we're getting in:", dimensions);
@ -284,6 +310,9 @@ class Resizer {
styleString += styleArray[i] + "; "; styleString += styleArray[i] + "; ";
this.setStyleString(styleString); this.setStyleString(styleString);
} }
setStyleString (styleString, count = 0) { setStyleString (styleString, count = 0) {
@ -322,21 +351,31 @@ class Resizer {
} }
} }
cssWatcher(){ cssWatcher(ths){
// this means we haven't set our CSS yet, or that we changed video. // this means we haven't set our CSS yet, or that we changed video.
if(! this.currentCss.top) if(! ths.currentCss.top)
return; return;
// this means video went missing. // this means video went missing. videoData will be re-initialized when the next video is found
if(! this.video) if(! ths.video){
ths.conf.destroy();
return; return;
}
// // our current css is fucky? Null, undefined and 0 are invalid values. // // our current css is fucky? Null, undefined and 0 are invalid values.
// if(! GlobalVars.currentCss.width || ! GlobalVars.currentCss.height ) // if(! GlobalVars.currentCss.width || ! GlobalVars.currentCss.height )
// return; // return;
var styleArrayStr = this.video.getAttribute('style'); // first, a quick test:
if (ths.currentVideoSettings.validFor == ths.conf.player.dimensions ){
if (ths.currentVideoSettings.videoWidth != ths.video.offsetWidth ||
ths.currentVideoSettings.videoHeight != ths.video.offsetHeight){
ths.restore();
return;
}
}
var styleArrayStr = ths.video.getAttribute('style');
if (styleArrayStr){ if (styleArrayStr){
var styleArray = styleArrayStr.split(";"); var styleArray = styleArrayStr.split(";");
@ -349,24 +388,24 @@ class Resizer {
if (styleArray[i].startsWith("top:")){ if (styleArray[i].startsWith("top:")){
// don't force css restore if currentCss.top is not defined // don't force css restore if currentCss.top is not defined
if(this.currentCss.top && styleArray[i] != this.currentCss.top){ if(ths.currentCss.top && styleArray[i] != ths.currentCss.top){
if(Debug.debug){ if(Debug.debug){
console.log("[Resizer::_res_antiCssOverride] SOMEBODY TOUCHED MA SPAGHETT (our CSS got overriden, restoring our css)"); console.log("[Resizer::_res_antiCssOverride] SOMEBODY TOUCHED MA SPAGHETT (our CSS got overriden, restoring our css)");
console.log("[Resizer::_res_antiCssOverride] MA SPAGHETT: top:", this.currentCss.top, "left:", this.currentCss.left, "thing that touched ma spaghett", styleArrayStr); console.log("[Resizer::_res_antiCssOverride] MA SPAGHETT: top:", ths.currentCss.top, "left:", ths.currentCss.left, "thing that touched ma spaghett", styleArrayStr);
} }
_res_restore(); ths.restore();
return; return;
} }
stuffChecked++; stuffChecked++;
} }
else if(styleArray[i].startsWith("left:")){ else if(styleArray[i].startsWith("left:")){
// don't force css restore if currentCss.left is not defined // don't force css restore if currentCss.left is not defined
if(this.currentCss.left && styleArray[i] != this.currentCss.left){ if(ths.currentCss.left && styleArray[i] != ths.currentCss.left){
if(Debug.debug){ if(Debug.debug){
console.log("[Resizer::_res_antiCssOverride] SOMEBODY TOUCHED MA SPAGHETT (our CSS got overriden, restoring our css)"); console.log("[Resizer::_res_antiCssOverride] SOMEBODY TOUCHED MA SPAGHETT (our CSS got overriden, restoring our css)");
console.log("[Resizer::_res_antiCssOverride] MA SPAGHETT: width:", this.currentCss.width, "height:", this.currentCss.height, "thing that touched ma spaghett", styleArrayStr); console.log("[Resizer::_res_antiCssOverride] MA SPAGHETT: width:", ths.currentCss.width, "height:", ths.currentCss.height, "thing that touched ma spaghett", styleArrayStr);
} }
_res_restore(); ths.restore();
return; return;
} }
stuffChecked++; stuffChecked++;

View File

@ -57,21 +57,7 @@ class Scaler {
} }
static calculateCrop(mode, video, playerDimensions) { static calculateCrop(mode, video, playerDimensions) {
// if 'ar' is string, we'll handle that in legacy wrapper
var ar = 0;
if(isNaN(mode)){
ar = this.modeToAr(mode);
} else {
ar = mode;
}
// handle fuckie-wuckies
if (! ar){
return null;
}
if(Debug.debug)
console.log("[Scaler::calculateCrop] trying to set ar. args are: ar->",ar,"; playerDimensions->",playerDimensions.width, "×", playerDimensions.height, "| obj:", playerDimensions);
if(!video || video.videoWidth == 0 || video.videoHeight == 0){ if(!video || video.videoWidth == 0 || video.videoHeight == 0){
if(Debug.debug) if(Debug.debug)
@ -79,6 +65,25 @@ class Scaler {
return {error: "no_video"}; return {error: "no_video"};
} }
// if 'ar' is string, we'll handle that in legacy wrapper
var ar = 0;
if(isNaN(mode)){
ar = Scaler.modeToAr(mode);
} else {
ar = mode;
}
// handle fuckie-wuckies
if (! ar){
if(Debug.debug)
console.log("[Scaler::calculateCrop] no ar?", ar, " -- we were given this mode:", mode);
return {error: "no_ar", ar: ar};
}
if(Debug.debug)
console.log("[Scaler::calculateCrop] trying to set ar. args are: ar->",ar,"; playerDimensions->",playerDimensions.width, "×", playerDimensions.height, "| obj:", playerDimensions);
if( (! playerDimensions) || playerDimensions.width === 0 || playerDimensions.height === 0 ){ if( (! playerDimensions) || playerDimensions.width === 0 || playerDimensions.height === 0 ){
if(Debug.debug) if(Debug.debug)
console.log("[Scaler::calculateCrop] ERROR — no (or invalid) playerDimensions:",playerDimensions); console.log("[Scaler::calculateCrop] ERROR — no (or invalid) playerDimensions:",playerDimensions);