/* * 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(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("\n"); stream->writeText("\n\n\n\n\n\n"); stream->writeText("\n"); } static void print_pixel_count(SkFILEWStream* stream, const DiffRecord& diff) { stream->writeText("
("); stream->writeDecAsText(static_cast(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(""); } static void print_label_cell(SkFILEWStream* stream, const DiffRecord& diff) { char metricBuf [20]; stream->writeText(""); 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("
Average color mismatch "); stream->writeDecAsText(static_cast(MAX3(diff.fAverageMismatchR, diff.fAverageMismatchG, diff.fAverageMismatchB))); stream->writeText("
Max color mismatch "); stream->writeDecAsText(MAX3(diff.fMaxMismatchR, diff.fMaxMismatchG, diff.fMaxMismatchB)); stream->writeText(""); break; case DiffRecord::kCouldNotCompare_Result: stream->writeText("Could not compare.
base: "); stream->writeText(DiffResource::getStatusDescription(diff.fBase.fStatus)); stream->writeText("
comparison: "); stream->writeText(DiffResource::getStatusDescription(diff.fComparison.fStatus)); stream->writeText(""); return; default: SkDEBUGFAIL("encountered DiffRecord with unknown result type"); return; } } static void print_image_cell(SkFILEWStream* stream, const SkString& path, int height) { stream->writeText("
"); } static void print_link_cell(SkFILEWStream* stream, const SkString& path, const char* text) { stream->writeText(""); } 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(""); 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("\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("\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( "\n\n" "\n" "\n\n\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( "
"); stream->writeText("select image"); 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("
"); } 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(".
"); stream->writeText("
"); stream->writeText("every different pixel shown in white"); stream->writeText(""); stream->writeText("color difference at each pixel"); stream->writeText("baseDir: "); stream->writeText(baseDir.c_str()); stream->writeText("comparisonDir: "); stream->writeText(comparisonDir.c_str()); stream->writeText("
writeText(diff.fBase.fFilename.c_str()); stream->writeText("\" checked=\"yes\">"); stream->writeText(diff.fBase.fFilename.c_str()); stream->writeText("
"); 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
writeText(path.c_str()); stream->writeText("\">writeText(path.c_str()); stream->writeText("\" height=\""); stream->writeDecAsText(height); stream->writeText("px\">writeText(path.c_str()); stream->writeText("\">"); stream->writeText(text); stream->writeText("N/A
\n" "\n" "
\n" "\n\n"); outputStream.flush(); }