Widgets: Remove the title attributes used in the Tag cloud widget.

- improves accessibility using an aria-label attribute to match the information conveyed visually with the one exposed to assistive technologies
- adds an option in the widget to display the item counts, consistently with what other widgets already do (Archives, Categories)

Props adamsoucie, emirpprime, Samantha Miller., MikeLittle, rianrietveld, sami.keijonen, adamsilverstein, westonruter, afercia.
See #24766.
Fixes #35566.


git-svn-id: https://develop.svn.wordpress.org/trunk@40816 602fd350-edb4-49c9-b593-d223f7449a82
This commit is contained in:
Andrea Fercia 2017-05-22 20:23:49 +00:00
parent 5ea3c91d57
commit 9e969efb89
4 changed files with 86 additions and 34 deletions

View File

@ -659,10 +659,10 @@ function wp_list_categories( $args = '' ) {
* be to return the top 45 tags in the tag cloud list. * be to return the top 45 tags in the tag cloud list.
* *
* The 'topic_count_text' argument is a nooped plural from _n_noop() to generate the * The 'topic_count_text' argument is a nooped plural from _n_noop() to generate the
* text for the tooltip of the tag link. * text for the tag link count.
* *
* The 'topic_count_text_callback' argument is a function, which given the count * The 'topic_count_text_callback' argument is a function, which given the count
* of the posts with that tag returns a text for the tooltip of the tag link. * of the posts with that tag returns a text for the tag link count.
* *
* The 'post_type' argument is used only when 'link' is set to 'edit'. It determines the post_type * The 'post_type' argument is used only when 'link' is set to 'edit'. It determines the post_type
* passed to edit.php for the popular tags edit links. * passed to edit.php for the popular tags edit links.
@ -671,6 +671,7 @@ function wp_list_categories( $args = '' ) {
* should be used, because only one will be used and the other ignored, if they are both set. * should be used, because only one will be used and the other ignored, if they are both set.
* *
* @since 2.3.0 * @since 2.3.0
* @since 4.8.0 Added the `show_count` argument.
* *
* @param array|string|null $args Optional. Override default arguments. * @param array|string|null $args Optional. Override default arguments.
* @return void|array Generated tag cloud, only if no failures and 'array' is set for the 'format' argument. * @return void|array Generated tag cloud, only if no failures and 'array' is set for the 'format' argument.
@ -680,7 +681,8 @@ function wp_tag_cloud( $args = '' ) {
$defaults = array( $defaults = array(
'smallest' => 8, 'largest' => 22, 'unit' => 'pt', 'number' => 45, 'smallest' => 8, 'largest' => 22, 'unit' => 'pt', 'number' => 45,
'format' => 'flat', 'separator' => "\n", 'orderby' => 'name', 'order' => 'ASC', 'format' => 'flat', 'separator' => "\n", 'orderby' => 'name', 'order' => 'ASC',
'exclude' => '', 'include' => '', 'link' => 'view', 'taxonomy' => 'post_tag', 'post_type' => '', 'echo' => true 'exclude' => '', 'include' => '', 'link' => 'view', 'taxonomy' => 'post_tag', 'post_type' => '', 'echo' => true,
'show_count' => 0,
); );
$args = wp_parse_args( $args, $defaults ); $args = wp_parse_args( $args, $defaults );
@ -736,6 +738,7 @@ function default_topic_count_scale( $count ) {
* *
* @todo Complete functionality. * @todo Complete functionality.
* @since 2.3.0 * @since 2.3.0
* @since 4.8.0 Added the `show_count` argument.
* *
* @param array $tags List of tags. * @param array $tags List of tags.
* @param string|array $args { * @param string|array $args {
@ -766,11 +769,13 @@ function default_topic_count_scale( $count ) {
* @type int|bool $filter Whether to enable filtering of the final output * @type int|bool $filter Whether to enable filtering of the final output
* via {@see 'wp_generate_tag_cloud'}. Default 1|true. * via {@see 'wp_generate_tag_cloud'}. Default 1|true.
* @type string $topic_count_text Nooped plural text from _n_noop() to supply to * @type string $topic_count_text Nooped plural text from _n_noop() to supply to
* tag tooltips. Default null. * tag counts. Default null.
* @type callable $topic_count_text_callback Callback used to generate nooped plural text for * @type callable $topic_count_text_callback Callback used to generate nooped plural text for
* tag tooltips based on the count. Default null. * tag counts based on the count. Default null.
* @type callable $topic_count_scale_callback Callback used to determine the tag count scaling * @type callable $topic_count_scale_callback Callback used to determine the tag count scaling
* value. Default default_topic_count_scale(). * value. Default default_topic_count_scale().
* @type bool|int $show_count Whether to display the tag counts. Default 0. Accepts
* 0, 1, or their bool equivalents.
* } * }
* @return string|array Tag cloud as a string or an array, depending on 'format' argument. * @return string|array Tag cloud as a string or an array, depending on 'format' argument.
*/ */
@ -780,6 +785,7 @@ function wp_generate_tag_cloud( $tags, $args = '' ) {
'format' => 'flat', 'separator' => "\n", 'orderby' => 'name', 'order' => 'ASC', 'format' => 'flat', 'separator' => "\n", 'orderby' => 'name', 'order' => 'ASC',
'topic_count_text' => null, 'topic_count_text_callback' => null, 'topic_count_text' => null, 'topic_count_text_callback' => null,
'topic_count_scale_callback' => 'default_topic_count_scale', 'filter' => 1, 'topic_count_scale_callback' => 'default_topic_count_scale', 'filter' => 1,
'show_count' => 0,
); );
$args = wp_parse_args( $args, $defaults ); $args = wp_parse_args( $args, $defaults );
@ -790,14 +796,14 @@ function wp_generate_tag_cloud( $tags, $args = '' ) {
return $return; return $return;
} }
// Juggle topic count tooltips: // Juggle topic counts.
if ( isset( $args['topic_count_text'] ) ) { if ( isset( $args['topic_count_text'] ) ) {
// First look for nooped plural support via topic_count_text. // First look for nooped plural support via topic_count_text.
$translate_nooped_plural = $args['topic_count_text']; $translate_nooped_plural = $args['topic_count_text'];
} elseif ( ! empty( $args['topic_count_text_callback'] ) ) { } elseif ( ! empty( $args['topic_count_text_callback'] ) ) {
// Look for the alternative callback style. Ignore the previous default. // Look for the alternative callback style. Ignore the previous default.
if ( $args['topic_count_text_callback'] === 'default_topic_count_text' ) { if ( $args['topic_count_text_callback'] === 'default_topic_count_text' ) {
$translate_nooped_plural = _n_noop( '%s topic', '%s topics' ); $translate_nooped_plural = _n_noop( '%s item', '%s items' );
} else { } else {
$translate_nooped_plural = false; $translate_nooped_plural = false;
} }
@ -806,7 +812,7 @@ function wp_generate_tag_cloud( $tags, $args = '' ) {
$translate_nooped_plural = _n_noop( $args['single_text'], $args['multiple_text'] ); $translate_nooped_plural = _n_noop( $args['single_text'], $args['multiple_text'] );
} else { } else {
// This is the default for when no callback, plural, or argument is passed in. // This is the default for when no callback, plural, or argument is passed in.
$translate_nooped_plural = _n_noop( '%s topic', '%s topics' ); $translate_nooped_plural = _n_noop( '%s item', '%s items' );
} }
/** /**
@ -861,6 +867,22 @@ function wp_generate_tag_cloud( $tags, $args = '' ) {
$font_spread = 1; $font_spread = 1;
$font_step = $font_spread / $spread; $font_step = $font_spread / $spread;
$aria_label = false;
/*
* Determine whether to output an 'aria-label' attribute with the tag name and count.
* When tags have a different font size, they visually convey an important information
* that should be available to assistive technologies too. On the other hand, sometimes
* themes set up the Tag Cloud to display all tags with the same font size (setting
* the 'smallest' and 'largest' arguments to the same value).
* In order to always serve the same content to all users, the 'aria-label' gets printed out:
* - when tags have a different size
* - when the tag count is displayed (for example when users check the checkbox in the
* Tag Cloud widget), regardless of the tags font size
*/
if ( $args['show_count'] || 0 !== $font_spread ) {
$aria_label = true;
}
// Assemble the data that will be used to generate the tag cloud markup. // Assemble the data that will be used to generate the tag cloud markup.
$tags_data = array(); $tags_data = array();
foreach ( $tags as $key => $tag ) { foreach ( $tags as $key => $tag ) {
@ -870,9 +892,9 @@ function wp_generate_tag_cloud( $tags, $args = '' ) {
$real_count = $real_counts[ $key ]; $real_count = $real_counts[ $key ];
if ( $translate_nooped_plural ) { if ( $translate_nooped_plural ) {
$title = sprintf( translate_nooped_plural( $translate_nooped_plural, $real_count ), number_format_i18n( $real_count ) ); $formatted_count = sprintf( translate_nooped_plural( $translate_nooped_plural, $real_count ), number_format_i18n( $real_count ) );
} else { } else {
$title = call_user_func( $args['topic_count_text_callback'], $real_count, $tag, $args ); $formatted_count = call_user_func( $args['topic_count_text_callback'], $real_count, $tag, $args );
} }
$tags_data[] = array( $tags_data[] = array(
@ -880,11 +902,13 @@ function wp_generate_tag_cloud( $tags, $args = '' ) {
'url' => '#' != $tag->link ? $tag->link : '#', 'url' => '#' != $tag->link ? $tag->link : '#',
'role' => '#' != $tag->link ? '' : ' role="button"', 'role' => '#' != $tag->link ? '' : ' role="button"',
'name' => $tag->name, 'name' => $tag->name,
'title' => $title, 'formatted_count' => $formatted_count,
'slug' => $tag->slug, 'slug' => $tag->slug,
'real_count' => $real_count, 'real_count' => $real_count,
'class' => 'tag-link-' . $tag_id, 'class' => 'tag-cloud-link tag-link-' . $tag_id,
'font_size' => $args['smallest'] + ( $count - $min_count ) * $font_step, 'font_size' => $args['smallest'] + ( $count - $min_count ) * $font_step,
'aria_label' => $aria_label ? sprintf( ' aria-label="%1$s (%2$s)"', esc_attr( $tag->name ), esc_attr( $formatted_count ) ) : '',
'show_count' => $args['show_count'] ? '<span class="tag-link-count"> (' . $real_count . ')</span>' : '',
); );
} }
@ -899,10 +923,19 @@ function wp_generate_tag_cloud( $tags, $args = '' ) {
$a = array(); $a = array();
// generate the output links array // Generate the output links array.
foreach ( $tags_data as $key => $tag_data ) { foreach ( $tags_data as $key => $tag_data ) {
$class = $tag_data['class'] . ' tag-link-position-' . ( $key + 1 ); $class = $tag_data['class'] . ' tag-link-position-' . ( $key + 1 );
$a[] = "<a href='" . esc_url( $tag_data['url'] ) . "'" . $tag_data['role'] . " class='" . esc_attr( $class ) . "' title='" . esc_attr( $tag_data['title'] ) . "' style='font-size: " . esc_attr( str_replace( ',', '.', $tag_data['font_size'] ) . $args['unit'] ) . ";'>" . esc_html( $tag_data['name'] ) . "</a>"; $a[] = sprintf(
'<a href="%1$s"%2$s class="%3$s" style="font-size: %4$s;"%5$s>%6$s%7$s</a>',
esc_url( $tag_data['url'] ),
$tag_data['role'],
esc_attr( $class ),
esc_attr( str_replace( ',', '.', $tag_data['font_size'] ) . $args['unit'] ),
$tag_data['aria_label'],
esc_html( $tag_data['name'] ),
$tag_data['show_count']
);
} }
switch ( $args['format'] ) { switch ( $args['format'] ) {
@ -910,7 +943,12 @@ function wp_generate_tag_cloud( $tags, $args = '' ) {
$return =& $a; $return =& $a;
break; break;
case 'list' : case 'list' :
$return = "<ul class='wp-tag-cloud'>\n\t<li>"; /*
* Force role="list", as some browsers (sic: Safari 10) don't expose to assistive
* technologies the default role when the list is styled with `list-style: none`.
* Note: this is redundant but doesn't harm.
*/
$return = "<ul class='wp-tag-cloud' role='list'>\n\t<li>";
$return .= join( "</li>\n\t<li>", $a ); $return .= join( "</li>\n\t<li>", $a );
$return .= "</li>\n</ul>\n"; $return .= "</li>\n</ul>\n";
break; break;

View File

@ -53,6 +53,8 @@ class WP_Widget_Tag_Cloud extends WP_Widget {
} }
} }
$show_count = ! empty( $instance['count'] );
/** /**
* Filters the taxonomy used in the Tag Cloud widget. * Filters the taxonomy used in the Tag Cloud widget.
* *
@ -65,7 +67,8 @@ class WP_Widget_Tag_Cloud extends WP_Widget {
*/ */
$tag_cloud = wp_tag_cloud( apply_filters( 'widget_tag_cloud_args', array( $tag_cloud = wp_tag_cloud( apply_filters( 'widget_tag_cloud_args', array(
'taxonomy' => $current_taxonomy, 'taxonomy' => $current_taxonomy,
'echo' => false 'echo' => false,
'show_count' => $show_count,
) ) ); ) ) );
if ( empty( $tag_cloud ) ) { if ( empty( $tag_cloud ) ) {
@ -102,6 +105,7 @@ class WP_Widget_Tag_Cloud extends WP_Widget {
public function update( $new_instance, $old_instance ) { public function update( $new_instance, $old_instance ) {
$instance = array(); $instance = array();
$instance['title'] = sanitize_text_field( $new_instance['title'] ); $instance['title'] = sanitize_text_field( $new_instance['title'] );
$instance['count'] = ! empty( $new_instance['count'] ) ? 1 : 0;
$instance['taxonomy'] = stripslashes($new_instance['taxonomy']); $instance['taxonomy'] = stripslashes($new_instance['taxonomy']);
return $instance; return $instance;
} }
@ -117,6 +121,7 @@ class WP_Widget_Tag_Cloud extends WP_Widget {
public function form( $instance ) { public function form( $instance ) {
$current_taxonomy = $this->_get_current_taxonomy($instance); $current_taxonomy = $this->_get_current_taxonomy($instance);
$title_id = $this->get_field_id( 'title' ); $title_id = $this->get_field_id( 'title' );
$count = isset( $instance['count'] ) ? (bool) $instance['count'] : false;
$instance['title'] = ! empty( $instance['title'] ) ? esc_attr( $instance['title'] ) : ''; $instance['title'] = ! empty( $instance['title'] ) ? esc_attr( $instance['title'] ) : '';
echo '<p><label for="' . $title_id .'">' . __( 'Title:' ) . '</label> echo '<p><label for="' . $title_id .'">' . __( 'Title:' ) . '</label>
@ -128,6 +133,14 @@ class WP_Widget_Tag_Cloud extends WP_Widget {
$name = $this->get_field_name( 'taxonomy' ); $name = $this->get_field_name( 'taxonomy' );
$input = '<input type="hidden" id="' . $id . '" name="' . $name . '" value="%s" />'; $input = '<input type="hidden" id="' . $id . '" name="' . $name . '" value="%s" />';
$count_checkbox = sprintf(
'<p><input type="checkbox" class="checkbox" id="%1$s" name="%2$s"%3$s /> <label for="%1$s">%4$s</label></p>',
$this->get_field_id( 'count' ),
$this->get_field_name( 'count' ),
checked( $count, true, false ),
__( 'Show tag counts' )
);
switch ( count( $taxonomies ) ) { switch ( count( $taxonomies ) ) {
// No tag cloud supporting taxonomies found, display error message // No tag cloud supporting taxonomies found, display error message
@ -136,14 +149,15 @@ class WP_Widget_Tag_Cloud extends WP_Widget {
printf( $input, '' ); printf( $input, '' );
break; break;
// Just a single tag cloud supporting taxonomy found, no need to display options // Just a single tag cloud supporting taxonomy found, no need to display a select.
case 1: case 1:
$keys = array_keys( $taxonomies ); $keys = array_keys( $taxonomies );
$taxonomy = reset( $keys ); $taxonomy = reset( $keys );
printf( $input, esc_attr( $taxonomy ) ); printf( $input, esc_attr( $taxonomy ) );
echo $count_checkbox;
break; break;
// More than one tag cloud supporting taxonomy found, display options // More than one tag cloud supporting taxonomy found, display a select.
default: default:
printf( printf(
'<p><label for="%1$s">%2$s</label>' . '<p><label for="%1$s">%2$s</label>' .
@ -162,7 +176,7 @@ class WP_Widget_Tag_Cloud extends WP_Widget {
); );
} }
echo '</select></p>'; echo '</select></p>' . $count_checkbox;
} }
} }

View File

@ -939,7 +939,7 @@ class Tests_Post extends WP_UnitTestCase {
'link' => 'edit' 'link' => 'edit'
) ); ) );
preg_match_all( "|href='([^']+)'|", $wp_tag_cloud, $matches ); preg_match_all( '|href="([^"]+)"|', $wp_tag_cloud, $matches );
$this->assertSame( 1, count( $matches[1] ) ); $this->assertSame( 1, count( $matches[1] ) );
$terms = get_terms( $tax ); $terms = get_terms( $tax );

View File

@ -119,7 +119,7 @@ class Tests_WP_Generate_Tag_Cloud extends WP_UnitTestCase {
'format' => 'list', 'format' => 'list',
) ); ) );
$this->assertRegExp( "|^<ul class='wp-tag-cloud'>|", $found ); $this->assertRegExp( "|^<ul class='wp-tag-cloud' role='list'>|", $found );
$this->assertRegExp( "|</ul>\n|", $found ); $this->assertRegExp( "|</ul>\n|", $found );
$this->assertContains( '>' . $tags[0]->name . '<', $found ); $this->assertContains( '>' . $tags[0]->name . '<', $found );
} }
@ -164,7 +164,7 @@ class Tests_WP_Generate_Tag_Cloud extends WP_UnitTestCase {
'format' => 'list', 'format' => 'list',
) ); ) );
$this->assertRegExp( "|^<ul class='wp-tag-cloud'>|", $found ); $this->assertRegExp( "|^<ul class='wp-tag-cloud' role='list'>|", $found );
$this->assertRegExp( "|</ul>\n|", $found ); $this->assertRegExp( "|</ul>\n|", $found );
foreach ( $tags as $tag ) { foreach ( $tags as $tag ) {
@ -198,8 +198,8 @@ class Tests_WP_Generate_Tag_Cloud extends WP_UnitTestCase {
), ),
) ); ) );
$this->assertContains( "title='Term has 1 post'", $actual[0] ); $this->assertContains( 'aria-label="' . $term_objects[0]->name . ' (Term has 1 post)"', $actual[0] );
$this->assertContains( "title='Term has 2 posts'", $actual[1] ); $this->assertContains( 'aria-label="' . $term_objects[1]->name . ' (Term has 2 posts)"', $actual[1] );
} }
public function test_topic_count_text_callback() { public function test_topic_count_text_callback() {
@ -223,8 +223,8 @@ class Tests_WP_Generate_Tag_Cloud extends WP_UnitTestCase {
'topic_count_text_callback' => array( $this, 'topic_count_text_callback' ), 'topic_count_text_callback' => array( $this, 'topic_count_text_callback' ),
) ); ) );
$this->assertContains( "title='1 foo'", $actual[0] ); $this->assertContains( 'aria-label="' . $term_objects[0]->name . ' (1 foo)"', $actual[0] );
$this->assertContains( "title='2 foo'", $actual[1] ); $this->assertContains( 'aria-label="' . $term_objects[1]->name . ' (2 foo)"', $actual[1] );
} }
/** /**