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' => 'janemichellepaulzack',
+ ),
+ 'the admin account included' => array(
+ 'args' => array(
+ 'exclude_admin' => false,
+ ),
+ 'expected' => 'adminjanemichellepaulzack',
+ ),
+ 'the full name of each user' => array(
+ 'args' => array(
+ 'show_fullname' => true,
+ ),
+ 'expected' => 'jane renomichelle jonespaul norriszack 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
' .
+ 'michelle
' .
+ 'paul
' .
+ 'zack
',
+ ),
+ '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' => 'janemichellepaulzack',
+ ),
+ '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' => '',
+ ),
+ );
+ }
+}