From 899856a1d41bfb7b3564364b3ffe6a949bafc3b0 Mon Sep 17 00:00:00 2001 From: lkassianik Date: Wed, 13 Apr 2016 09:20:15 -0700 Subject: [PATCH] Links on badge card should be accessible Summary: Ref T10710 Test Plan: Open user profile with badge, flip badge card, open awarder link. Reviewers: epriestley, chad, #blessed_reviewers Reviewed By: epriestley, #blessed_reviewers Subscribers: chad, Korvin Maniphest Tasks: T10710 Differential Revision: https://secure.phabricator.com/D15699 --- resources/celerity/map.php | 27 +++++++----- src/view/phui/PHUIBadgeView.php | 4 +- webroot/rsrc/css/phui/phui-badge.css | 4 ++ webroot/rsrc/js/core/behavior-badge-view.js | 41 +++++++++++++++++++ webroot/rsrc/js/core/behavior-toggle-class.js | 4 +- 5 files changed, 67 insertions(+), 13 deletions(-) create mode 100644 webroot/rsrc/js/core/behavior-badge-view.js diff --git a/resources/celerity/map.php b/resources/celerity/map.php index 52ba475631..005fc1363a 100644 --- a/resources/celerity/map.php +++ b/resources/celerity/map.php @@ -8,7 +8,7 @@ return array( 'names' => array( 'core.pkg.css' => 'ce06b6f6', - 'core.pkg.js' => 'e526f428', + 'core.pkg.js' => '37344f3c', 'darkconsole.pkg.js' => 'e7393ebb', 'differential.pkg.css' => '7ba78475', 'differential.pkg.js' => 'd0cd0df6', @@ -121,7 +121,7 @@ return array( 'rsrc/css/phui/calendar/phui-calendar.css' => 'ccabe893', 'rsrc/css/phui/phui-action-list.css' => 'c5eba19d', 'rsrc/css/phui/phui-action-panel.css' => '91c7b835', - 'rsrc/css/phui/phui-badge.css' => 'f25c3476', + 'rsrc/css/phui/phui-badge.css' => '3baef8db', 'rsrc/css/phui/phui-big-info-view.css' => 'bd903741', 'rsrc/css/phui/phui-box.css' => 'd909ea3d', 'rsrc/css/phui/phui-button.css' => 'a64a8de6', @@ -472,6 +472,7 @@ return array( 'rsrc/js/core/behavior-active-nav.js' => 'e379b58e', 'rsrc/js/core/behavior-audio-source.js' => '59b251eb', 'rsrc/js/core/behavior-autofocus.js' => '7319e029', + 'rsrc/js/core/behavior-badge-view.js' => '8ff5e24c', 'rsrc/js/core/behavior-choose-control.js' => '327a00d1', 'rsrc/js/core/behavior-crop.js' => 'fa0f4fc2', 'rsrc/js/core/behavior-dark-console.js' => 'f411b6ae', @@ -504,7 +505,7 @@ return array( 'rsrc/js/core/behavior-search-typeahead.js' => '06c32383', 'rsrc/js/core/behavior-select-on-click.js' => '4e3e79a6', 'rsrc/js/core/behavior-time-typeahead.js' => 'f80d6bf0', - 'rsrc/js/core/behavior-toggle-class.js' => '5d7c9f33', + 'rsrc/js/core/behavior-toggle-class.js' => '92b9ec77', 'rsrc/js/core/behavior-tokenizer.js' => 'b3a4b884', 'rsrc/js/core/behavior-tooltip.js' => '42fcb747', 'rsrc/js/core/behavior-watch-anchor.js' => '9f36c42d', @@ -580,6 +581,7 @@ return array( 'javelin-behavior-aphront-more' => 'a80d0378', 'javelin-behavior-audio-source' => '59b251eb', 'javelin-behavior-audit-preview' => 'd835b03a', + 'javelin-behavior-badge-view' => '8ff5e24c', 'javelin-behavior-bulk-job-reload' => 'edf8a145', 'javelin-behavior-choose-control' => '327a00d1', 'javelin-behavior-comment-actions' => '06460e71', @@ -687,7 +689,7 @@ return array( 'javelin-behavior-stripe-payment-form' => '3f5d6dbf', 'javelin-behavior-test-payment-form' => 'fc91ab6c', 'javelin-behavior-time-typeahead' => 'f80d6bf0', - 'javelin-behavior-toggle-class' => '5d7c9f33', + 'javelin-behavior-toggle-class' => '92b9ec77', 'javelin-behavior-typeahead-browse' => '635de1ec', 'javelin-behavior-typeahead-search' => '93d0c9e3', 'javelin-behavior-view-placeholder' => '47830651', @@ -807,7 +809,7 @@ return array( 'phrequent-css' => 'ffc185ad', 'phriction-document-css' => 'd1861e06', 'phui-action-panel-css' => '91c7b835', - 'phui-badge-view-css' => 'f25c3476', + 'phui-badge-view-css' => '3baef8db', 'phui-big-info-view-css' => 'bd903741', 'phui-box-css' => 'd909ea3d', 'phui-button-css' => 'a64a8de6', @@ -1335,11 +1337,6 @@ return array( 'javelin-stratcom', 'javelin-dom', ), - '5d7c9f33' => array( - 'javelin-behavior', - 'javelin-stratcom', - 'javelin-dom', - ), '5e9f347c' => array( 'javelin-behavior', 'multirow-row-manager', @@ -1575,6 +1572,11 @@ return array( 'javelin-stratcom', 'javelin-install', ), + '8ff5e24c' => array( + 'javelin-behavior', + 'javelin-stratcom', + 'javelin-dom', + ), '901935ef' => array( 'javelin-behavior', 'javelin-dom', @@ -1589,6 +1591,11 @@ return array( 92197373 => array( 'phui-workcard-view-css', ), + '92b9ec77' => array( + 'javelin-behavior', + 'javelin-stratcom', + 'javelin-dom', + ), '93d0c9e3' => array( 'javelin-behavior', 'javelin-stratcom', diff --git a/src/view/phui/PHUIBadgeView.php b/src/view/phui/PHUIBadgeView.php index 7594b6cb30..5ff6e297b0 100644 --- a/src/view/phui/PHUIBadgeView.php +++ b/src/view/phui/PHUIBadgeView.php @@ -61,6 +61,8 @@ final class PHUIBadgeView extends AphrontTagView { require_celerity_resource('phui-badge-view-css'); $id = celerity_generate_unique_node_id(); + Javelin::initBehavior('badge-view', array()); + $classes = array(); $classes[] = 'phui-badge-view'; if ($this->quality) { @@ -70,7 +72,7 @@ final class PHUIBadgeView extends AphrontTagView { return array( 'class' => implode(' ', $classes), - 'sigil' => 'jx-toggle-class', + 'sigil' => 'jx-badge-view', 'id' => $id, 'meta' => array( 'map' => array( diff --git a/webroot/rsrc/css/phui/phui-badge.css b/webroot/rsrc/css/phui/phui-badge.css index fdb396b539..b702d3e6c5 100644 --- a/webroot/rsrc/css/phui/phui-badge.css +++ b/webroot/rsrc/css/phui/phui-badge.css @@ -50,6 +50,10 @@ background-color: {$lightbluebackground}; } +.phui-badge-card a { + color: {$darkbluetext}; +} + .card-flipped .phui-badge-card-container { transform: translateX( -100% ) rotateY( -180deg ); -webkit-transform: translateX( -100% ) rotateY( -180deg ); diff --git a/webroot/rsrc/js/core/behavior-badge-view.js b/webroot/rsrc/js/core/behavior-badge-view.js new file mode 100644 index 0000000000..92c7623e9f --- /dev/null +++ b/webroot/rsrc/js/core/behavior-badge-view.js @@ -0,0 +1,41 @@ +/** + * @provides javelin-behavior-badge-view + * @requires javelin-behavior + * javelin-stratcom + * javelin-dom + */ + +/** + * Toggle CSS classes when an element is clicked. This behavior is activated + * by adding the sigil `jx-badge-view` to an element, and a key `map` to its + * data. The `map` should be a map from element IDs to the classes that should + * be toggled on them. + * + * Optionally, you may provide a `state` key to set the default state of the + * element. + */ +JX.behavior('badge-view', function(config, statics) { + function install() { + JX.Stratcom.listen( + ['click'], + 'jx-badge-view', + function(e) { + if (e.getNode('tag:a')) { + // If the event has a 'tag:a' node on it, that means the user + // either clicked a link or some other node inside a link. + return; + } + + var t = e.getNodeData('jx-badge-view'); + t.state = !t.state; + for (var k in t.map) { + JX.DOM.alterClass(JX.$(k), t.map[k], t.state); + } + e.kill(); + }); + + return true; + } + + statics.install = statics.install || install(); +}); diff --git a/webroot/rsrc/js/core/behavior-toggle-class.js b/webroot/rsrc/js/core/behavior-toggle-class.js index d8f7ef5bb5..d4756eb6bb 100644 --- a/webroot/rsrc/js/core/behavior-toggle-class.js +++ b/webroot/rsrc/js/core/behavior-toggle-class.js @@ -15,8 +15,6 @@ * element. */ JX.behavior('toggle-class', function(config, statics) { - statics.install = statics.install || install(); - function install() { JX.Stratcom.listen( ['touchstart', 'mousedown'], @@ -42,4 +40,6 @@ JX.behavior('toggle-class', function(config, statics) { return true; } + + statics.install = statics.install || install(); });