[ Index ] |
PHP Cross Reference of BuddyPress |
[Summary view] [Print] [Text view]
1 <?php 2 /** 3 * BuddyPress Groups component admin screen. 4 * 5 * Props to WordPress core for the Comments admin screen, and its contextual 6 * help text, on which this implementation is heavily based. 7 * 8 * @package BuddyPress 9 * @subpackage Groups 10 * @since 1.7.0 11 */ 12 13 // Exit if accessed directly. 14 defined( 'ABSPATH' ) || exit; 15 16 // Include WP's list table class. 17 if ( ! class_exists( 'WP_List_Table' ) ) { 18 require ABSPATH . 'wp-admin/includes/class-wp-list-table.php'; 19 } 20 21 // The per_page screen option. Has to be hooked in extremely early. 22 if ( is_admin() && ! empty( $_REQUEST['page'] ) && 'bp-groups' == $_REQUEST['page'] ) { 23 add_filter( 'set-screen-option', 'bp_groups_admin_screen_options', 10, 3 ); 24 } 25 26 /** 27 * Register the Groups component admin screen. 28 * 29 * @since 1.7.0 30 */ 31 function bp_groups_add_admin_menu() { 32 33 // Add our screen. 34 $hook = add_menu_page( 35 _x( 'Groups', 'Admin Groups page title', 'buddypress' ), 36 _x( 'Groups', 'Admin Groups menu', 'buddypress' ), 37 'bp_moderate', 38 'bp-groups', 39 'bp_groups_admin', 40 'div' 41 ); 42 43 // Hook into early actions to load custom CSS and our init handler. 44 add_action( "load-$hook", 'bp_groups_admin_load' ); 45 } 46 add_action( bp_core_admin_hook(), 'bp_groups_add_admin_menu' ); 47 48 /** 49 * Redirects the user on the Goups network admin screen when BuddyPress is network activated. 50 * 51 * @since 7.0.0 52 */ 53 function bp_group_site_admin_network_admin_redirect() { 54 wp_safe_redirect( add_query_arg( 'page', 'bp-groups', network_admin_url( 'admin.php' ) ) ); 55 exit(); 56 } 57 58 /** 59 * Create Groups submenu to manage BuddyPress types. 60 * 61 * @since 7.0.0 62 */ 63 function bp_groups_admin_types_menu() { 64 if ( ! bp_is_root_blog() ) { 65 return; 66 } 67 68 if ( bp_is_network_activated() && ! bp_is_multiblog_mode() && is_network_admin() ) { 69 // Adds a 'bp-groups' submenu to go to the root blog Group types screen. 70 $group_type_admin_url = add_query_arg( 'taxonomy', 'bp_group_type', get_admin_url( bp_get_root_blog_id(), 'edit-tags.php' ) ); 71 add_submenu_page( 72 'bp-groups', 73 __( 'Group Types', 'buddypress' ), 74 __( 'Group Types', 'buddypress' ), 75 'bp_moderate', 76 esc_url( $group_type_admin_url ) 77 ); 78 } elseif ( ! is_network_admin() ) { 79 if ( bp_is_network_activated() && ! bp_is_multiblog_mode() ) { 80 // Adds a 'bp-groups' menu to the root blog menu. 81 $redirect_hook = add_menu_page( 82 _x( 'Groups', 'Admin Groups page title', 'buddypress' ), 83 _x( 'Groups', 'Admin Groups menu', 'buddypress' ), 84 'bp_moderate', 85 'bp-groups', 86 '__return_empty_string', 87 'div' 88 ); 89 90 add_action( "load-{$redirect_hook}", 'bp_group_site_admin_network_admin_redirect' ); 91 } 92 93 // Add the submenu to manage Group Types. 94 add_submenu_page( 95 'bp-groups', 96 __( 'Group Types', 'buddypress' ), 97 __( 'Group Types', 'buddypress' ), 98 'bp_moderate', 99 basename( add_query_arg( 'taxonomy', 'bp_group_type', bp_get_admin_url( 'edit-tags.php' ) ) ) 100 ); 101 } 102 } 103 add_action( 'bp_admin_menu', 'bp_groups_admin_types_menu' ); 104 105 /** 106 * Add groups component to custom menus array. 107 * 108 * This ensures that the Groups menu item appears in the proper order on the 109 * main Dashboard menu. 110 * 111 * @since 1.7.0 112 * 113 * @param array $custom_menus Array of BP top-level menu items. 114 * @return array Menu item array, with Groups added. 115 */ 116 function bp_groups_admin_menu_order( $custom_menus = array() ) { 117 array_push( $custom_menus, 'bp-groups' ); 118 return $custom_menus; 119 } 120 add_filter( 'bp_admin_menu_order', 'bp_groups_admin_menu_order' ); 121 122 /** 123 * Set up the Groups admin page. 124 * 125 * Loaded before the page is rendered, this function does all initial setup, 126 * including: processing form requests, registering contextual help, and 127 * setting up screen options. 128 * 129 * @since 1.7.0 130 * 131 * @global BP_Groups_List_Table $bp_groups_list_table Groups screen list table. 132 */ 133 function bp_groups_admin_load() { 134 global $bp_groups_list_table; 135 136 // Build redirection URL. 137 $redirect_to = remove_query_arg( array( 'action', 'action2', 'gid', 'deleted', 'error', 'updated', 'success_new', 'error_new', 'success_modified', 'error_modified' ), $_SERVER['REQUEST_URI'] ); 138 139 $doaction = bp_admin_list_table_current_bulk_action(); 140 $min = bp_core_get_minified_asset_suffix(); 141 142 /** 143 * Fires at top of groups admin page. 144 * 145 * @since 1.7.0 146 * 147 * @param string $doaction Current $_GET action being performed in admin screen. 148 */ 149 do_action( 'bp_groups_admin_load', $doaction ); 150 151 // Edit screen. 152 if ( 'do_delete' == $doaction && ! empty( $_GET['gid'] ) ) { 153 154 check_admin_referer( 'bp-groups-delete' ); 155 156 $group_ids = wp_parse_id_list( $_GET['gid'] ); 157 158 $count = 0; 159 foreach ( $group_ids as $group_id ) { 160 if ( groups_delete_group( $group_id ) ) { 161 $count++; 162 } 163 } 164 165 $redirect_to = add_query_arg( 'deleted', $count, $redirect_to ); 166 167 bp_core_redirect( $redirect_to ); 168 169 } elseif ( 'edit' == $doaction && ! empty( $_GET['gid'] ) ) { 170 // Columns screen option. 171 add_screen_option( 'layout_columns', array( 'default' => 2, 'max' => 2, ) ); 172 173 get_current_screen()->add_help_tab( array( 174 'id' => 'bp-group-edit-overview', 175 'title' => __( 'Overview', 'buddypress' ), 176 'content' => 177 '<p>' . __( 'This page is a convenient way to edit the details associated with one of your groups.', 'buddypress' ) . '</p>' . 178 '<p>' . __( 'The Name and Description box is fixed in place, but you can reposition all the other boxes using drag and drop, and can minimize or expand them by clicking the title bar of each box. Use the Screen Options tab to hide or unhide, or to choose a 1- or 2-column layout for this screen.', 'buddypress' ) . '</p>' 179 ) ); 180 181 // Help panel - sidebar links. 182 get_current_screen()->set_help_sidebar( 183 '<p><strong>' . __( 'For more information:', 'buddypress' ) . '</strong></p>' . 184 '<p><a href="https://buddypress.org/support">' . __( 'Support Forums', 'buddypress' ) . '</a></p>' 185 ); 186 187 // Register metaboxes for the edit screen. 188 add_meta_box( 'submitdiv', _x( 'Save', 'group admin edit screen', 'buddypress' ), 'bp_groups_admin_edit_metabox_status', get_current_screen()->id, 'side', 'high' ); 189 add_meta_box( 'bp_group_settings', _x( 'Settings', 'group admin edit screen', 'buddypress' ), 'bp_groups_admin_edit_metabox_settings', get_current_screen()->id, 'side', 'core' ); 190 add_meta_box( 'bp_group_add_members', _x( 'Add New Members', 'group admin edit screen', 'buddypress' ), 'bp_groups_admin_edit_metabox_add_new_members', get_current_screen()->id, 'normal', 'core' ); 191 add_meta_box( 'bp_group_members', _x( 'Manage Members', 'group admin edit screen', 'buddypress' ), 'bp_groups_admin_edit_metabox_members', get_current_screen()->id, 'normal', 'core' ); 192 193 // Group Type metabox. Only added if group types have been registered. 194 $group_types = bp_groups_get_group_types(); 195 if ( ! empty( $group_types ) ) { 196 add_meta_box( 197 'bp_groups_admin_group_type', 198 _x( 'Group Type', 'groups admin edit screen', 'buddypress' ), 199 'bp_groups_admin_edit_metabox_group_type', 200 get_current_screen()->id, 201 'side', 202 'core' 203 ); 204 } 205 206 /** 207 * Fires after the registration of all of the default group meta boxes. 208 * 209 * @since 1.7.0 210 */ 211 do_action( 'bp_groups_admin_meta_boxes' ); 212 213 // Enqueue JavaScript files. 214 wp_enqueue_script( 'postbox' ); 215 wp_enqueue_script( 'dashboard' ); 216 217 // Index screen. 218 } else { 219 // Create the Groups screen list table. 220 $bp_groups_list_table = new BP_Groups_List_Table(); 221 222 // The per_page screen option. 223 add_screen_option( 'per_page', array( 'label' => _x( 'Groups', 'Groups per page (screen options)', 'buddypress' )) ); 224 225 // Help panel - overview text. 226 get_current_screen()->add_help_tab( array( 227 'id' => 'bp-groups-overview', 228 'title' => __( 'Overview', 'buddypress' ), 229 'content' => 230 '<p>' . __( 'You can manage groups much like you can manage comments and other content. This screen is customizable in the same ways as other management screens, and you can act on groups by using the on-hover action links or the Bulk Actions.', 'buddypress' ) . '</p>', 231 ) ); 232 233 get_current_screen()->add_help_tab( array( 234 'id' => 'bp-groups-overview-actions', 235 'title' => __( 'Group Actions', 'buddypress' ), 236 'content' => 237 '<p>' . __( 'Clicking "Visit" will take you to the group’s public page. Use this link to see what the group looks like on the front end of your site.', 'buddypress' ) . '</p>' . 238 '<p>' . __( 'Clicking "Edit" will take you to a Dashboard panel where you can manage various details about the group, such as its name and description, its members, and other settings.', 'buddypress' ) . '</p>' . 239 '<p>' . __( 'If you click "Delete" under a specific group, or select a number of groups and then choose Delete from the Bulk Actions menu, you will be led to a page where you’ll be asked to confirm the permanent deletion of the group(s).', 'buddypress' ) . '</p>', 240 ) ); 241 242 // Help panel - sidebar links. 243 get_current_screen()->set_help_sidebar( 244 '<p><strong>' . __( 'For more information:', 'buddypress' ) . '</strong></p>' . 245 '<p>' . __( '<a href="https://buddypress.org/support/">Support Forums</a>', 'buddypress' ) . '</p>' 246 ); 247 248 // Add accessible hidden heading and text for Groups screen pagination. 249 get_current_screen()->set_screen_reader_content( array( 250 /* translators: accessibility text */ 251 'heading_pagination' => __( 'Groups list navigation', 'buddypress' ), 252 ) ); 253 } 254 255 $bp = buddypress(); 256 257 // Enqueue CSS and JavaScript. 258 wp_enqueue_script( 'bp_groups_admin_js', $bp->plugin_url . "bp-groups/admin/js/admin{$min}.js", array( 'jquery', 'wp-ajax-response', 'jquery-ui-autocomplete' ), bp_get_version(), true ); 259 wp_localize_script( 'bp_groups_admin_js', 'BP_Group_Admin', array( 260 'add_member_placeholder' => __( 'Start typing a username to add a new member.', 'buddypress' ), 261 'warn_on_leave' => __( 'If you leave this page, you will lose any unsaved changes you have made to the group.', 'buddypress' ), 262 ) ); 263 wp_enqueue_style( 'bp_groups_admin_css', $bp->plugin_url . "bp-groups/admin/css/admin{$min}.css", array(), bp_get_version() ); 264 265 wp_style_add_data( 'bp_groups_admin_css', 'rtl', 'replace' ); 266 if ( $min ) { 267 wp_style_add_data( 'bp_groups_admin_css', 'suffix', $min ); 268 } 269 270 271 if ( $doaction && 'save' == $doaction ) { 272 // Get group ID. 273 $group_id = isset( $_REQUEST['gid'] ) ? (int) $_REQUEST['gid'] : ''; 274 275 $redirect_to = add_query_arg( array( 276 'gid' => (int) $group_id, 277 'action' => 'edit' 278 ), $redirect_to ); 279 280 // Check this is a valid form submission. 281 check_admin_referer( 'edit-group_' . $group_id ); 282 283 // Get the group from the database. 284 $group = groups_get_group( $group_id ); 285 286 // If the group doesn't exist, just redirect back to the index. 287 if ( empty( $group->slug ) ) { 288 wp_redirect( $redirect_to ); 289 exit; 290 } 291 292 // Check the form for the updated properties. 293 // Store errors. 294 $error = 0; 295 $success_new = $error_new = $success_modified = $error_modified = array(); 296 297 // Name, description and slug must not be empty. 298 if ( empty( $_POST['bp-groups-name'] ) ) { 299 $error = $error - 1; 300 } 301 if ( empty( $_POST['bp-groups-description'] ) ) { 302 $error = $error - 2; 303 } 304 if ( empty( $_POST['bp-groups-slug'] ) ) { 305 $error = $error - 4; 306 } 307 308 /* 309 * Group name, slug, and description are handled with 310 * groups_edit_base_group_details(). 311 */ 312 if ( ! $error && ! groups_edit_base_group_details( array( 313 'group_id' => $group_id, 314 'name' => $_POST['bp-groups-name'], 315 'slug' => $_POST['bp-groups-slug'], 316 'description' => $_POST['bp-groups-description'], 317 'notify_members' => false, 318 ) ) ) { 319 $error = $group_id; 320 } 321 322 // Enable discussion forum. 323 $enable_forum = ( isset( $_POST['group-show-forum'] ) ) ? 1 : 0; 324 325 /** 326 * Filters the allowed status values for the group. 327 * 328 * @since 1.0.2 329 * 330 * @param array $value Array of allowed group statuses. 331 */ 332 $allowed_status = apply_filters( 'groups_allowed_status', array( 'public', 'private', 'hidden' ) ); 333 $status = ( in_array( $_POST['group-status'], (array) $allowed_status ) ) ? $_POST['group-status'] : 'public'; 334 335 /** 336 * Filters the allowed invite status values for the group. 337 * 338 * @since 1.5.0 339 * 340 * @param array $value Array of allowed invite statuses. 341 */ 342 $allowed_invite_status = apply_filters( 'groups_allowed_invite_status', array( 'members', 'mods', 'admins' ) ); 343 $invite_status = in_array( $_POST['group-invite-status'], (array) $allowed_invite_status ) ? $_POST['group-invite-status'] : 'members'; 344 345 if ( !groups_edit_group_settings( $group_id, $enable_forum, $status, $invite_status ) ) { 346 $error = $group_id; 347 } 348 349 // Process new members. 350 $user_names = array(); 351 352 if ( ! empty( $_POST['bp-groups-new-members'] ) ) { 353 $user_names = array_merge( $user_names, explode( ',', $_POST['bp-groups-new-members'] ) ); 354 } 355 356 if ( ! empty( $user_names ) ) { 357 358 foreach( array_values( $user_names ) as $user_name ) { 359 $un = trim( $user_name ); 360 361 // Make sure the user exists before attempting 362 // to add to the group. 363 $user = get_user_by( 'slug', $un ); 364 365 if ( empty( $user ) ) { 366 $error_new[] = $un; 367 } else { 368 if ( ! groups_join_group( $group_id, $user->ID ) ) { 369 $error_new[] = $un; 370 } else { 371 $success_new[] = $un; 372 } 373 } 374 } 375 } 376 377 // Process member role changes. 378 if ( ! empty( $_POST['bp-groups-role'] ) && ! empty( $_POST['bp-groups-existing-role'] ) ) { 379 380 // Before processing anything, make sure you're not 381 // attempting to remove the all user admins. 382 $admin_count = 0; 383 foreach ( (array) $_POST['bp-groups-role'] as $new_role ) { 384 if ( 'admin' == $new_role ) { 385 $admin_count++; 386 break; 387 } 388 } 389 390 if ( ! $admin_count ) { 391 392 $redirect_to = add_query_arg( 'no_admins', 1, $redirect_to ); 393 $error = $group_id; 394 395 } else { 396 397 // Process only those users who have had their roles changed. 398 foreach ( (array) $_POST['bp-groups-role'] as $user_id => $new_role ) { 399 $user_id = (int) $user_id; 400 401 $existing_role = isset( $_POST['bp-groups-existing-role'][$user_id] ) ? $_POST['bp-groups-existing-role'][$user_id] : ''; 402 403 if ( $existing_role != $new_role ) { 404 $result = false; 405 406 switch ( $new_role ) { 407 case 'mod' : 408 // Admin to mod is a demotion. Demote to 409 // member, then fall through. 410 if ( 'admin' == $existing_role ) { 411 groups_demote_member( $user_id, $group_id ); 412 } 413 414 case 'admin' : 415 // If the user was banned, we must 416 // unban first. 417 if ( 'banned' == $existing_role ) { 418 groups_unban_member( $user_id, $group_id ); 419 } 420 421 // At this point, each existing_role 422 // is a member, so promote. 423 $result = groups_promote_member( $user_id, $group_id, $new_role ); 424 425 break; 426 427 case 'member' : 428 429 if ( 'admin' == $existing_role || 'mod' == $existing_role ) { 430 $result = groups_demote_member( $user_id, $group_id ); 431 } elseif ( 'banned' == $existing_role ) { 432 $result = groups_unban_member( $user_id, $group_id ); 433 } 434 435 break; 436 437 case 'banned' : 438 439 $result = groups_ban_member( $user_id, $group_id ); 440 441 break; 442 443 case 'remove' : 444 445 $result = groups_remove_member( $user_id, $group_id ); 446 447 break; 448 } 449 450 // Store the success or failure. 451 if ( $result ) { 452 $success_modified[] = $user_id; 453 } else { 454 $error_modified[] = $user_id; 455 } 456 } 457 } 458 } 459 } 460 461 /** 462 * Fires before redirect so plugins can do something first on save action. 463 * 464 * @since 1.6.0 465 * 466 * @param int $group_id ID of the group being edited. 467 */ 468 do_action( 'bp_group_admin_edit_after', $group_id ); 469 470 // Create the redirect URL. 471 if ( $error ) { 472 // This means there was an error updating group details. 473 $redirect_to = add_query_arg( 'error', (int) $error, $redirect_to ); 474 } else { 475 // Group details were update successfully. 476 $redirect_to = add_query_arg( 'updated', 1, $redirect_to ); 477 } 478 479 if ( !empty( $success_new ) ) { 480 $success_new = implode( ',', array_filter( $success_new, 'urlencode' ) ); 481 $redirect_to = add_query_arg( 'success_new', $success_new, $redirect_to ); 482 } 483 484 if ( !empty( $error_new ) ) { 485 $error_new = implode( ',', array_filter( $error_new, 'urlencode' ) ); 486 $redirect_to = add_query_arg( 'error_new', $error_new, $redirect_to ); 487 } 488 489 if ( !empty( $success_modified ) ) { 490 $success_modified = implode( ',', array_filter( $success_modified, 'urlencode' ) ); 491 $redirect_to = add_query_arg( 'success_modified', $success_modified, $redirect_to ); 492 } 493 494 if ( !empty( $error_modified ) ) { 495 $error_modified = implode( ',', array_filter( $error_modified, 'urlencode' ) ); 496 $redirect_to = add_query_arg( 'error_modified', $error_modified, $redirect_to ); 497 } 498 499 /** 500 * Filters the URL to redirect to after successfully editing a group. 501 * 502 * @since 1.7.0 503 * 504 * @param string $redirect_to URL to redirect user to. 505 */ 506 wp_redirect( apply_filters( 'bp_group_admin_edit_redirect', $redirect_to ) ); 507 exit; 508 509 510 // If a referrer and a nonce is supplied, but no action, redirect back. 511 } elseif ( ! empty( $_GET['_wp_http_referer'] ) ) { 512 wp_redirect( remove_query_arg( array( '_wp_http_referer', '_wpnonce' ), stripslashes( $_SERVER['REQUEST_URI'] ) ) ); 513 exit; 514 } 515 } 516 517 /** 518 * Handle save/update of screen options for the Groups component admin screen. 519 * 520 * @since 1.7.0 521 * 522 * @param string $value Will always be false unless another plugin filters it first. 523 * @param string $option Screen option name. 524 * @param string $new_value Screen option form value. 525 * @return string|int Option value. False to abandon update. 526 */ 527 function bp_groups_admin_screen_options( $value, $option, $new_value ) { 528 if ( 'toplevel_page_bp_groups_per_page' != $option && 'toplevel_page_bp_groups_network_per_page' != $option ) 529 return $value; 530 531 // Per page. 532 $new_value = (int) $new_value; 533 if ( $new_value < 1 || $new_value > 999 ) 534 return $value; 535 536 return $new_value; 537 } 538 539 /** 540 * Select the appropriate Groups admin screen, and output it. 541 * 542 * @since 1.7.0 543 */ 544 function bp_groups_admin() { 545 // Decide whether to load the index or edit screen. 546 $doaction = bp_admin_list_table_current_bulk_action(); 547 548 // Display the single group edit screen. 549 if ( 'edit' == $doaction && ! empty( $_GET['gid'] ) ) { 550 bp_groups_admin_edit(); 551 552 // Display the group deletion confirmation screen. 553 } elseif ( 'delete' == $doaction && ! empty( $_GET['gid'] ) ) { 554 bp_groups_admin_delete(); 555 556 // Otherwise, display the groups index screen. 557 } else { 558 bp_groups_admin_index(); 559 } 560 } 561 562 /** 563 * Display the single groups edit screen. 564 * 565 * @since 1.7.0 566 */ 567 function bp_groups_admin_edit() { 568 569 if ( ! bp_current_user_can( 'bp_moderate' ) ) 570 die( '-1' ); 571 572 $messages = array(); 573 574 // If the user has just made a change to a group, build status messages. 575 if ( !empty( $_REQUEST['no_admins'] ) || ! empty( $_REQUEST['error'] ) || ! empty( $_REQUEST['updated'] ) || ! empty( $_REQUEST['error_new'] ) || ! empty( $_REQUEST['success_new'] ) || ! empty( $_REQUEST['error_modified'] ) || ! empty( $_REQUEST['success_modified'] ) ) { 576 $no_admins = ! empty( $_REQUEST['no_admins'] ) ? 1 : 0; 577 $errors = ! empty( $_REQUEST['error'] ) ? $_REQUEST['error'] : ''; 578 $updated = ! empty( $_REQUEST['updated'] ) ? $_REQUEST['updated'] : ''; 579 $error_new = ! empty( $_REQUEST['error_new'] ) ? explode( ',', $_REQUEST['error_new'] ) : array(); 580 $success_new = ! empty( $_REQUEST['success_new'] ) ? explode( ',', $_REQUEST['success_new'] ) : array(); 581 $error_modified = ! empty( $_REQUEST['error_modified'] ) ? explode( ',', $_REQUEST['error_modified'] ) : array(); 582 $success_modified = ! empty( $_REQUEST['success_modified'] ) ? explode( ',', $_REQUEST['success_modified'] ) : array(); 583 584 if ( ! empty( $no_admins ) ) { 585 $messages[] = __( 'You cannot remove all administrators from a group.', 'buddypress' ); 586 } 587 588 if ( ! empty( $errors ) ) { 589 if ( $errors < 0 ) { 590 $messages[] = __( 'Group name, slug, and description are all required fields.', 'buddypress' ); 591 } else { 592 $messages[] = __( 'An error occurred when trying to update your group details.', 'buddypress' ); 593 } 594 595 } elseif ( ! empty( $updated ) ) { 596 $messages[] = __( 'The group has been updated successfully.', 'buddypress' ); 597 } 598 599 if ( ! empty( $error_new ) ) { 600 /* translators: %s: comma separated list of usernames */ 601 $messages[] = sprintf( __( 'The following users could not be added to the group: %s', 'buddypress' ), '<em>' . esc_html( implode( ', ', $error_new ) ) . '</em>' ); 602 } 603 604 if ( ! empty( $success_new ) ) { 605 /* translators: %s: comma separated list of usernames */ 606 $messages[] = sprintf( __( 'The following users were successfully added to the group: %s', 'buddypress' ), '<em>' . esc_html( implode( ', ', $success_new ) ) . '</em>' ); 607 } 608 609 if ( ! empty( $error_modified ) ) { 610 $error_modified = bp_groups_admin_get_usernames_from_ids( $error_modified ); 611 /* translators: %s: comma separated list of usernames */ 612 $messages[] = sprintf( __( 'An error occurred when trying to modify the following members: %s', 'buddypress' ), '<em>' . esc_html( implode( ', ', $error_modified ) ) . '</em>' ); 613 } 614 615 if ( ! empty( $success_modified ) ) { 616 $success_modified = bp_groups_admin_get_usernames_from_ids( $success_modified ); 617 /* translators: %s: comma separated list of usernames */ 618 $messages[] = sprintf( __( 'The following members were successfully modified: %s', 'buddypress' ), '<em>' . esc_html( implode( ', ', $success_modified ) ) . '</em>' ); 619 } 620 } 621 622 $is_error = ! empty( $no_admins ) || ! empty( $errors ) || ! empty( $error_new ) || ! empty( $error_modified ); 623 624 // Get the group from the database. 625 $group = groups_get_group( (int) $_GET['gid'] ); 626 627 $group_name = isset( $group->name ) ? bp_get_group_name( $group ) : ''; 628 629 // Construct URL for form. 630 $form_url = remove_query_arg( array( 'action', 'deleted', 'no_admins', 'error', 'error_new', 'success_new', 'error_modified', 'success_modified' ), $_SERVER['REQUEST_URI'] ); 631 $form_url = add_query_arg( 'action', 'save', $form_url ); 632 633 /** 634 * Fires before the display of the edit form. 635 * 636 * Useful for plugins to modify the group before display. 637 * 638 * @since 1.7.0 639 * 640 * @param BP_Groups_Group $this Instance of the current group being edited. Passed by reference. 641 */ 642 do_action_ref_array( 'bp_groups_admin_edit', array( &$group ) ); ?> 643 644 <div class="wrap"> 645 <h1 class="wp-heading-inline"><?php _e( 'Edit Group', 'buddypress' ); ?></h1> 646 647 <?php if ( is_user_logged_in() && bp_user_can_create_groups() ) : ?> 648 <a class="page-title-action" href="<?php echo trailingslashit( bp_get_groups_directory_permalink() . 'create' ); ?>"><?php _e( 'Add New', 'buddypress' ); ?></a> 649 <?php endif; ?> 650 651 <hr class="wp-header-end"> 652 653 <?php // If the user has just made a change to an group, display the status messages. ?> 654 <?php if ( !empty( $messages ) ) : ?> 655 <div id="moderated" class="<?php echo ( $is_error ) ? 'error' : 'updated'; ?> notice is-dismissible"><p><?php echo implode( "</p><p>", $messages ); ?></p></div> 656 <?php endif; ?> 657 658 <?php if ( $group->id ) : ?> 659 660 <form action="<?php echo esc_url( $form_url ); ?>" id="bp-groups-edit-form" method="post"> 661 <div id="poststuff"> 662 663 <div id="post-body" class="metabox-holder columns-<?php echo 1 == get_current_screen()->get_columns() ? '1' : '2'; ?>"> 664 <div id="post-body-content"> 665 <div id="postdiv"> 666 <div id="bp_groups_name" class="groupbox"> 667 <h2><?php _e( 'Name and Description', 'buddypress' ); ?></h2> 668 <div class="inside"> 669 <label for="bp-groups-name" class="screen-reader-text"><?php 670 /* translators: accessibility text */ 671 _e( 'Group Name', 'buddypress' ); 672 ?></label> 673 <input type="text" name="bp-groups-name" id="bp-groups-name" value="<?php echo esc_attr( stripslashes( $group_name ) ) ?>" /> 674 <div id="bp-groups-permalink-box"> 675 <strong><?php esc_html_e( 'Permalink:', 'buddypress' ) ?></strong> 676 <span id="bp-groups-permalink"> 677 <?php bp_groups_directory_permalink(); ?> <input type="text" id="bp-groups-slug" name="bp-groups-slug" value="<?php bp_group_slug( $group ); ?>" autocomplete="off"> / 678 </span> 679 <a href="<?php echo bp_group_permalink( $group ) ?>" class="button button-small" id="bp-groups-visit-group"><?php esc_html_e( 'Visit Group', 'buddypress' ) ?></a> 680 </div> 681 682 <label for="bp-groups-description" class="screen-reader-text"><?php 683 /* translators: accessibility text */ 684 _e( 'Group Description', 'buddypress' ); 685 ?></label> 686 <?php wp_editor( stripslashes( $group->description ), 'bp-groups-description', array( 'media_buttons' => false, 'teeny' => true, 'textarea_rows' => 5, 'quicktags' => array( 'buttons' => 'strong,em,link,block,del,ins,img,code,spell,close' ) ) ); ?> 687 </div> 688 </div> 689 </div> 690 </div><!-- #post-body-content --> 691 692 <div id="postbox-container-1" class="postbox-container"> 693 <?php do_meta_boxes( get_current_screen()->id, 'side', $group ); ?> 694 </div> 695 696 <div id="postbox-container-2" class="postbox-container"> 697 <?php do_meta_boxes( get_current_screen()->id, 'normal', $group ); ?> 698 <?php do_meta_boxes( get_current_screen()->id, 'advanced', $group ); ?> 699 </div> 700 </div><!-- #post-body --> 701 702 </div><!-- #poststuff --> 703 <?php wp_nonce_field( 'closedpostboxes', 'closedpostboxesnonce', false ); ?> 704 <?php wp_nonce_field( 'meta-box-order', 'meta-box-order-nonce', false ); ?> 705 <?php wp_nonce_field( 'edit-group_' . $group->id ); ?> 706 </form> 707 708 <?php else : ?> 709 710 <p><?php 711 printf( 712 '%1$s <a href="%2$s">%3$s</a>', 713 __( 'No group found with this ID.', 'buddypress' ), 714 esc_url( bp_get_admin_url( 'admin.php?page=bp-groups' ) ), 715 __( 'Go back and try again.', 'buddypress' ) 716 ); 717 ?></p> 718 719 <?php endif; ?> 720 721 </div><!-- .wrap --> 722 723 <?php 724 } 725 726 /** 727 * Display the Group delete confirmation screen. 728 * 729 * We include a separate confirmation because group deletion is truly 730 * irreversible. 731 * 732 * @since 1.7.0 733 */ 734 function bp_groups_admin_delete() { 735 736 if ( ! bp_current_user_can( 'bp_moderate' ) ) { 737 die( '-1' ); 738 } 739 740 $group_ids = isset( $_REQUEST['gid'] ) ? $_REQUEST['gid'] : 0; 741 if ( ! is_array( $group_ids ) ) { 742 $group_ids = explode( ',', $group_ids ); 743 } 744 $group_ids = wp_parse_id_list( $group_ids ); 745 $groups = groups_get_groups( array( 746 'include' => $group_ids, 747 'show_hidden' => true, 748 'per_page' => null, // Return all results. 749 ) ); 750 751 // Create a new list of group ids, based on those that actually exist. 752 $gids = array(); 753 foreach ( $groups['groups'] as $group ) { 754 $gids[] = $group->id; 755 } 756 757 $base_url = remove_query_arg( array( 'action', 'action2', 'paged', 's', '_wpnonce', 'gid' ), $_SERVER['REQUEST_URI'] ); ?> 758 759 <div class="wrap"> 760 <h1 class="wp-heading-inline"><?php _e( 'Delete Groups', 'buddypress' ) ?></h1> 761 <hr class="wp-header-end"> 762 763 <p><?php _e( 'You are about to delete the following groups:', 'buddypress' ) ?></p> 764 765 <ul class="bp-group-delete-list"> 766 <?php foreach ( $groups['groups'] as $group ) : ?> 767 <li><?php echo esc_html( bp_get_group_name( $group ) ); ?></li> 768 <?php endforeach; ?> 769 </ul> 770 771 <p><strong><?php _e( 'This action cannot be undone.', 'buddypress' ) ?></strong></p> 772 773 <a class="button-primary" href="<?php echo esc_url( wp_nonce_url( add_query_arg( array( 'action' => 'do_delete', 'gid' => implode( ',', $gids ) ), $base_url ), 'bp-groups-delete' ) ); ?>"><?php _e( 'Delete Permanently', 'buddypress' ) ?></a> 774 <a class="button" href="<?php echo esc_attr( $base_url ); ?>"><?php _e( 'Cancel', 'buddypress' ) ?></a> 775 </div> 776 777 <?php 778 } 779 780 /** 781 * Display the Groups admin index screen. 782 * 783 * This screen contains a list of all BuddyPress groups. 784 * 785 * @since 1.7.0 786 * 787 * @global BP_Groups_List_Table $bp_groups_list_table Group screen list table. 788 * @global string $plugin_page Currently viewed plugin page. 789 */ 790 function bp_groups_admin_index() { 791 global $bp_groups_list_table, $plugin_page; 792 793 $messages = array(); 794 795 // If the user has just made a change to a group, build status messages. 796 if ( ! empty( $_REQUEST['deleted'] ) ) { 797 $deleted = ! empty( $_REQUEST['deleted'] ) ? (int) $_REQUEST['deleted'] : 0; 798 799 if ( $deleted > 0 ) { 800 /* translators: %s: number of deleted groups */ 801 $messages[] = sprintf( _n( '%s group has been permanently deleted.', '%s groups have been permanently deleted.', $deleted, 'buddypress' ), number_format_i18n( $deleted ) ); 802 } 803 } 804 805 // Prepare the group items for display. 806 $bp_groups_list_table->prepare_items(); 807 808 /** 809 * Fires before the display of messages for the edit form. 810 * 811 * Useful for plugins to modify the messages before display. 812 * 813 * @since 1.7.0 814 * 815 * @param array $messages Array of messages to be displayed. 816 */ 817 do_action( 'bp_groups_admin_index', $messages ); ?> 818 819 <div class="wrap"> 820 821 <h1 class="wp-heading-inline"><?php _e( 'Groups', 'buddypress' ); ?></h1> 822 823 <?php if ( is_user_logged_in() && bp_user_can_create_groups() ) : ?> 824 <a class="page-title-action" href="<?php echo trailingslashit( bp_get_groups_directory_permalink() . 'create' ); ?>"><?php _e( 'Add New', 'buddypress' ); ?></a> 825 <?php endif; ?> 826 827 <?php if ( !empty( $_REQUEST['s'] ) ) : ?> 828 <span class="subtitle"><?php printf( __( 'Search results for “%s”', 'buddypress' ), wp_html_excerpt( esc_html( stripslashes( $_REQUEST['s'] ) ), 50 ) ); ?></span> 829 <?php endif; ?> 830 831 <hr class="wp-header-end"> 832 833 <?php // If the user has just made a change to an group, display the status messages. ?> 834 <?php if ( !empty( $messages ) ) : ?> 835 <div id="moderated" class="<?php echo ( ! empty( $_REQUEST['error'] ) ) ? 'error' : 'updated'; ?> notice is-dismissible"><p><?php echo implode( "<br/>\n", $messages ); ?></p></div> 836 <?php endif; ?> 837 838 <?php // Display each group on its own row. ?> 839 <?php $bp_groups_list_table->views(); ?> 840 841 <form id="bp-groups-form" action="" method="get"> 842 <?php $bp_groups_list_table->search_box( __( 'Search all Groups', 'buddypress' ), 'bp-groups' ); ?> 843 <input type="hidden" name="page" value="<?php echo esc_attr( $plugin_page ); ?>" /> 844 <?php $bp_groups_list_table->display(); ?> 845 </form> 846 847 </div> 848 849 <?php 850 } 851 852 /** 853 * Markup for the single group's Settings metabox. 854 * 855 * @since 1.7.0 856 * 857 * @param object $item Information about the current group. 858 */ 859 function bp_groups_admin_edit_metabox_settings( $item ) { 860 861 $invite_status = bp_group_get_invite_status( $item->id ); ?> 862 863 <?php if ( bp_is_active( 'forums' ) ) : ?> 864 <div class="bp-groups-settings-section" id="bp-groups-settings-section-forum"> 865 <label for="group-show-forum"><input type="checkbox" name="group-show-forum" id="group-show-forum" <?php checked( $item->enable_forum ) ?> /> <?php _e( 'Enable discussion forum', 'buddypress' ) ?></label> 866 </div> 867 <?php endif; ?> 868 869 <div class="bp-groups-settings-section" id="bp-groups-settings-section-status"> 870 <fieldset> 871 <legend><?php _e( 'Privacy', 'buddypress' ); ?></legend> 872 873 <label for="bp-group-status-public"><input type="radio" name="group-status" id="bp-group-status-public" value="public" <?php checked( $item->status, 'public' ) ?> /><?php _e( 'Public', 'buddypress' ) ?></label> 874 <label for="bp-group-status-private"><input type="radio" name="group-status" id="bp-group-status-private" value="private" <?php checked( $item->status, 'private' ) ?> /><?php _e( 'Private', 'buddypress' ) ?></label> 875 <label for="bp-group-status-hidden"><input type="radio" name="group-status" id="bp-group-status-hidden" value="hidden" <?php checked( $item->status, 'hidden' ) ?> /><?php _e( 'Hidden', 'buddypress' ) ?></label> 876 </fieldset> 877 </div> 878 879 <div class="bp-groups-settings-section" id="bp-groups-settings-section-invite-status"> 880 <fieldset> 881 <legend><?php _e( 'Who can invite others to this group?', 'buddypress' ); ?></legend> 882 883 <label for="bp-group-invite-status-members"><input type="radio" name="group-invite-status" id="bp-group-invite-status-members" value="members" <?php checked( $invite_status, 'members' ) ?> /><?php _e( 'All group members', 'buddypress' ) ?></label> 884 <label for="bp-group-invite-status-mods"><input type="radio" name="group-invite-status" id="bp-group-invite-status-mods" value="mods" <?php checked( $invite_status, 'mods' ) ?> /><?php _e( 'Group admins and mods only', 'buddypress' ) ?></label> 885 <label for="bp-group-invite-status-admins"><input type="radio" name="group-invite-status" id="bp-group-invite-status-admins" value="admins" <?php checked( $invite_status, 'admins' ) ?> /><?php _e( 'Group admins only', 'buddypress' ) ?></label> 886 </fieldset> 887 </div> 888 889 <?php 890 } 891 892 /** 893 * Output the markup for a single group's Add New Members metabox. 894 * 895 * @since 1.7.0 896 * 897 * @param BP_Groups_Group $item The BP_Groups_Group object for the current group. 898 */ 899 function bp_groups_admin_edit_metabox_add_new_members( $item ) { 900 if ( bp_is_large_install() ) { 901 $class = ''; 902 $notice = __( 'Enter a comma-separated list of user logins.', 'buddypress' ); 903 } else { 904 $class = 'bp-suggest-user'; 905 $notice = ''; 906 } 907 908 ?> 909 910 <label for="bp-groups-new-members" class="screen-reader-text"><?php 911 /* translators: accessibility text */ 912 _e( 'Add new members', 'buddypress' ); 913 ?></label> 914 <input name="bp-groups-new-members" type="text" id="bp-groups-new-members" class="<?php echo esc_attr( $class ); ?>" placeholder="" /> 915 <?php if ( $notice ) : ?> 916 <p class="description"><?php echo esc_html( $notice ); ?></p> 917 <?php endif; ?> 918 <ul id="bp-groups-new-members-list"></ul> 919 <?php 920 } 921 922 /** 923 * Renders the Members metabox on single group pages. 924 * 925 * @since 1.7.0 926 * 927 * @param BP_Groups_Group $item The BP_Groups_Group object for the current group. 928 */ 929 function bp_groups_admin_edit_metabox_members( $item ) { 930 // Use the BP REST API if it supported. 931 if ( bp_rest_api_is_available() && bp_groups_has_manage_group_members_templates() ) { 932 wp_enqueue_script( 'bp-group-manage-members' ); 933 wp_localize_script( 934 'bp-group-manage-members', 935 'bpGroupManageMembersSettings', 936 bp_groups_get_group_manage_members_script_data( $item->id ) 937 ); 938 939 // Inject the Group Manage Members interface. 940 bp_groups_manage_group_members_interface( 'admin_footer' ); 941 942 /** 943 * Echo out the JavaScript variable. 944 * This seems to be required by the autocompleter, leaving this here for now... 945 */ 946 echo '<script type="text/javascript">var group_id = "' . esc_js( $item->id ) . '";</script>'; 947 return; 948 } 949 950 // Pull up a list of group members, so we can separate out the types 951 // We'll also keep track of group members here to place them into a 952 // JavaScript variable, which will help with group member autocomplete. 953 $members = array( 954 'admin' => array(), 955 'mod' => array(), 956 'member' => array(), 957 'banned' => array(), 958 ); 959 960 $pagination = array( 961 'admin' => array(), 962 'mod' => array(), 963 'member' => array(), 964 'banned' => array(), 965 ); 966 967 foreach ( $members as $type => &$member_type_users ) { 968 $page_qs_key = $type . '_page'; 969 $current_type_page = isset( $_GET[ $page_qs_key ] ) ? absint( $_GET[ $page_qs_key ] ) : 1; 970 $member_type_query = new BP_Group_Member_Query( array( 971 'group_id' => $item->id, 972 'group_role' => array( $type ), 973 'type' => 'alphabetical', 974 /** 975 * Filters the admin members type per page value. 976 * 977 * @since 2.8.0 978 * 979 * @param int $value Member types per page. Default 10. 980 * @param string $type Member type. 981 */ 982 'per_page' => apply_filters( 'bp_groups_admin_members_type_per_page', 10, $type ), 983 'page' => $current_type_page, 984 ) ); 985 986 $member_type_users = $member_type_query->results; 987 $pagination[ $type ] = bp_groups_admin_create_pagination_links( $member_type_query, $type ); 988 } 989 990 // Echo out the JavaScript variable. 991 echo '<script type="text/javascript">var group_id = "' . esc_js( $item->id ) . '";</script>'; 992 993 // Loop through each member type. 994 foreach ( $members as $member_type => $type_users ) : ?> 995 996 <div class="bp-groups-member-type" id="bp-groups-member-type-<?php echo esc_attr( $member_type ) ?>"> 997 998 <h3><?php switch ( $member_type ) : 999 case 'admin' : esc_html_e( 'Administrators', 'buddypress' ); break; 1000 case 'mod' : esc_html_e( 'Moderators', 'buddypress' ); break; 1001 case 'member' : esc_html_e( 'Members', 'buddypress' ); break; 1002 case 'banned' : esc_html_e( 'Banned Members', 'buddypress' ); break; 1003 endswitch; ?></h3> 1004 1005 <div class="bp-group-admin-pagination table-top"> 1006 <?php echo $pagination[ $member_type ] ?> 1007 </div> 1008 1009 <?php if ( !empty( $type_users ) ) : ?> 1010 1011 <table class="widefat bp-group-members"> 1012 <thead> 1013 <tr> 1014 <th scope="col" class="uid-column"><?php _ex( 'ID', 'Group member user_id in group admin', 'buddypress' ); ?></th> 1015 <th scope="col" class="uname-column"><?php _ex( 'Name', 'Group member name in group admin', 'buddypress' ); ?></th> 1016 <th scope="col" class="urole-column"><?php _ex( 'Group Role', 'Group member role in group admin', 'buddypress' ); ?></th> 1017 </tr> 1018 </thead> 1019 1020 <tbody> 1021 1022 <?php foreach ( $type_users as $type_user ) : ?> 1023 <tr> 1024 <th scope="row" class="uid-column"><?php echo esc_html( $type_user->ID ); ?></th> 1025 1026 <td class="uname-column"> 1027 <a style="float: left;" href="<?php echo bp_core_get_user_domain( $type_user->ID ); ?>"><?php echo bp_core_fetch_avatar( array( 1028 'item_id' => $type_user->ID, 1029 'width' => '32', 1030 'height' => '32' 1031 ) ); ?></a> 1032 1033 <span style="margin: 8px; float: left;"><?php echo bp_core_get_userlink( $type_user->ID ); ?></span> 1034 </td> 1035 1036 <td class="urole-column"> 1037 <label for="bp-groups-role-<?php echo esc_attr( $type_user->ID ); ?>" class="screen-reader-text"><?php 1038 /* translators: accessibility text */ 1039 _e( 'Select group role for member', 'buddypress' ); 1040 ?></label> 1041 <select class="bp-groups-role" id="bp-groups-role-<?php echo esc_attr( $type_user->ID ); ?>" name="bp-groups-role[<?php echo esc_attr( $type_user->ID ); ?>]"> 1042 <optgroup label="<?php esc_attr_e( 'Roles', 'buddypress' ); ?>"> 1043 <option class="admin" value="admin" <?php selected( 'admin', $member_type ); ?>><?php esc_html_e( 'Administrator', 'buddypress' ); ?></option> 1044 <option class="mod" value="mod" <?php selected( 'mod', $member_type ); ?>><?php esc_html_e( 'Moderator', 'buddypress' ); ?></option> 1045 <option class="member" value="member" <?php selected( 'member', $member_type ); ?>><?php esc_html_e( 'Member', 'buddypress' ); ?></option> 1046 <?php if ( 'banned' === $member_type ) : ?> 1047 <option class="banned" value="banned" <?php selected( 'banned', $member_type ); ?>><?php esc_html_e( 'Banned', 'buddypress' ); ?></option> 1048 <?php endif; ?> 1049 </optgroup> 1050 <optgroup label="<?php esc_attr_e( 'Actions', 'buddypress' ); ?>"> 1051 <option class="remove" value="remove"><?php esc_html_e( 'Remove', 'buddypress' ); ?></option> 1052 <?php if ( 'banned' !== $member_type ) : ?> 1053 <option class="banned" value="banned"><?php esc_html_e( 'Ban', 'buddypress' ); ?></option> 1054 <?php endif; ?> 1055 </optgroup> 1056 </select> 1057 1058 <?php 1059 /** 1060 * Store the current role for this user, 1061 * so we can easily detect changes. 1062 * 1063 * @todo remove this, and do database detection on save 1064 */ 1065 ?> 1066 <input type="hidden" name="bp-groups-existing-role[<?php echo esc_attr( $type_user->ID ); ?>]" value="<?php echo esc_attr( $member_type ); ?>" /> 1067 </td> 1068 </tr> 1069 1070 <?php if ( has_filter( 'bp_groups_admin_manage_member_row' ) ) : ?> 1071 <tr> 1072 <td colspan="3"> 1073 <?php 1074 1075 /** 1076 * Fires after the listing of a single row for members in a group on the group edit screen. 1077 * 1078 * @since 1.8.0 1079 * 1080 * @param int $ID ID of the user being rendered. 1081 * @param BP_Groups_Group $item Object for the current group. 1082 */ 1083 do_action( 'bp_groups_admin_manage_member_row', $type_user->ID, $item ); ?> 1084 </td> 1085 </tr> 1086 <?php endif; ?> 1087 1088 <?php endforeach; ?> 1089 1090 </tbody> 1091 </table> 1092 1093 <?php else : ?> 1094 1095 <p class="bp-groups-no-members description"><?php esc_html_e( 'No members of this type', 'buddypress' ); ?></p> 1096 1097 <?php endif; ?> 1098 1099 </div><!-- .bp-groups-member-type --> 1100 1101 <?php endforeach; 1102 } 1103 1104 /** 1105 * Renders the Status metabox for the Groups admin edit screen. 1106 * 1107 * @since 1.7.0 1108 * 1109 * @param object $item Information about the currently displayed group. 1110 */ 1111 function bp_groups_admin_edit_metabox_status( $item ) { 1112 $base_url = add_query_arg( array( 1113 'page' => 'bp-groups', 1114 'gid' => $item->id 1115 ), bp_get_admin_url( 'admin.php' ) ); ?> 1116 1117 <div id="submitcomment" class="submitbox"> 1118 <div id="major-publishing-actions"> 1119 <div id="delete-action"> 1120 <a class="submitdelete deletion" href="<?php echo esc_url( wp_nonce_url( add_query_arg( 'action', 'delete', $base_url ), 'bp-groups-delete' ) ); ?>"><?php _e( 'Delete Group', 'buddypress' ) ?></a> 1121 </div> 1122 1123 <div id="publishing-action"> 1124 <?php submit_button( __( 'Save Changes', 'buddypress' ), 'primary', 'save', false ); ?> 1125 </div> 1126 <div class="clear"></div> 1127 </div><!-- #major-publishing-actions --> 1128 </div><!-- #submitcomment --> 1129 1130 <?php 1131 } 1132 1133 /** 1134 * Render the Group Type metabox. 1135 * 1136 * @since 2.6.0 1137 * 1138 * @param BP_Groups_Group|null $group The BP_Groups_Group object corresponding to the group being edited. 1139 */ 1140 function bp_groups_admin_edit_metabox_group_type( BP_Groups_Group $group = null ) { 1141 1142 // Bail if no group ID. 1143 if ( empty( $group->id ) ) { 1144 return; 1145 } 1146 1147 $types = bp_groups_get_group_types( array(), 'objects' ); 1148 $current_types = (array) bp_groups_get_group_type( $group->id, false ); 1149 $backend_only = bp_groups_get_group_types( array( 'show_in_create_screen' => false ) ); 1150 ?> 1151 1152 <label for="bp-groups-group-type" class="screen-reader-text"><?php 1153 /* translators: accessibility text */ 1154 esc_html_e( 'Select group type', 'buddypress' ); 1155 ?></label> 1156 1157 <ul class="categorychecklist form-no-clear"> 1158 <?php foreach ( $types as $type ) : ?> 1159 <li> 1160 <label class="selectit"><input value="<?php echo esc_attr( $type->name ) ?>" name="bp-groups-group-type[]" type="checkbox" <?php checked( true, in_array( $type->name, $current_types ) ); ?>> 1161 <?php 1162 echo esc_html( $type->labels['singular_name'] ); 1163 if ( in_array( $type->name, $backend_only ) ) { 1164 printf( ' <span class="description">%s</span>', esc_html__( '(Not available on the front end)', 'buddypress' ) ); 1165 } 1166 ?> 1167 1168 </label> 1169 </li> 1170 1171 <?php endforeach; ?> 1172 1173 </ul> 1174 1175 <?php 1176 1177 wp_nonce_field( 'bp-group-type-change-' . $group->id, 'bp-group-type-nonce' ); 1178 } 1179 1180 /** 1181 * Process changes from the Group Type metabox. 1182 * 1183 * @since 2.6.0 1184 * 1185 * @param int $group_id Group ID. 1186 */ 1187 function bp_groups_process_group_type_update( $group_id ) { 1188 if ( ! isset( $_POST['bp-group-type-nonce'] ) ) { 1189 return; 1190 } 1191 1192 check_admin_referer( 'bp-group-type-change-' . $group_id, 'bp-group-type-nonce' ); 1193 1194 // Permission check. 1195 if ( ! bp_current_user_can( 'bp_moderate' ) ) { 1196 return; 1197 } 1198 1199 $group_types = ! empty( $_POST['bp-groups-group-type'] ) ? wp_unslash( $_POST['bp-groups-group-type'] ) : array(); 1200 1201 /* 1202 * If an invalid group type is passed, someone's doing something 1203 * fishy with the POST request, so we can fail silently. 1204 */ 1205 if ( bp_groups_set_group_type( $group_id, $group_types ) ) { 1206 // @todo Success messages can't be posted because other stuff happens on the page load. 1207 } 1208 } 1209 add_action( 'bp_group_admin_edit_after', 'bp_groups_process_group_type_update' ); 1210 1211 /** 1212 * Create pagination links out of a BP_Group_Member_Query. 1213 * 1214 * This function is intended to create pagination links for use under the 1215 * Manage Members section of the Groups Admin Dashboard pages. It is a stopgap 1216 * measure until a more general pagination solution is in place for BuddyPress. 1217 * Plugin authors should not use this function, as it is likely to be 1218 * deprecated soon. 1219 * 1220 * @since 1.8.0 1221 * 1222 * @param BP_Group_Member_Query $query A BP_Group_Member_Query object. 1223 * @param string $member_type member|mod|admin|banned. 1224 * @return string Pagination links HTML. 1225 */ 1226 function bp_groups_admin_create_pagination_links( BP_Group_Member_Query $query, $member_type ) { 1227 $pagination = ''; 1228 1229 if ( ! in_array( $member_type, array( 'admin', 'mod', 'member', 'banned' ) ) ) { 1230 return $pagination; 1231 } 1232 1233 // The key used to paginate this member type in the $_GET global. 1234 $qs_key = $member_type . '_page'; 1235 $url_base = remove_query_arg( array( $qs_key, 'updated', 'success_modified' ), $_SERVER['REQUEST_URI'] ); 1236 1237 $page = isset( $_GET[ $qs_key ] ) ? absint( $_GET[ $qs_key ] ) : 1; 1238 1239 /** 1240 * Filters the number of members per member type that is displayed in group editing admin area. 1241 * 1242 * @since 2.8.0 1243 * 1244 * @param string $member_type Member type, which is a group role (admin, mod etc). 1245 */ 1246 $per_page = apply_filters( 'bp_groups_admin_members_type_per_page', 10, $member_type ); 1247 1248 // Don't show anything if there's no pagination. 1249 if ( 1 === $page && $query->total_users <= $per_page ) { 1250 return $pagination; 1251 } 1252 1253 $current_page_start = ( ( $page - 1 ) * $per_page ) + 1; 1254 $current_page_end = $page * $per_page > intval( $query->total_users ) ? $query->total_users : $page * $per_page; 1255 1256 $pag_links = paginate_links( array( 1257 'base' => add_query_arg( $qs_key, '%#%', $url_base ), 1258 'format' => '', 1259 'prev_text' => __( '«', 'buddypress' ), 1260 'next_text' => __( '»', 'buddypress' ), 1261 'total' => ceil( $query->total_users / $per_page ), 1262 'current' => $page, 1263 ) ); 1264 1265 if ( 1 == $query->total_users ) { 1266 $viewing_text = __( 'Viewing 1 member', 'buddypress' ); 1267 } else { 1268 $viewing_text = sprintf( 1269 /* translators: 1: group member from number. 2: group member to number. 3: total group members. */ 1270 _nx( 'Viewing %1$s - %2$s of %3$s member', 'Viewing %1$s - %2$s of %3$s members', $query->total_users, 'Group members pagination in group admin', 'buddypress' ), 1271 bp_core_number_format( $current_page_start ), 1272 bp_core_number_format( $current_page_end ), 1273 bp_core_number_format( $query->total_users ) 1274 ); 1275 } 1276 1277 $pagination .= '<span class="bp-group-admin-pagination-viewing">' . $viewing_text . '</span>'; 1278 $pagination .= '<span class="bp-group-admin-pagination-links">' . $pag_links . '</span>'; 1279 1280 return $pagination; 1281 } 1282 1283 /** 1284 * Get a set of usernames corresponding to a set of user IDs. 1285 * 1286 * @since 1.7.0 1287 * 1288 * @param array $user_ids Array of user IDs. 1289 * @return array Array of user_logins corresponding to $user_ids. 1290 */ 1291 function bp_groups_admin_get_usernames_from_ids( $user_ids = array() ) { 1292 1293 $usernames = array(); 1294 $users = new WP_User_Query( array( 'blog_id' => 0, 'include' => $user_ids ) ); 1295 1296 foreach ( (array) $users->results as $user ) { 1297 $usernames[] = $user->user_login; 1298 } 1299 1300 return $usernames; 1301 } 1302 1303 /** 1304 * AJAX handler for group member autocomplete requests. 1305 * 1306 * @since 1.7.0 1307 */ 1308 function bp_groups_admin_autocomplete_handler() { 1309 1310 // Bail if user user shouldn't be here, or is a large network. 1311 if ( ! bp_current_user_can( 'bp_moderate' ) || bp_is_large_install() ) { 1312 wp_die( -1 ); 1313 } 1314 1315 $term = isset( $_GET['term'] ) ? sanitize_text_field( $_GET['term'] ) : ''; 1316 $group_id = isset( $_GET['group_id'] ) ? absint( $_GET['group_id'] ) : 0; 1317 1318 if ( ! $term || ! $group_id ) { 1319 wp_die( -1 ); 1320 } 1321 1322 $suggestions = bp_core_get_suggestions( array( 1323 'group_id' => -$group_id, // A negative value will exclude this group's members from the suggestions. 1324 'limit' => 10, 1325 'term' => $term, 1326 'type' => 'members', 1327 ) ); 1328 1329 $matches = array(); 1330 1331 if ( $suggestions && ! is_wp_error( $suggestions ) ) { 1332 foreach ( $suggestions as $user ) { 1333 1334 $matches[] = array( 1335 // Translators: 1: user_login, 2: user_email. 1336 'label' => sprintf( __( '%1$s (%2$s)', 'buddypress' ), $user->name, $user->ID ), 1337 'value' => $user->ID, 1338 ); 1339 } 1340 } 1341 1342 wp_die( json_encode( $matches ) ); 1343 } 1344 add_action( 'wp_ajax_bp_group_admin_member_autocomplete', 'bp_groups_admin_autocomplete_handler' ); 1345 1346 /** 1347 * Process input from the Group Type bulk change select. 1348 * 1349 * @since 2.7.0 1350 * 1351 * @param string $doaction Current $_GET action being performed in admin screen. 1352 */ 1353 function bp_groups_admin_process_group_type_bulk_changes( $doaction ) { 1354 // Bail if no groups are specified or if this isn't a relevant action. 1355 if ( empty( $_REQUEST['gid'] ) 1356 || ( empty( $_REQUEST['bp_change_type'] ) && empty( $_REQUEST['bp_change_type2'] ) ) 1357 || empty( $_REQUEST['bp_change_group_type'] ) 1358 ) { 1359 return; 1360 } 1361 1362 // Bail if nonce check fails. 1363 check_admin_referer( 'bp-bulk-groups-change-type-' . bp_loggedin_user_id(), 'bp-bulk-groups-change-type-nonce' ); 1364 1365 if ( ! bp_current_user_can( 'bp_moderate' ) ) { 1366 return; 1367 } 1368 1369 $new_type = ''; 1370 if ( ! empty( $_REQUEST['bp_change_type2'] ) ) { 1371 $new_type = sanitize_text_field( $_REQUEST['bp_change_type2'] ); 1372 } elseif ( ! empty( $_REQUEST['bp_change_type'] ) ) { 1373 $new_type = sanitize_text_field( $_REQUEST['bp_change_type'] ); 1374 } 1375 1376 // Check that the selected type actually exists. 1377 if ( 'remove_group_type' !== $new_type && null === bp_groups_get_group_type_object( $new_type ) ) { 1378 $error = true; 1379 } else { 1380 // Run through group ids. 1381 $error = false; 1382 foreach ( (array) $_REQUEST['gid'] as $group_id ) { 1383 $group_id = (int) $group_id; 1384 1385 // Get the old group type to check against. 1386 $current_types = bp_groups_get_group_type( $group_id, false ); 1387 1388 if ( $current_types && 'remove_group_type' === $new_type ) { 1389 $group_types = array(); 1390 } elseif ( ! $current_types || 1 !== count( $current_types ) || $new_type !== $current_types[0] ) { 1391 $group_types = array( $new_type ); 1392 } 1393 1394 // Set the new group type. 1395 if ( isset( $group_types ) ) { 1396 $set = bp_groups_set_group_type( $group_id, $group_types ); 1397 if ( false === $set || is_wp_error( $set ) ) { 1398 $error = true; 1399 } 1400 unset( $group_types ); 1401 } 1402 } 1403 } 1404 1405 // If there were any errors, show the error message. 1406 if ( $error ) { 1407 $redirect = add_query_arg( array( 'updated' => 'group-type-change-error' ), wp_get_referer() ); 1408 } else { 1409 $redirect = add_query_arg( array( 'updated' => 'group-type-change-success' ), wp_get_referer() ); 1410 } 1411 1412 wp_redirect( $redirect ); 1413 exit(); 1414 } 1415 add_action( 'bp_groups_admin_load', 'bp_groups_admin_process_group_type_bulk_changes' ); 1416 1417 /** 1418 * Display an admin notice upon group type bulk update. 1419 * 1420 * @since 2.7.0 1421 */ 1422 function bp_groups_admin_groups_type_change_notice() { 1423 $updated = isset( $_REQUEST['updated'] ) ? $_REQUEST['updated'] : false; 1424 1425 // Display feedback. 1426 if ( $updated && in_array( $updated, array( 'group-type-change-error', 'group-type-change-success' ), true ) ) { 1427 1428 if ( 'group-type-change-error' === $updated ) { 1429 $notice = __( 'There was an error while changing group type. Please try again.', 'buddypress' ); 1430 $type = 'error'; 1431 } else { 1432 $notice = __( 'Group type was changed successfully.', 'buddypress' ); 1433 $type = 'updated'; 1434 } 1435 1436 bp_core_add_admin_notice( $notice, $type ); 1437 } 1438 } 1439 add_action( bp_core_admin_hook(), 'bp_groups_admin_groups_type_change_notice' ); 1440 1441 /** 1442 * Checks whether a group type already exists. 1443 * 1444 * @since 7.0.0 1445 * 1446 * @param boolean $exists True if the group type already exists. False otherwise. 1447 * @param string $type_id The group type identifier. 1448 * @return boolean True if the group type already exists. False otherwise. 1449 */ 1450 function bp_groups_type_admin_type_exists( $exists = false, $type_id = '' ) { 1451 if ( ! $type_id ) { 1452 return $exists; 1453 } 1454 1455 return ! is_null( bp_groups_get_group_type_object( $type_id ) ); 1456 } 1457 add_filter( bp_get_group_type_tax_name() . '_check_existing_type', 'bp_groups_type_admin_type_exists', 1, 2 ); 1458 1459 /** 1460 * Set the feedback messages for the Group Types Admin actions. 1461 * 1462 * @since 7.0.0 1463 * 1464 * @param array $messages The feedback messages. 1465 * @return array The feedback messages including the ones for the Group Types Admin actions. 1466 */ 1467 function bp_groups_type_admin_updated_messages( $messages = array() ) { 1468 $type_taxonomy = bp_get_group_type_tax_name(); 1469 1470 $messages[ $type_taxonomy ] = array( 1471 0 => '', 1472 1 => __( 'Please define the Group Type ID field.', 'buddypress' ), 1473 2 => __( 'Group type successfully added.', 'buddypress' ), 1474 3 => __( 'Sorry, there was an error and the Group type wasn’t added.', 'buddypress' ), 1475 // The following one needs to be != 5. 1476 4 => __( 'Group type successfully updated.', 'buddypress' ), 1477 5 => __( 'Sorry, this Group type already exists.', 'buddypress' ), 1478 6 => __( 'Sorry, the Group type was not deleted: it does not exist.', 'buddypress' ), 1479 7 => __( 'Sorry, This Group type is registered using code, deactivate the plugin or remove the custom code before trying to delete it again.', 'buddypress' ), 1480 8 => __( 'Sorry, there was an error while trying to delete this Group type.', 'buddypress' ), 1481 9 => __( 'Group type successfully deleted.', 'buddypress' ), 1482 10 => __( 'Group type could not be updated due to missing required information.', 'buddypress' ), 1483 ); 1484 1485 return $messages; 1486 } 1487 add_filter( 'term_updated_messages', 'bp_groups_type_admin_updated_messages' );
title
Description
Body
title
Description
Body
title
Description
Body
title
Body
Generated: Tue Sep 10 01:00:52 2024 | Cross-referenced by PHPXref 0.7.1 |