aboutsummaryrefslogtreecommitdiffhomepage
diff options
context:
space:
mode:
authorGravatar zachr@google.com <zachr@google.com@2bbb7eff-a529-9590-31e7-b0007b416f81>2013-08-07 15:43:04 +0000
committerGravatar zachr@google.com <zachr@google.com@2bbb7eff-a529-9590-31e7-b0007b416f81>2013-08-07 15:43:04 +0000
commit6f8e2c5e87bd6620ff5290a9a8b6d5b570366336 (patch)
tree0be8e8411705ba7bf2d1327a46785d30a4c81c1a
parent2865676b84ec77de7a8af34b0d919b29f8b1f836 (diff)
download and rebaseline images using server
BUG= R=epoger@google.com Review URL: https://codereview.chromium.org/20654006 git-svn-id: http://skia.googlecode.com/svn/trunk@10607 2bbb7eff-a529-9590-31e7-b0007b416f81
-rwxr-xr-xtools/jsondiff.py48
-rw-r--r--tools/skpdiff/diff_viewer.js37
-rwxr-xr-xtools/skpdiff/skpdiff_server.py370
-rw-r--r--tools/skpdiff/viewer.html23
-rw-r--r--tools/skpdiff/viewer_style.css30
5 files changed, 378 insertions, 130 deletions
diff --git a/tools/jsondiff.py b/tools/jsondiff.py
index dd89c6d8dc..76958c6428 100755
--- a/tools/jsondiff.py
+++ b/tools/jsondiff.py
@@ -48,14 +48,17 @@ class GMDiffer(object):
def _GetFileContentsAsString(self, filepath):
"""Returns the full contents of a file, as a single string.
- If the filename looks like a URL, download its contents..."""
- if filepath.startswith('http:') or filepath.startswith('https:'):
+ If the filename looks like a URL, download its contents.
+ If the filename is None, return None."""
+ if filepath is None:
+ return None
+ elif filepath.startswith('http:') or filepath.startswith('https:'):
return urllib2.urlopen(filepath).read()
else:
return open(filepath, 'r').read()
- def _GetExpectedResults(self, filepath):
- """Returns the dictionary of expected results from a JSON file,
+ def _GetExpectedResults(self, contents):
+ """Returns the dictionary of expected results from a JSON string,
in this form:
{
@@ -75,7 +78,6 @@ class GMDiffer(object):
returned dictionary.
"""
result_dict = {}
- contents = self._GetFileContentsAsString(filepath)
json_dict = gm_json.LoadFromString(contents)
all_expectations = json_dict[gm_json.JSONKEY_EXPECTEDRESULTS]
for test_name in all_expectations.keys():
@@ -86,18 +88,18 @@ class GMDiffer(object):
num_allowed_digests = len(allowed_digests)
if num_allowed_digests > 1:
raise ValueError(
- 'test %s in file %s has %d allowed digests' % (
- test_name, filepath, num_allowed_digests))
+ 'test %s has %d allowed digests' % (
+ test_name, num_allowed_digests))
digest_pair = allowed_digests[0]
if digest_pair[0] != gm_json.JSONKEY_HASHTYPE_BITMAP_64BITMD5:
raise ValueError(
- 'test %s in file %s has unsupported hashtype %s' % (
- test_name, filepath, digest_pair[0]))
+ 'test %s has unsupported hashtype %s' % (
+ test_name, digest_pair[0]))
result_dict[test_name] = digest_pair[1]
return result_dict
- def _GetActualResults(self, filepath):
- """Returns the dictionary of actual results from a JSON file,
+ def _GetActualResults(self, contents):
+ """Returns the dictionary of actual results from a JSON string,
in this form:
{
@@ -116,7 +118,6 @@ class GMDiffer(object):
returned dictionary.
"""
result_dict = {}
- contents = self._GetFileContentsAsString(filepath)
json_dict = gm_json.LoadFromString(contents)
all_result_types = json_dict[gm_json.JSONKEY_ACTUALRESULTS]
for result_type in all_result_types.keys():
@@ -126,8 +127,8 @@ class GMDiffer(object):
digest_pair = results_of_this_type[test_name]
if digest_pair[0] != gm_json.JSONKEY_HASHTYPE_BITMAP_64BITMD5:
raise ValueError(
- 'test %s in file %s has unsupported hashtype %s' % (
- test_name, filepath, digest_pair[0]))
+ 'test %s has unsupported hashtype %s' % (
+ test_name, digest_pair[0]))
result_dict[test_name] = digest_pair[1]
return result_dict
@@ -152,11 +153,22 @@ class GMDiffer(object):
If newfile is not specified, then 'new' is the actual results within
oldfile.
"""
- old_results = self._GetExpectedResults(oldfile)
- if newfile:
- new_results = self._GetExpectedResults(newfile)
+ return self.GenerateDiffDictFromStrings(self._GetFileContentsAsString(oldfile),
+ self._GetFileContentsAsString(newfile))
+
+ def GenerateDiffDictFromStrings(self, oldjson, newjson=None):
+ """Generate a dictionary showing the diffs:
+ old = expectations within oldjson
+ new = expectations within newjson
+
+ If newfile is not specified, then 'new' is the actual results within
+ oldfile.
+ """
+ old_results = self._GetExpectedResults(oldjson)
+ if newjson:
+ new_results = self._GetExpectedResults(newjson)
else:
- new_results = self._GetActualResults(oldfile)
+ new_results = self._GetActualResults(oldjson)
return self._DictionaryDiff(old_results, new_results)
diff --git a/tools/skpdiff/diff_viewer.js b/tools/skpdiff/diff_viewer.js
index 700bf4b0e4..e7156b359f 100644
--- a/tools/skpdiff/diff_viewer.js
+++ b/tools/skpdiff/diff_viewer.js
@@ -1,3 +1,5 @@
+var MAX_SWAP_IMG_SIZE = 400;
+
angular.module('diff_viewer', []).
config(['$routeProvider', function($routeProvider) {
// Show the list of differences by default
@@ -28,10 +30,10 @@ directive('swapImg', function() {
image = rightImage;
}
- // Make it so the maximum size of an image is 500, and the images are scaled
- // down in halves.
+ // Make it so the maximum size of an image is MAX_SWAP_IMG_SIZE, and the images are
+ // scaled down in halves.
var divisor = 1;
- while ((image.width / divisor) > 500) {
+ while ((image.width / divisor) > MAX_SWAP_IMG_SIZE) {
divisor *= 2;
}
@@ -74,7 +76,7 @@ directive('swapImg', function() {
};
});
-function DiffListController($scope) {
+function DiffListController($scope, $http, $timeout) {
// Label each kind of differ for the sort buttons.
$scope.differs = [
{
@@ -100,4 +102,31 @@ function DiffListController($scope) {
$scope.sortingDiffer = function(record) {
return record.diffs[$scope.sortIndex].result;
};
+
+ // Flash status indicators on the rows, and then remove them so the style can potentially be
+ // reapplied later.
+ $scope.flashRowStatus = function(success, record) {
+ var flashStyle = success ? "success-flash" : "failure-flash";
+ var flashDurationMillis = success ? 500 : 800;
+
+ // Store the style in the record. The row will pick up the style this way instead of through
+ // index because index can change with sort order.
+ record.cssClasses = flashStyle;
+
+ // The animation cannot be repeated unless the class is removed the element.
+ $timeout(function() {
+ record.cssClasses = "";
+ }, flashDurationMillis);
+ }
+
+ $scope.setHashOf = function(imagePath, record) {
+ $http.post("/set_hash", {
+ "path": imagePath
+ }).success(function(data) {
+ $scope.flashRowStatus(data.success, record);
+ }).error(function() {
+ $scope.flashRowStatus(false, record);
+ });
+
+ };
}
diff --git a/tools/skpdiff/skpdiff_server.py b/tools/skpdiff/skpdiff_server.py
index 79ffc839e2..14035403a2 100755
--- a/tools/skpdiff/skpdiff_server.py
+++ b/tools/skpdiff/skpdiff_server.py
@@ -8,6 +8,7 @@ import json
import os
import os.path
import re
+import subprocess
import sys
import tempfile
import urllib2
@@ -99,7 +100,7 @@ def download_gm_image(image_name, image_path, hash_val):
@param hash_val The hash value of the image.
"""
- # Seperate the test name from a image name
+ # Separate the test name from a image name
image_match = IMAGE_FILENAME_RE.match(image_name)
test_name = image_match.group(1)
@@ -111,71 +112,6 @@ def download_gm_image(image_name, image_path, hash_val):
download_file(image_url, image_path)
-def download_changed_images(expectations_dir, expected_name, updated_name,
- expected_image_dir, actual_image_dir):
-
- """Download the expected and actual GMs that changed into the given paths.
- Determining what changed will be done by comparing each expected_name JSON
- results file to its corresponding updated_name JSON results file if it
- exists.
-
- @param expectations_dir The directory to traverse for results files. This
- should resmble expectations/gm in the Skia trunk.
- @param expected_name The name of the expected result files. These are
- in the format of expected-results.json.
- @param updated_name The name of the updated expected result files.
- Normally this matches --expectations-filename-output for the
- rebaseline.py tool.
- @param expected_image_dir The directory to place downloaded expected images
- into.
- @param actual_image_dir The directory to place downloaded actual images
- into.
- """
-
- differ = jsondiff.GMDiffer()
-
- # Look through expectations for hashes that changed
- for root, dirs, files in os.walk(expectations_dir):
- for expectation_file in files:
- # There are many files in the expectations directory. We only care
- # about expected results.
- if expectation_file != expected_name:
- continue
-
- # Get the name of the results file, and be sure there is an updated
- # result to compare against. If there is not, there is no point in
- # diffing this device.
- expected_file_path = os.path.join(root, expected_name)
- updated_file_path = os.path.join(root, updated_name)
- if not os.path.isfile(updated_file_path):
- continue
-
- # Find all expectations that did not match.
- expected_diff = differ.GenerateDiffDict(expected_file_path,
- updated_file_path)
-
- # The name of the device corresponds to the name of the folder we
- # are in.
- device_name = os.path.basename(root)
-
- # Create name prefixes to store the devices old and new GM results
- expected_image_prefix = os.path.join(expected_image_dir,
- device_name) + '-'
-
- actual_image_prefix = os.path.join(actual_image_dir,
- device_name) + '-'
-
- # Download each image that had a differing result
- for image_name, hashes in expected_diff.iteritems():
- print('Downloading', image_name, 'for device', device_name)
- download_gm_image(image_name,
- expected_image_prefix + image_name,
- hashes['old'])
- download_gm_image(image_name,
- actual_image_prefix + image_name,
- hashes['new'])
-
-
def get_image_set_from_skpdiff(skpdiff_records):
"""Get the set of all images references in the given records.
@@ -186,6 +122,249 @@ def get_image_set_from_skpdiff(skpdiff_records):
return expected_set | actual_set
+def set_expected_hash_in_json(expected_results_json, image_name, hash_value):
+ """Set the expected hash for the object extracted from
+ expected-results.json. Note that this only work with bitmap-64bitMD5 hash
+ types.
+
+ @param expected_results_json The Python dictionary with the results to
+ modify.
+ @param image_name The name of the image to set the hash of.
+ @param hash_value The hash to set for the image.
+ """
+ expected_results = expected_results_json[gm_json.JSONKEY_EXPECTEDRESULTS]
+
+ if image_name in expected_results:
+ expected_results[image_name][gm_json.JSONKEY_EXPECTEDRESULTS_ALLOWEDDIGESTS][0][1] = hash_value
+ else:
+ expected_results[image_name] = {
+ gm_json.JSONKEY_EXPECTEDRESULTS_ALLOWEDDIGESTS:
+ [
+ [
+ gm_json.JSONKEY_HASHTYPE_BITMAP_64BITMD5,
+ hash_value
+ ]
+ ]
+ }
+
+
+def get_head_version(path):
+ """Get the version of the file at the given path stored inside the HEAD of
+ the git repository. It is returned as a string.
+
+ @param path The path of the file whose HEAD is returned. It is assumed the
+ path is inside a git repo rooted at SKIA_ROOT_DIR.
+ """
+
+ # git-show will not work with absolute paths. This ensures we give it a path
+ # relative to the skia root.
+ git_path = os.path.relpath(path, SKIA_ROOT_DIR)
+ git_show_proc = subprocess.Popen(['git', 'show', 'HEAD:' + git_path],
+ stdout=subprocess.PIPE)
+
+ # When invoked outside a shell, git will output the last committed version
+ # of the file directly to stdout.
+ git_version_content, _ = git_show_proc.communicate()
+ return git_version_content
+
+
+class GMInstance:
+ """Information about a GM test result on a specific device:
+ - device_name = the name of the device that rendered it
+ - image_name = the GM test name and config
+ - expected_hash = the current expected hash value
+ - actual_hash = the actual hash value
+ """
+ def __init__(self,
+ device_name, image_name,
+ expected_hash, actual_hash):
+ self.device_name = device_name
+ self.image_name = image_name
+ self.expected_hash = expected_hash
+ self.actual_hash = actual_hash
+
+
+class ExpectationsManager:
+ def __init__(self, expectations_dir, expected_name, updated_name,
+ skpdiff_path):
+ """
+ @param expectations_dir The directory to traverse for results files.
+ This should resemble expectations/gm in the Skia trunk.
+ @param expected_name The name of the expected result files. These
+ are in the format of expected-results.json.
+ @param updated_name The name of the updated expected result files.
+ Normally this matches --expectations-filename-output for the
+ rebaseline.py tool.
+ @param skpdiff_path The path used to execute the skpdiff command.
+ """
+ self._expectations_dir = expectations_dir
+ self._expected_name = expected_name
+ self._updated_name = updated_name
+ self._skpdiff_path = skpdiff_path
+ self._generate_gm_comparison()
+
+ def _generate_gm_comparison(self):
+ """Generate all the data needed to compare GMs:
+ - determine which GMs changed
+ - download the changed images
+ - compare them with skpdiff
+ """
+
+ # Get the expectations and compare them with actual hashes
+ self._get_expectations()
+
+
+ # Create a temporary file tree that makes sense for skpdiff to operate
+ # on.
+ image_output_dir = tempfile.mkdtemp('skpdiff')
+ expected_image_dir = os.path.join(image_output_dir, 'expected')
+ actual_image_dir = os.path.join(image_output_dir, 'actual')
+ os.mkdir(expected_image_dir)
+ os.mkdir(actual_image_dir)
+
+ # Download expected and actual images that differed into the temporary
+ # file tree.
+ self._download_expectation_images(expected_image_dir, actual_image_dir)
+
+ # Invoke skpdiff with our downloaded images and place its results in the
+ # temporary directory.
+ self.skpdiff_output_path = os.path.join(image_output_dir,
+ 'skpdiff_output.json')
+ skpdiff_cmd = SKPDIFF_INVOKE_FORMAT.format(self._skpdiff_path,
+ self.skpdiff_output_path,
+ expected_image_dir,
+ actual_image_dir)
+ os.system(skpdiff_cmd)
+
+
+ def _get_expectations(self):
+ """Fills self._expectations with GMInstance objects for each test whose
+ expectation is different between the following two files:
+ - the local filesystem's updated results file
+ - git's head version of the expected results file
+ """
+ differ = jsondiff.GMDiffer()
+ self._expectations = []
+ for root, dirs, files in os.walk(self._expectations_dir):
+ for expectation_file in files:
+ # There are many files in the expectations directory. We only
+ # care about expected results.
+ if expectation_file != self._expected_name:
+ continue
+
+ # Get the name of the results file, and be sure there is an
+ # updated result to compare against. If there is not, there is
+ # no point in diffing this device.
+ expected_file_path = os.path.join(root, self._expected_name)
+ updated_file_path = os.path.join(root, self._updated_name)
+ if not os.path.isfile(updated_file_path):
+ continue
+
+ # Always get the expected results from git because we may have
+ # changed them in a previous instance of the server.
+ expected_contents = get_head_version(expected_file_path)
+ updated_contents = None
+ with open(updated_file_path, 'rb') as updated_file:
+ updated_contents = updated_file.read()
+
+ # Find all expectations that did not match.
+ expected_diff = differ.GenerateDiffDictFromStrings(
+ expected_contents,
+ updated_contents)
+
+ # The name of the device corresponds to the name of the folder
+ # we are in.
+ device_name = os.path.basename(root)
+
+ # Store old and new versions of the expectation for each GM
+ for image_name, hashes in expected_diff.iteritems():
+ self._expectations.append(
+ GMInstance(device_name, image_name,
+ hashes['old'], hashes['new']))
+
+
+ def _download_expectation_images(self, expected_image_dir, actual_image_dir):
+ """Download the expected and actual images for the _expectations array.
+
+ @param expected_image_dir The directory to download expected images
+ into.
+ @param actual_image_dir The directory to download actual images into.
+ """
+ image_map = {}
+
+ # Look through expectations and download their images.
+ for expectation in self._expectations:
+ # Build appropriate paths to download the images into.
+ expected_image_path = os.path.join(expected_image_dir,
+ expectation.device_name + '-' +
+ expectation.image_name)
+
+ actual_image_path = os.path.join(actual_image_dir,
+ expectation.device_name + '-' +
+ expectation.image_name)
+
+ print('Downloading %s for device %s' % (
+ expectation.image_name, expectation.device_name))
+
+ # Download images
+ download_gm_image(expectation.image_name,
+ expected_image_path,
+ expectation.expected_hash)
+
+ download_gm_image(expectation.image_name,
+ actual_image_path,
+ expectation.actual_hash)
+
+ # Annotate the expectations with where the images were downloaded
+ # to.
+ expectation.expected_image_path = expected_image_path
+ expectation.actual_image_path = actual_image_path
+
+ # Map the image paths back to the expectations.
+ image_map[expected_image_path] = (False, expectation)
+ image_map[actual_image_path] = (True, expectation)
+
+ self.image_map = image_map
+
+ def _set_expected_hash(self, device_name, image_name, hash_value):
+ """Set the expected hash for the image of the given device. This always
+ writes directly to the expected results file of the given device
+
+ @param device_name The name of the device to write the hash to.
+ @param image_name The name of the image whose hash to set.
+ @param hash_value The value of the hash to set.
+ """
+
+ # Retrieve the expected results file as it is in the working tree
+ json_path = os.path.join(self._expectations_dir, device_name,
+ self._expected_name)
+ expectations = gm_json.LoadFromFile(json_path)
+
+ # Set the specified hash.
+ set_expected_hash_in_json(expectations, image_name, hash_value)
+
+ # Write it out to disk using gm_json to keep the formatting consistent.
+ gm_json.WriteToFile(expectations, json_path)
+
+ def use_hash_of(self, image_path):
+ """Determine the hash of the image at the path using the records, and
+ set it as the expected hash for its device and image config.
+
+ @param image_path The path of the image as it was stored in the output
+ of skpdiff_path
+ """
+
+ # Get the metadata about the image at the path.
+ is_actual, expectation = self.image_map[image_path]
+
+ expectation_hash = expectation.actual_hash if is_actual else\
+ expectation.expected_hash
+
+ # Write out that image's hash directly to the expected results file.
+ self._set_expected_hash(expectation.device_name, expectation.image_name,
+ expectation_hash)
+
+
class SkPDiffHandler(BaseHTTPServer.BaseHTTPRequestHandler):
def send_file(self, file_path):
# Grab the extension if there is one
@@ -211,7 +390,6 @@ class SkPDiffHandler(BaseHTTPServer.BaseHTTPRequestHandler):
# under the dir_path. This is to prevent accidentally serving files
# outside the directory intended using symlinks, or '../'.
real_path = os.path.normpath(os.path.join(dir_path, file_path))
- print(repr(real_path))
if os.path.commonprefix([real_path, dir_path]) == dir_path:
if os.path.isfile(real_path):
self.send_file(real_path)
@@ -248,12 +426,26 @@ class SkPDiffHandler(BaseHTTPServer.BaseHTTPRequestHandler):
# If no file to send was found, just give the standard 404
self.send_error(404)
+ def do_POST(self):
+ if self.path == '/set_hash':
+ content_length = int(self.headers['Content-length'])
+ request_data = json.loads(self.rfile.read(content_length))
+ self.server.expectations_manager.use_hash_of(request_data['path'])
+ self.send_response(200)
+ self.send_header('Content-type', 'application/json')
+ self.end_headers()
+ self.wfile.write('{"success":true}')
+ return
-def run_server(skpdiff_output_path, port=8080):
+ # If the we have no handler for this path, give em' the 404
+ self.send_error(404)
+
+
+def run_server(expectations_manager, port=8080):
# Preload the skpdiff results file. This is so we can perform some
# processing on it.
skpdiff_output_json = ''
- with open(skpdiff_output_path, 'rb') as records_file:
+ with open(expectations_manager.skpdiff_output_path, 'rb') as records_file:
skpdiff_output_json = records_file.read()
# It's important to parse the results file so that we can make a set of
@@ -264,7 +456,8 @@ def run_server(skpdiff_output_path, port=8080):
# Add JSONP padding to the JSON because the web page expects it. It expects
# it because it was designed to run with or without a web server. Without a
# web server, the only way to load JSON is with JSONP.
- skpdiff_output_json = 'var SkPDiffRecords = ' + skpdiff_output_json
+ skpdiff_output_json = ('var SkPDiffRecords = ' +
+ json.dumps({'records': skpdiff_records}) + ';')
# Do not bind to interfaces other than localhost because the server will
# attempt to serve files relative to the root directory as a last resort
@@ -274,7 +467,8 @@ def run_server(skpdiff_output_path, port=8080):
http_server = BaseHTTPServer.HTTPServer(server_address, SkPDiffHandler)
http_server.image_set = image_set
http_server.skpdiff_output_json = skpdiff_output_json
- print('Navigate thine browser to: http://{}:{}'.format(*server_address))
+ http_server.expectations_manager = expectations_manager
+ print('Navigate thine browser to: http://{}:{}/'.format(*server_address))
http_server.serve_forever()
@@ -320,39 +514,19 @@ def main():
if skpdiff_path is None:
sys.exit(1)
- # Create a temporary file tree that makes sense for skpdiff.to operate on
- image_output_dir = tempfile.mkdtemp('skpdiff')
- expected_image_dir = os.path.join(image_output_dir, 'expected')
- actual_image_dir = os.path.join(image_output_dir, 'actual')
- os.mkdir(expected_image_dir)
- os.mkdir(actual_image_dir)
-
# Print out the paths of things for easier debugging
print('script dir :', SCRIPT_DIR)
print('tools dir :', TOOLS_DIR)
print('root dir :', SKIA_ROOT_DIR)
print('expectations dir :', args['expectations_dir'])
print('skpdiff path :', skpdiff_path)
- print('tmp dir :', image_output_dir)
- print('expected image dir :', expected_image_dir)
- print('actual image dir :', actual_image_dir)
-
- # Download expected and actual images that differed into the temporary file
- # tree.
- download_changed_images(args['expectations_dir'],
- args['expected'], args['updated'],
- expected_image_dir, actual_image_dir)
-
- # Invoke skpdiff with our downloaded images and place its results in the
- # temporary directory.
- skpdiff_output_path = os.path.join(image_output_dir, 'skpdiff_output.json')
- skpdiff_cmd = SKPDIFF_INVOKE_FORMAT.format(skpdiff_path,
- skpdiff_output_path,
- expected_image_dir,
- actual_image_dir)
- os.system(skpdiff_cmd)
-
- run_server(skpdiff_output_path, port=args['port'])
+
+ expectations_manager = ExpectationsManager(args['expectations_dir'],
+ args['expected'],
+ args['updated'],
+ skpdiff_path)
+
+ run_server(expectations_manager, port=args['port'])
if __name__ == '__main__':
main()
diff --git a/tools/skpdiff/viewer.html b/tools/skpdiff/viewer.html
index bd198b24b4..46c5871ae1 100644
--- a/tools/skpdiff/viewer.html
+++ b/tools/skpdiff/viewer.html
@@ -27,7 +27,7 @@
<table>
<thead>
<tr>
- <td>Baseline Image</td>
+ <td>Expected Image</td>
<td>Actual Image</td>
<td>Results</td>
</tr>
@@ -37,15 +37,18 @@
Loops through every record and crates a row for it. This sorts based on the whichever
metric the user chose, and places a limit on the max number of records to show.
-->
- <tr ng-repeat="record in records | orderBy:sortingDiffer | limitTo:500">
- <td><swap-img left-src="{{ record.baselinePath }}"
- right-src="{{ record.testPath }}"
- side="left"
- class="gm-image left-image" /></td>
- <td><swap-img left-src="{{ record.baselinePath }}"
- right-src="{{ record.testPath }}"
- side="right"
- class="gm-image right-image" /></td>
+ <tr ng-repeat="record in records | orderBy:sortingDiffer | limitTo:500"
+ class="{{ record.cssClasses }}">
+ <td ng-click="setHashOf(record.baselinePath, record)">
+ <swap-img left-src="{{ record.baselinePath }}"
+ right-src="{{ record.testPath }}"
+ side="left"
+ class="gm-image left-image" /></td>
+ <td ng-click="setHashOf(record.testPath, record)">
+ <swap-img left-src="{{ record.baselinePath }}"
+ right-src="{{ record.testPath }}"
+ side="right"
+ class="gm-image right-image" /></td>
<td>
<div ng-repeat="diff in record.diffs"
ng-mouseover="highlight(diff.differName)"
diff --git a/tools/skpdiff/viewer_style.css b/tools/skpdiff/viewer_style.css
index 140deaa55b..81b4f906c9 100644
--- a/tools/skpdiff/viewer_style.css
+++ b/tools/skpdiff/viewer_style.css
@@ -35,6 +35,36 @@ thead > tr > td {
text-align: right;
}
+.success-flash {
+ -webkit-animation-duration: 0.5s;
+ -webkit-animation-name: greenflash;
+}
+
+.failure-flash {
+ -webkit-animation-duration: 0.8s;
+ -webkit-animation-name: redflash;
+}
+
+@-webkit-keyframes greenflash {
+ from {
+ background-color: #8F8;
+ }
+
+ to {
+ background-color: #FFF
+ }
+}
+
+@-webkit-keyframes redflash {
+ from {
+ background-color: #F88;
+ }
+
+ to {
+ background-color: #FFF
+ }
+}
+
.result {
padding: 8px;
cursor: default;