Отсутствие множественных ролей раздражало меня долгое время, так как базовый класс WP_User поддерживает несколько ролей. Я даже подумывал о поиске альтернативного программного решения. @lpryor - после прочтения вашего поста у меня появилась мотивация реализовать его самостоятельно.
Это заняло удивительно короткое количество строк, хотя мне пришлось взломать файл users.php, так как я был слишком ленив, чтобы создать отдельный плагин, чтобы сделать это для меня. Очевидно, что это неправильный способ сделать это, поэтому, если я буду достаточно мотивирован в будущем, я могу попытаться сделать это правильно.
Если вас не волнует возможность обновления до последней версии Wordpress (что вам и нужно) - вы можете реализовать несколько ролей с помощью фрагментов кода ниже. Пожалуйста, имейте в виду, что я не эксперт по WordPress. Я просто открыл соответствующие файлы и внес изменения, не пытаясь понять все последствия того, что я делал. Код кажется мне разумным, но я бы не стал доверять ему свою жизнь.
(Я использую 3.2, поэтому ваши номера строк могут отличаться). В class-wp-users-list-table.php непосредственно перед строкой 150 добавьте что-то вроде следующего:
<div class="alignleft actions">
<label class="screen-reader-text" for="remove_role"><?php _e( 'Remove role …' ) ?></label>
<select name="remove_role" id="remove_role">
<option value=''><?php _e( 'Remove role …' ) ?></option>
<?php wp_dropdown_roles(); ?>
</select>
<?php submit_button( __( 'Remove' ), 'secondary', 'changeit', false ); ?>
</div>
затем измените функцию current_account, чтобы она выглядела примерно так
function current_action() {
if ( isset($_REQUEST['changeit']) ) {
if ( !empty($_REQUEST['new_role']) )
return 'promote';
elseif ( !empty($_REQUEST['remove_role']) )
return 'remove_role';
}
return parent::current_action();
}
Сейчас в users.php закомментируйте строки 71-76
/*
if ( $id == $current_user->ID && !$wp_roles->role_objects[$_REQUEST['new_role']]->has_cap('promote_users') ) {
$update = 'err_admin_role';
continue;
}
*/
Замените set_role в строке 83 на add_role
$user->add_role($_REQUEST['new_role']);
В строке 92 добавьте следующее (Это просто слегка отредактированное копирование и вставка из действия по продвижению - я не проверил, чтобы возможность promo_user подходила для удаления ролей)
case 'remove_role':
check_admin_referer('bulk-users');
if ( ! current_user_can( 'promote_users' ) )
wp_die( __( 'You can’t edit that user.' ) );
if ( empty($_REQUEST['users']) ) {
wp_redirect($redirect);
exit();
}
$editable_roles = get_editable_roles();
if ( empty( $editable_roles[$_REQUEST['remove_role']] ) )
wp_die(__('You can’t remove that role'));
$userids = $_REQUEST['users'];
$update = 'remove_role';
foreach ( $userids as $id ) {
$id = (int) $id;
if ( ! current_user_can('promote_user', $id) )
wp_die(__('You can’t edit that user.'));
// The new role of the current user must also have promote_users caps
// Need to think this through
/*
if ( $id == $current_user->ID && !$wp_roles->role_objects[$_REQUEST['new_role']]->has_cap('promote_users') ) {
$update = 'err_admin_role';
continue;
}
*/
// If the user doesn't already belong to the blog, bail.
if ( is_multisite() && !is_user_member_of_blog( $id ) )
wp_die(__('Cheatin’ uh?'));
$user = new WP_User($id);
$user->remove_role($_REQUEST['remove_role']);
}
wp_redirect(add_query_arg('update', $update, $redirect));
exit();
В строке 370 добавить следующее
case 'remove_role':
$messages[] = '<div id="message" class="updated"><p>' . __('Removed role.') . '</p></div>';
break;