Js проверяет, были ли усеченные/поврежденные данные изображения

Я ищу, как определить, что данные изображения усечены\повреждены. Например, эта картина: введите описание изображения здесь

изображение данных не является полным (это более ощутимо для IE, а его отмечено как предупреждение в консоли firefox), но img.onerror не запускается, а img.completed - true.

demo: https://jsfiddle.net/7dd0ybb4/

var img = document.getElementById('MyPicture');

img.onerror = () => alert('error img');  
img.onload = () =>  console.log(img.complete); //true

img.src = "/img/54ef555013d7ed996aa23f1ae7b67f39.jpg";

Я хочу узнать об этом. если изображение имеет недопустимые данные.

Ответ 1

Вы можете декодировать изображение в массив байтов:

var src = 'here comes your base64 data'
var imageData = Uint8Array.from(atob(src.replace('data:image/jpeg;base64,', '')), c => c.charCodeAt(0))

JPEG должны начинать с байтов FF D8 и заканчиваться FF D9, поэтому мы проверяем, являются ли последние два элемента созданного буфера массива 255 и 217. См. живой пример.

var imageCorrupted = ((imageData[imageData.length - 1] === 217) && (imageData[imageData.length - 2] === 255));

Мы можем использовать аналогичную проверку для PNG (живой пример), который заканчивается блоком IEND, содержащим эту последовательность байтов:

// in hex: 00 00 00 00 49 45 4e 44 ae 42 60 82
var sequence = [0, 0, 0, 0, 73, 69, 78, 68, 174, 66, 96, 130];

Ответ 2

вы можете проверить поврежденные пиксели на холсте, как этот код

var ctx = document.getElementById('canvas').getContext('2d');
var img = new Image();
img.onload = function(){
    ctx.drawImage(img,0,0);
    var height=this.height;
    var width=this.width;
    var hCenter=Math.round(width/2);
    var corruptedRowsCount=0;
    for(var row=height;row>0;row--){
      var c = ctx.getImageData(hCenter, row, 1, 1).data;
      if(c[0]==0 && c[1]==0 && c[2]==0 && c[3]==0)
          corruptedRowsCount++;
      else 
          break;
    }
    console.log(Math.round(corruptedRowsCount*100/height)+" precent of image is corrupted");
};

img.src="";
<canvas id="canvas">