diff --git a/src/wp-includes/user.php b/src/wp-includes/user.php index fb01d38dad..49f4f3fa7d 100644 --- a/src/wp-includes/user.php +++ b/src/wp-includes/user.php @@ -765,6 +765,126 @@ function get_users( $args = array() ) { return (array) $user_search->get_results(); } +/** + * List all the users of the site, with several options available. + * + * @since 5.9.0 + * + * @param string|array $args { + * Optional. Array or string of default arguments. + * + * @type string $orderby How to sort the users. Accepts 'nicename', 'email', 'url', 'registered', + * 'user_nicename', 'user_email', 'user_url', 'user_registered', 'name', + * 'display_name', 'post_count', 'ID', 'meta_value', 'user_login'. Default 'name'. + * @type string $order Sorting direction for $orderby. Accepts 'ASC', 'DESC'. Default 'ASC'. + * @type int $number Maximum users to return or display. Default empty (all users). + * @type bool $exclude_admin Whether to exclude the 'admin' account, if it exists. Default false. + * @type bool $show_fullname Whether to show the user's full name. Default false. + * @type string $feed If not empty, show a link to the user's feed and use this text as the alt + * parameter of the link. Default empty. + * @type string $feed_image If not empty, show a link to the user's feed and use this image URL as + * clickable anchor. Default empty. + * @type string $feed_type The feed type to link to, such as 'rss2'. Defaults to default feed type. + * @type bool $echo Whether to output the result or instead return it. Default true. + * @type string $style If 'list', each user is wrapped in an `
  • ` element, otherwise the users + * will be separated by commas. + * @type bool $html Whether to list the items in HTML form or plaintext. Default true. + * @type string $exclude An array, comma-, or space-separated list of user IDs to exclude. Default empty. + * @type string $include An array, comma-, or space-separated list of user IDs to include. Default empty. + * } + * @return string|null The output if echo is false. Otherwise null. + */ +function wp_list_users( $args = array() ) { + $defaults = array( + 'orderby' => 'name', + 'order' => 'ASC', + 'number' => '', + 'exclude_admin' => true, + 'show_fullname' => false, + 'feed' => '', + 'feed_image' => '', + 'feed_type' => '', + 'echo' => true, + 'style' => 'list', + 'html' => true, + 'exclude' => '', + 'include' => '', + ); + + $args = wp_parse_args( $args, $defaults ); + + $return = ''; + + $query_args = wp_array_slice_assoc( $args, array( 'orderby', 'order', 'number', 'exclude', 'include' ) ); + $query_args['fields'] = 'ids'; + $users = get_users( $query_args ); + + foreach ( $users as $user_id ) { + $user = get_userdata( $user_id ); + + if ( $args['exclude_admin'] && 'admin' === $user->display_name ) { + continue; + } + + if ( $args['show_fullname'] && '' !== $user->first_name && '' !== $user->last_name ) { + $name = "$user->first_name $user->last_name"; + } else { + $name = $user->display_name; + } + + if ( ! $args['html'] ) { + $return .= $name . ', '; + + continue; // No need to go further to process HTML. + } + + if ( 'list' === $args['style'] ) { + $return .= '
  • '; + } + + $row = $name; + + if ( ! empty( $args['feed_image'] ) || ! empty( $args['feed'] ) ) { + $row .= ' '; + if ( empty( $args['feed_image'] ) ) { + $row .= '('; + } + + $row .= ''; + } else { + $row .= $name; + } + + $row .= ''; + + if ( empty( $args['feed_image'] ) ) { + $row .= ')'; + } + } + + $return .= $row; + $return .= ( 'list' === $args['style'] ) ? '
  • ' : ', '; + } + + $return = rtrim( $return, ', ' ); + + if ( ! $args['echo'] ) { + return $return; + } + echo $return; +} + /** * Get the sites a user belongs to. * diff --git a/tests/phpunit/tests/user/wpListUsers.php b/tests/phpunit/tests/user/wpListUsers.php new file mode 100644 index 0000000000..32d30699c1 --- /dev/null +++ b/tests/phpunit/tests/user/wpListUsers.php @@ -0,0 +1,202 @@ +user->create( + array( + 'user_login' => 'zack', + 'display_name' => 'zack', + 'role' => 'subscriber', + 'first_name' => 'zack', + 'last_name' => 'moon', + 'user_email' => 'm.zack@example.com', + 'user_url' => 'http://moonzack.fake', + ) + ); + + self::$user_ids[] = $factory->user->create( + array( + 'user_login' => 'jane', + 'display_name' => 'jane', + 'role' => 'contributor', + 'first_name' => 'jane', + 'last_name' => 'reno', + 'user_email' => 'r.jane@example.com', + 'user_url' => 'http://janereno.fake', + ) + ); + + self::$user_ids[] = $factory->user->create( + array( + 'user_login' => 'michelle', + 'display_name' => 'michelle', + 'role' => 'subscriber', + 'first_name' => 'michelle', + 'last_name' => 'jones', + 'user_email' => 'j.michelle@example.com', + 'user_url' => 'http://lemichellejones.fake', + ) + ); + + self::$user_ids[] = $factory->user->create( + array( + 'user_login' => 'paul', + 'display_name' => 'paul', + 'role' => 'subscriber', + 'first_name' => 'paul', + 'last_name' => 'norris', + 'user_email' => 'n.paul@example.com', + 'user_url' => 'http://awildpaulappeared.fake', + ) + ); + + foreach ( self::$user_ids as $user ) { + $factory->post->create( + array( + 'post_type' => 'post', + 'post_author' => $user, + ) + ); + } + } + + /** + * Test that wp_list_users() creates the expected list of users. + * + * @dataProvider data_should_create_a_user_list + * + * @ticket 15145 + * + * @param array|string $args The arguments to create a list of users. + * @param string $expected The expected result. + */ + public function test_should_create_a_user_list( $args, $expected ) { + $actual = wp_list_users( $args ); + + $expected = str_replace( + array( 'AUTHOR_ID_zack', 'AUTHOR_ID_jane', 'AUTHOR_ID_michelle', 'AUTHOR_ID_paul' ), + array( self::$user_ids[0], self::$user_ids[1], self::$user_ids[2], self::$user_ids[3] ), + $expected + ); + + if ( null === $actual ) { + $this->expectOutputString( $expected ); + } else { + $this->assertSame( $expected, $actual ); + } + } + + /** + * Data provider. + * + * @return array + */ + public function data_should_create_a_user_list() { + return array( + 'defaults when no args are supplied' => array( + 'args' => '', + 'expected' => '
  • jane
  • michelle
  • paul
  • zack
  • ', + ), + 'the admin account included' => array( + 'args' => array( + 'exclude_admin' => false, + ), + 'expected' => '
  • admin
  • jane
  • michelle
  • paul
  • zack
  • ', + ), + 'the full name of each user' => array( + 'args' => array( + 'show_fullname' => true, + ), + 'expected' => '
  • jane reno
  • michelle jones
  • paul norris
  • zack moon
  • ', + ), + 'the feed of each user' => array( + 'args' => array( + 'feed' => 'User feed', + ), + 'expected' => '
  • jane (User feed)
  • ' . + '
  • michelle (User feed)
  • ' . + '
  • paul (User feed)
  • ' . + '
  • zack (User feed)
  • ', + ), + 'the feed of each user and an image' => array( + 'args' => array( + 'feed' => 'User feed with image', + 'feed_image' => 'http://example.org/image.jpg', + ), + 'expected' => '
  • jane User feed with image
  • ' . + '
  • michelle User feed with image
  • ' . + '
  • paul User feed with image
  • ' . + '
  • zack User feed with image
  • ', + ), + 'a feed of the specified type' => array( + 'args' => array( + 'feed' => 'User feed as atom', + 'feed_type' => 'atom', + ), + 'expected' => '
  • jane (User feed as atom)
  • ' . + '
  • michelle (User feed as atom)
  • ' . + '
  • paul (User feed as atom)
  • ' . + '
  • zack (User feed as atom)
  • ', + ), + 'no output via echo' => array( + 'args' => array( + 'echo' => false, + ), + 'expected' => '
  • jane
  • michelle
  • paul
  • zack
  • ', + ), + 'commas separating each user' => array( + 'args' => array( + 'style' => '', + ), + 'expected' => 'jane, michelle, paul, zack', + ), + 'plain text format' => array( + 'args' => array( + 'html' => false, + ), + 'expected' => 'jane, michelle, paul, zack', + ), + ); + } + + /** + * Tests that wp_list_users() does not create a user list. + * + * @dataProvider data_should_not_create_a_user_list + * + * @ticket 15145 + * + * @param array|string $args The arguments to create a list of users. + */ + public function test_should_not_create_a_user_list( $args ) { + $actual = wp_list_users( $args ); + + if ( null === $actual ) { + $this->expectOutputString( '', 'wp_list_users() did not output an empty string.' ); + } else { + $this->assertSame( $actual, 'wp_list_users() did not return an empty string.' ); + } + } + + /** + * Data provider. + * + * @return array + */ + public function data_should_not_create_a_user_list() { + return array( + 'an empty user query result' => array( + 'args' => array( + 'include' => array( 9999 ), + ), + 'expected' => '', + ), + ); + } +}