layout/tools/reftest/reftest-analyzer.xhtml

changeset 0
6474c204b198
     1.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     1.2 +++ b/layout/tools/reftest/reftest-analyzer.xhtml	Wed Dec 31 06:09:35 2014 +0100
     1.3 @@ -0,0 +1,579 @@
     1.4 +<?xml version="1.0" encoding="UTF-8"?>
     1.5 +<!-- -*- Mode: HTML; tab-width: 2; indent-tabs-mode: nil; -*- -->
     1.6 +<!-- vim: set shiftwidth=2 tabstop=2 autoindent expandtab: -->
     1.7 +<!-- This Source Code Form is subject to the terms of the Mozilla Public
     1.8 +   - License, v. 2.0. If a copy of the MPL was not distributed with this
     1.9 +   - file, You can obtain one at http://mozilla.org/MPL/2.0/. -->
    1.10 +<!--
    1.11 +
    1.12 +Features to add:
    1.13 +* make the left and right parts of the viewer independently scrollable
    1.14 +* make the test list filterable
    1.15 +** default to only showing unexpecteds
    1.16 +* add other ways to highlight differences other than circling?
    1.17 +* add zoom/pan to images
    1.18 +* Add ability to load log via XMLHttpRequest (also triggered via URL param)
    1.19 +* color the test list based on pass/fail and expected/unexpected/random/skip
    1.20 +* ability to load multiple logs ?
    1.21 +** rename them by clicking on the name and editing
    1.22 +** turn the test list into a collapsing tree view
    1.23 +** move log loading into popup from viewer UI
    1.24 +
    1.25 +-->
    1.26 +<!DOCTYPE html>
    1.27 +<html lang="en-US" xml:lang="en-US" xmlns="http://www.w3.org/1999/xhtml">
    1.28 +<head>
    1.29 +  <title>Reftest analyzer</title>
    1.30 +  <style type="text/css"><![CDATA[
    1.31 +
    1.32 +  html, body { margin: 0; }
    1.33 +  html { padding: 0; }
    1.34 +  body { padding: 4px; }
    1.35 +
    1.36 +  #pixelarea, #itemlist, #images { position: absolute; }
    1.37 +  #itemlist, #images { overflow: auto; }
    1.38 +  #pixelarea { top: 0; left: 0; width: 320px; height: 84px; overflow: visible }
    1.39 +  #itemlist { top: 84px; left: 0; width: 320px; bottom: 0; }
    1.40 +  #images { top: 0; bottom: 0; left: 320px; right: 0; }
    1.41 +
    1.42 +  #leftpane { width: 320px; }
    1.43 +  #images { position: fixed; top: 10px; left: 340px; }
    1.44 +
    1.45 +  form#imgcontrols { margin: 0; display: block; }
    1.46 +
    1.47 +  #itemlist > table { border-collapse: collapse; }
    1.48 +  #itemlist > table > tbody > tr > td { border: 1px solid; padding: 1px; }
    1.49 +
    1.50 +  /*
    1.51 +  #itemlist > table > tbody > tr.pass > td.url { background: lime; }
    1.52 +  #itemlist > table > tbody > tr.fail > td.url { background: red; }
    1.53 +  */
    1.54 +
    1.55 +  #magnification > svg { display: block; width: 84px; height: 84px; }
    1.56 +
    1.57 +  #pixelinfo { font: small sans-serif; position: absolute; width: 200px; left: 84px; }
    1.58 +  #pixelinfo table { border-collapse: collapse; }
    1.59 +  #pixelinfo table th { white-space: nowrap; text-align: left; padding: 0; }
    1.60 +  #pixelinfo table td { font-family: monospace; padding: 0 0 0 0.25em; }
    1.61 +
    1.62 +  #pixelhint { display: inline; color: #88f; cursor: help; }
    1.63 +  #pixelhint > * { display: none; position: absolute; margin: 8px 0 0 8px; padding: 4px; width: 400px; background: #ffa; color: black; box-shadow: 3px 3px 2px #888; z-index: 1; }
    1.64 +  #pixelhint:hover { color: #000; }
    1.65 +  #pixelhint:hover > * { display: block; }
    1.66 +  #pixelhint p { margin: 0; }
    1.67 +  #pixelhint p + p { margin-top: 1em; }
    1.68 +
    1.69 +  ]]></style>
    1.70 +  <script type="text/javascript"><![CDATA[
    1.71 +
    1.72 +var XLINK_NS = "http://www.w3.org/1999/xlink";
    1.73 +var SVG_NS = "http://www.w3.org/2000/svg";
    1.74 +var IMAGE_NOT_AVAILABLE = "";
    1.75 +
    1.76 +var gPhases = null;
    1.77 +
    1.78 +var gIDCache = {};
    1.79 +
    1.80 +var gMagPixPaths = [];     // 2D array of array-of-two <path> objects used in the pixel magnifier
    1.81 +var gMagWidth = 5;         // number of zoomed in pixels to show horizontally
    1.82 +var gMagHeight = 5;        // number of zoomed in pixels to show vertically
    1.83 +var gMagZoom = 16;         // size of the zoomed in pixels
    1.84 +var gImage1Data;           // ImageData object for the reference image
    1.85 +var gImage2Data;           // ImageData object for the test output image
    1.86 +var gFlashingPixels = [];  // array of <path> objects that should be flashed due to pixel color mismatch
    1.87 +
    1.88 +function ID(id) {
    1.89 +  if (!(id in gIDCache))
    1.90 +    gIDCache[id] = document.getElementById(id);
    1.91 +  return gIDCache[id];
    1.92 +}
    1.93 +
    1.94 +function hash_parameters() {
    1.95 +  var result = { };
    1.96 +  var params = window.location.hash.substr(1).split(/[&;]/);
    1.97 +  for (var i = 0; i < params.length; i++) {
    1.98 +    var parts = params[i].split("=");
    1.99 +    result[parts[0]] = unescape(unescape(parts[1]));
   1.100 +  }
   1.101 +  return result;
   1.102 +}
   1.103 +
   1.104 +function load() {
   1.105 +  gPhases = [ ID("entry"), ID("loading"), ID("viewer") ];
   1.106 +  build_mag();
   1.107 +  var params = hash_parameters();
   1.108 +  if (params.log) {
   1.109 +    ID("logentry").value = params.log;
   1.110 +    log_pasted();
   1.111 +  }
   1.112 +  window.addEventListener('keypress', maybe_load_image, false);
   1.113 +  ID("image1").addEventListener('error', image_load_error, false);
   1.114 +  ID("image2").addEventListener('error', image_load_error, false);
   1.115 +}
   1.116 +
   1.117 +function image_load_error(e) {
   1.118 +  e.target.setAttributeNS(XLINK_NS, "xlink:href", IMAGE_NOT_AVAILABLE);
   1.119 +}
   1.120 +
   1.121 +function build_mag() {
   1.122 +  var mag = ID("mag");
   1.123 +
   1.124 +  var r = document.createElementNS(SVG_NS, "rect");
   1.125 +  r.setAttribute("x", gMagZoom * -gMagWidth / 2);
   1.126 +  r.setAttribute("y", gMagZoom * -gMagHeight / 2);
   1.127 +  r.setAttribute("width", gMagZoom * gMagWidth);
   1.128 +  r.setAttribute("height", gMagZoom * gMagHeight);
   1.129 +  mag.appendChild(r);
   1.130 +
   1.131 +  mag.setAttribute("transform", "translate(" + (gMagZoom * (gMagWidth / 2) + 1) + "," + (gMagZoom * (gMagHeight / 2) + 1) + ")");
   1.132 +
   1.133 +  for (var x = 0; x < gMagWidth; x++) {
   1.134 +    gMagPixPaths[x] = [];
   1.135 +    for (var y = 0; y < gMagHeight; y++) {
   1.136 +      var p1 = document.createElementNS(SVG_NS, "path");
   1.137 +      p1.setAttribute("d", "M" + ((x - gMagWidth / 2) + 1) * gMagZoom + "," + (y - gMagHeight / 2) * gMagZoom + "h" + -gMagZoom + "v" + gMagZoom);
   1.138 +      p1.setAttribute("stroke", "black");
   1.139 +      p1.setAttribute("stroke-width", "1px");
   1.140 +      p1.setAttribute("fill", "#aaa");
   1.141 +
   1.142 +      var p2 = document.createElementNS(SVG_NS, "path");
   1.143 +      p2.setAttribute("d", "M" + ((x - gMagWidth / 2) + 1) * gMagZoom + "," + (y - gMagHeight / 2) * gMagZoom + "v" + gMagZoom + "h" + -gMagZoom);
   1.144 +      p2.setAttribute("stroke", "black");
   1.145 +      p2.setAttribute("stroke-width", "1px");
   1.146 +      p2.setAttribute("fill", "#888");
   1.147 +
   1.148 +      mag.appendChild(p1);
   1.149 +      mag.appendChild(p2);
   1.150 +      gMagPixPaths[x][y] = [p1, p2];
   1.151 +    }
   1.152 +  }
   1.153 +
   1.154 +  var flashedOn = false;
   1.155 +  setInterval(function() {
   1.156 +    flashedOn = !flashedOn;
   1.157 +    flash_pixels(flashedOn);
   1.158 +  }, 500);
   1.159 +}
   1.160 +
   1.161 +function show_phase(phaseid) {
   1.162 +  for (var i in gPhases) {
   1.163 +    var phase = gPhases[i];
   1.164 +    phase.style.display = (phase.id == phaseid) ? "" : "none";
   1.165 +  }
   1.166 +
   1.167 +  if (phase == "viewer")
   1.168 +    ID("images").style.display = "none";
   1.169 +}
   1.170 +
   1.171 +function fileentry_changed() {
   1.172 +  show_phase("loading");
   1.173 +  var input = ID("fileentry");
   1.174 +  var files = input.files;
   1.175 +  if (files.length > 0) {
   1.176 +    // Only handle the first file; don't handle multiple selection.
   1.177 +    // The parts of the log we care about are ASCII-only.  Since we
   1.178 +    // can ignore lines we don't care about, best to read in as
   1.179 +    // iso-8859-1, which guarantees we don't get decoding errors.
   1.180 +    var fileReader = new FileReader();
   1.181 +    fileReader.onload = function(e) {
   1.182 +      var log = null;
   1.183 +
   1.184 +      log = e.target.result;
   1.185 +
   1.186 +      if (log)
   1.187 +        process_log(log);
   1.188 +      else
   1.189 +        show_phase("entry");
   1.190 +    }
   1.191 +    fileReader.readAsText(files[0], "iso-8859-1");
   1.192 +  }
   1.193 +  // So the user can process the same filename again (after
   1.194 +  // overwriting the log), clear the value on the form input so we
   1.195 +  // will always get an onchange event.
   1.196 +  input.value = "";
   1.197 +}
   1.198 +
   1.199 +function log_pasted() {
   1.200 +  show_phase("loading");
   1.201 +  var entry = ID("logentry");
   1.202 +  var log = entry.value;
   1.203 +  entry.value = "";
   1.204 +  process_log(log);
   1.205 +}
   1.206 +
   1.207 +var gTestItems;
   1.208 +
   1.209 +function process_log(contents) {
   1.210 +  var lines = contents.split(/[\r\n]+/);
   1.211 +  gTestItems = [];
   1.212 +  for (var j in lines) {
   1.213 +    var line = lines[j];
   1.214 +    var match = line.match(/^.*?REFTEST (.*)$/);
   1.215 +    if (!match)
   1.216 +      continue;
   1.217 +    line = match[1];
   1.218 +    match = line.match(/^(TEST-PASS|TEST-UNEXPECTED-PASS|TEST-KNOWN-FAIL|TEST-UNEXPECTED-FAIL|TEST-DEBUG-INFO)(\(EXPECTED RANDOM\)|) \| ([^\|]+) \|(.*)/);
   1.219 +    if (match) {
   1.220 +      var state = match[1];
   1.221 +      var random = match[2];
   1.222 +      var url = match[3];
   1.223 +      var extra = match[4];
   1.224 +      gTestItems.push(
   1.225 +        {
   1.226 +          pass: !state.match(/DEBUG-INFO$|FAIL$/),
   1.227 +          // only one of the following three should ever be true
   1.228 +          unexpected: !!state.match(/^TEST-UNEXPECTED/),
   1.229 +          random: (random == "(EXPECTED RANDOM)"),
   1.230 +          skip: (extra == " (SKIP)"),
   1.231 +          url: url,
   1.232 +          images: []
   1.233 +        });
   1.234 +      continue;
   1.235 +    }
   1.236 +    match = line.match(/^  IMAGE[^:]*: (.*)$/);
   1.237 +    if (match) {
   1.238 +      var item = gTestItems[gTestItems.length - 1];
   1.239 +      item.images.push(match[1]);
   1.240 +    }
   1.241 +  }
   1.242 +
   1.243 +  build_viewer();
   1.244 +}
   1.245 +
   1.246 +function build_viewer() {
   1.247 +  if (gTestItems.length == 0) {
   1.248 +    show_phase("entry");
   1.249 +    return;
   1.250 +  }
   1.251 +
   1.252 +  var cell = ID("itemlist");
   1.253 +  while (cell.childNodes.length > 0)
   1.254 +    cell.removeChild(cell.childNodes[cell.childNodes.length - 1]);
   1.255 +
   1.256 +  var table = document.createElement("table");
   1.257 +  var tbody = document.createElement("tbody");
   1.258 +  table.appendChild(tbody);
   1.259 +
   1.260 +  for (var i in gTestItems) {
   1.261 +    var item = gTestItems[i];
   1.262 +
   1.263 +    // XXX skip expected pass items until we have filtering UI
   1.264 +    if (item.pass && !item.unexpected)
   1.265 +      continue;
   1.266 +
   1.267 +    var tr = document.createElement("tr");
   1.268 +    var rowclass = item.pass ? "pass" : "fail";
   1.269 +    var td;
   1.270 +    var text;
   1.271 +
   1.272 +    td = document.createElement("td");
   1.273 +    text = "";
   1.274 +    if (item.unexpected) { text += "!"; rowclass += " unexpected"; }
   1.275 +    if (item.random) { text += "R"; rowclass += " random"; }
   1.276 +    if (item.skip) { text += "S"; rowclass += " skip"; }
   1.277 +    td.appendChild(document.createTextNode(text));
   1.278 +    tr.appendChild(td);
   1.279 +
   1.280 +    td = document.createElement("td");
   1.281 +    td.className = "url";
   1.282 +    // Only display part of URL after "/mozilla/".
   1.283 +    var match = item.url.match(/\/mozilla\/(.*)/);
   1.284 +    text = document.createTextNode(match ? match[1] : item.url);
   1.285 +    if (item.images.length > 0) {
   1.286 +      var a = document.createElement("a");
   1.287 +      a.href = "javascript:show_images(" + i + ")";
   1.288 +      a.appendChild(text);
   1.289 +      td.appendChild(a);
   1.290 +    } else {
   1.291 +      td.appendChild(text);
   1.292 +    }
   1.293 +    tr.appendChild(td);
   1.294 +
   1.295 +    tbody.appendChild(tr);
   1.296 +  }
   1.297 +
   1.298 +  cell.appendChild(table);
   1.299 +
   1.300 +  show_phase("viewer");
   1.301 +}
   1.302 +
   1.303 +function get_image_data(src, whenReady) {
   1.304 +  var img = new Image();
   1.305 +  img.onload = function() {
   1.306 +    var canvas = document.createElement("canvas");
   1.307 +    canvas.width = 800;
   1.308 +    canvas.height = 1000;
   1.309 +
   1.310 +    var ctx = canvas.getContext("2d");
   1.311 +    ctx.drawImage(img, 0, 0);
   1.312 +
   1.313 +    whenReady(ctx.getImageData(0, 0, 800, 1000));
   1.314 +  };
   1.315 +  img.src = src;
   1.316 +}
   1.317 +
   1.318 +function show_images(i) {
   1.319 +  var item = gTestItems[i];
   1.320 +  var cell = ID("images");
   1.321 +
   1.322 +  ID("image1").style.display = "";
   1.323 +  ID("image2").style.display = "none";
   1.324 +  ID("diffrect").style.display = "none";
   1.325 +  ID("imgcontrols").reset();
   1.326 +
   1.327 +  ID("image1").setAttributeNS(XLINK_NS, "xlink:href", item.images[0]);
   1.328 +  // Making the href be #image1 doesn't seem to work
   1.329 +  ID("feimage1").setAttributeNS(XLINK_NS, "xlink:href", item.images[0]);
   1.330 +  if (item.images.length == 1) {
   1.331 +    ID("imgcontrols").style.display = "none";
   1.332 +  } else {
   1.333 +    ID("imgcontrols").style.display = "";
   1.334 +
   1.335 +    ID("image2").setAttributeNS(XLINK_NS, "xlink:href", item.images[1]);
   1.336 +    // Making the href be #image2 doesn't seem to work
   1.337 +    ID("feimage2").setAttributeNS(XLINK_NS, "xlink:href", item.images[1]);
   1.338 +  }
   1.339 +
   1.340 +  cell.style.display = "";
   1.341 +
   1.342 +  get_image_data(item.images[0], function(data) { gImage1Data = data });
   1.343 +  get_image_data(item.images[1], function(data) { gImage2Data = data });
   1.344 +}
   1.345 +
   1.346 +function show_image(i) {
   1.347 +  if (i == 1) {
   1.348 +    ID("image1").style.display = "";
   1.349 +    ID("image2").style.display = "none";
   1.350 +  } else {
   1.351 +    ID("image1").style.display = "none";
   1.352 +    ID("image2").style.display = "";
   1.353 +  }
   1.354 +}
   1.355 +
   1.356 +function maybe_load_image(event) {
   1.357 +  switch (event.charCode) {
   1.358 +  case 49: // "1" key
   1.359 +    document.getElementById("radio1").checked = true;
   1.360 +    show_image(1);
   1.361 +    break;
   1.362 +  case 50: // "2" key
   1.363 +    document.getElementById("radio2").checked = true;
   1.364 +    show_image(2);
   1.365 +    break;
   1.366 +  }
   1.367 +}
   1.368 +
   1.369 +function show_differences(cb) {
   1.370 +  ID("diffrect").style.display = cb.checked ? "" : "none";
   1.371 +}
   1.372 +
   1.373 +function flash_pixels(on) {
   1.374 +  var stroke = on ? "red" : "black";
   1.375 +  var strokeWidth = on ? "2px" : "1px";
   1.376 +  for (var i = 0; i < gFlashingPixels.length; i++) {
   1.377 +    gFlashingPixels[i].setAttribute("stroke", stroke);
   1.378 +    gFlashingPixels[i].setAttribute("stroke-width", strokeWidth);
   1.379 +  }
   1.380 +}
   1.381 +
   1.382 +function cursor_point(evt) {
   1.383 +  var m = evt.target.getScreenCTM().inverse();
   1.384 +  var p = ID("svg").createSVGPoint();
   1.385 +  p.x = evt.clientX;
   1.386 +  p.y = evt.clientY;
   1.387 +  p = p.matrixTransform(m);
   1.388 +  return { x: Math.floor(p.x), y: Math.floor(p.y) };
   1.389 +}
   1.390 +
   1.391 +function hex2(i) {
   1.392 +  return (i < 16 ? "0" : "") + i.toString(16);
   1.393 +}
   1.394 +
   1.395 +function canvas_pixel_as_hex(data, x, y) {
   1.396 +  var offset = (y * data.width + x) * 4;
   1.397 +  var r = data.data[offset];
   1.398 +  var g = data.data[offset + 1];
   1.399 +  var b = data.data[offset + 2];
   1.400 +  return "#" + hex2(r) + hex2(g) + hex2(b);
   1.401 +}
   1.402 +
   1.403 +function hex_as_rgb(hex) {
   1.404 +  return "rgb(" + [parseInt(hex.substring(1, 3), 16), parseInt(hex.substring(3, 5), 16), parseInt(hex.substring(5, 7), 16)] + ")";
   1.405 +}
   1.406 +
   1.407 +function magnify(evt) {
   1.408 +  var { x: x, y: y } = cursor_point(evt);
   1.409 +  var centerPixelColor1, centerPixelColor2;
   1.410 +
   1.411 +  var dx_lo = -Math.floor(gMagWidth / 2);
   1.412 +  var dx_hi = Math.floor(gMagWidth / 2);
   1.413 +  var dy_lo = -Math.floor(gMagHeight / 2);
   1.414 +  var dy_hi = Math.floor(gMagHeight / 2);
   1.415 +
   1.416 +  flash_pixels(false);
   1.417 +  gFlashingPixels = [];
   1.418 +  for (var j = dy_lo; j <= dy_hi; j++) {
   1.419 +    for (var i = dx_lo; i <= dx_hi; i++) {
   1.420 +      var px = x + i;
   1.421 +      var py = y + j;
   1.422 +      var p1 = gMagPixPaths[i + dx_hi][j + dy_hi][0];
   1.423 +      var p2 = gMagPixPaths[i + dx_hi][j + dy_hi][1];
   1.424 +      if (px < 0 || py < 0 || px >= 800 || py >= 1000) {
   1.425 +        p1.setAttribute("fill", "#aaa");
   1.426 +        p2.setAttribute("fill", "#888");
   1.427 +      } else {
   1.428 +        var color1 = canvas_pixel_as_hex(gImage1Data, x + i, y + j);
   1.429 +        var color2 = canvas_pixel_as_hex(gImage2Data, x + i, y + j);
   1.430 +        p1.setAttribute("fill", color1);
   1.431 +        p2.setAttribute("fill", color2);
   1.432 +        if (color1 != color2) {
   1.433 +          gFlashingPixels.push(p1, p2);
   1.434 +          p1.parentNode.appendChild(p1);
   1.435 +          p2.parentNode.appendChild(p2);
   1.436 +        }
   1.437 +        if (i == 0 && j == 0) {
   1.438 +          centerPixelColor1 = color1;
   1.439 +          centerPixelColor2 = color2;
   1.440 +        }
   1.441 +      }
   1.442 +    }
   1.443 +  }
   1.444 +  flash_pixels(true);
   1.445 +  show_pixelinfo(x, y, centerPixelColor1, hex_as_rgb(centerPixelColor1), centerPixelColor2, hex_as_rgb(centerPixelColor2));
   1.446 +}
   1.447 +
   1.448 +function show_pixelinfo(x, y, pix1rgb, pix1hex, pix2rgb, pix2hex) {
   1.449 +  var pixelinfo = ID("pixelinfo");
   1.450 +  ID("coords").textContent = [x, y];
   1.451 +  ID("pix1hex").textContent = pix1hex;
   1.452 +  ID("pix1rgb").textContent = pix1rgb;
   1.453 +  ID("pix2hex").textContent = pix2hex;
   1.454 +  ID("pix2rgb").textContent = pix2rgb;
   1.455 +}
   1.456 +
   1.457 +  ]]></script>
   1.458 +
   1.459 +</head>
   1.460 +<body onload="load()">
   1.461 +
   1.462 +<div id="entry">
   1.463 +
   1.464 +<h1>Reftest analyzer: load reftest log</h1>
   1.465 +
   1.466 +<p>Either paste your log into this textarea:<br />
   1.467 +<textarea cols="80" rows="10" id="logentry"/><br/>
   1.468 +<input type="button" value="Process pasted log" onclick="log_pasted()" /></p>
   1.469 +
   1.470 +<p>... or load it from a file:<br/>
   1.471 +<input type="file" id="fileentry" onchange="fileentry_changed()" />
   1.472 +</p>
   1.473 +</div>
   1.474 +
   1.475 +<div id="loading" style="display:none">Loading log...</div>
   1.476 +
   1.477 +<div id="viewer" style="display:none">
   1.478 +  <div id="pixelarea">
   1.479 +    <div id="pixelinfo">
   1.480 +      <table>
   1.481 +        <tbody>
   1.482 +          <tr><th>Pixel at:</th><td colspan="2" id="coords"/></tr>
   1.483 +          <tr><th>Image 1:</th><td id="pix1rgb"></td><td id="pix1hex"></td></tr>
   1.484 +          <tr><th>Image 2:</th><td id="pix2rgb"></td><td id="pix2hex"></td></tr>
   1.485 +        </tbody>
   1.486 +      </table>
   1.487 +      <div>
   1.488 +        <div id="pixelhint">★
   1.489 +          <div>
   1.490 +            <p>Move the mouse over the reftest image on the right to show
   1.491 +            magnified pixels on the left.  The color information above is for
   1.492 +            the pixel centered in the magnified view.</p>
   1.493 +            <p>Image 1 is shown in the upper triangle of each pixel and Image 2
   1.494 +            is shown in the lower triangle.</p>
   1.495 +          </div>
   1.496 +        </div>
   1.497 +      </div>
   1.498 +    </div>
   1.499 +    <div id="magnification">
   1.500 +      <svg xmlns="http://www.w3.org/2000/svg" width="84" height="84" shape-rendering="optimizeSpeed">
   1.501 +        <g id="mag"/>
   1.502 +      </svg>
   1.503 +    </div>
   1.504 +  </div>
   1.505 +  <div id="itemlist"></div>
   1.506 +  <div id="images" style="display:none">
   1.507 +    <form id="imgcontrols">
   1.508 +    <label title="1"><input id="radio1" type="radio" name="which" value="0" onchange="show_image(1)" checked="checked" />Image 1</label>
   1.509 +    <label title="2"><input id="radio2" type="radio" name="which" value="1" onchange="show_image(2)" />Image 2</label>
   1.510 +    <label><input type="checkbox" onchange="show_differences(this)" />Circle differences</label>
   1.511 +    </form>
   1.512 +    <svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" version="1.1" width="800px" height="1000px" viewBox="0 0 800 1000" id="svg">
   1.513 +      <defs>
   1.514 +        <!-- use sRGB to avoid loss of data -->
   1.515 +        <filter id="showDifferences" x="0%" y="0%" width="100%" height="100%"
   1.516 +                style="color-interpolation-filters: sRGB">
   1.517 +          <feImage id="feimage1" result="img1" xlink:href="#image1" />
   1.518 +          <feImage id="feimage2" result="img2" xlink:href="#image2" />
   1.519 +          <!-- inv1 and inv2 are the images with RGB inverted -->
   1.520 +          <feComponentTransfer result="inv1" in="img1">
   1.521 +            <feFuncR type="linear" slope="-1" intercept="1" />
   1.522 +            <feFuncG type="linear" slope="-1" intercept="1" />
   1.523 +            <feFuncB type="linear" slope="-1" intercept="1" />
   1.524 +          </feComponentTransfer>
   1.525 +          <feComponentTransfer result="inv2" in="img2">
   1.526 +            <feFuncR type="linear" slope="-1" intercept="1" />
   1.527 +            <feFuncG type="linear" slope="-1" intercept="1" />
   1.528 +            <feFuncB type="linear" slope="-1" intercept="1" />
   1.529 +          </feComponentTransfer>
   1.530 +          <!-- w1 will have non-white pixels anywhere that img2
   1.531 +               is brighter than img1, and w2 for the reverse.
   1.532 +               It would be nice not to have to go through these
   1.533 +               intermediate states, but feComposite
   1.534 +               type="arithmetic" can't transform the RGB channels
   1.535 +               and leave the alpha channel untouched. -->
   1.536 +          <feComposite result="w1" in="img1" in2="inv2" operator="arithmetic" k2="1" k3="1" />
   1.537 +          <feComposite result="w2" in="img2" in2="inv1" operator="arithmetic" k2="1" k3="1" />
   1.538 +          <!-- c1 will have non-black pixels anywhere that img2
   1.539 +               is brighter than img1, and c2 for the reverse -->
   1.540 +          <feComponentTransfer result="c1" in="w1">
   1.541 +            <feFuncR type="linear" slope="-1" intercept="1" />
   1.542 +            <feFuncG type="linear" slope="-1" intercept="1" />
   1.543 +            <feFuncB type="linear" slope="-1" intercept="1" />
   1.544 +          </feComponentTransfer>
   1.545 +          <feComponentTransfer result="c2" in="w2">
   1.546 +            <feFuncR type="linear" slope="-1" intercept="1" />
   1.547 +            <feFuncG type="linear" slope="-1" intercept="1" />
   1.548 +            <feFuncB type="linear" slope="-1" intercept="1" />
   1.549 +          </feComponentTransfer>
   1.550 +          <!-- c will be nonblack (and fully on) for every pixel+component where there are differences -->
   1.551 +          <feComposite result="c" in="c1" in2="c2" operator="arithmetic" k2="255" k3="255" />
   1.552 +          <!-- a will be opaque for every pixel with differences and transparent for all others -->
   1.553 +          <feColorMatrix result="a" type="matrix" values="0 0 0 0 0  0 0 0 0 0  0 0 0 0 0  1 1 1 0 0" />
   1.554 +
   1.555 +          <!-- a, dilated by 1 pixel -->
   1.556 +          <feMorphology result="dila1" in="a" operator="dilate" radius="1" />
   1.557 +          <!-- a, dilated by 2 pixels -->
   1.558 +          <feMorphology result="dila2" in="dila1" operator="dilate" radius="1" />
   1.559 +
   1.560 +          <!-- all the pixels in the 2-pixel dilation of a but not in the 1-pixel dilation, to highlight the diffs -->
   1.561 +          <feComposite result="highlight" in="dila2" in2="dila1" operator="out" />
   1.562 +
   1.563 +          <feFlood result="red" flood-color="red" />
   1.564 +          <feComposite result="redhighlight" in="red" in2="highlight" operator="in" />
   1.565 +          <feFlood result="black" flood-color="black" flood-opacity="0.5" />
   1.566 +          <feMerge>
   1.567 +            <feMergeNode in="black" />
   1.568 +            <feMergeNode in="redhighlight" />
   1.569 +          </feMerge>
   1.570 +        </filter>
   1.571 +      </defs>
   1.572 +      <g onmousemove="magnify(evt)">
   1.573 +        <image x="0" y="0" width="100%" height="100%" id="image1" />
   1.574 +        <image x="0" y="0" width="100%" height="100%" id="image2" />
   1.575 +      </g>
   1.576 +      <rect id="diffrect" filter="url(#showDifferences)" pointer-events="none" x="0" y="0" width="100%" height="100%" />
   1.577 +    </svg>
   1.578 +  </div>
   1.579 +</div>
   1.580 +
   1.581 +</body>
   1.582 +</html>

mercurial