<?xml version="1.0" encoding="UTF-8"?> <!-- -*- Mode: HTML; tab-width: 2; indent-tabs-mode: nil; -*- --> <!-- vim: set shiftwidth=2 tabstop=2 autoindent expandtab: --> <!-- This Source Code Form is subject to the terms of the Mozilla Public - License, v. 2.0. If a copy of the MPL was not distributed with this - file, You can obtain one at http://mozilla.org/MPL/2.0/. --> <!-- Features to add: * make the left and right parts of the viewer independently scrollable * make the test list filterable ** default to only showing unexpecteds * add other ways to highlight differences other than circling? * add zoom/pan to images * Add ability to load log via XMLHttpRequest (also triggered via URL param) * color the test list based on pass/fail and expected/unexpected/random/skip * ability to load multiple logs ? ** rename them by clicking on the name and editing ** turn the test list into a collapsing tree view ** move log loading into popup from viewer UI --> <!DOCTYPE html> <html lang="en-US" xml:lang="en-US" xmlns="http://www.w3.org/1999/xhtml"> <head> <title>Reftest analyzer</title> <style type="text/css"><![CDATA[ html, body { margin: 0; } html { padding: 0; } body { padding: 4px; } #pixelarea, #itemlist, #images { position: absolute; } #itemlist, #images { overflow: auto; } #pixelarea { top: 0; left: 0; width: 320px; height: 84px; overflow: visible } #itemlist { top: 84px; left: 0; width: 320px; bottom: 0; } #images { top: 0; bottom: 0; left: 320px; right: 0; } #leftpane { width: 320px; } #images { position: fixed; top: 10px; left: 340px; } form#imgcontrols { margin: 0; display: block; } #itemlist > table { border-collapse: collapse; } #itemlist > table > tbody > tr > td { border: 1px solid; padding: 1px; } #itemlist td.activeitem { background-color: yellow; } /* #itemlist > table > tbody > tr.pass > td.url { background: lime; } #itemlist > table > tbody > tr.fail > td.url { background: red; } */ #magnification > svg { display: block; width: 84px; height: 84px; } #pixelinfo { font: small sans-serif; position: absolute; width: 200px; left: 84px; } #pixelinfo table { border-collapse: collapse; } #pixelinfo table th { white-space: nowrap; text-align: left; padding: 0; } #pixelinfo table td { font-family: monospace; padding: 0 0 0 0.25em; } #pixelhint { display: inline; color: #88f; cursor: help; } #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; } #pixelhint:hover { color: #000; } #pixelhint:hover > * { display: block; } #pixelhint p { margin: 0; } #pixelhint p + p { margin-top: 1em; } ]]></style> <script type="text/javascript"><![CDATA[ var XLINK_NS = "http://www.w3.org/1999/xlink"; var SVG_NS = "http://www.w3.org/2000/svg"; var IMAGE_NOT_AVAILABLE = ""; var gPhases = null; var gIDCache = {}; var gMagPixPaths = []; // 2D array of array-of-two <path> objects used in the pixel magnifier var gMagWidth = 5; // number of zoomed in pixels to show horizontally var gMagHeight = 5; // number of zoomed in pixels to show vertically var gMagZoom = 16; // size of the zoomed in pixels var gImage1Data; // ImageData object for the reference image var gImage2Data; // ImageData object for the test output image var gFlashingPixels = []; // array of <path> objects that should be flashed due to pixel color mismatch var gParams; function ID(id) { if (!(id in gIDCache)) gIDCache[id] = document.getElementById(id); return gIDCache[id]; } function hash_parameters() { var result = { }; var params = window.location.hash.substr(1).split(/[&;]/); for (var i = 0; i < params.length; i++) { var parts = params[i].split("="); result[parts[0]] = unescape(unescape(parts[1])); } return result; } function load() { gPhases = [ ID("entry"), ID("loading"), ID("viewer") ]; build_mag(); gParams = hash_parameters(); if (gParams.log) { show_phase("loading"); process_log(gParams.log); } else if (gParams.logurl) { show_phase("loading"); var req = new XMLHttpRequest(); req.onreadystatechange = function() { if (req.readyState === 4) { process_log(req.responseText); } }; req.open('GET', gParams.logurl, true); req.send(); } window.addEventListener('keypress', handle_keyboard_shortcut, false); ID("image1").addEventListener('error', image_load_error, false); ID("image2").addEventListener('error', image_load_error, false); } function image_load_error(e) { e.target.setAttributeNS(XLINK_NS, "xlink:href", IMAGE_NOT_AVAILABLE); } function build_mag() { var mag = ID("mag"); var r = document.createElementNS(SVG_NS, "rect"); r.setAttribute("x", gMagZoom * -gMagWidth / 2); r.setAttribute("y", gMagZoom * -gMagHeight / 2); r.setAttribute("width", gMagZoom * gMagWidth); r.setAttribute("height", gMagZoom * gMagHeight); mag.appendChild(r); mag.setAttribute("transform", "translate(" + (gMagZoom * (gMagWidth / 2) + 1) + "," + (gMagZoom * (gMagHeight / 2) + 1) + ")"); for (var x = 0; x < gMagWidth; x++) { gMagPixPaths[x] = []; for (var y = 0; y < gMagHeight; y++) { var p1 = document.createElementNS(SVG_NS, "path"); p1.setAttribute("d", "M" + ((x - gMagWidth / 2) + 1) * gMagZoom + "," + (y - gMagHeight / 2) * gMagZoom + "h" + -gMagZoom + "v" + gMagZoom); p1.setAttribute("stroke", "black"); p1.setAttribute("stroke-width", "1px"); p1.setAttribute("fill", "#aaa"); var p2 = document.createElementNS(SVG_NS, "path"); p2.setAttribute("d", "M" + ((x - gMagWidth / 2) + 1) * gMagZoom + "," + (y - gMagHeight / 2) * gMagZoom + "v" + gMagZoom + "h" + -gMagZoom); p2.setAttribute("stroke", "black"); p2.setAttribute("stroke-width", "1px"); p2.setAttribute("fill", "#888"); mag.appendChild(p1); mag.appendChild(p2); gMagPixPaths[x][y] = [p1, p2]; } } var flashedOn = false; setInterval(function() { flashedOn = !flashedOn; flash_pixels(flashedOn); }, 500); } function show_phase(phaseid) { for (var i in gPhases) { var phase = gPhases[i]; phase.style.display = (phase.id == phaseid) ? "" : "none"; } if (phase == "viewer") ID("images").style.display = "none"; } function fileentry_changed() { show_phase("loading"); var input = ID("fileentry"); var files = input.files; if (files.length > 0) { // Only handle the first file; don't handle multiple selection. // The parts of the log we care about are ASCII-only. Since we // can ignore lines we don't care about, best to read in as // iso-8859-1, which guarantees we don't get decoding errors. var fileReader = new FileReader(); fileReader.onload = function(e) { var log = null; log = e.target.result; if (log) process_log(log); else show_phase("entry"); } fileReader.readAsText(files[0], "iso-8859-1"); } // So the user can process the same filename again (after // overwriting the log), clear the value on the form input so we // will always get an onchange event. input.value = ""; } function log_pasted() { show_phase("loading"); var entry = ID("logentry"); var log = entry.value; entry.value = ""; process_log(log); } var gTestItems; function process_log(contents) { var lines = contents.split(/[\r\n]+/); gTestItems = []; for (var j in lines) { var line = lines[j]; // Ignore duplicated output in logcat. if (line.match(/I\/Gecko.*?REFTEST/)) continue; var match = line.match(/^.*?REFTEST (.*)$/); if (!match) continue; line = match[1]; match = line.match(/^(TEST-PASS|TEST-UNEXPECTED-PASS|TEST-KNOWN-FAIL|TEST-UNEXPECTED-FAIL|TEST-DEBUG-INFO)(\(EXPECTED RANDOM\)|) \| ([^\|]+) \|(.*)/); if (match) { var state = match[1]; var random = match[2]; var url = match[3]; var extra = match[4]; gTestItems.push( { pass: !state.match(/DEBUG-INFO$|FAIL$/), // only one of the following three should ever be true unexpected: !!state.match(/^TEST-UNEXPECTED/), random: (random == "(EXPECTED RANDOM)"), skip: (extra == " (SKIP)"), url: url, images: [], imageLabels: [] }); continue; } match = line.match(/IMAGE([^:]*): (data:.*)$/); if (match) { var item = gTestItems[gTestItems.length - 1]; item.images.push(match[2]); item.imageLabels.push(match[1]); } } build_viewer(); } function build_viewer() { if (gTestItems.length == 0) { show_phase("entry"); return; } var cell = ID("itemlist"); while (cell.childNodes.length > 0) cell.removeChild(cell.childNodes[cell.childNodes.length - 1]); var table = document.createElement("table"); var tbody = document.createElement("tbody"); table.appendChild(tbody); for (var i in gTestItems) { var item = gTestItems[i]; // optional url filter for only showing unexpected results if (parseInt(gParams.only_show_unexpected) && !item.unexpected) continue; // XXX regardless skip expected pass items until we have filtering UI if (item.pass && !item.unexpected) continue; var tr = document.createElement("tr"); var rowclass = item.pass ? "pass" : "fail"; var td; var text; td = document.createElement("td"); text = ""; if (item.unexpected) { text += "!"; rowclass += " unexpected"; } if (item.random) { text += "R"; rowclass += " random"; } if (item.skip) { text += "S"; rowclass += " skip"; } td.appendChild(document.createTextNode(text)); tr.appendChild(td); td = document.createElement("td"); td.id = "item" + i; td.className = "url"; // Only display part of URL after "/mozilla/". var match = item.url.match(/\/mozilla\/(.*)/); text = document.createTextNode(match ? match[1] : item.url); if (item.images.length > 0) { var a = document.createElement("a"); a.href = "javascript:show_images(" + i + ")"; a.appendChild(text); td.appendChild(a); } else { td.appendChild(text); } tr.appendChild(td); tbody.appendChild(tr); } cell.appendChild(table); show_phase("viewer"); } function get_image_data(src, whenReady) { var img = new Image(); img.onload = function() { var canvas = document.createElement("canvas"); canvas.width = img.naturalWidth; canvas.height = img.naturalHeight; var ctx = canvas.getContext("2d"); ctx.drawImage(img, 0, 0); whenReady(ctx.getImageData(0, 0, img.naturalWidth, img.naturalHeight)); }; img.src = src; } function sync_svg_size(imageData) { // We need the size of the 'svg' and its 'image' elements to match the size // of the ImageData objects that we're going to read pixels from or else our // magnify() function will be very broken. ID("svg").setAttribute("width", imageData.width); ID("svg").setAttribute("height", imageData.height); } function show_images(i) { var item = gTestItems[i]; var cell = ID("images"); // Remove activeitem class from any existing elements var activeItems = document.querySelectorAll(".activeitem"); for (var activeItemIdx = activeItems.length; activeItemIdx-- != 0;) { activeItems[activeItemIdx].classList.remove("activeitem"); } ID("item" + i).classList.add("activeitem"); ID("image1").style.display = ""; ID("image2").style.display = "none"; ID("diffrect").style.display = "none"; ID("imgcontrols").reset(); ID("image1").setAttributeNS(XLINK_NS, "xlink:href", item.images[0]); // Making the href be #image1 doesn't seem to work ID("feimage1").setAttributeNS(XLINK_NS, "xlink:href", item.images[0]); if (item.images.length == 1) { ID("imgcontrols").style.display = "none"; } else { ID("imgcontrols").style.display = ""; ID("image2").setAttributeNS(XLINK_NS, "xlink:href", item.images[1]); // Making the href be #image2 doesn't seem to work ID("feimage2").setAttributeNS(XLINK_NS, "xlink:href", item.images[1]); ID("label1").textContent = 'Image ' + item.imageLabels[0]; ID("label2").textContent = 'Image ' + item.imageLabels[1]; } cell.style.display = ""; get_image_data(item.images[0], function(data) { gImage1Data = data; sync_svg_size(gImage1Data); }); get_image_data(item.images[1], function(data) { gImage2Data = data }); } function show_image(i) { if (i == 1) { ID("image1").style.display = ""; ID("image2").style.display = "none"; } else { ID("image1").style.display = "none"; ID("image2").style.display = ""; } } function handle_keyboard_shortcut(event) { switch (event.charCode) { case 49: // "1" key document.getElementById("radio1").checked = true; show_image(1); break; case 50: // "2" key document.getElementById("radio2").checked = true; show_image(2); break; case 100: // "d" key document.getElementById("differences").click(); break; case 112: // "p" key shift_images(-1); break; case 110: // "n" key shift_images(1); break; } } function shift_images(dir) { var activeItem = document.querySelector(".activeitem"); if (!activeItem) { return; } for (var elm = activeItem; elm; elm = elm.parentElement) { if (elm.tagName != "tr") { continue; } elm = dir > 0 ? elm.nextElementSibling : elm.previousElementSibling; if (elm) { elm.getElementsByTagName("a")[0].click(); } return; } } function show_differences(cb) { ID("diffrect").style.display = cb.checked ? "" : "none"; } function flash_pixels(on) { var stroke = on ? "red" : "black"; var strokeWidth = on ? "2px" : "1px"; for (var i = 0; i < gFlashingPixels.length; i++) { gFlashingPixels[i].setAttribute("stroke", stroke); gFlashingPixels[i].setAttribute("stroke-width", strokeWidth); } } function cursor_point(evt) { var m = evt.target.getScreenCTM().inverse(); var p = ID("svg").createSVGPoint(); p.x = evt.clientX; p.y = evt.clientY; p = p.matrixTransform(m); return { x: Math.floor(p.x), y: Math.floor(p.y) }; } function hex2(i) { return (i < 16 ? "0" : "") + i.toString(16); } function canvas_pixel_as_hex(data, x, y) { var offset = (y * data.width + x) * 4; var r = data.data[offset]; var g = data.data[offset + 1]; var b = data.data[offset + 2]; return "#" + hex2(r) + hex2(g) + hex2(b); } function hex_as_rgb(hex) { return "rgb(" + [parseInt(hex.substring(1, 3), 16), parseInt(hex.substring(3, 5), 16), parseInt(hex.substring(5, 7), 16)] + ")"; } function magnify(evt) { var { x: x, y: y } = cursor_point(evt); var centerPixelColor1, centerPixelColor2; var dx_lo = -Math.floor(gMagWidth / 2); var dx_hi = Math.floor(gMagWidth / 2); var dy_lo = -Math.floor(gMagHeight / 2); var dy_hi = Math.floor(gMagHeight / 2); flash_pixels(false); gFlashingPixels = []; for (var j = dy_lo; j <= dy_hi; j++) { for (var i = dx_lo; i <= dx_hi; i++) { var px = x + i; var py = y + j; var p1 = gMagPixPaths[i + dx_hi][j + dy_hi][0]; var p2 = gMagPixPaths[i + dx_hi][j + dy_hi][1]; // Here we just use the dimensions of gImage1Data since we expect test // and reference to have the same dimensions. if (px < 0 || py < 0 || px >= gImage1Data.width || py >= gImage1Data.height) { p1.setAttribute("fill", "#aaa"); p2.setAttribute("fill", "#888"); } else { var color1 = canvas_pixel_as_hex(gImage1Data, x + i, y + j); var color2 = canvas_pixel_as_hex(gImage2Data, x + i, y + j); p1.setAttribute("fill", color1); p2.setAttribute("fill", color2); if (color1 != color2) { gFlashingPixels.push(p1, p2); p1.parentNode.appendChild(p1); p2.parentNode.appendChild(p2); } if (i == 0 && j == 0) { centerPixelColor1 = color1; centerPixelColor2 = color2; } } } } flash_pixels(true); show_pixelinfo(x, y, centerPixelColor1, hex_as_rgb(centerPixelColor1), centerPixelColor2, hex_as_rgb(centerPixelColor2)); } function show_pixelinfo(x, y, pix1rgb, pix1hex, pix2rgb, pix2hex) { var pixelinfo = ID("pixelinfo"); ID("coords").textContent = [x, y]; ID("pix1hex").textContent = pix1hex; ID("pix1rgb").textContent = pix1rgb; ID("pix2hex").textContent = pix2hex; ID("pix2rgb").textContent = pix2rgb; } ]]></script> </head> <body onload="load()"> <div id="entry"> <h1>Reftest analyzer: load reftest log</h1> <p>Either paste your log into this textarea:<br /> <textarea cols="80" rows="10" id="logentry"/><br/> <input type="button" value="Process pasted log" onclick="log_pasted()" /></p> <p>... or load it from a file:<br/> <input type="file" id="fileentry" onchange="fileentry_changed()" /> </p> </div> <div id="loading" style="display:none">Loading log...</div> <div id="viewer" style="display:none"> <div id="pixelarea"> <div id="pixelinfo"> <table> <tbody> <tr><th>Pixel at:</th><td colspan="2" id="coords"/></tr> <tr><th>Image 1:</th><td id="pix1rgb"></td><td id="pix1hex"></td></tr> <tr><th>Image 2:</th><td id="pix2rgb"></td><td id="pix2hex"></td></tr> </tbody> </table> <div> <div id="pixelhint">★ <div> <p>Move the mouse over the reftest image on the right to show magnified pixels on the left. The color information above is for the pixel centered in the magnified view.</p> <p>Image 1 is shown in the upper triangle of each pixel and Image 2 is shown in the lower triangle.</p> </div> </div> </div> </div> <div id="magnification"> <svg xmlns="http://www.w3.org/2000/svg" width="84" height="84" shape-rendering="optimizeSpeed"> <g id="mag"/> </svg> </div> </div> <div id="itemlist"></div> <div id="images" style="display:none"> <form id="imgcontrols"> <input id="radio1" type="radio" name="which" value="0" onchange="show_image(1)" checked="checked" /><label id="label1" title="1" for="radio1">Image 1</label> <input id="radio2" type="radio" name="which" value="1" onchange="show_image(2)" /><label id="label2" title="2" for="radio2">Image 2</label> <label><input id="differences" type="checkbox" onchange="show_differences(this)" />Circle differences</label> </form> <svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" version="1.1" width="800" height="1000" id="svg"> <defs> <!-- use sRGB to avoid loss of data --> <filter id="showDifferences" x="0%" y="0%" width="100%" height="100%" style="color-interpolation-filters: sRGB"> <feImage id="feimage1" result="img1" xlink:href="#image1" /> <feImage id="feimage2" result="img2" xlink:href="#image2" /> <!-- inv1 and inv2 are the images with RGB inverted --> <feComponentTransfer result="inv1" in="img1"> <feFuncR type="linear" slope="-1" intercept="1" /> <feFuncG type="linear" slope="-1" intercept="1" /> <feFuncB type="linear" slope="-1" intercept="1" /> </feComponentTransfer> <feComponentTransfer result="inv2" in="img2"> <feFuncR type="linear" slope="-1" intercept="1" /> <feFuncG type="linear" slope="-1" intercept="1" /> <feFuncB type="linear" slope="-1" intercept="1" /> </feComponentTransfer> <!-- w1 will have non-white pixels anywhere that img2 is brighter than img1, and w2 for the reverse. It would be nice not to have to go through these intermediate states, but feComposite type="arithmetic" can't transform the RGB channels and leave the alpha channel untouched. --> <feComposite result="w1" in="img1" in2="inv2" operator="arithmetic" k2="1" k3="1" /> <feComposite result="w2" in="img2" in2="inv1" operator="arithmetic" k2="1" k3="1" /> <!-- c1 will have non-black pixels anywhere that img2 is brighter than img1, and c2 for the reverse --> <feComponentTransfer result="c1" in="w1"> <feFuncR type="linear" slope="-1" intercept="1" /> <feFuncG type="linear" slope="-1" intercept="1" /> <feFuncB type="linear" slope="-1" intercept="1" /> </feComponentTransfer> <feComponentTransfer result="c2" in="w2"> <feFuncR type="linear" slope="-1" intercept="1" /> <feFuncG type="linear" slope="-1" intercept="1" /> <feFuncB type="linear" slope="-1" intercept="1" /> </feComponentTransfer> <!-- c will be nonblack (and fully on) for every pixel+component where there are differences --> <feComposite result="c" in="c1" in2="c2" operator="arithmetic" k2="255" k3="255" /> <!-- a will be opaque for every pixel with differences and transparent for all others --> <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" /> <!-- a, dilated by 1 pixel --> <feMorphology result="dila1" in="a" operator="dilate" radius="1" /> <!-- a, dilated by 2 pixels --> <feMorphology result="dila2" in="dila1" operator="dilate" radius="1" /> <!-- all the pixels in the 2-pixel dilation of a but not in the 1-pixel dilation, to highlight the diffs --> <feComposite result="highlight" in="dila2" in2="dila1" operator="out" /> <feFlood result="red" flood-color="red" /> <feComposite result="redhighlight" in="red" in2="highlight" operator="in" /> <feFlood result="black" flood-color="black" flood-opacity="0.5" /> <feMerge> <feMergeNode in="black" /> <feMergeNode in="redhighlight" /> </feMerge> </filter> </defs> <g onmousemove="magnify(evt)"> <image x="0" y="0" width="100%" height="100%" id="image1" /> <image x="0" y="0" width="100%" height="100%" id="image2" /> </g> <rect id="diffrect" filter="url(#showDifferences)" pointer-events="none" x="0" y="0" width="100%" height="100%" /> </svg> </div> </div> </body> </html>