PHP Classes

PHP Image Correllation: Get the similarity between images with correlation

Recommend this page to a friend!
  Info   Example   Screenshots   View files Files   Install with Composer Install with Composer   Download Download   Reputation   Support forum   Blog    
Ratings Unique User Downloads Download Rankings
Not enough user ratingsTotal: 52 All time: 10,661 This week: 524Up
Version License PHP version Categories
correlimages 1.0.0Free for non-comm...5Algorithms, PHP 5, Graphics
Description 

Author

This package can get the similarity between images with correlation.

It provides a script that serves JavaScript code to perform image analysis calculations to compare two images and determine their similarity when calculating correlation coefficients.

The correlation coefficient is calculated using a number between 0 and 100 to show how similar the two images are.

Innovation Award
PHP Programming Innovation award nominee
August 2023
Number 2
Comparing images can be a valuable activity for many types of applications.

For instance, an application can determine if an image is mostly a copy of another embodiment.

Another application can compare two movie images to determine if there are significant changes between the two scenes.

This package implements an image comparison approach calculating a correlation factor using JavaScript and PHP code.

Manuel Lemos
Picture of Dantigny francois
  Performance   Level  
Name: Dantigny francois <contact>
Classes: 3 packages by
Country: France France
Age: ???
All time rank: 341489 in France France
Week rank: 180 Up12 in France France Up
Innovation award
Innovation award
Nominee: 2x

Winner: 1x

Example

<?php
//----------------------------------------
// versionning
// CorrelImages : 1.6 : 2023/02/06
// rajout image dans partie documentation pour mieux illustrer (une image vaut 1000 mots)
// CorrelImages : 1.5 : 2023/02/03
// partie "Help" détaillée avec le processus global et les paramètres
// CorrelImages : 1.4 : 2023/02/03
// rajout du nombre de tests sauvegardé dans localStorage (après bouton effacer le localStorage)
// CorrelImages : 1.3 : 2023/02/02
// lègère modification pour prise en compte (couleurs) du cas ou Nombres de Matches insuffisant
// CorrelImages : 1.2 : 2023/01/31
// CorrelImages : 1.0 : 2023/01/29
// CorrelImages : 0.1 : 2023/01/27
// fichier PHP type pour une application
//----------------------------------------


ini_set("default_charset", 'utf-8');
ini_set("display_errors", 1);
ini_set("error_reporting", E_ALL | E_STRICT);
date_default_timezone_set('Europe/Paris');
ini_set("always_populate_raw_post_data", "-1");
set_time_limit(0);
session_start();

/* fonction pour debugging */
function file_ecrit($filename,$data){ // pour gestion des erreurs côté serveur
   
if($fp = fopen($filename,'a')){ // mode ajout !!
       
$ok = fwrite($fp,$data);
       
fclose($fp);
        return
$ok;
    }
    else return
false;
}

/* Le compteur */
 
$db = new PDO('sqlite:compteur.sqlite');
 
$pdo_result=$db->query("SELECT nb FROM compte WHERE seul=1");
  if (!
$pdo_result){
     
$sqlreq = 'CREATE TABLE "compte" ( "nb" INTEGER DEFAULT (0), "seul" INTEGER DEFAULT (1)) WITHOUT ROWID';
     
$db->exec($sqlreq);
  } else {
     
$pdo_result->setFetchMode(PDO::FETCH_BOTH);
     
$lastnb=$pdo_result->fetchColumn();
     
$pdo_result->closeCursor();
     
$lastnb=intval($lastnb);
  }

 
/* gestion des cookies, et donc du compteur ! */
 
if(!@$_COOKIE["correlimages"]) { //absence de cookie
    
setcookie("correlimages","nimp",time()+(60*60*24*90),"/correlimages/");
    
$cook = "nimp"; // cookie par défaut
     // Incrémente le compteur
    
$lastnb++;
    
//file_ecrit('compte.txt','**** (+1='.$lastnb.")"."\n");
     //Met à jour le compteur
    
$pdo_result=$db->exec("UPDATE compte set nb=".$lastnb." WHERE seul=1");
    
//file_ecrit('compte.txt','**** yes! '.$lastnb." le ".date("d/m/y-H:i:s")."\n");
 
} else { // sinon, on le récupère !
     //file_ecrit('compte.txt','**** ne compte pas ! '.$lastnb." le ".date("d/m/y-H:i:s")."\n");
    
$cook=$_COOKIE['correlimages'];
  }
 
$db = Null;

/* variable pour affichage titre / version / nbre de vue */
 
$Versiondu = 'V 1.3 du 06/02/2023';

?>
<!DOCTYPE html>
    <html xmlns="http://www.w3.org/1999/xhtml" xml:lang="fr" lang="fr">

    <head>
        <meta http-equiv="X-UA-Compatible" content="IE=edge"/>
        <meta http-equiv="content-type" content="text/html; charset=UTF-8"/>
        <meta http-equiv="cache-control" content="no-cache, must-revalidate"/>
        <meta http-equiv="Pragma" content="no-cache"/>
        <meta http-equiv="Expires" content="0"/>
        <meta name="DC.Language" content="fr"/>
        <meta name="description" content="indice coefficient de corrélation entre deux images"/>
        <meta name="author" content="ArouG" />
        <meta name="keywords" content="corrélation images tailles différentes"/>
        <meta name="date" content="2023/01/27"/>
        <meta name="robots" content="nofollow"/>
        <title>Correlimages</title>
        <style>
        html {
            font: 1.1em sans-serif;
        }
       
        body {
            display: block;
            background-color: black;
            margin: 8px;
        }
       
        .top-box {
            width: 1400px;
            height: 18px;
            margin-bottom: -18px;
            position: relative;
        }
       
        #bidon {
            width: 1400px;
            height: 20px;
            background-color: #000000;
            float: left;
        }
       
        #entete {
            width: 1400px;
            margin-top: 0;
            margin-bottom: 0;
            margin-left: 0;
            margin-right: 0;
            line-height: 5px;
            background-color: #600c0c;
            padding-top: 0;
            z-index: 50;
        }
       
        #main, #help, #NameRes {
            min-height: 500px;
            margin-left: 20px;
            margin-right: 20px;
        }
        #help{
            text-align: left;
            font: 1.1em sans-serif;
        }
        #help h1{
            padding-left : 0px;
            font: 1.2em sans-serif;
            font-weight: bold;
        }
        #help h2{
            padding-left : 15px;
            font: 1.15em sans-serif;
            font-weight: bold;
        }
        #help p{
            padding-left : 30px;
        }

        #appelHelp {
            position : absolute;
            background-color : #600c0c;
            right:20px;
            top:20px;
            color : white;
        }

        #nomResult {
            position : absolute;
            background-color : #600c0c;
            left:20px;
            top:20px;
            color : white;
        }

        #cornleft {
            float: left;
            width: 150px;
            height: 75px;
            position: relative;
            background-color: #600c0c;
        }
       
        #titre {
            float: left;
            width: 1100px;
            position: relative;
            margin-top: 0;
            height: 75px;
            background-color: #600c0c;
        }

        #titre p {
            color: #f0e39e;
            font-family: Georgia, "Bitstream Vera Serif", Norasi, serif;
            font-size: 0.8em;
            font-style: italic;
            line-height: 0.2em;
        }
       
        #titre a {
            margin-top: 10px;
            color: white;
            font-family: Georgia, "Bitstream Vera Serif", Norasi, serif;
            font-style: italic;
            font-size: 1em;
            font-style: italic;
        }

        #titre h2 {
            color: #f0e39e;
            font-family: Georgia, "Bitstream Vera Serif", Norasi, serif;
            font-style: italic;
            font-size: 1.1em;
            font-style: italic;
            text-align: center;
        }
       
        #cornright {
            float: left;
            width: 150px;
            height: 75px;
            position: relative;
            background-color: #600c0c;
        }
       
        #menu {
            text-align: center;
            background-color: #FFDEAD;
            width: 1400px;
            margin: auto;
            padding: 0;
        }
        #visu{
            margin: auto;
        }
        #paramsTab, #outputsTab{
            padding-top: 20px;
            margin: auto;
            border-collapse: collapse;
            border: 1px solid black;
            text-align: center;
            font-size: 0.7rem;
        }
        td{
            border: 1px solid black;
        }
        #TH1inp, #TH2inp, #FCNB1inp, #FCNB2inp, #OSS1inp, #OSS2inp, #RanTinp {
            max-width:20px;
        }
        #MatchV, #RanItinp, #MatVinp, #RatioSinp{
            max-width:30px;
        }
        #TmsQG, #TmsT{
            background-color: #00FFF0;
        }
        .QCG {
            background-color: rgba(0, 220, 0, 1);
            color: white;
        }
        .bad{ /* pour QC et NbGM */
            background-color: rgba(255, 0, 0, 1);
            color: white;
        }
        .goodQC{
            background-color: rgba(0, 220, 0, 1);
            color: black;
        }
        .goodNbGM{
            background-color: #FFDEAD;
            color: black;
        }
        /*
        .infobulle{
            color:#C00;
            text-decoration:none;
            border-bottom: 1px dotted;
            font-weight: bold;
        } */
        #basdepage {
            margin: 0;
            padding: 0;
            font-size: 0.55em;
            background-color: #600c0c;
            width: 1400px;
            float: left;
        }
        #gauche {
            text-align: left;
            float: left;
        }
       
        #droite {
            text-align: right;
            float: left;
        }
       
        #centrebas {
            float: left;
            width: 1224px;
            text-align: center;
            margin: auto;
            padding: 0;
            color: white;
            font-family: Georgia, "Bitstream Vera Serif", Norasi, serif;
            font-style: italic;
            font-size: 18px;
            font-style: italic;
        }
       
        .styledvp {
            border: 0;
            line-height: 1.5;
            padding: 0;
            font-size: 1rem;
            text-align: center;
            color: #fff;
            min-width:100px;
            text-shadow: 1px 1px 1px #000;
            border-radius: 10px;
            background-color: rgba(0, 180, 0, 1);
            background-image: linear-gradient(to top left,
                                              rgba(0, 0, 0, .2),
                                              rgba(0, 0, 0, .2) 30%,
                                              rgba(0, 0, 0, 0));
            box-shadow: inset 2px 2px 3px rgba(255, 255, 255, .6),
                        inset -2px -2px 3px rgba(0, 0, 0, .6);
        }

        .styledvp:hover {
            background-color: rgba(0, 235, 0, 1);
        }
        .styledvp:focus{
            outline: 0px;
        }

        .styleddr {
            border: 0;
            line-height: 1.5;
            padding: 0;
            font-size: 1rem;
            text-align: center;
            color: #fff;
            min-width:80px;
            text-shadow: 1px 1px 1px #000;
            border-radius: 10px;
            background-color: rgba(220, 0, 0, 1);
            background-image: linear-gradient(to top left,
                                              rgba(0, 0, 0, .2),
                                              rgba(0, 0, 0, .2) 30%,
                                              rgba(0, 0, 0, 0));
            box-shadow: inset 2px 2px 3px rgba(255, 255, 255, .6),
                        inset -2px -2px 3px rgba(0, 0, 0, .6);
        }

        .styleddr:hover {
            background-color: rgba(255, 0, 0, 1);
        }
        .styleddr:focus{
            outline: 0px;
        }

        .centre{
            text-align: center;
            padding-left:270px;
        }

        #selectfics {
            text-align: center;
            margin-inline: auto;
            margin-top: 15px;
            margin-bottom: 15px;
        }

        </style>
        <!-- source JSFEAT : https://github.com/inspirit/jsfeat -->
        <!--script src="./js/jsfeat.js"></script-->
        <script src="./js/jsfeat-min.js"></script>
        <script src="./js/modernizr-custom.js"></script>
        <script>

        "use strict";

        // https://gist.github.com/juliocesar/926500#file-best-localstorage-polyfill-evar-js
        if (!Modernizr.localstorage) {
          window.localStorage = {
            _data : {},
            setItem : function(id, val) { return this._data[id] = String(val); },
            getItem : function(id) { return this._data.hasOwnProperty(id) ? this._data[id] : undefined; },
            removeItem : function(id) { return delete this._data[id]; },
            clear : function() { return this._data = {}; }
          };
        }

        var img1, img2;
        var myCanv, myCont, canvW, canvH, canv1W, canv1H, canv2W, canv2H, canv1L, canv1T, canv2L, canv2T, canvWmax, myCt1, myCt2, myC1, myC2;
        var descr1, descr2, corners1, corners2, ssc_key_points1, ssc_key_points2, corners_selected1, corners_selected2, matches, good_matches;
        var Parinit = {};
        var Result = {};
        var DatDeb;

        function storageAvailable(type) {
            try {
                var storage = window[type],
                    x = '__storage_test__';
                storage.setItem(x, x);
                storage.removeItem(x);
                return true;
            }
            catch(e) {
                return e instanceof DOMException && (
                    // everything except Firefox
                    e.code === 22 ||
                    // Firefox
                    e.code === 1014 ||
                    // test name field too, because code might not be present
                    // everything except Firefox
                    e.name === 'QuotaExceededError' ||
                    // Firefox
                    e.name === 'NS_ERROR_DOM_QUOTA_REACHED') &&
                    // acknowledge QuotaExceededError only if there's something already stored
                    storage.length !== 0;
            }
        }

        // for debug only : need datasave.php
        function DataToFile(filename, data) {
            var xhr = new XMLHttpRequest();
            var arr = new Uint8Array(data);
            xhr.open("post", "datasave.php", true); // Sync open file
            xhr.setRequestHeader("Content-Type", "application/octet-stream");
            xhr.setRequestHeader("X-FILE-NAME", filename); // file name
            xhr.setRequestHeader("X-FILE-SIZE", data.length); // file size
            xhr.send(data); // Send
        }

        async function downloadFile(data, name = 'file', type = 'text/plain') {
            // https://mindsers.blog/fr/telechargement-fichier-javascript/
            const { createElement } = document
            const { URL: { createObjectURL, revokeObjectURL }, setTimeout } = window

            const blob = new Blob([data], { type })
            const url = createObjectURL(blob)

            const anchor = document.createElement('a')
            anchor.setAttribute('href', url)
            anchor.setAttribute('download', name)
            anchor.click()
         
            setTimeout(() => { revokeObjectURL(url) }, 100)
        }

        async function downFile(){
            var auj = new Date();

            var dd = auj.getDate();
            var mm = auj.getMonth() + 1;
            var yyyy = auj.getFullYear();
            var hh = auj.getHours();
            var mn = auj.getMinutes();
            var ss = auj.getSeconds();
            if (dd < 10) {
                dd = '0' + dd;
            }
            if (mm < 10) {
                mm = '0' + mm;
            }
            if (hh < 10){
                hh = '0' + hh;
            }
            if (mn < 10){
                mn = '0' + mn;
            }
            if (ss < 10){
                ss = '0' + ss;
            }
            var name = 'CorrelImagesTest_'+yyyy+'_'+mm+'_'+dd+'_'+hh+'_'+mn+'_'+ss+'.csv';
            var ligne = localStorage.getItem('Correltest');
            await downloadFile(ligne, name);
        }

        var previewPicture = function(e) {
            if(e.files) {
                var pict = e.files[0];
                if((pict.type == 'image/jpeg') || (pict.type == 'image/gif') || (pict.type == 'image/png')) {
                    var pictId = e.id;
                    var reader = new FileReader();
                    reader.onload = function(e) {
                        if(pictId == 'inp1') {
                            document.getElementById("img1").src = e.target.result;
                        } else document.getElementById("img2").src = e.target.result;
                    }
                    reader.readAsDataURL(pict);
                } else alert('It is not an image file !');
            }
        }

        function help() {
            if(document.querySelector("#HelpButt").firstChild.textContent == 'Help') {
                document.querySelector("#HelpButt").firstChild.textContent = "Back";
                document.querySelector("#main").style.display = 'none';
                document.querySelector("#NameResButt").style.display = 'none';
                document.querySelector("#help").style.display = 'block';
            } else {
                document.querySelector("#HelpButt").firstChild.textContent = "Help";
                document.querySelector("#help").style.display = 'none';
                document.querySelector("#main").style.display = 'block';
                document.querySelector("#NameResButt").style.display = 'block';
            }
        }

        function NameRes() {
            if(document.querySelector("#NameResButt").firstChild.textContent == 'Sauvegarde') {
                document.querySelector("#NameResButt").firstChild.textContent = "Back";
                document.querySelector("#main").style.display = 'none';
                document.querySelector("#HelpButt").style.display = 'none';
                document.querySelector("#NameRes").style.display = 'block';
            } else {
                document.querySelector("#NameResButt").firstChild.textContent = "Sauvegarde";
                document.querySelector("#NameRes").style.display = 'none';
                document.querySelector("#main").style.display = 'block';
                document.querySelector("#HelpButt").style.display = 'block';
            }
        }
        // non zero bits count
        function popcnt32(n) {
            n -= ((n >> 1) & 0x55555555);
            n = (n & 0x33333333) + ((n >> 2) & 0x33333333);
            return(((n + (n >> 4)) & 0xF0F0F0F) * 0x1010101) >> 24;
        }

        // adapté from https://gist.github.com/nandor/7e74368a449924483173
        function my_ransac(matches, thres, iter) {
            var ret = {};
            var bd = 456555;
            //var good_matches=[];
            var count = 0;
            var best = {};
            var bestCnt = 0;
            // hypothèse : x2 = x0 + ax * x1; y2 = y0 + ay * y1 avec, normalement : x0 et y0 petits et ax=ay pas loin de 1 !
            // ax0*x0 + ax1*x1 + cx = 0 et ay0*y0+ay1*y1+cy = 0
            // 4 paramètres à déterminer 2 points sont nécessaires
            var ax0, ax1, cx, ay0, ay1, cy;
            for(var it = 0; it < iter; it += 1) {
                var i0 = ~~(Math.random() * (matches.length - 1)),
                    i1;
                do {
                    i1 = ~~(Math.random() * (matches.length - 1));
                } while (i1 == i0);
                var p0 = matches[i0],
                    p1 = matches[i1];
                // Calcule des 6 paramètres à partir des 2 points tirés au hasard
                var slope = (p1.x2 - p0.x2) / (p1.x1 - p0.x1);
                ax0 = -1.0;
                ax1 = 1.0 / slope;
                cx = p0.x1 - p0.x2 / slope;
                slope = (p1.y2 - p0.y2) / (p1.y1 - p0.y1);
                ay0 = -1.0;
                ay1 = 1.0 / slope;
                cy = p0.y1 - p0.y2 / slope;
                // Détermine le nombre de points correspondant au modèle basé sur les 2 points tirés au hasard
                count = 0;
                good_matches = [];
                for(var i = 0; i < matches.length; i += 1) {
                    var p = matches[i];
                    // valeurs théorique de x2, y2 soit donc x'2 et y'2
                    var xp2 = (p.x1 - cx) / ax1;
                    var yp2 = (p.y1 - cy) / ay1;
                    var dist = Math.sqrt((xp2 - p.x2) * (xp2 - p.x2) + (yp2 - p.y2) * (yp2 - p.y2));
                    if(dist < thres) {
                        if(dist < bd) {
                            bd = dist;
                        }
                        good_matches.push(matches[i]);
                        count += 1;
                    }
                }
                // Si on a trouvé plus de points, on sauvegarde les paramètres de notre nouveau modèle
                if(bestCnt < count) {
                    bestCnt = count;
                    best.ax0 = ax0;
                    best.ax1 = ax1;
                    best.cx = cx;
                    best.ay0 = ay0;
                    best.ay1 = ay1;
                    best.cy = cy;
                    best.cnt = count;
                    best.gm = good_matches;
                    best.dist = bd;
                }
            }
            Result.ax0 = best.ax0;
            Result.ax1 = best.ax1;
            Result.cx = best.cx;
            Result.ay0 = best.ay0;
            Result.ay1 = best.ay1;
            Result.cy = best.cy;
            Result.ransac_bestcount = best.cnt;
            Result.ransac_bestdist = best.dist;
            return best;
        }

        function match_pattern(descriptors1, descriptors2, typ, par, ssc1, ssc2, matches) {
            var match_threshold = 30;
            var match_frac = 0.8;
            if(typ == 't') {
                var match_threshold = parseInt(par);
            } else {
                var match_frac = parseFloat(par);
            }
            var q_cnt = descriptors1.rows;
            var query_du8 = descriptors1.data;
            var query_u32 = descriptors1.buffer.i32; // cast to integer buffer
            var qd_off = 0;
            var qidx = 0;
            var num_matches = 0;
            for(qidx = 0; qidx < q_cnt; ++qidx) {
                var best_dist = 256;
                var best_dist2 = 256;
                var best_idx = -1;
                var ld_cnt = descriptors2.rows;
                var ld_i32 = descriptors2.buffer.i32; // cast to integer buffer
                var ld_off = 0;
                for(var pidx = 0; pidx < ld_cnt; ++pidx) {
                    var curr_d = 0;
                    for(var k = 0; k < 8; ++k) {
                        curr_d += popcnt32(query_u32[qd_off + k] ^ ld_i32[ld_off + k]); // effectue un XOR (a XOR b = 1 <=> a != b)
                    }
                    if(curr_d < best_dist) {
                        best_dist2 = best_dist;
                        best_dist = curr_d;
                        best_idx = pidx;
                    } else if(curr_d < best_dist2) {
                        best_dist2 = curr_d;
                    }
                    ld_off += 8; // next descriptor
                }
                // filter if typ == 't'
                if(typ == 't') {
                    if(best_dist < match_threshold) {
                        matches.push({
                            'nm': num_matches,
                            'id1': qidx,
                            'id2': best_idx,
                            'dist': curr_d,
                            'x1': ssc1[qidx].x,
                            'y1': ssc1[qidx].y,
                            'x2': ssc2[best_idx].x,
                            'y2': ssc2[best_idx].y
                        });
                        num_matches++;
                    }
                // filter if typ == 'f'
                } else {
                    if(best_dist < match_frac * best_dist2) {
                        matches.push({
                            'nm': num_matches,
                            'id1': qidx,
                            'id2': best_idx,
                            'dist': curr_d,
                            'x1': ssc1[qidx].x,
                            'y1': ssc1[qidx].y,
                            'x2': ssc2[best_idx].x,
                            'y2': ssc2[best_idx].y
                        });
                        num_matches++;
                    }
                }
                qd_off += 8; // next query descriptor
            }
            return num_matches;
        }

        // adapté en javascript à partir du code Python ; https://github.com/BAILOOL/ANMS-Codes
        function ssc(keypoints, num_ret_points, tolerance, cols, rows) {
            var width, c, num_cell_cols, num_cell_rows, covered_vec, row, col, row_min, row_max, col_min, col_max, rtoc, ctoc;
            var exp1 = rows + cols + 2 * num_ret_points;
            var exp2 = (4 * cols + 4 * num_ret_points + 4 * rows * num_ret_points + rows * rows + cols * cols - 2 * rows * cols + 4 * rows * cols * num_ret_points);
            var exp3 = Math.sqrt(exp2);
            var exp4 = num_ret_points - 1;
            var sol1 = -Math.round((exp1 + exp3) / exp4);
            var sol2 = -1 * Math.round((exp1 - exp3) / exp4);
            var high = sol1;
            if(sol2 >= sol1) high = sol2;
            var low = Math.floor(Math.sqrt(keypoints.length) / num_ret_points);
            var prev_width = -1;
            var selected_keypoints = [];
            var result_list = [];
            var result = [];
            var complete = false;
            var k = num_ret_points
            var k_min = Math.round(k - (k * tolerance));
            var k_max = Math.round(k + (k * tolerance));
            while(!complete) {
                width = low + (high - low) / 2;
                if((width == prev_width) || (low > high)) {
                    result_list = result;
                    break;
                }
                c = width / 2;
                num_cell_cols = parseInt(Math.floor(cols / c));
                num_cell_rows = parseInt(Math.floor(rows / c));
                covered_vec = [];
                for(var ii = 0; ii < num_cell_cols; ii += 1) {
                    covered_vec[ii] = [];
                    for(var jj = 0; jj < num_cell_rows; jj += 1) {
                        covered_vec[ii][jj] = false;
                    }
                }
                result = [];
                for(var i = 1; i < keypoints.length; i += 1) {
                    row = parseInt(Math.floor(keypoints[i].y / c));
                    col = parseInt(Math.floor(keypoints[i].x / c));
                    //if ((covered_vec[row] !== undefined) && (covered_vec[row][col] !== undefined) && (!covered_vec[row][col])) {
                    if ((covered_vec[row] !== undefined) && (!covered_vec[row][col])) {
                    //if (!covered_vec[row][col]) {
                        result.push(i);
                        row_min = parseInt(row - Math.floor(width / c));
                        row_min = Math.max(row_min, 0);
                        row_max = parseInt(row + Math.floor(width / c));
                        row_max = Math.min(row_max, num_cell_rows);
                        col_min = parseInt(col - Math.floor(width / c));
                        col_min = Math.max(col_min, 0);
                        col_max = parseInt(col + Math.floor(width / c));
                        col_max = Math.min(col_max, num_cell_cols);
                        for(rtoc = row_min; rtoc < row_max + 1; rtoc += 1) {
                            for(ctoc = col_min; ctoc < col_max + 1; ctoc += 1) {
                                //if ((covered_vec[rtoc] !== undefined) && (covered_vec[rtoc][ctoc] !== undefined) && (!covered_vec[rtoc][ctoc])) covered_vec[rtoc][ctoc] = true;
                                if ((covered_vec[rtoc] !== undefined) && (!covered_vec[rtoc][ctoc])) covered_vec[rtoc][ctoc] = true;
                                //if (!covered_vec[rtoc][ctoc]) covered_vec[rtoc][ctoc] = true;
                            }
                        }
                    }
                }
                if((k_min <= result.length) && (result.length <= k_max)) {
                    result_list = result;
                    complete = true;
                } else if(result.length < k_min) {
                    high = width - 1;
                } else {
                    low = width + 1;
                }
                prev_width = width;
            }
            for(i = 0; i < result_list.length; i += 1) selected_keypoints.push(keypoints[result_list[i]]);
            return selected_keypoints;
        }

        function render() {
            var x, y, x1, y1, x2, y2;
            // image 1 : en vert, tous les points obtenus par fast_corners ... seront repeints en bleus
            myCont.strokeStyle = "rgb(0,255,0)";
            for(var i = 0; i < corners_selected1.length; i++) {
                x = Math.ceil(corners_selected1[i].x * canv1W / img1.width) + canv1L;
                y = Math.ceil(corners_selected1[i].y * canv1H / img1.height) + canv1T;
                myCont.beginPath();
                myCont.moveTo(x, y - 1);
                myCont.lineTo(x, y + 1);
                myCont.moveTo(x - 1, y);
                myCont.lineTo(x + 1, y);
                myCont.lineWidth = 1;
                myCont.stroke();
            }
            // image 1 : ... seront repeints en violet ceux sélectionnés par la méthode SSC
            myCont.strokeStyle = "rgb(255,0,255)";
            for(var i = 0; i < ssc_key_points1.length; i++) {
                x = Math.ceil(ssc_key_points1[i].x * canv1W / img1.width) + canv1L;
                y = Math.ceil(ssc_key_points1[i].y * canv1H / img1.height) + canv1T;
                myCont.beginPath();
                myCont.moveTo(x, y);
                myCont.lineTo(x - 1, y - 1);
                myCont.lineTo(x + 1, y - 1);
                myCont.lineTo(x + 1, y + 1);
                myCont.lineTo(x - 1, y + 1);
                myCont.lineTo(x - 1, y - 1);
                myCont.lineWidth = 1;
                myCont.stroke();
            }
            // image 2 : en vert, tous les points obtenus par fast_corners ... seront repeints en bleus
            myCont.strokeStyle = "rgb(0,255,0)";
            for(var i = 0; i < corners_selected2.length; i++) {
                x = Math.ceil(corners_selected2[i].x * canv2W / img2.width) + canv2L;
                y = Math.ceil(corners_selected2[i].y * canv2H / img2.height) + canv2T;
                myCont.beginPath();
                myCont.moveTo(x, y - 1);
                myCont.lineTo(x, y + 1);
                myCont.moveTo(x - 1, y);
                myCont.lineTo(x + 1, y);
                myCont.lineWidth = 1;
                myCont.stroke();
            }
            // image 2 : ... seront repeints en violet ceux sélectionnés par la méthode SSC
            myCont.strokeStyle = "rgb(255,0,255)";
            for(var i = 0; i < ssc_key_points2.length; i++) {
                x = Math.ceil(ssc_key_points2[i].x * canv2W / img2.width) + canv2L;
                y = Math.ceil(ssc_key_points2[i].y * canv2H / img2.height) + canv2T;
                myCont.beginPath();
                myCont.moveTo(x, y);
                myCont.lineTo(x - 1, y - 1);
                myCont.lineTo(x + 1, y - 1);
                myCont.lineTo(x + 1, y + 1);
                myCont.lineTo(x - 1, y + 1);
                myCont.lineTo(x - 1, y - 1);
                myCont.lineWidth = 1;
                myCont.stroke();
            }
            // visualisation de la pseudo homothétie : relie en bleu ciel les bons points récupérés par RANSAC
            myCont.strokeStyle = "rgb(128,255,255)";
            for(var i = 0; i < good_matches.gm.length; i++) {
                x1 = Math.ceil(good_matches.gm[i].x1 * canv1W / img1.width) + canv1L;
                y1 = Math.ceil(good_matches.gm[i].y1 * canv1H / img1.height) + canv1T;
                x2 = Math.ceil(good_matches.gm[i].x2 * canv2W / img2.width) + canv2L;
                y2 = Math.ceil(good_matches.gm[i].y2 * canv2H / img2.height) + canv2T;
                myCont.beginPath();
                myCont.moveTo(x1, y1);
                myCont.lineTo(x2, y2);
                myCont.lineWidth = 1;
                myCont.stroke();
            }
        }

        function renderRect(RO,RD){
            myCont.strokeStyle = "rgb(255,255,0)";
            var x0,y0,x2,y2;
            x0 = Math.ceil(RO.xO * canv1W / img1.width) + canv1L;
            y0 = Math.ceil(RO.yO * canv1H / img1.height) + canv1T;
            x2 = Math.ceil(RO.xD * canv1W / img1.width) + canv1L;
            y2 = Math.ceil(RO.yD * canv1H / img1.height) + canv1T;
            myCont.beginPath();
            myCont.moveTo(x0, y0);
            myCont.lineTo(x2, y0);
            myCont.lineTo(x2, y2);
            myCont.lineTo(x0, y2);
            myCont.lineTo(x0, y0);
            myCont.lineWidth = 1;
            myCont.stroke();
            x0 = Math.ceil(RD.xO * canv2W / img2.width) + canv2L;
            y0 = Math.ceil(RD.yO * canv2H / img2.height) + canv2T;
            x2 = Math.ceil(RD.xD * canv2W / img2.width) + canv2L;
            y2 = Math.ceil(RD.yD * canv2H / img2.height) + canv2T;
            myCont.beginPath();
            myCont.moveTo(x0, y0);
            myCont.lineTo(x2, y0);
            myCont.lineTo(x2, y2);
            myCont.lineTo(x0, y2);
            myCont.lineTo(x0, y0);
            myCont.lineWidth = 1;
            myCont.stroke();
        }

        function initialise_parametres_initiaux() {
            Parinit.egalise_hist1 = 0;
            if(document.querySelector("#EH1").checked) {
                Parinit.egalise_hist1 = 1;
            }
            Parinit.egalise_hist2 = 0;
            if(document.querySelector("#EH2").checked) {
                Parinit.egalise_hist2 = 1;
            }
            Parinit.th1 = parseInt(document.querySelector("#TH1inp").value);
            Parinit.th2 = parseInt(document.querySelector("#TH2inp").value);
            Parinit.FastCNb1 = parseInt(document.querySelector("#FCNB1inp").value);
            Parinit.FastCNb2 = parseInt(document.querySelector("#FCNB2inp").value);
            Parinit.ObjSSC1 = parseInt(document.querySelector("#OSS1inp").value);
            Parinit.ObjSSC2 = parseInt(document.querySelector("#OSS2inp").value);
            Parinit.Matval = parseInt(document.querySelector("#MatchV").value);
            if(Parinit.Matval > 1) {
                Parinit.Mattyp = 't';
            } else {
                Parinit.Mattyp = 'f';
                Parinit.Matval = parseFloat(document.querySelector("#MatchV").value);
            }
            Parinit.RanT = parseInt(document.querySelector("#RanTinp").value);
            Parinit.RanIter = parseInt(document.querySelector("#RanItinp").value);
            Parinit.ratioS = parseFloat(document.querySelector("#RatioSinp").value);
            if (localStorage.length == 0){
                var ligne = 'nom_img1;largeur_img1;hauteur_img1;nom_img2;largeur_img2;hauteur_img2;P_EH1;P_EH2;Seuil1;Seuil2;FastCounterNb1;';
                ligne += 'FastCounterNb2;ObjSSC1;ObjSSC2;MatchType;MatchVal;RansacSeuil;RansacIter;RatioSurf;TmsG1;TmsG2;TmsEH1;TmsEH2;';
                ligne += 'TmsFC1;TmsFC2;TmsSSC1;TmsSSC2;TmsMatch;TmsGM;TmsQG;TmsTot;FCnb1;FCnb2;SSCNb1;SSCNb2;MatchNb;GMNb;CoefC;CoefG;';
                ligne += 'FxAx;FxCx;FyAy;FyCy;BoxOrigxO;BoxOrigyO;BoxOrigxD;BorOrigyD;BoxDestxO;BoxDestyO;BoxDestxD;BoxDestyD'+'\n';
                localStorage.setItem('Correltest', ligne);
            }
            document.querySelector("#nblig").textContent = localStorage.getItem("Correltest").split('\n').length-2;
            document.querySelector('#QC').className = 'QCG';
            document.querySelector('#QG').className = 'QCG';
            document.querySelector('#NbGM').className = 'goodNbGM';
        }

        function affiche_result() {
            document.querySelector("#Tms1G").textContent = parseInt(Result.Tms1G) + 'ms';
            document.querySelector("#Tms2G").textContent = parseInt(Result.Tms2G) + 'ms';
            document.querySelector("#Tms1Eg").textContent = parseInt(Result.Tms1Eg) + 'ms';
            document.querySelector("#Tms2Eg").textContent = parseInt(Result.Tms2Eg) + 'ms';
            document.querySelector("#TmsFC1").textContent = parseInt(Result.TmsFC1) + 'ms';
            document.querySelector("#TmsFC2").textContent = parseInt(Result.TmsFC2) + 'ms';
            document.querySelector("#FCDnb1").textContent = parseInt(Result.FCDnb1);
            document.querySelector("#FCDnb2").textContent = parseInt(Result.FCDnb2);
            document.querySelector("#TmsSSC1").textContent = parseInt(Result.TmsSSC1) + 'ms';
            document.querySelector("#TmsSSC2").textContent = parseInt(Result.TmsSSC2) + 'ms';
            document.querySelector("#NbSSC1").textContent = parseInt(Result.NbSSC1);
            document.querySelector("#NbSSC2").textContent = parseInt(Result.NbSSC2);
            document.querySelector("#TmsM").textContent = parseInt(Result.TmsM) + 'ms';
            document.querySelector("#NbMat").textContent = parseInt(Result.NbMat);
            document.querySelector("#TmsGM").textContent = parseInt(Result.TmsGM) + 'ms';
            document.querySelector("#NbGM").textContent = parseInt(Result.NbGM);
            Result.TmsT = Result.Tms1G + Result.Tms2G + Result.Tms1Eg + Result.Tms2Eg + Result.TmsFC1 + Result.TmsFC2 + Result.TmsSSC1 + Result.TmsSSC2 + Result.TmsM + Result.TmsGM + Result.TmsQG;
            document.querySelector("#TmsT").textContent = parseInt(Result.TmsT) + 'ms';
            document.querySelector("#QC").textContent = parseInt(Result.QC);
            document.querySelector("#Homax1").textContent = Math.ceil(parseFloat(Result.ax1) * 100) / 100;
            document.querySelector("#Homcx").textContent = parseInt(Result.cx);
            document.querySelector("#Homay1").textContent = Math.ceil(parseFloat(Result.ay1) * 100) / 100;
            document.querySelector("#Homcy").textContent = parseInt(Result.cy);
            document.querySelector("#RectOOut").textContent = 'Orig ('+img1.width+'*'+img1.height+')';
            document.querySelector("#RectDOut").textContent = 'Dest ('+img2.width+'*'+img2.height+')';
            document.querySelector("#OxO").textContent = Result.ROxO;
            document.querySelector("#OyO").textContent = Result.ROyO;
            document.querySelector("#OxD").textContent = Result.ROxD;
            document.querySelector("#OyD").textContent = Result.ROyD;
            document.querySelector("#DxO").textContent = Result.RDxO;
            document.querySelector("#DyO").textContent = Result.RDyO;
            document.querySelector("#DxD").textContent = Result.RDxD;
            document.querySelector("#DyD").textContent = Result.RDyD;
            document.querySelector("#TmsQG").textContent = parseInt(Result.TmsQG) + 'ms';
            document.querySelector("#QG").textContent = Math.ceil(parseFloat(Result.QG) * 100);
            if (Result.QG == 0){
                // pas suffisament de Good Matches !
                if (Result.QC == 0){
                    document.querySelector("#QC").className = "bad";
                    document.querySelector("#NbGM").className = "bad";
                } else {
                // rapport surfaces inférieur à 0.5
                }
            }
            var ligne=localStorage.getItem('Correltest');
            ligne += document.querySelector("#inp1").files[0].name+';'+img1.width+';'+img1.height+';'+document.querySelector("#inp2").files[0].name+';'+img2.width+';'+img2.height+';'+Parinit.egalise_hist1+';'+Parinit.egalise_hist2+';'+Parinit.th1+';'+Parinit.th2+';'+Parinit.FastCNb1+';'+Parinit.FastCNb2+';'+Parinit.ObjSSC1+';'+Parinit.ObjSSC2+';'+Parinit.Mattyp+';'+Parinit.Matval.toString().replace(".",",")+';'+Parinit.RanT+';'+Parinit.RanIter+';'+Parinit.ratioS+';'+parseInt(Result.Tms1G)+';'+parseInt(Result.Tms2G)+';'+parseInt(Result.Tms1Eg)+';'+parseInt(Result.Tms2Eg)+';'+parseInt(Result.TmsFC1)+';'+parseInt(Result.TmsFC2)+';'+parseInt(Result.TmsSSC1)+';'+parseInt(Result.TmsSSC2)+';'+parseInt(Result.TmsM)+';'+parseInt(Result.TmsGM)+';'+parseInt(Result.TmsQG)+';'+parseInt(Result.TmsT)+';'+parseInt(Result.FCDnb1)+';'+parseInt(Result.FCDnb2)+';'+parseInt(Result.NbSSC1)+';'+parseInt(Result.NbSSC2)+';'+parseInt(Result.NbMat)+';'+parseInt(Result.NbGM)+';'+parseInt(Result.QC)+';'+Math.ceil(parseFloat(Result.QG) * 100)+';'+parseFloat(Result.ax1)+';'+parseFloat(Result.cx)+';'+parseFloat(Result.ay1)+';'+parseFloat(Result.cy)+';'+Result.ROxO+';'+Result.ROyO+';'+Result.ROxD+';'+Result.ROyD+';'+Result.RDxO+';'+Result.RDyO+';'+Result.RDxD+';'+Result.RDyD+'\n';
            localStorage.setItem('Correltest',ligne);
            document.querySelector("#nblig").textContent = localStorage.getItem("Correltest").split('\n').length-2;
        }

        async function save_result(){
            var auj = new Date();

            var dd = auj.getDate();
            var mm = auj.getMonth() + 1;
            var yyyy = auj.getFullYear();
            var hh = auj.getHours();
            var mn = auj.getMinutes();
            var ss = auj.getSeconds();
            if (dd < 10) {
                dd = '0' + dd;
            }
            if (mm < 10) {
                mm = '0' + mm;
            }
            if (hh < 10){
                hh = '0' + hh;
            }
            if (mn < 10){
                mn = '0' + mn;
            }
            if (ss < 10){
                ss = '0' + ss;
            }
            var name = 'CorrelImagesTest_'+yyyy+'_'+mm+'_'+dd+'_'+hh+'_'+mn+'_'+ss+'.csv';
            var ligne = localStorage.getItem('Correltest');
            await downloadFile(ligne, name);
        }

        function ClearStorage(){
            localStorage.removeItem('Correltest');
            var ligne = 'nom_img1;largeur_img1;hauteur_img1;nom_img2;largeur_img2;hauteur_img2;P_EH1;P_EH2;Seuil1;Seuil2;FastCounterNb1;';
            ligne += 'FastCounterNb2;ObjSSC1;ObjSSC2;MatchType;MatchVal;RansacSeuil;RansacIter;RatioSurf;TmsG1;TmsG2;TmsEH1;TmsEH2;';
            ligne += 'TmsFC1;TmsFC2;TmsSSC1;TmsSSC2;TmsMatch;TmsGM;TmsQG;TmsTot;FCnb1;FCnb2;SSCNb1;SSCNb2;MatchNb;GMNb;CoefC;CoefG;';
            ligne += 'FxAx;FxCx;FyAy;FyCy;BoxOrigxO;BoxOrigyO;BoxOrigxD;BorOrigyD;BoxDestxO;BoxDestyO;BoxDestxD;BoxDestyD'+'\n';
            localStorage.setItem('Correltest', ligne);
            document.querySelector("#nblig").textContent = localStorage.getItem("Correltest").split('\n').length-2;
        }

        function init() {
            document.querySelector("#nblig").textContent = localStorage.getItem("Correltest").split('\n').length-2;
            document.querySelector("#f_ok").addEventListener('click', async function(e) {
                e.stopImmediatePropagation();
                document.querySelector("#f_ok").style.visibility = "hidden";
                img1 = document.getElementById("img1");
                img2 = document.getElementById("img2");
                let mainW = document.querySelector('#main').clientWidth;
                if((img1.src != '') && (img2.src != '')) {

                    // traitement principal - les canvas et leurs dimensions
                    myCanv = document.querySelector("#visu");
                    myCanv.width = mainW;
                    myCont = myCanv.getContext('2d');
                    canvWmax = Math.max(img1.width, img2.width);
                    if(canvWmax > mainW) {
                        canv1W = Math.ceil(mainW * img1.width / canvWmax);
                        canv2W = Math.ceil(mainW * img2.width / canvWmax);
                        canv1H = Math.ceil(img1.height * canv1W / img1.width);
                        canv2H = Math.ceil(img2.height * canv2W / img2.width);
                    } else {
                        canv1W = img1.width;
                        canv2W = img2.width;
                        canv1H = img1.height;
                        canv2H = img2.height;
                        myCanv.width = canvWmax;
                    }
                    canv1T = 0;
                    canv1L = Math.floor((Math.max(canv1W, canv2W) - canv1W) / 2);
                    canv2T = canv1H;
                    canv2L = Math.floor((Math.max(canv1W, canv2W) - canv2W) / 2);
                    myCanv.height = canv1H + canv2H;

                    myCont.drawImage(img1, 0, 0, img1.width, img1.height, canv1L, canv1T, canv1W, canv1H);
                    myCont.drawImage(img2, 0, 0, img2.width, img2.height, canv2L, canv2T, canv2W, canv2H);

                    let ok = initialise_parametres_initiaux();
                    myC1 = document.querySelector("#Cv1");
                    myC1.width = img1.width;
                    myC1.height = img1.height;
                    myCt1 = myC1.getContext('2d');
                    myCt1.drawImage(img1, 0, 0, img1.width, img1.height);
                    myC2 = document.querySelector("#Cv2");
                    myC2.width = img2.width;
                    myC2.height = img2.height;
                    myCt2 = myC2.getContext('2d');
                    myCt2.drawImage(img2, 0, 0, img2.width, img2.height);

                    // IMAGE 1
                    // 0 initialise img1_u8 et corners en fonction de la taille du canvas
                    var img1_u8 = new jsfeat.matrix_t(img1.width, img1.height, jsfeat.U8_t | jsfeat.C1_t);
                    corners1 = [];
                    var i = img1.width * img1.height;
                    while(--i >= 0) {
                        corners1[i] = new jsfeat.keypoint_t(0, 0, 0, 0);
                    }

                    // on effectue le grayscale : 0/ img1_u8 recevra l'image en N&B
                    var imageData1 = myCt1.getImageData(0, 0, img1.width, img1.height);
                    DatDeb = Date.now();
                    jsfeat.imgproc.grayscale(imageData1.data, img1.width, img1.height, img1_u8);
                    Result.Tms1G = Date.now() - DatDeb;

                    // éxécution de l'égalisation de l'histogramme si souhaité :
                    if(Parinit.egalise_hist1 == 1) {
                        DatDeb = Date.now();
                        jsfeat.imgproc.equalize_histogram(img1_u8, img1_u8);
                        Result.Tms1Eg = Date.now() - DatDeb;
                    } else Result.Tms1Eg = 0;

                    // initialisation du threshold pour fast_corners ----- Param1
                    var threshold = Parinit.th1;
                    jsfeat.fast_corners.set_threshold(threshold);

                    // fast_corners avec objectif d'obtenir au moins Parinit.FastCNb1 points remarquables
                    DatDeb = Date.now();
                    var count1 = 0;
                    do {
                        count1 = jsfeat.fast_corners.detect(img1_u8, corners1, 5);
                        Result.ratio1 = ((100 * count1) / (img1.width * img1.height));
                        threshold = threshold / 2;
                        jsfeat.fast_corners.set_threshold(threshold);
                    } while ((count1 < Parinit.FastCNb1) && (threshold > 1)); // <----- éviter de boucler à l'infini !?
                    Result.TmsFC1 = Date.now() - DatDeb;
                    Result.FCDnb1 = count1;

                    // ssc pour obtention d'une distribution spatiale homogène des points clés ()
                    DatDeb = Date.now();
                    corners_selected1 = corners1.slice(0, count1);
                    corners_selected1.sort(function (a, b) { return a.score > b.score ? -1 : 1 });
                    // attention : ssc nécessite que les points à filtrer soient classés par ordre décroissant de force / score
                    ssc_key_points1 = ssc(corners_selected1, Parinit.ObjSSC1, 0.1, img1.width, img1.height); // Param3 objectif nb_points
                    Result.TmsSSC1 = Date.now() - DatDeb;
                    Result.NbSSC1 = ssc_key_points1.length;

                    // création des "descriptors" pour chacun des points retenus après SSC limités en nombre à 500 !
                    descr1 = new jsfeat.matrix_t(32, 500, jsfeat.U8_t | jsfeat.C1_t);
                    jsfeat.orb.describe(img1_u8, ssc_key_points1, ssc_key_points1.length, descr1);

                    // IMAGE 2
                    // 0 initialise img2_u8 et corners en fonction de la taille du canvas
                    var img2_u8 = new jsfeat.matrix_t(img2.width, img2.height, jsfeat.U8_t | jsfeat.C1_t);
                    corners2 = [];
                    var i = img2.width * img2.height;
                    while(--i >= 0) {
                        corners2[i] = new jsfeat.keypoint_t(0, 0, 0, 0);
                    }

                    // on effectue le grayscale : img2_u8 recevra l'image en N&B
                    var imageData2 = myCt2.getImageData(0, 0, img2.width, img2.height);
                    DatDeb = Date.now();
                    jsfeat.imgproc.grayscale(imageData2.data, img2.width, img2.height, img2_u8);
                    Result.Tms2G = Date.now() - DatDeb;

                    // éxécution de l'égalisation de l'histogramme si souhaité :
                    if(Parinit.egalise_hist2 == 1) {
                        DatDeb = Date.now();
                        jsfeat.imgproc.equalize_histogram(img2_u8, img2_u8);
                        Result.Tms2Eg = Date.now() - DatDeb;
                    } else Result.Tms2Eg = 0;

                    // initialisation du threshold pour fast_corners2
                    threshold = Parinit.th2;
                    jsfeat.fast_corners.set_threshold(threshold);

                    // fast_corners avec objectif d'obtenir au moins Parinit.FastCNb2 points remarquables
                    var count2 = 0;
                    DatDeb = Date.now();
                    do {
                        count2 = jsfeat.fast_corners.detect(img2_u8, corners2, 5);
                        Result.ratio2 = ((100 * count2) / (img2.width * img2.height));
                        threshold = threshold / 2;
                        jsfeat.fast_corners.set_threshold(threshold);
                    } while ((count2 < Parinit.FastCNb2) && (threshold > 1)); // ----------------------- Param2
                    Result.TmsFC2 = Date.now() - DatDeb;
                    Result.FCDnb2 = count2;

                    // ssc pour obtention d'une distribution spatiale homogène des points clés ()
                    DatDeb = Date.now();
                    corners_selected2 = corners2.slice(0, count2);
                    corners_selected2.sort(function (a, b) { return a.score > b.score ? -1 : 1 });
                    // attention : ssc nécessite que les points à filtrer soient classés par ordre décroissant de force / score
                    ssc_key_points2 = ssc(corners_selected2, 450, 0.1, img2.width, img2.height); // Param3 objectif nb_points
                    Result.TmsSSC2 = Date.now() - DatDeb;
                    Result.NbSSC2 = ssc_key_points2.length;

                    // création des "descriptors" pour chacun des points retenus après SSC limités en nombre à 500 !
                    descr2 = new jsfeat.matrix_t(32, 500, jsfeat.U8_t | jsfeat.C1_t);
                    jsfeat.orb.describe(img2_u8, ssc_key_points2, ssc_key_points2.length, descr2);

                    // On commence à rechercher les points ayant des descripteurs "assez proches"
                    var num_matches = 0;
                    good_matches = {};
                    matches = [];

                    /* --------------------------------------------------------------------------------------------------------
                     Recherche des descripteurs "proches" : nous avons le choix entre deux méthodes :
                      a)soit à partir d'une distance / Threshold maxi (Parinit.Mattyp = 't', Parinit.Matval = valeur max)
                        Ex : num_matches = match_pattern(descr1, descr2, 't', 200 , ssc_key_points1, ssc_key_points2);
                      b)soit en diminuant de + en + cette distance max (Parinit.Mattyp = 'f', Parinit.Matval = coefficient < 1)
                        Ex : num_matches = match_pattern(descr1, descr2, 'f', 0.9 , ssc_key_points1, ssc_key_points2);
                    /* --------------------------------------------------------------------------------------------------------*/
                    DatDeb = Date.now();
                    num_matches = match_pattern(descr1, descr2, Parinit.Mattyp, Parinit.Matval, ssc_key_points1, ssc_key_points2, matches);
                    Result.TmsM = Date.now() - DatDeb;
                    Result.NbMat = matches.length;
                                            /* -----------------sauvegarde du fichier pour debug------------- *//*
                                            var outmg='';
                                            for (var is=0; is<matches.length; is++){ outmg += 'x1,y1,x2,y2='+matches[is].x1+','+matches[is].y1+','+matches[is].x2+','+matches[is].y2+'\n'};
                                            await downloadFile(outmg, 'matches.txt'); outmg='';
                                            /* --------------------------------------------------------------- */

                    // Recherche d'une pseudo homothétie par l'algorithme de Ransac si nous avons au moins 10 bonnes correspondances
                    if(matches.length > 10) {
                        DatDeb = Date.now();
                        // params5 et 6
                        good_matches = my_ransac(matches, Parinit.RanT, Parinit.RanIter); // threshold 20, 3000 itérations
                        Result.TmsGM = Date.now() - DatDeb;
                        Result.NbGM = good_matches.gm.length;
                                            /* -----------------sauvegarde du fichier pour debug------------- *//*
                                            var outmg='';
                                            for (var is=0; is<good_matches.gm.length; is++){ outmg += 'x1,y1,x2,y2='+good_matches.gm[is].x1+','+good_matches.gm[is].y1+','+good_matches.gm[is].x2+','+good_matches.gm[is].y2+'\n'};
                                            await downloadFile(outmg, 'good_matches.txt'); outmg='';
                                            /* --------------------------------------------------------------- */
                        // résultat recherché <=> Coefficient de corrélation !
                        var QCorr = Math.ceil(100 * Result.NbGM / Result.NbMat);

                        // affichage visuel des points correspondants
                        render();

                        // calcul maintenant de la "distance en gris" entre les 2 images. Rappel -x1 + ax1*x2 +cx = 0 et -y1 + ay1*y2 + cy = 0
                        var xp0, yp0, xp2, yp2;
                        var rectO = {}; var rectD = {};
                        DatDeb = Date.now();
                        xp0 = fx1(0); yp0 = fy1(0); xp2 = fx1(img1.width); yp2 = fy1(img1.height); //Diagonale de haut à gauche vers bas à àroite
                        if ((xp0 < 0) && (yp0 < 0)){
                            rectO.xO = fx0(0); rectO.yO = fy0(0); rectD.xO = 0; rectD.yO = 0;
                        } else if ((xp0 >= 0) && (yp0 >=0)){
                            rectO.xO = 0; rectO.yO = 0; rectD.xO = fx1(0); rectD.yO = fy1(0);
                        } else if ((xp0 >= 0) && (yp0 < 0)){
                            rectO.xO = 0; rectO.yO = fy0(0); rectD.xO = fx1(0); rectD.yO = 0;
                        } else { // xp0 < 0 et yp0 >= 0
                            rectO.xO = fx0(0); rectO.yO = 0; rectD.xO = 0; rectD.yO = fy1(0);
                        }

                        if ((xp2 <= img2.width-1) && (yp2 <= img2.height-1)){
                            rectO.xD= img1.width-1; rectO.yD= img1.height-1; rectD.xD= fx1(img1.width-1); rectD.yD= fy1(img1.height-1);
                        } else if ((xp2 <= img2.width-1) && (yp2 > img2.height-1)){
                            rectO.xD= img1.width-1; rectO.yD= fy0(img2.height-1); rectD.xD= fx1(img1.width-1); rectD.yD= img2.height-1;
                        } else if ((xp2 > img2.width-1) && (yp2 <= img2.height-1)){
                            rectO.xD= fx0(img2.width-1); rectO.yD= img1.height-1; rectD.xD= img2.width-1; rectD.yD= fy1(img1.height-1);
                        } else { // xp2 > img2.width-1 et yp2 > img2.heidth-1
                            rectO.xD= fx0(img2.width-1); rectO.yD= fy0(img2.height-1); rectD.xD= img2.width-1; rectD.yD= img2.height-1;
                        }
                        Result.ROxO = rectO.xO; Result.ROyO = rectO.yO; Result.ROxD = rectO.xD; Result.ROyD = rectO.yD;
                        Result.RDxO = rectD.xO; Result.RDyO = rectD.yO; Result.RDxD = rectD.xD; Result.RDyD = rectD.yD;

                        // CONTINUE si et seulement si la surface de chaque rectangle est supérieure au ratio minimum d'image !
                        if (((Math.abs((Result.ROxD-Result.ROxO) * (Result.ROyD-Result.ROyO)) / (img1.width*img1.height)) >= Parinit.ratioS) && ((Math.abs((Result.RDxD-Result.RDxO) * (Result.RDyD-Result.RDyO)) / (img2.width*img2.height)) >= Parinit.ratioS)){
                            var x1, y1, x2, y2;
                            // Choissons le plus petit rectangle pour miniiser les calculs ... et donc le temps !
                            if (Math.abs(rectD.xD - rectD.xO) > Math.abs(rectO.xD - rectO.xO)){
                                var som = 0;
                                for (x1=rectO.xO; x1<rectO.xD; x1++){
                                    for (y1=rectO.yO; y1<rectO.yD; y1++){
                                        x2=fx1(x1); y2=fy1(y1);
                                        som += Math.abs((img1_u8.data[y1*img1.width + x1] - img2_u8.data[y2*img2.width + x2]));
                                    }
                                }
                                som = som / (Math.abs((rectO.xD - rectO.xO) * (rectO.yD - rectO.yO))* 255);
                            } else {
                                var x1, y1, x2, y2;
                                var som = 0;
                                for (x1=rectD.xO; x1<rectD.xD; x1++){
                                    for (y1=rectD.yO; y1<rectD.yD; y1++){
                                        x2=fx0(x1); y2=fy0(y1);
                                        som += Math.abs((img2_u8.data[y1*img2.width + x1] - img1_u8.data[y2*img1.width + x2]));
                                    }
                                }
                                som = som / (Math.abs((rectO.xD - rectO.xO) * (rectO.yD - rectO.yO))* 255);
                            }
                            Result.TmsQG = Date.now() - DatDeb;
                            Result.QG = 1-som;
                            Result.QC = QCorr;
                            renderRect(rectO, rectD);
                        } else {
                            Result.QC = QCorr;
                            Result.QG = 0;
                            Result.TmsQG = Date.now() - DatDeb;
                        }
                    } else {
                        var QCorr = 0;
                        Result.NbGM = 0;
                        Result.ax0 = 0;
                        Result.ax1 = 0;
                        Result.cx = 0;
                        Result.ay0 = 0;
                        Result.ay1 = 0;
                        Result.cy = 0;
                        Result.ransac_bestcount = 0;
                        Result.ransac_bestdist = 'infinity';
                        Result.ROxO = '-'; Result.ROyO = '-'; Result.ROxD = '-'; Result.ROyD = '-';
                        Result.RDxO = '-'; Result.RDyO = '-'; Result.RDxD = '-'; Result.RDyD = '-';
                        Result.axex = '-'; Result.axey = '-';
                        Result.QG = 0;
                        Result.QC = 0;
                        Result.TmsQG = 0;
                    }

                    // affichages des résultats dans le tableau
                    affiche_result();
                } else alert('2 images doivent préalablement avoir été sélectionnées');
                document.querySelector("#f_ok").style.visibility = "visible";
            },true);
        }

        function fx0(x1){ // rappel : -x0 + Result.ax1.x1 + Result.cx = 0
            var x0 = Math.ceil((Result.ax1 * x1) + Result.cx);
            return x0;
        }
        function fx1(x0){ // rappel : -x0 + Result.ax1.x1 + Result.cx = 0
            var x1 = Math.ceil((x0 -Result.cx)/Result.ax1);
            return x1;
        }
        function fy0(y1){ // rappel : -y0 + Result.ay1.y1 + Result.cy = 0
            var y0 = Math.ceil((Result.ay1 * y1) + Result.cy);
            return y0;
        }
        function fy1(y0){ // rappel : -y0 + Result.ay1.y1 + Result.cy = 0
            var y1 = Math.ceil((y0 -Result.cy)/Result.ay1);
            return y1;
        }
        </script>
    </head>

    <body onload="init();">
        <div id="menu">
            <div id="top_of_box" class="top-box"> </div>
            <div id="bidon"></div>
            <div id="entete"> <img id="cornleft" src="" alt="coingauche" />

            <div id="titre">
                <h2><a>CorrelImages by ArouG</a> - visiteurs : <a><?php echo $lastnb; ?></a> depuis le 27/01/2023</h2>
                <p style="text-align:center;">--------------------------------</p>
                <p style="text-align:center;"><?php echo $Versiondu; ?> - Contact : ArouG at turbosudoku dot fr</p> <!-- params -->
                <hr />
                <div id="appelHelp">
                    <button id='HelpButt' class="styledvp" onclick="help();">Help</button>
                </div>
                <div id="nomResult">
                    <button id='NameResButt' class="styledvp" onclick="NameRes();">Sauvegarde</button>
                </div>
            </div><!-- div titre -->

            <img id="cornright" src="" alt="coindroit" />
                <div><span class="bidon0">&nbsp;</span></div>
            </div> <!-- entete -->
            <div id="NameRes" style="display:none">
                <p>CorrelImages a aussi été conçu pour permettre la détermination, en fonction de la paire d'images présentées, des "meilleurs paramètres d'entrée possibles". Si le nombre de descripteurs est fixé à 500 (pour chaque image), il reste beaucoup d'autres paramètres d'entrée !</p>
                <p>Aussi m'était-il indispensable d'offrir la possibilité de stocker les différentes valeurs obtenues en fonction des paramètres</p>
                <p>J'ai choisi le format .csv car l'un des plus facile à utiliser et permettant un traitement postérieur avec des outils du type tableur : Excel de MicrosoftOffice , calcs d'OpenOffice mais toute base de données que vous utilisez ;-)</p>
                <p>Afin de permettre la conservation des données d'une séance à l'autre, j'utilise le 'localStorage'. Attention : cette technologie est propre à chaque navigateur ! Si vous débutez avec firefox, ne vous attendez pas à retrouver vos données sous Chrome ou Edge !</p>
                <p>La "récupération" de vos donnéees s'effectuera par téléchargement et il sera donc nécessaire de récupérer vos données &lt;=> le fichier résultant sous le répertoire dédié (C:\downloads ou C:\Téléchargements pour les utilisateurs de Windows)</p>
                <p> Le fichier téléchargé se nommera CorrelImagesTest_date_et_heure.csv. Deux commandes sont à votre disposition :</p>
                <br>
                <p> <button id="DownButt" class="styleddr" type="button" onClick="save_result();">Télécharger le fichier</button> et, de temps en temps, <button id="ClearButt" class="styleddr" type="button" onClick="ClearStorage();">Vider vos données du localStorage</button>&nbsp;&nbsp;&nbsp;Nombre de tests enregistrés en cours :&nbsp;<span id="nblig">&nbsp;</span></p>

            </div>
            <div id="help" style="display:none">
                <h1>I - Objectif</h1>
                <p>CorrelImages a pour objectif de mettre au point un nombre, compris entre 0 et 100, permettant d'indiquer le degré de corrélation entre 2 images de dimensions éventuellement différentes, en recherchant une relation telle que, pour tout point P1(x1,y1) (du moins un très grand nombre) de la première image, on puisse associer un point P2(x2,y2) de la seconde tels que :</p>
                <span class="centre">a.x1 + b.x2 + c = 0 ET d.y1 + e.y2 + f = 0</span>
                <P> autrement dit que les deux images soient plus ou moins homothétiques.</p>
                <p>CorrelImages utilise / met en oeuvre plusieurs techniques liées au traitement d'images. Il utilise principalement la librairie <a href="https://github.com/inspirit/jsfeat" target="_blank">JSFeat.js</a> (passage couleur en b&B, égalisation d'histogramme, détection rapides des points remarquables d'une image, descripteurs ORB et match_pattern) mais aussi des techniques non incluses dans cette librairie : diminution des points remarquables par suppression des non maximaux en vue d'obtenir une distribution "homogène" et l'algorithme de RANSAC pour ce qui concerne la recherche de la "pseudo homothétie"</p>
                <br>
                <h1>II - Algorithme d'ensemble et paramètres</h1>
                <h2>a) passage d'une image couleur en image à nuance de gris</h2>
                <p>(aucun paramètre)</p>
                <h2>b) égalisation éventuelle de l'histogramme (normalisation d'image)</h2>
                <p>Paramètres : '=1' et '=2' précisent si cet normalisation sera réalisée ou pas</p>
                <h2>c) détermination des points remarquables par l'algorithme "Fast Corners" (Jsfeat)</h2>
                <p>Paramètres : 'Th1' (resp. Th2) représente le seuil initial requis et 'FCNb1' (resp. FCNb2) le nombre minimal de points remarquables attendus, espérés au terme de ce process. Le seuil initial sera diminué tant que le nombre minimum de points ne sera pas atteint. ATTENTION : CorrelImages limite le nombre de descripteurs à 500 par image ... rien ne sert de rechercher 1000 point remarquables !</p>
                <h2>d) diminution des points remarquables avec homogéinisation dans l'espace (ANMS)</h2>
                <p>Paramètres : 'OSC1' (resp. 'OSC2') représente un objectif (+/- 10%) de réduction des points obtenus par "Fast Corners" (avec homogéinisation dans l'image). Rappel : 500 descripteurs par image; donc OSC1 doit être inférieur ou égal à 450 !</p>
                <h2>e) créations de descripteurs ORB pour permettre </h2>
                <p>(aucun paramètre)</p>
                <h2>f) de "matcher" les descripteurs / points remarquables des deux images</h2>
                <p>Paramètre : 'MatchP' : 2 méthodes sont implémentées pour la détermination des points remarquables "assimilables" : soit en fixant un seuil "max" (supérieur à 1, initiaement 200) permettant de sélectionner les bonnes paires de points des autres, soit par un facteur compris entre 0 < MatchP < 1 (se conforter au code pour plus de compréhension)</p>
                <h2>g) et utilisation de Ransac pour filtrer parmi les paires de points récupérées à l'étape précédente celles résultant "au mieux" d'une transformation reliant les 2 images</h2>
                <p>2 paramètres pour ce process : 'RanT' qui est un seuil maximal (seront sélectionnées les paires de points dont la "distance au modèle sélectionné au hasard" sera inférieure à ce seuil) et 'RanIter' : nombre d'itérations (de modèles) pris au hasard avant de sélectionner le meilleur !</p>
                <h2>h) Notion de ratio de surface - détermination des 2 blocs communs :</h2>
                <p>Principalement pour certains type de photos / images (dessins ou images comportant des lettres), le procédé de Ransac aboutit à de fausse corrélations. Généralement, ces faux positifs se rencontrent lorsque l'une ou l'autre des images focalise les points sur une toute petite zône. Or CorrelImages a pour objectif de traiter des images de tailles 'pas trop dissemblables'. Aussi, afin d'écarter / d'éliminer ces faux positifs, on ne conservera des images pour lesquelles les blocs en corrélation seront 'suffisament grands'. D'où ce ratio d'image tel que 0 < RatioS < 1 (initialement = 0.5)</p>
                <h2>i) calcul de la distance entre ces deux blocks.</h2>
                <p>(aucun paramètre) : la distance entre les deux images se calculera par la distance moyenne entre les valeurs de gris de ces deux blocs déterminés précédement</p>
                <br>
                <h1>III - Illustration :</h1>
                <img src="help.png" alt="image">
                <p>Points mauves + points verts = Fnb</p>
                <p>Points mauves seuls = NbC</p>
                <p>Lignes bleues = NbGM</p>
                <p>Carrés jaunes : Orig et Dest</p>
                <br>
                <h1>IV - Politique de cookies :</h1>
                <p>Correlmages n'utilise qu'un seul cookie 'forcé' (nom = CorrelImages, valeur = nimp) permettant le comptage des visites sur le site</p>
            </div>
            <div id="main">
                <div id="paramsDiv">
                    <table id="paramsTab"><tbody>
                        <tr><td id="parentree" colspan="12">paramètres d'entrée</td>
                        </tr><tr>
                            <td>=1</td><td>=2</td><td>Th1</td><td>Th2</td><td>FCNb1</td><td>FCNb2</td><td>OSC1</td><td>OSC2</td><td>MatchP</td><td>RanT</td><td>RanIter</td><td>RatioS</td>
                        </tr><tr>
                            <td id="egalise_hist1"><input id="EH1" title="égalisation de l'histogramme 1 ?" type="checkbox" checked></td><td id="egalise_hist2"><input id="EH2" title="égalisation de l'histogramme 2 ?" type="checkbox" checked></td><td id="th1"><input title="valeur du seuil1" type="text" id="TH1inp" minlength="1" maxlength="3" value="30"></td><td id="th2"><input title="valeur du seuil2" type="text" id="TH2inp" minlength="1" maxlength="3" value="30"></td> <td id="FastCNb1"><input type="text" id="FCNB1inp" title="nombre de points minimum de fast.counter1" minlength="1" maxlength="3" value="600"></td><td id="FastCNb2"><input type="text" id="FCNB2inp" title="nombre de points minimum de fast.counter2" minlength="1" maxlength="3" value="600"></td><td id="ObjSSC1"><input title="évaluation (+/-10%) du nombre de descripteurs résultant (<450)" type="text" id="OSS1inp" minlength="1" maxlength="3" value="450"></td><td id="ObjSSC2"><input title="évaluation (+/-10%) du nombre de descripteurs résultant (<450)" type="text" id="OSS2inp" minlength="1" maxlength="3" value="450"></td><td id="MatchP"><input title="méthode de criblage pour les descripteurs correspondants" type="text" id="MatchV" minlength="1" maxlength="4" value="200"></td><td id="RanT"><input title="seuil pour recherche des 'bonnes ressemblances'" type="text" id="RanTinp" minlength="1" maxlength="3" value="25"></td><td id="RanIter"><input title="nombre d'itérations à effectuer pour obtention des 'bonnes ressemblances'" type="text" id="RanItinp" minlength="1" maxlength="4" value="2500"></td><td id="RatioS"><input title="ratio minimal de surface pour accepter correspondance" type="text" id="RatioSinp" minlength="1" maxlength="4" value="0.50"></td>
                        </tr>
                    </tbody></table>
                    <table id="outputsTab"><tbody>
                        <tr><td id="timesOut" colspan="12">temps (ms)</td><td id="QuantitesOut" colspan="6">nombre de points</td><td id="CoeffOut" colspan="2">Coeffs</td><td id="TransfOut" colspan="4">Trans</td><td id="RectOOut" colspan="4">Orig</td><td id="RectDOut" colspan="4">Dest</td>
                        </tr><tr>
                            <td>T1G</td><td>T1Eg</td><td>T2G</td><td>T2Eg</td><td>TFC1</td><td>TFC2</td><td>TSSC1</td><td>TSSC2</td><td>TmsM</td><td>TmsGM</td><td>TmsQG</td><td>TmsT</td><td>Fnb1</td><td>Fnb2</td><td>NbC1</td><td>NbC2</td><td>NbMat</td><td>NbGM</td><td>QC</td><td>QG</td><td>ax1</td><td>cx</td><td>ay1</td><td>cy</td><td>O.x0</td><td>O.y0</td><td>O.xD</td><td>O.yD</td><td>D.x0</td><td>D.y0</td><td>D.xD</td><td>D.yD</td>
                        </tr><tr>
                            <td id="Tms1G">0</td><td id="Tms1Eg">0</td><td id="Tms2G">0</td><td id="Tms2Eg">0</td><td id="TmsFC1">0</td><td id="TmsFC2">0</td><td id="TmsSSC1">0</td><td id="TmsSSC2">0</td><td id="TmsM">0</td><td id="TmsGM">0</td><td id="TmsQG">0</td><td id="TmsT">0</td><td id="FCDnb1">0</td><td id="FCDnb2">0</td><td id="NbSSC1">0</td><td id="NbSSC2">0</td><td id="NbMat">0</td><td id="NbGM">0</td><td id="QC">0</td><td id="QG">0</td><td id="Homax1">0</td><td id="Homcx">0</td><td id="Homay1">0</td><td id="Homcy">0</td><td id="OxO">0</td><td id="OyO">0</td><td id="OxD">0</td><td id="OyD">0</td><td id="DxO">0</td><td id="DyO">0</td><td id="DxD">0</td><td id="DyD">0</td>
                        </tr>
                    </tbody></table>
                </div>
                <div id="ficselect">
                    <table id="selectfics">
                    <tr>
                        <td><input id="inp1" type="file" onchange="previewPicture(this)" accept=".jpg, .png, .gif"></td>
                        <td><button id="f_ok" class="styledvp">ok</button></td>
                        <td><input id="inp2" type="file" onchange="previewPicture(this)" accept=".jpg, .png, .gif"></td>
                    </tr></table>
                </div>
                <div>
                    <canvas id="visu"></canvas>
                </div>
            </div>
            <div id="basdepage">
                <hr />
                <div id="gauche">
                    <a href="https://validator.w3.org/check?uri=https://aroug.eu/correlimages/index.php"> <img src="https://www.w3.org/Icons/valid-xhtml10" alt="Valid XHTML 1.0 Transitional" height="31" width="88" /> </a>
                </div>
                <div id="centrebas">Document soumis à licence <a href="https://creativecommons.org/licenses/by/2.0/fr/">Creative Commons "by"</a></div>
                <div id="droite">
                    <a href="https://jigsaw.w3.org/css-validator/validator?uri=https://aroug.eu/correlimages/index.php"> <img style="border:0;width:88px;height:31px" src="https://jigsaw.w3.org/css-validator/images/vcss" alt="CSS Valide !" /> </a>
                    <!--a href="https://jigsaw.w3.org/css-validator/validator?uri=https://aroug.eu/correlimages/index.php"> <img style="border:0;width:88px;height:31px" src="../vcss.gif" alt="CSS Valide !" /> </a-->
                </div>
            </div>
        </div>
        <img id="img1" src="onepoint.png" alt="image1" style="display:none"><img id="img2" src="onepoint.png" alt="image2" style="display:none">
        <canvas id="Cv1" style="display:none"></canvas><canvas id="Cv2" style="display:none"></canvas>
    </body>

    </html>


Details

CorrelImages

Try to build a number between 0 and 100 for matching 2 images ...

Contexte

Je travaille sur un projet plus ambitieux visant à comparer deux vidéos d'un même film, éventuellement avec des tailles différentes, des framerates différents et des longueurs différentes pour en extraire les différences (portions rajoutées ou supprimées entre les deux versions). Dans cet objectif, je suis amené - après avoir "nettoyé" les éventuelles bandes noires", à comparer 2 images. 
Mes premières tentatives ont consisté à "mesurer" l'écart en nuances de gris entre deux images; pour deux versions "de mêmes dimensions d'images", cela est assez concluant. Malheureusement, pour deux images de tailles différentes - souffrant parfois d'un "resizing" et d'une éventuelle translation, les résultats sont nettement moins concluants. 
Je me suis donc orienté vers la recherche de points remarquables entre deux images pour mettre en évidence une fonction "un peu homothétique" entre ces deux images me permettant de définir deux blocs - chacun dans une image - à partir desquels la "différence de gris" est efficace, concluante.
Il est bien évident que les notions de points remarquables existent déjà (reconnaissance de formes, de visages, création d'images panoraùiques, SLAM, ...) mais je recherchais quelques techniques (technologies) "efficaces". CorrelImages permet non seulement de mettre en place ce Coefficient de Corrélation entre deux images mais permet surtout de "jouer" avec les différents paramètres d'entrée et de sauvegarder les tests réalisés afin de "peaufiner" l'incidence des paramètres tant au niveau de la qualité du résultat que de son efficacité en matière de temps.
Afin d'effectuer une majorité des traitements, je me suis aidé d'une bibliothèque déjà existante concernant le traitement d'images : JSFEAT ( https://github.com/inspirit/jsfeat ) mais j'ai du porter en javascript un algorithme visant à diminuer le nombre de points remarquables d'une image tout en respectant une certaine distribution spatiale homogène ( https://github.com/BAILOOL/ANMS-Codes ) et adapter à ma sauce l'algorithme de Ransac ( https://gist.github.com/nandor/7e74368a449924483173 ) permettant une bonne détermination de la transformation (qualifiée d'un peu homothétique plus haut) permettant de passer d'une image à l'autre.

Algorithme d'ensemble :

a) passage d'une image couleur en image à nuance de gris

b) égalisation éventuelle de l'histogramme (normalisation d'image)

c) détermination des points remarquables par l'algorithme "Fast Corners" (Jsfeat)

d) diminution des points remarquables avec homogéinisation dans l'espace (ANMS)

e) créations de descripteurs ORB pour les points résultants permettant (Jsfeat) :

f) de "matcher" les descripteurs / points remarquables des deux images ( Jsfeat )

g) et utilisation de Ransac pour filtrer parmi mes points récupérés à l'étape précédente ceux résultant "au mieux" d'une transformation reliant les 2 images

finalement :

h) détermination de 2 blocs "communs" et

i) calcul de la distance entre ces deux blocks.

Améliorations :

Citons le filtrage des paramètres d'entrée ... (function initialise_parametres_initiaux ) et ajout d'un dernier paramètre : le nombre minimal de "good matches" nécessaires avant de continuer le process (fixé à 10 dans CorrelImages à ce jour)

Utilisation :

https://aroug.eu/CorrelImages


Screenshots (1)  
  • help.png
  Files folder image Files (7)  
File Role Description
Accessible without login Plain text file index.php Example Example script
Accessible without login Plain text file jsfeat-min.js Data Auxiliary data
Accessible without login Plain text file jsfeat.js Data Auxiliary data
Accessible without login Plain text file LICENSE Lic. License text
Accessible without login Image file onepoint.png Icon Icon image
Accessible without login Plain text file README.md Doc. Documentation

The PHP Classes site has supported package installation using the Composer tool since 2013, as you may verify by reading this instructions page.
Install with Composer Install with Composer
 Version Control Unique User Downloads Download Rankings  
 100%
Total:52
This week:0
All time:10,661
This week:524Up