/*
* Copyright 2012 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include "skdiff.h"
#include "skdiff_html.h"
#include "SkStream.h"
#include "SkTime.h"
/// Make layout more consistent by scaling image to 240 height, 360 width,
/// or natural size, whichever is smallest.
static int compute_image_height(int height, int width) {
int retval = 240;
if (height < retval) {
retval = height;
}
float scale = (float) retval / height;
if (width * scale > 360) {
scale = (float) 360 / width;
retval = static_cast<int>(height * scale);
}
return retval;
}
static void print_table_header(SkFILEWStream* stream,
const int matchCount,
const int colorThreshold,
const RecordArray& differences,
const SkString &baseDir,
const SkString &comparisonDir,
bool doOutputDate = false) {
stream->writeText("<table>\n");
stream->writeText("<tr><th>");
stream->writeText("select image</th>\n<th>");
if (doOutputDate) {
SkTime::DateTime dt;
SkTime::GetDateTime(&dt);
stream->writeText("SkDiff run at ");
stream->writeDecAsText(dt.fHour);
stream->writeText(":");
if (dt.fMinute < 10) {
stream->writeText("0");
}
stream->writeDecAsText(dt.fMinute);
stream->writeText(":");
if (dt.fSecond < 10) {
stream->writeText("0");
}
stream->writeDecAsText(dt.fSecond);
stream->writeText("<br>");
}
stream->writeDecAsText(matchCount);
stream->writeText(" of ");
stream->writeDecAsText(differences.count());
stream->writeText(" diffs matched ");
if (colorThreshold == 0) {
stream->writeText("exactly");
} else {
stream->writeText("within ");
stream->writeDecAsText(colorThreshold);
stream->writeText(" color units per component");
}
stream->writeText(".<br>");
stream->writeText("</th>\n<th>");
stream->writeText("every different pixel shown in white");
stream->writeText("</th>\n<th>");
stream->writeText("color difference at each pixel");
stream->writeText("</th>\n<th>baseDir: ");
stream->writeText(baseDir.c_str());
stream->writeText("</th>\n<th>comparisonDir: ");
stream->writeText(comparisonDir.c_str());
stream->writeText("</th>\n");
stream->writeText("</tr>\n");
}
static void print_pixel_count(SkFILEWStream* stream, const DiffRecord& diff) {
stream->writeText("<br>(");
stream->writeDecAsText(static_cast<int>(diff.fFractionDifference *
diff.fBase.fBitmap.width() *
diff.fBase.fBitmap.height()));
stream->writeText(" pixels)");
/*
stream->writeDecAsText(diff.fWeightedFraction *
diff.fBaseWidth *
diff.fBaseHeight);
stream->writeText(" weighted pixels)");
*/
}
static void print_checkbox_cell(SkFILEWStream* stream, const DiffRecord& diff) {
stream->writeText("<td><input type=\"checkbox\" name=\"");
stream->writeText(diff.fBase.fFilename.c_str());
stream->writeText("\" checked=\"yes\"></td>");
}
static void print_label_cell(SkFILEWStream* stream, const DiffRecord& diff) {
char metricBuf [20];
stream->writeText("<td><b>");
stream->writeText(diff.fBase.fFilename.c_str());
stream->writeText("</b><br>");
switch (diff.fResult) {
case DiffRecord::kEqualBits_Result:
SkDEBUGFAIL("should not encounter DiffRecord with kEqualBits here");
return;
case DiffRecord::kEqualPixels_Result:
SkDEBUGFAIL("should not encounter DiffRecord with kEqualPixels here");
return;
case DiffRecord::kDifferentSizes_Result:
stream->writeText("Image sizes differ</td>");
return;
case DiffRecord::kDifferentPixels_Result:
sprintf(metricBuf, "%.4f%%", 100 * diff.fFractionDifference);
stream->writeText(metricBuf);
stream->writeText(" of pixels differ");
stream->writeText("\n (");
sprintf(metricBuf, "%.4f%%", 100 * diff.fWeightedFraction);
stream->writeText(metricBuf);
stream->writeText(" weighted)");
// Write the actual number of pixels that differ if it's < 1%
if (diff.fFractionDifference < 0.01) {
print_pixel_count(stream, diff);
}
stream->writeText("<br>");
if (SkScalarRoundToInt(diff.fAverageMismatchA) > 0) {
stream->writeText("<br>Average alpha channel mismatch ");
stream->writeDecAsText(SkScalarRoundToInt(diff.fAverageMismatchA));
}
stream->writeText("<br>Max alpha channel mismatch ");
stream->writeDecAsText(SkScalarRoundToInt(diff.fMaxMismatchA));
stream->writeText("<br>Total alpha channel mismatch ");
stream->writeDecAsText(static_cast<int>(diff.fTotalMismatchA));
stream->writeText("<br>");
stream->writeText("<br>Average color mismatch ");
stream->writeDecAsText(SkScalarRoundToInt(MAX3(diff.fAverageMismatchR,
diff.fAverageMismatchG,
diff.fAverageMismatchB)));
stream->writeText("<br>Max color mismatch ");
stream->writeDecAsText(MAX3(diff.fMaxMismatchR,
diff.fMaxMismatchG,
diff.fMaxMismatchB));
stream->writeText("</td>");
break;
case DiffRecord::kCouldNotCompare_Result:
stream->writeText("Could not compare.<br>base: ");
stream->writeText(DiffResource::getStatusDescription(diff.fBase.fStatus));
stream->writeText("<br>comparison: ");
stream->writeText(DiffResource::getStatusDescription(diff.fComparison.fStatus));
stream->writeText("</td>");
return;
default:
SkDEBUGFAIL("encountered DiffRecord with unknown result type");
return;
}
}
static void print_image_cell(SkFILEWStream* stream, const SkString& path, int height) {
stream->writeText("<td><a href=\"");
stream->writeText(path.c_str());
stream->writeText("\"><img src=\"");
stream->writeText(path.c_str());
stream->writeText("\" height=\"");
stream->writeDecAsText(height);
stream->writeText("px\"></a></td>");
}
static void print_link_cell(SkFILEWStream* stream, const SkString& path, const char* text) {
stream->writeText("<td><a href=\"");
stream->writeText(path.c_str());
stream->writeText("\">");
stream->writeText(text);
stream->writeText("</a></td>");
}
static void print_diff_resource_cell(SkFILEWStream* stream, DiffResource& resource,
const SkString& relativePath, bool local) {
if (resource.fBitmap.empty()) {
if (DiffResource::kCouldNotDecode_Status == resource.fStatus) {
if (local && !resource.fFilename.isEmpty()) {
print_link_cell(stream, resource.fFilename, "N/A");
return;
}
if (!resource.fFullPath.isEmpty()) {
if (!resource.fFullPath.startsWith(PATH_DIV_STR)) {
resource.fFullPath.prepend(relativePath);
}
print_link_cell(stream, resource.fFullPath, "N/A");
return;
}
}
stream->writeText("<td>N/A</td>");
return;
}
int height = compute_image_height(resource.fBitmap.height(), resource.fBitmap.width());
if (local) {
print_image_cell(stream, resource.fFilename, height);
return;
}
if (!resource.fFullPath.startsWith(PATH_DIV_STR)) {
resource.fFullPath.prepend(relativePath);
}
print_image_cell(stream, resource.fFullPath, height);
}
static void print_diff_row(SkFILEWStream* stream, DiffRecord& diff, const SkString& relativePath) {
stream->writeText("<tr>\n");
print_checkbox_cell(stream, diff);
print_label_cell(stream, diff);
print_diff_resource_cell(stream, diff.fWhite, relativePath, true);
print_diff_resource_cell(stream, diff.fDifference, relativePath, true);
print_diff_resource_cell(stream, diff.fBase, relativePath, false);
print_diff_resource_cell(stream, diff.fComparison, relativePath, false);
stream->writeText("</tr>\n");
stream->flush();
}
void print_diff_page(const int matchCount,
const int colorThreshold,
const RecordArray& differences,
const SkString& baseDir,
const SkString& comparisonDir,
const SkString& outputDir) {
SkASSERT(!baseDir.isEmpty());
SkASSERT(!comparisonDir.isEmpty());
SkASSERT(!outputDir.isEmpty());
SkString outputPath(outputDir);
outputPath.append("index.html");
//SkFILEWStream outputStream ("index.html");
SkFILEWStream outputStream(outputPath.c_str());
// Need to convert paths from relative-to-cwd to relative-to-outputDir
// FIXME this doesn't work if there are '..' inside the outputDir
bool isPathAbsolute = false;
// On Windows or Linux, a path starting with PATH_DIV_CHAR is absolute.
if (outputDir.size() > 0 && PATH_DIV_CHAR == outputDir[0]) {
isPathAbsolute = true;
}
#ifdef SK_BUILD_FOR_WIN32
// On Windows, absolute paths can also start with "x:", where x is any
// drive letter.
if (outputDir.size() > 1 && ':' == outputDir[1]) {
isPathAbsolute = true;
}
#endif
SkString relativePath;
if (!isPathAbsolute) {
unsigned int ui;
for (ui = 0; ui < outputDir.size(); ui++) {
if (outputDir[ui] == PATH_DIV_CHAR) {
relativePath.append(".." PATH_DIV_STR);
}
}
}
outputStream.writeText(
"<html>\n<head>\n"
"<script src=\"https://ajax.googleapis.com/ajax/"
"libs/jquery/1.7.2/jquery.min.js\"></script>\n"
"<script type=\"text/javascript\">\n"
"function generateCheckedList() {\n"
"var boxes = $(\":checkbox:checked\");\n"
"var fileCmdLineString = '';\n"
"var fileMultiLineString = '';\n"
"for (var i = 0; i < boxes.length; i++) {\n"
"fileMultiLineString += boxes[i].name + '<br>';\n"
"fileCmdLineString += boxes[i].name + ' ';\n"
"}\n"
"$(\"#checkedList\").html(fileCmdLineString + "
"'<br><br>' + fileMultiLineString);\n"
"}\n"
"</script>\n</head>\n<body>\n");
print_table_header(&outputStream, matchCount, colorThreshold, differences,
baseDir, comparisonDir);
int i;
for (i = 0; i < differences.count(); i++) {
DiffRecord* diff = differences[i];
switch (diff->fResult) {
// Cases in which there is no diff to report.
case DiffRecord::kEqualBits_Result:
case DiffRecord::kEqualPixels_Result:
continue;
// Cases in which we want a detailed pixel diff.
case DiffRecord::kDifferentPixels_Result:
case DiffRecord::kDifferentSizes_Result:
case DiffRecord::kCouldNotCompare_Result:
print_diff_row(&outputStream, *diff, relativePath);
continue;
default:
SkDEBUGFAIL("encountered DiffRecord with unknown result type");
continue;
}
}
outputStream.writeText(
"</table>\n"
"<input type=\"button\" "
"onclick=\"generateCheckedList()\" "
"value=\"Create Rebaseline List\">\n"
"<div id=\"checkedList\"></div>\n"
"</body>\n</html>\n");
outputStream.flush();
}