[ Index ]

PHP Cross Reference of WordPress Trunk (Updated Daily)

Search

title

Body

[close]

/wp-admin/includes/ -> dashboard.php (source)

   1  <?php
   2  /**
   3   * WordPress Dashboard Widget Administration Screen API
   4   *
   5   * @package WordPress
   6   * @subpackage Administration
   7   */
   8  
   9  /**
  10   * Registers dashboard widgets.
  11   *
  12   * Handles POST data, sets up filters.
  13   *
  14   * @since 2.5.0
  15   *
  16   * @global array $wp_registered_widgets
  17   * @global array $wp_registered_widget_controls
  18   * @global callable[] $wp_dashboard_control_callbacks
  19   */
  20  function wp_dashboard_setup() {
  21      global $wp_registered_widgets, $wp_registered_widget_controls, $wp_dashboard_control_callbacks;
  22  
  23      $screen = get_current_screen();
  24  
  25      /* Register Widgets and Controls */
  26      $wp_dashboard_control_callbacks = array();
  27  
  28      // Browser version
  29      $check_browser = wp_check_browser_version();
  30  
  31      if ( $check_browser && $check_browser['upgrade'] ) {
  32          add_filter( 'postbox_classes_dashboard_dashboard_browser_nag', 'dashboard_browser_nag_class' );
  33  
  34          if ( $check_browser['insecure'] ) {
  35              wp_add_dashboard_widget( 'dashboard_browser_nag', __( 'You are using an insecure browser!' ), 'wp_dashboard_browser_nag' );
  36          } else {
  37              wp_add_dashboard_widget( 'dashboard_browser_nag', __( 'Your browser is out of date!' ), 'wp_dashboard_browser_nag' );
  38          }
  39      }
  40  
  41      // PHP Version.
  42      $check_php = wp_check_php_version();
  43  
  44      if ( $check_php && current_user_can( 'update_php' ) ) {
  45          // If "not acceptable" the widget will be shown.
  46          if ( isset( $check_php['is_acceptable'] ) && ! $check_php['is_acceptable'] ) {
  47              add_filter( 'postbox_classes_dashboard_dashboard_php_nag', 'dashboard_php_nag_class' );
  48  
  49              if ( $check_php['is_lower_than_future_minimum'] ) {
  50                  wp_add_dashboard_widget( 'dashboard_php_nag', __( 'PHP Update Required' ), 'wp_dashboard_php_nag' );
  51              } else {
  52                  wp_add_dashboard_widget( 'dashboard_php_nag', __( 'PHP Update Recommended' ), 'wp_dashboard_php_nag' );
  53              }
  54          }
  55      }
  56  
  57      // Site Health.
  58      if ( current_user_can( 'view_site_health_checks' ) && ! is_network_admin() ) {
  59          if ( ! class_exists( 'WP_Site_Health' ) ) {
  60              require_once  ABSPATH . 'wp-admin/includes/class-wp-site-health.php';
  61          }
  62  
  63          WP_Site_Health::get_instance();
  64  
  65          wp_enqueue_style( 'site-health' );
  66          wp_enqueue_script( 'site-health' );
  67  
  68          wp_add_dashboard_widget( 'dashboard_site_health', __( 'Site Health Status' ), 'wp_dashboard_site_health' );
  69      }
  70  
  71      // Right Now.
  72      if ( is_blog_admin() && current_user_can( 'edit_posts' ) ) {
  73          wp_add_dashboard_widget( 'dashboard_right_now', __( 'At a Glance' ), 'wp_dashboard_right_now' );
  74      }
  75  
  76      if ( is_network_admin() ) {
  77          wp_add_dashboard_widget( 'network_dashboard_right_now', __( 'Right Now' ), 'wp_network_dashboard_right_now' );
  78      }
  79  
  80      // Activity Widget.
  81      if ( is_blog_admin() ) {
  82          wp_add_dashboard_widget( 'dashboard_activity', __( 'Activity' ), 'wp_dashboard_site_activity' );
  83      }
  84  
  85      // QuickPress Widget.
  86      if ( is_blog_admin() && current_user_can( get_post_type_object( 'post' )->cap->create_posts ) ) {
  87          $quick_draft_title = sprintf( '<span class="hide-if-no-js">%1$s</span> <span class="hide-if-js">%2$s</span>', __( 'Quick Draft' ), __( 'Your Recent Drafts' ) );
  88          wp_add_dashboard_widget( 'dashboard_quick_press', $quick_draft_title, 'wp_dashboard_quick_press' );
  89      }
  90  
  91      // WordPress Events and News.
  92      wp_add_dashboard_widget( 'dashboard_primary', __( 'WordPress Events and News' ), 'wp_dashboard_events_news' );
  93  
  94      if ( is_network_admin() ) {
  95  
  96          /**
  97           * Fires after core widgets for the Network Admin dashboard have been registered.
  98           *
  99           * @since 3.1.0
 100           */
 101          do_action( 'wp_network_dashboard_setup' );
 102  
 103          /**
 104           * Filters the list of widgets to load for the Network Admin dashboard.
 105           *
 106           * @since 3.1.0
 107           *
 108           * @param string[] $dashboard_widgets An array of dashboard widget IDs.
 109           */
 110          $dashboard_widgets = apply_filters( 'wp_network_dashboard_widgets', array() );
 111      } elseif ( is_user_admin() ) {
 112  
 113          /**
 114           * Fires after core widgets for the User Admin dashboard have been registered.
 115           *
 116           * @since 3.1.0
 117           */
 118          do_action( 'wp_user_dashboard_setup' );
 119  
 120          /**
 121           * Filters the list of widgets to load for the User Admin dashboard.
 122           *
 123           * @since 3.1.0
 124           *
 125           * @param string[] $dashboard_widgets An array of dashboard widget IDs.
 126           */
 127          $dashboard_widgets = apply_filters( 'wp_user_dashboard_widgets', array() );
 128      } else {
 129  
 130          /**
 131           * Fires after core widgets for the admin dashboard have been registered.
 132           *
 133           * @since 2.5.0
 134           */
 135          do_action( 'wp_dashboard_setup' );
 136  
 137          /**
 138           * Filters the list of widgets to load for the admin dashboard.
 139           *
 140           * @since 2.5.0
 141           *
 142           * @param string[] $dashboard_widgets An array of dashboard widget IDs.
 143           */
 144          $dashboard_widgets = apply_filters( 'wp_dashboard_widgets', array() );
 145      }
 146  
 147      foreach ( $dashboard_widgets as $widget_id ) {
 148          $name = empty( $wp_registered_widgets[ $widget_id ]['all_link'] ) ? $wp_registered_widgets[ $widget_id ]['name'] : $wp_registered_widgets[ $widget_id ]['name'] . " <a href='{$wp_registered_widgets[$widget_id]['all_link']}' class='edit-box open-box'>" . __( 'View all' ) . '</a>';
 149          wp_add_dashboard_widget( $widget_id, $name, $wp_registered_widgets[ $widget_id ]['callback'], $wp_registered_widget_controls[ $widget_id ]['callback'] );
 150      }
 151  
 152      if ( 'POST' === $_SERVER['REQUEST_METHOD'] && isset( $_POST['widget_id'] ) ) {
 153          check_admin_referer( 'edit-dashboard-widget_' . $_POST['widget_id'], 'dashboard-widget-nonce' );
 154          ob_start(); // Hack - but the same hack wp-admin/widgets.php uses.
 155          wp_dashboard_trigger_widget_control( $_POST['widget_id'] );
 156          ob_end_clean();
 157          wp_redirect( remove_query_arg( 'edit' ) );
 158          exit;
 159      }
 160  
 161      /** This action is documented in wp-admin/includes/meta-boxes.php */
 162      do_action( 'do_meta_boxes', $screen->id, 'normal', '' );
 163  
 164      /** This action is documented in wp-admin/includes/meta-boxes.php */
 165      do_action( 'do_meta_boxes', $screen->id, 'side', '' );
 166  }
 167  
 168  /**
 169   * Adds a new dashboard widget.
 170   *
 171   * @since 2.7.0
 172   * @since 5.6.0 The `$context` and `$priority` parameters were added.
 173   *
 174   * @global callable[] $wp_dashboard_control_callbacks
 175   *
 176   * @param string   $widget_id        Widget ID  (used in the 'id' attribute for the widget).
 177   * @param string   $widget_name      Title of the widget.
 178   * @param callable $callback         Function that fills the widget with the desired content.
 179   *                                   The function should echo its output.
 180   * @param callable $control_callback Optional. Function that outputs controls for the widget. Default null.
 181   * @param array    $callback_args    Optional. Data that should be set as the $args property of the widget array
 182   *                                   (which is the second parameter passed to your callback). Default null.
 183   * @param string   $context          Optional. The context within the screen where the box should display.
 184   *                                   Accepts 'normal', 'side', 'column3', or 'column4'. Default 'normal'.
 185   * @param string   $priority         Optional. The priority within the context where the box should show.
 186   *                                   Accepts 'high', 'core', 'default', or 'low'. Default 'core'.
 187   */
 188  function wp_add_dashboard_widget( $widget_id, $widget_name, $callback, $control_callback = null, $callback_args = null, $context = 'normal', $priority = 'core' ) {
 189      global $wp_dashboard_control_callbacks;
 190  
 191      $screen = get_current_screen();
 192  
 193      $private_callback_args = array( '__widget_basename' => $widget_name );
 194  
 195      if ( is_null( $callback_args ) ) {
 196          $callback_args = $private_callback_args;
 197      } elseif ( is_array( $callback_args ) ) {
 198          $callback_args = array_merge( $callback_args, $private_callback_args );
 199      }
 200  
 201      if ( $control_callback && is_callable( $control_callback ) && current_user_can( 'edit_dashboard' ) ) {
 202          $wp_dashboard_control_callbacks[ $widget_id ] = $control_callback;
 203  
 204          if ( isset( $_GET['edit'] ) && $widget_id === $_GET['edit'] ) {
 205              list($url)    = explode( '#', add_query_arg( 'edit', false ), 2 );
 206              $widget_name .= ' <span class="postbox-title-action"><a href="' . esc_url( $url ) . '">' . __( 'Cancel' ) . '</a></span>';
 207              $callback     = '_wp_dashboard_control_callback';
 208          } else {
 209              list($url)    = explode( '#', add_query_arg( 'edit', $widget_id ), 2 );
 210              $widget_name .= ' <span class="postbox-title-action"><a href="' . esc_url( "$url#$widget_id" ) . '" class="edit-box open-box">' . __( 'Configure' ) . '</a></span>';
 211          }
 212      }
 213  
 214      $side_widgets = array( 'dashboard_quick_press', 'dashboard_primary' );
 215  
 216      if ( in_array( $widget_id, $side_widgets, true ) ) {
 217          $context = 'side';
 218      }
 219  
 220      $high_priority_widgets = array( 'dashboard_browser_nag', 'dashboard_php_nag' );
 221  
 222      if ( in_array( $widget_id, $high_priority_widgets, true ) ) {
 223          $priority = 'high';
 224      }
 225  
 226      if ( empty( $context ) ) {
 227          $context = 'normal';
 228      }
 229  
 230      if ( empty( $priority ) ) {
 231          $priority = 'core';
 232      }
 233  
 234      add_meta_box( $widget_id, $widget_name, $callback, $screen, $context, $priority, $callback_args );
 235  }
 236  
 237  /**
 238   * Outputs controls for the current dashboard widget.
 239   *
 240   * @access private
 241   * @since 2.7.0
 242   *
 243   * @param mixed $dashboard
 244   * @param array $meta_box
 245   */
 246  function _wp_dashboard_control_callback( $dashboard, $meta_box ) {
 247      echo '<form method="post" class="dashboard-widget-control-form wp-clearfix">';
 248      wp_dashboard_trigger_widget_control( $meta_box['id'] );
 249      wp_nonce_field( 'edit-dashboard-widget_' . $meta_box['id'], 'dashboard-widget-nonce' );
 250      echo '<input type="hidden" name="widget_id" value="' . esc_attr( $meta_box['id'] ) . '" />';
 251      submit_button( __( 'Save Changes' ) );
 252      echo '</form>';
 253  }
 254  
 255  /**
 256   * Displays the dashboard.
 257   *
 258   * @since 2.5.0
 259   */
 260  function wp_dashboard() {
 261      $screen      = get_current_screen();
 262      $columns     = absint( $screen->get_columns() );
 263      $columns_css = '';
 264  
 265      if ( $columns ) {
 266          $columns_css = " columns-$columns";
 267      }
 268      ?>
 269  <div id="dashboard-widgets" class="metabox-holder<?php echo $columns_css; ?>">
 270      <div id="postbox-container-1" class="postbox-container">
 271      <?php do_meta_boxes( $screen->id, 'normal', '' ); ?>
 272      </div>
 273      <div id="postbox-container-2" class="postbox-container">
 274      <?php do_meta_boxes( $screen->id, 'side', '' ); ?>
 275      </div>
 276      <div id="postbox-container-3" class="postbox-container">
 277      <?php do_meta_boxes( $screen->id, 'column3', '' ); ?>
 278      </div>
 279      <div id="postbox-container-4" class="postbox-container">
 280      <?php do_meta_boxes( $screen->id, 'column4', '' ); ?>
 281      </div>
 282  </div>
 283  
 284      <?php
 285      wp_nonce_field( 'closedpostboxes', 'closedpostboxesnonce', false );
 286      wp_nonce_field( 'meta-box-order', 'meta-box-order-nonce', false );
 287  }
 288  
 289  //
 290  // Dashboard Widgets.
 291  //
 292  
 293  /**
 294   * Dashboard widget that displays some basic stats about the site.
 295   *
 296   * Formerly 'Right Now'. A streamlined 'At a Glance' as of 3.8.
 297   *
 298   * @since 2.7.0
 299   */
 300  function wp_dashboard_right_now() {
 301      ?>
 302      <div class="main">
 303      <ul>
 304      <?php
 305      // Posts and Pages.
 306      foreach ( array( 'post', 'page' ) as $post_type ) {
 307          $num_posts = wp_count_posts( $post_type );
 308  
 309          if ( $num_posts && $num_posts->publish ) {
 310              if ( 'post' === $post_type ) {
 311                  /* translators: %s: Number of posts. */
 312                  $text = _n( '%s Published post', '%s Published posts', $num_posts->publish );
 313              } else {
 314                  /* translators: %s: Number of pages. */
 315                  $text = _n( '%s Published page', '%s Published pages', $num_posts->publish );
 316              }
 317  
 318              $text             = sprintf( $text, number_format_i18n( $num_posts->publish ) );
 319              $post_type_object = get_post_type_object( $post_type );
 320  
 321              if ( $post_type_object && current_user_can( $post_type_object->cap->edit_posts ) ) {
 322                  $url = add_query_arg(
 323                      array(
 324                          'post_status' => 'publish',
 325                          'post_type'   => $post_type,
 326                      ),
 327                      admin_url( 'edit.php' )
 328                  );
 329                  printf( '<li class="%1$s-count"><a href="%2$s">%3$s</a></li>', $post_type, esc_url( $url ), esc_html( $text ) );
 330              } else {
 331                  printf( '<li class="%1$s-count"><span>%2$s</span></li>', $post_type, $text );
 332              }
 333          }
 334      }
 335  
 336      // Comments.
 337      $num_comm = wp_count_comments();
 338  
 339      if ( $num_comm && ( $num_comm->approved || $num_comm->moderated ) ) {
 340          /* translators: %s: Number of comments. */
 341          $text = sprintf( _n( '%s Comment', '%s Comments', $num_comm->approved ), number_format_i18n( $num_comm->approved ) );
 342          ?>
 343          <li class="comment-count">
 344              <a href="edit-comments.php"><?php echo $text; ?></a>
 345          </li>
 346          <?php
 347          $moderated_comments_count_i18n = number_format_i18n( $num_comm->moderated );
 348          /* translators: %s: Number of comments. */
 349          $text = sprintf( _n( '%s Comment in moderation', '%s Comments in moderation', $num_comm->moderated ), $moderated_comments_count_i18n );
 350          ?>
 351          <li class="comment-mod-count<?php echo ! $num_comm->moderated ? ' hidden' : ''; ?>">
 352              <a href="edit-comments.php?comment_status=moderated" class="comments-in-moderation-text"><?php echo $text; ?></a>
 353          </li>
 354          <?php
 355      }
 356  
 357      /**
 358       * Filters the array of extra elements to list in the 'At a Glance'
 359       * dashboard widget.
 360       *
 361       * Prior to 3.8.0, the widget was named 'Right Now'. Each element
 362       * is wrapped in list-item tags on output.
 363       *
 364       * @since 3.8.0
 365       *
 366       * @param string[] $items Array of extra 'At a Glance' widget items.
 367       */
 368      $elements = apply_filters( 'dashboard_glance_items', array() );
 369  
 370      if ( $elements ) {
 371          echo '<li>' . implode( "</li>\n<li>", $elements ) . "</li>\n";
 372      }
 373  
 374      ?>
 375      </ul>
 376      <?php
 377      update_right_now_message();
 378  
 379      // Check if search engines are asked not to index this site.
 380      if ( ! is_network_admin() && ! is_user_admin()
 381          && current_user_can( 'manage_options' ) && ! get_option( 'blog_public' )
 382      ) {
 383  
 384          /**
 385           * Filters the link title attribute for the 'Search engines discouraged'
 386           * message displayed in the 'At a Glance' dashboard widget.
 387           *
 388           * Prior to 3.8.0, the widget was named 'Right Now'.
 389           *
 390           * @since 3.0.0
 391           * @since 4.5.0 The default for `$title` was updated to an empty string.
 392           *
 393           * @param string $title Default attribute text.
 394           */
 395          $title = apply_filters( 'privacy_on_link_title', '' );
 396  
 397          /**
 398           * Filters the link label for the 'Search engines discouraged' message
 399           * displayed in the 'At a Glance' dashboard widget.
 400           *
 401           * Prior to 3.8.0, the widget was named 'Right Now'.
 402           *
 403           * @since 3.0.0
 404           *
 405           * @param string $content Default text.
 406           */
 407          $content = apply_filters( 'privacy_on_link_text', __( 'Search engines discouraged' ) );
 408  
 409          $title_attr = '' === $title ? '' : " title='$title'";
 410  
 411          echo "<p class='search-engines-info'><a href='options-reading.php'$title_attr>$content</a></p>";
 412      }
 413      ?>
 414      </div>
 415      <?php
 416      /*
 417       * activity_box_end has a core action, but only prints content when multisite.
 418       * Using an output buffer is the only way to really check if anything's displayed here.
 419       */
 420      ob_start();
 421  
 422      /**
 423       * Fires at the end of the 'At a Glance' dashboard widget.
 424       *
 425       * Prior to 3.8.0, the widget was named 'Right Now'.
 426       *
 427       * @since 2.5.0
 428       */
 429      do_action( 'rightnow_end' );
 430  
 431      /**
 432       * Fires at the end of the 'At a Glance' dashboard widget.
 433       *
 434       * Prior to 3.8.0, the widget was named 'Right Now'.
 435       *
 436       * @since 2.0.0
 437       */
 438      do_action( 'activity_box_end' );
 439  
 440      $actions = ob_get_clean();
 441  
 442      if ( ! empty( $actions ) ) :
 443          ?>
 444      <div class="sub">
 445          <?php echo $actions; ?>
 446      </div>
 447          <?php
 448      endif;
 449  }
 450  
 451  /**
 452   * @since 3.1.0
 453   */
 454  function wp_network_dashboard_right_now() {
 455      $actions = array();
 456  
 457      if ( current_user_can( 'create_sites' ) ) {
 458          $actions['create-site'] = '<a href="' . network_admin_url( 'site-new.php' ) . '">' . __( 'Create a New Site' ) . '</a>';
 459      }
 460      if ( current_user_can( 'create_users' ) ) {
 461          $actions['create-user'] = '<a href="' . network_admin_url( 'user-new.php' ) . '">' . __( 'Create a New User' ) . '</a>';
 462      }
 463  
 464      $c_users = get_user_count();
 465      $c_blogs = get_blog_count();
 466  
 467      /* translators: %s: Number of users on the network. */
 468      $user_text = sprintf( _n( '%s user', '%s users', $c_users ), number_format_i18n( $c_users ) );
 469      /* translators: %s: Number of sites on the network. */
 470      $blog_text = sprintf( _n( '%s site', '%s sites', $c_blogs ), number_format_i18n( $c_blogs ) );
 471  
 472      /* translators: 1: Text indicating the number of sites on the network, 2: Text indicating the number of users on the network. */
 473      $sentence = sprintf( __( 'You have %1$s and %2$s.' ), $blog_text, $user_text );
 474  
 475      if ( $actions ) {
 476          echo '<ul class="subsubsub">';
 477          foreach ( $actions as $class => $action ) {
 478              $actions[ $class ] = "\t<li class='$class'>$action";
 479          }
 480          echo implode( " |</li>\n", $actions ) . "</li>\n";
 481          echo '</ul>';
 482      }
 483      ?>
 484      <br class="clear" />
 485  
 486      <p class="youhave"><?php echo $sentence; ?></p>
 487  
 488  
 489      <?php
 490          /**
 491           * Fires in the Network Admin 'Right Now' dashboard widget
 492           * just before the user and site search form fields.
 493           *
 494           * @since MU (3.0.0)
 495           */
 496          do_action( 'wpmuadminresult' );
 497      ?>
 498  
 499      <form action="<?php echo esc_url( network_admin_url( 'users.php' ) ); ?>" method="get">
 500          <p>
 501              <label class="screen-reader-text" for="search-users">
 502                  <?php
 503                  /* translators: Hidden accessibility text. */
 504                  _e( 'Search Users' );
 505                  ?>
 506              </label>
 507              <input type="search" name="s" value="" size="30" autocomplete="off" id="search-users" />
 508              <?php submit_button( __( 'Search Users' ), '', false, false, array( 'id' => 'submit_users' ) ); ?>
 509          </p>
 510      </form>
 511  
 512      <form action="<?php echo esc_url( network_admin_url( 'sites.php' ) ); ?>" method="get">
 513          <p>
 514              <label class="screen-reader-text" for="search-sites">
 515                  <?php
 516                  /* translators: Hidden accessibility text. */
 517                  _e( 'Search Sites' );
 518                  ?>
 519              </label>
 520              <input type="search" name="s" value="" size="30" autocomplete="off" id="search-sites" />
 521              <?php submit_button( __( 'Search Sites' ), '', false, false, array( 'id' => 'submit_sites' ) ); ?>
 522          </p>
 523      </form>
 524      <?php
 525      /**
 526       * Fires at the end of the 'Right Now' widget in the Network Admin dashboard.
 527       *
 528       * @since MU (3.0.0)
 529       */
 530      do_action( 'mu_rightnow_end' );
 531  
 532      /**
 533       * Fires at the end of the 'Right Now' widget in the Network Admin dashboard.
 534       *
 535       * @since MU (3.0.0)
 536       */
 537      do_action( 'mu_activity_box_end' );
 538  }
 539  
 540  /**
 541   * Displays the Quick Draft widget.
 542   *
 543   * @since 3.8.0
 544   *
 545   * @global int $post_ID
 546   *
 547   * @param string|false $error_msg Optional. Error message. Default false.
 548   */
 549  function wp_dashboard_quick_press( $error_msg = false ) {
 550      global $post_ID;
 551  
 552      if ( ! current_user_can( 'edit_posts' ) ) {
 553          return;
 554      }
 555  
 556      // Check if a new auto-draft (= no new post_ID) is needed or if the old can be used.
 557      $last_post_id = (int) get_user_option( 'dashboard_quick_press_last_post_id' ); // Get the last post_ID.
 558  
 559      if ( $last_post_id ) {
 560          $post = get_post( $last_post_id );
 561  
 562          if ( empty( $post ) || 'auto-draft' !== $post->post_status ) { // auto-draft doesn't exist anymore.
 563              $post = get_default_post_to_edit( 'post', true );
 564              update_user_option( get_current_user_id(), 'dashboard_quick_press_last_post_id', (int) $post->ID ); // Save post_ID.
 565          } else {
 566              $post->post_title = ''; // Remove the auto draft title.
 567          }
 568      } else {
 569          $post    = get_default_post_to_edit( 'post', true );
 570          $user_id = get_current_user_id();
 571  
 572          // Don't create an option if this is a super admin who does not belong to this site.
 573          if ( in_array( get_current_blog_id(), array_keys( get_blogs_of_user( $user_id ) ), true ) ) {
 574              update_user_option( $user_id, 'dashboard_quick_press_last_post_id', (int) $post->ID ); // Save post_ID.
 575          }
 576      }
 577  
 578      $post_ID = (int) $post->ID;
 579      ?>
 580  
 581      <form name="post" action="<?php echo esc_url( admin_url( 'post.php' ) ); ?>" method="post" id="quick-press" class="initial-form hide-if-no-js">
 582  
 583          <?php
 584          if ( $error_msg ) {
 585              wp_admin_notice(
 586                  $error_msg,
 587                  array(
 588                      'additional_classes' => array( 'error' ),
 589                  )
 590              );
 591          }
 592          ?>
 593  
 594          <div class="input-text-wrap" id="title-wrap">
 595              <label for="title">
 596                  <?php
 597                  /** This filter is documented in wp-admin/edit-form-advanced.php */
 598                  echo apply_filters( 'enter_title_here', __( 'Title' ), $post );
 599                  ?>
 600              </label>
 601              <input type="text" name="post_title" id="title" autocomplete="off" />
 602          </div>
 603  
 604          <div class="textarea-wrap" id="description-wrap">
 605              <label for="content"><?php _e( 'Content' ); ?></label>
 606              <textarea name="content" id="content" placeholder="<?php esc_attr_e( 'What&#8217;s on your mind?' ); ?>" class="mceEditor" rows="3" cols="15" autocomplete="off"></textarea>
 607          </div>
 608  
 609          <p class="submit">
 610              <input type="hidden" name="action" id="quickpost-action" value="post-quickdraft-save" />
 611              <input type="hidden" name="post_ID" value="<?php echo $post_ID; ?>" />
 612              <input type="hidden" name="post_type" value="post" />
 613              <?php wp_nonce_field( 'add-post' ); ?>
 614              <?php submit_button( __( 'Save Draft' ), 'primary', 'save', false, array( 'id' => 'save-post' ) ); ?>
 615              <br class="clear" />
 616          </p>
 617  
 618      </form>
 619      <?php
 620      wp_dashboard_recent_drafts();
 621  }
 622  
 623  /**
 624   * Show recent drafts of the user on the dashboard.
 625   *
 626   * @since 2.7.0
 627   *
 628   * @param WP_Post[]|false $drafts Optional. Array of posts to display. Default false.
 629   */
 630  function wp_dashboard_recent_drafts( $drafts = false ) {
 631      if ( ! $drafts ) {
 632          $query_args = array(
 633              'post_type'      => 'post',
 634              'post_status'    => 'draft',
 635              'author'         => get_current_user_id(),
 636              'posts_per_page' => 4,
 637              'orderby'        => 'modified',
 638              'order'          => 'DESC',
 639          );
 640  
 641          /**
 642           * Filters the post query arguments for the 'Recent Drafts' dashboard widget.
 643           *
 644           * @since 4.4.0
 645           *
 646           * @param array $query_args The query arguments for the 'Recent Drafts' dashboard widget.
 647           */
 648          $query_args = apply_filters( 'dashboard_recent_drafts_query_args', $query_args );
 649  
 650          $drafts = get_posts( $query_args );
 651          if ( ! $drafts ) {
 652              return;
 653          }
 654      }
 655  
 656      echo '<div class="drafts">';
 657  
 658      if ( count( $drafts ) > 3 ) {
 659          printf(
 660              '<p class="view-all"><a href="%s">%s</a></p>' . "\n",
 661              esc_url( admin_url( 'edit.php?post_status=draft' ) ),
 662              __( 'View all drafts' )
 663          );
 664      }
 665  
 666      echo '<h2 class="hide-if-no-js">' . __( 'Your Recent Drafts' ) . "</h2>\n";
 667      echo '<ul>';
 668  
 669      /* translators: Maximum number of words used in a preview of a draft on the dashboard. */
 670      $draft_length = (int) _x( '10', 'draft_length' );
 671  
 672      $drafts = array_slice( $drafts, 0, 3 );
 673      foreach ( $drafts as $draft ) {
 674          $url   = get_edit_post_link( $draft->ID );
 675          $title = _draft_or_post_title( $draft->ID );
 676  
 677          echo "<li>\n";
 678          printf(
 679              '<div class="draft-title"><a href="%s" aria-label="%s">%s</a><time datetime="%s">%s</time></div>',
 680              esc_url( $url ),
 681              /* translators: %s: Post title. */
 682              esc_attr( sprintf( __( 'Edit &#8220;%s&#8221;' ), $title ) ),
 683              esc_html( $title ),
 684              get_the_time( 'c', $draft ),
 685              get_the_time( __( 'F j, Y' ), $draft )
 686          );
 687  
 688          $the_content = wp_trim_words( $draft->post_content, $draft_length );
 689  
 690          if ( $the_content ) {
 691              echo '<p>' . $the_content . '</p>';
 692          }
 693          echo "</li>\n";
 694      }
 695  
 696      echo "</ul>\n";
 697      echo '</div>';
 698  }
 699  
 700  /**
 701   * Outputs a row for the Recent Comments widget.
 702   *
 703   * @access private
 704   * @since 2.7.0
 705   *
 706   * @global WP_Comment $comment Global comment object.
 707   *
 708   * @param WP_Comment $comment   The current comment.
 709   * @param bool       $show_date Optional. Whether to display the date.
 710   */
 711  function _wp_dashboard_recent_comments_row( &$comment, $show_date = true ) {
 712      $GLOBALS['comment'] = clone $comment;
 713  
 714      if ( $comment->comment_post_ID > 0 ) {
 715          $comment_post_title = _draft_or_post_title( $comment->comment_post_ID );
 716          $comment_post_url   = get_the_permalink( $comment->comment_post_ID );
 717          $comment_post_link  = '<a href="' . esc_url( $comment_post_url ) . '">' . $comment_post_title . '</a>';
 718      } else {
 719          $comment_post_link = '';
 720      }
 721  
 722      $actions_string = '';
 723      if ( current_user_can( 'edit_comment', $comment->comment_ID ) ) {
 724          // Pre-order it: Approve | Reply | Edit | Spam | Trash.
 725          $actions = array(
 726              'approve'   => '',
 727              'unapprove' => '',
 728              'reply'     => '',
 729              'edit'      => '',
 730              'spam'      => '',
 731              'trash'     => '',
 732              'delete'    => '',
 733              'view'      => '',
 734          );
 735  
 736          $approve_nonce = esc_html( '_wpnonce=' . wp_create_nonce( 'approve-comment_' . $comment->comment_ID ) );
 737          $del_nonce     = esc_html( '_wpnonce=' . wp_create_nonce( 'delete-comment_' . $comment->comment_ID ) );
 738  
 739          $action_string = 'comment.php?action=%s&p=' . $comment->comment_post_ID . '&c=' . $comment->comment_ID . '&%s';
 740  
 741          $approve_url   = sprintf( $action_string, 'approvecomment', $approve_nonce );
 742          $unapprove_url = sprintf( $action_string, 'unapprovecomment', $approve_nonce );
 743          $spam_url      = sprintf( $action_string, 'spamcomment', $del_nonce );
 744          $trash_url     = sprintf( $action_string, 'trashcomment', $del_nonce );
 745          $delete_url    = sprintf( $action_string, 'deletecomment', $del_nonce );
 746  
 747          $actions['approve'] = sprintf(
 748              '<a href="%s" data-wp-lists="%s" class="vim-a aria-button-if-js" aria-label="%s">%s</a>',
 749              esc_url( $approve_url ),
 750              "dim:the-comment-list:comment-{$comment->comment_ID}:unapproved:e7e7d3:e7e7d3:new=approved",
 751              esc_attr__( 'Approve this comment' ),
 752              __( 'Approve' )
 753          );
 754  
 755          $actions['unapprove'] = sprintf(
 756              '<a href="%s" data-wp-lists="%s" class="vim-u aria-button-if-js" aria-label="%s">%s</a>',
 757              esc_url( $unapprove_url ),
 758              "dim:the-comment-list:comment-{$comment->comment_ID}:unapproved:e7e7d3:e7e7d3:new=unapproved",
 759              esc_attr__( 'Unapprove this comment' ),
 760              __( 'Unapprove' )
 761          );
 762  
 763          $actions['edit'] = sprintf(
 764              '<a href="%s" aria-label="%s">%s</a>',
 765              "comment.php?action=editcomment&amp;c={$comment->comment_ID}",
 766              esc_attr__( 'Edit this comment' ),
 767              __( 'Edit' )
 768          );
 769  
 770          $actions['reply'] = sprintf(
 771              '<button type="button" onclick="window.commentReply && commentReply.open(\'%s\',\'%s\');" class="vim-r button-link hide-if-no-js" aria-label="%s">%s</button>',
 772              $comment->comment_ID,
 773              $comment->comment_post_ID,
 774              esc_attr__( 'Reply to this comment' ),
 775              __( 'Reply' )
 776          );
 777  
 778          $actions['spam'] = sprintf(
 779              '<a href="%s" data-wp-lists="%s" class="vim-s vim-destructive aria-button-if-js" aria-label="%s">%s</a>',
 780              esc_url( $spam_url ),
 781              "delete:the-comment-list:comment-{$comment->comment_ID}::spam=1",
 782              esc_attr__( 'Mark this comment as spam' ),
 783              /* translators: "Mark as spam" link. */
 784              _x( 'Spam', 'verb' )
 785          );
 786  
 787          if ( ! EMPTY_TRASH_DAYS ) {
 788              $actions['delete'] = sprintf(
 789                  '<a href="%s" data-wp-lists="%s" class="delete vim-d vim-destructive aria-button-if-js" aria-label="%s">%s</a>',
 790                  esc_url( $delete_url ),
 791                  "delete:the-comment-list:comment-{$comment->comment_ID}::trash=1",
 792                  esc_attr__( 'Delete this comment permanently' ),
 793                  __( 'Delete Permanently' )
 794              );
 795          } else {
 796              $actions['trash'] = sprintf(
 797                  '<a href="%s" data-wp-lists="%s" class="delete vim-d vim-destructive aria-button-if-js" aria-label="%s">%s</a>',
 798                  esc_url( $trash_url ),
 799                  "delete:the-comment-list:comment-{$comment->comment_ID}::trash=1",
 800                  esc_attr__( 'Move this comment to the Trash' ),
 801                  _x( 'Trash', 'verb' )
 802              );
 803          }
 804  
 805          $actions['view'] = sprintf(
 806              '<a class="comment-link" href="%s" aria-label="%s">%s</a>',
 807              esc_url( get_comment_link( $comment ) ),
 808              esc_attr__( 'View this comment' ),
 809              __( 'View' )
 810          );
 811  
 812          /** This filter is documented in wp-admin/includes/class-wp-comments-list-table.php */
 813          $actions = apply_filters( 'comment_row_actions', array_filter( $actions ), $comment );
 814  
 815          $i = 0;
 816  
 817          foreach ( $actions as $action => $link ) {
 818              ++$i;
 819  
 820              if ( ( ( 'approve' === $action || 'unapprove' === $action ) && 2 === $i )
 821                  || 1 === $i
 822              ) {
 823                  $separator = '';
 824              } else {
 825                  $separator = ' | ';
 826              }
 827  
 828              // Reply and quickedit need a hide-if-no-js span.
 829              if ( 'reply' === $action || 'quickedit' === $action ) {
 830                  $action .= ' hide-if-no-js';
 831              }
 832  
 833              if ( 'view' === $action && '1' !== $comment->comment_approved ) {
 834                  $action .= ' hidden';
 835              }
 836  
 837              $actions_string .= "<span class='$action'>{$separator}{$link}</span>";
 838          }
 839      }
 840      ?>
 841  
 842          <li id="comment-<?php echo $comment->comment_ID; ?>" <?php comment_class( array( 'comment-item', wp_get_comment_status( $comment ) ), $comment ); ?>>
 843  
 844              <?php
 845              $comment_row_class = '';
 846  
 847              if ( get_option( 'show_avatars' ) ) {
 848                  echo get_avatar( $comment, 50, 'mystery' );
 849                  $comment_row_class .= ' has-avatar';
 850              }
 851              ?>
 852  
 853              <?php if ( ! $comment->comment_type || 'comment' === $comment->comment_type ) : ?>
 854  
 855              <div class="dashboard-comment-wrap has-row-actions <?php echo $comment_row_class; ?>">
 856              <p class="comment-meta">
 857                  <?php
 858                  // Comments might not have a post they relate to, e.g. programmatically created ones.
 859                  if ( $comment_post_link ) {
 860                      printf(
 861                          /* translators: 1: Comment author, 2: Post link, 3: Notification if the comment is pending. */
 862                          __( 'From %1$s on %2$s %3$s' ),
 863                          '<cite class="comment-author">' . get_comment_author_link( $comment ) . '</cite>',
 864                          $comment_post_link,
 865                          '<span class="approve">' . __( '[Pending]' ) . '</span>'
 866                      );
 867                  } else {
 868                      printf(
 869                          /* translators: 1: Comment author, 2: Notification if the comment is pending. */
 870                          __( 'From %1$s %2$s' ),
 871                          '<cite class="comment-author">' . get_comment_author_link( $comment ) . '</cite>',
 872                          '<span class="approve">' . __( '[Pending]' ) . '</span>'
 873                      );
 874                  }
 875                  ?>
 876              </p>
 877  
 878                  <?php
 879              else :
 880                  switch ( $comment->comment_type ) {
 881                      case 'pingback':
 882                          $type = __( 'Pingback' );
 883                          break;
 884                      case 'trackback':
 885                          $type = __( 'Trackback' );
 886                          break;
 887                      default:
 888                          $type = ucwords( $comment->comment_type );
 889                  }
 890                  $type = esc_html( $type );
 891                  ?>
 892              <div class="dashboard-comment-wrap has-row-actions">
 893              <p class="comment-meta">
 894                  <?php
 895                  // Pingbacks, Trackbacks or custom comment types might not have a post they relate to, e.g. programmatically created ones.
 896                  if ( $comment_post_link ) {
 897                      printf(
 898                          /* translators: 1: Type of comment, 2: Post link, 3: Notification if the comment is pending. */
 899                          _x( '%1$s on %2$s %3$s', 'dashboard' ),
 900                          "<strong>$type</strong>",
 901                          $comment_post_link,
 902                          '<span class="approve">' . __( '[Pending]' ) . '</span>'
 903                      );
 904                  } else {
 905                      printf(
 906                          /* translators: 1: Type of comment, 2: Notification if the comment is pending. */
 907                          _x( '%1$s %2$s', 'dashboard' ),
 908                          "<strong>$type</strong>",
 909                          '<span class="approve">' . __( '[Pending]' ) . '</span>'
 910                      );
 911                  }
 912                  ?>
 913              </p>
 914              <p class="comment-author"><?php comment_author_link( $comment ); ?></p>
 915  
 916              <?php endif; // comment_type ?>
 917              <blockquote><p><?php comment_excerpt( $comment ); ?></p></blockquote>
 918              <?php if ( $actions_string ) : ?>
 919              <p class="row-actions"><?php echo $actions_string; ?></p>
 920              <?php endif; ?>
 921              </div>
 922          </li>
 923      <?php
 924      $GLOBALS['comment'] = null;
 925  }
 926  
 927  /**
 928   * Outputs the Activity widget.
 929   *
 930   * Callback function for {@see 'dashboard_activity'}.
 931   *
 932   * @since 3.8.0
 933   */
 934  function wp_dashboard_site_activity() {
 935  
 936      echo '<div id="activity-widget">';
 937  
 938      $future_posts = wp_dashboard_recent_posts(
 939          array(
 940              'max'    => 5,
 941              'status' => 'future',
 942              'order'  => 'ASC',
 943              'title'  => __( 'Publishing Soon' ),
 944              'id'     => 'future-posts',
 945          )
 946      );
 947      $recent_posts = wp_dashboard_recent_posts(
 948          array(
 949              'max'    => 5,
 950              'status' => 'publish',
 951              'order'  => 'DESC',
 952              'title'  => __( 'Recently Published' ),
 953              'id'     => 'published-posts',
 954          )
 955      );
 956  
 957      $recent_comments = wp_dashboard_recent_comments();
 958  
 959      if ( ! $future_posts && ! $recent_posts && ! $recent_comments ) {
 960          echo '<div class="no-activity">';
 961          echo '<p>' . __( 'No activity yet!' ) . '</p>';
 962          echo '</div>';
 963      }
 964  
 965      echo '</div>';
 966  }
 967  
 968  /**
 969   * Generates Publishing Soon and Recently Published sections.
 970   *
 971   * @since 3.8.0
 972   *
 973   * @param array $args {
 974   *     An array of query and display arguments.
 975   *
 976   *     @type int    $max     Number of posts to display.
 977   *     @type string $status  Post status.
 978   *     @type string $order   Designates ascending ('ASC') or descending ('DESC') order.
 979   *     @type string $title   Section title.
 980   *     @type string $id      The container id.
 981   * }
 982   * @return bool False if no posts were found. True otherwise.
 983   */
 984  function wp_dashboard_recent_posts( $args ) {
 985      $query_args = array(
 986          'post_type'      => 'post',
 987          'post_status'    => $args['status'],
 988          'orderby'        => 'date',
 989          'order'          => $args['order'],
 990          'posts_per_page' => (int) $args['max'],
 991          'no_found_rows'  => true,
 992          'cache_results'  => true,
 993          'perm'           => ( 'future' === $args['status'] ) ? 'editable' : 'readable',
 994      );
 995  
 996      /**
 997       * Filters the query arguments used for the Recent Posts widget.
 998       *
 999       * @since 4.2.0
1000       *
1001       * @param array $query_args The arguments passed to WP_Query to produce the list of posts.
1002       */
1003      $query_args = apply_filters( 'dashboard_recent_posts_query_args', $query_args );
1004  
1005      $posts = new WP_Query( $query_args );
1006  
1007      if ( $posts->have_posts() ) {
1008  
1009          echo '<div id="' . $args['id'] . '" class="activity-block">';
1010  
1011          echo '<h3>' . $args['title'] . '</h3>';
1012  
1013          echo '<ul>';
1014  
1015          $today    = current_time( 'Y-m-d' );
1016          $tomorrow = current_datetime()->modify( '+1 day' )->format( 'Y-m-d' );
1017          $year     = current_time( 'Y' );
1018  
1019          while ( $posts->have_posts() ) {
1020              $posts->the_post();
1021  
1022              $time = get_the_time( 'U' );
1023  
1024              if ( ! is_int( $time ) ) {
1025                  /* translators: Date and time format for recent posts on the dashboard, from a different calendar year, see https://www.php.net/manual/datetime.format.php */
1026                  $date = get_the_date( __( 'M jS Y' ) );
1027              } elseif ( gmdate( 'Y-m-d', $time ) === $today ) {
1028                  $date = __( 'Today' );
1029              } elseif ( gmdate( 'Y-m-d', $time ) === $tomorrow ) {
1030                  $date = __( 'Tomorrow' );
1031              } elseif ( gmdate( 'Y', $time ) !== $year ) {
1032                  /* translators: Date and time format for recent posts on the dashboard, from a different calendar year, see https://www.php.net/manual/datetime.format.php */
1033                  $date = date_i18n( __( 'M jS Y' ), $time );
1034              } else {
1035                  /* translators: Date and time format for recent posts on the dashboard, see https://www.php.net/manual/datetime.format.php */
1036                  $date = date_i18n( __( 'M jS' ), $time );
1037              }
1038  
1039              // Use the post edit link for those who can edit, the permalink otherwise.
1040              $recent_post_link = current_user_can( 'edit_post', get_the_ID() ) ? get_edit_post_link() : get_permalink();
1041  
1042              $draft_or_post_title = _draft_or_post_title();
1043              printf(
1044                  '<li><span>%1$s</span> <a href="%2$s" aria-label="%3$s">%4$s</a></li>',
1045                  /* translators: 1: Relative date, 2: Time. */
1046                  sprintf( _x( '%1$s, %2$s', 'dashboard' ), $date, get_the_time() ),
1047                  $recent_post_link,
1048                  /* translators: %s: Post title. */
1049                  esc_attr( sprintf( __( 'Edit &#8220;%s&#8221;' ), $draft_or_post_title ) ),
1050                  $draft_or_post_title
1051              );
1052          }
1053  
1054          echo '</ul>';
1055          echo '</div>';
1056  
1057      } else {
1058          return false;
1059      }
1060  
1061      wp_reset_postdata();
1062  
1063      return true;
1064  }
1065  
1066  /**
1067   * Show Comments section.
1068   *
1069   * @since 3.8.0
1070   *
1071   * @param int $total_items Optional. Number of comments to query. Default 5.
1072   * @return bool False if no comments were found. True otherwise.
1073   */
1074  function wp_dashboard_recent_comments( $total_items = 5 ) {
1075      // Select all comment types and filter out spam later for better query performance.
1076      $comments = array();
1077  
1078      $comments_query = array(
1079          'number' => $total_items * 5,
1080          'offset' => 0,
1081      );
1082  
1083      if ( ! current_user_can( 'edit_posts' ) ) {
1084          $comments_query['status'] = 'approve';
1085      }
1086  
1087      $comments_count = 0;
1088      do {
1089          $possible = get_comments( $comments_query );
1090  
1091          if ( empty( $possible ) || ! is_array( $possible ) ) {
1092              break;
1093          }
1094  
1095          foreach ( $possible as $comment ) {
1096              if ( ! current_user_can( 'edit_post', $comment->comment_post_ID )
1097                  && ( post_password_required( $comment->comment_post_ID )
1098                      || ! current_user_can( 'read_post', $comment->comment_post_ID ) )
1099              ) {
1100                  // The user has no access to the post and thus cannot see the comments.
1101                  continue;
1102              }
1103  
1104              $comments[]     = $comment;
1105              $comments_count = count( $comments );
1106  
1107              if ( $comments_count === $total_items ) {
1108                  break 2;
1109              }
1110          }
1111  
1112          $comments_query['offset'] += $comments_query['number'];
1113          $comments_query['number']  = $total_items * 10;
1114      } while ( $comments_count < $total_items );
1115  
1116      if ( $comments ) {
1117          echo '<div id="latest-comments" class="activity-block table-view-list">';
1118          echo '<h3>' . __( 'Recent Comments' ) . '</h3>';
1119  
1120          echo '<ul id="the-comment-list" data-wp-lists="list:comment">';
1121          foreach ( $comments as $comment ) {
1122              _wp_dashboard_recent_comments_row( $comment );
1123          }
1124          echo '</ul>';
1125  
1126          if ( current_user_can( 'edit_posts' ) ) {
1127              echo '<h3 class="screen-reader-text">' .
1128                  /* translators: Hidden accessibility text. */
1129                  __( 'View more comments' ) .
1130              '</h3>';
1131              _get_list_table( 'WP_Comments_List_Table' )->views();
1132          }
1133  
1134          wp_comment_reply( -1, false, 'dashboard', false );
1135          wp_comment_trashnotice();
1136  
1137          echo '</div>';
1138      } else {
1139          return false;
1140      }
1141      return true;
1142  }
1143  
1144  /**
1145   * Display generic dashboard RSS widget feed.
1146   *
1147   * @since 2.5.0
1148   *
1149   * @param string $widget_id
1150   */
1151  function wp_dashboard_rss_output( $widget_id ) {
1152      $widgets = get_option( 'dashboard_widget_options' );
1153      echo '<div class="rss-widget">';
1154      wp_widget_rss_output( $widgets[ $widget_id ] );
1155      echo '</div>';
1156  }
1157  
1158  /**
1159   * Checks to see if all of the feed url in $check_urls are cached.
1160   *
1161   * If $check_urls is empty, look for the rss feed url found in the dashboard
1162   * widget options of $widget_id. If cached, call $callback, a function that
1163   * echoes out output for this widget. If not cache, echo a "Loading..." stub
1164   * which is later replaced by Ajax call (see top of /wp-admin/index.php)
1165   *
1166   * @since 2.5.0
1167   * @since 5.3.0 Formalized the existing and already documented `...$args` parameter
1168   *              by adding it to the function signature.
1169   *
1170   * @param string   $widget_id  The widget ID.
1171   * @param callable $callback   The callback function used to display each feed.
1172   * @param array    $check_urls RSS feeds.
1173   * @param mixed    ...$args    Optional additional parameters to pass to the callback function.
1174   * @return bool True on success, false on failure.
1175   */
1176  function wp_dashboard_cached_rss_widget( $widget_id, $callback, $check_urls = array(), ...$args ) {
1177      $doing_ajax = wp_doing_ajax();
1178      $loading    = '<p class="widget-loading hide-if-no-js">' . __( 'Loading&hellip;' ) . '</p>';
1179      $loading   .= wp_get_admin_notice(
1180          __( 'This widget requires JavaScript.' ),
1181          array(
1182              'type'               => 'error',
1183              'additional_classes' => array( 'inline', 'hide-if-js' ),
1184          )
1185      );
1186  
1187      if ( empty( $check_urls ) ) {
1188          $widgets = get_option( 'dashboard_widget_options' );
1189  
1190          if ( empty( $widgets[ $widget_id ]['url'] ) && ! $doing_ajax ) {
1191              echo $loading;
1192              return false;
1193          }
1194  
1195          $check_urls = array( $widgets[ $widget_id ]['url'] );
1196      }
1197  
1198      $locale    = get_user_locale();
1199      $cache_key = 'dash_v2_' . md5( $widget_id . '_' . $locale );
1200      $output    = get_transient( $cache_key );
1201  
1202      if ( false !== $output ) {
1203          echo $output;
1204          return true;
1205      }
1206  
1207      if ( ! $doing_ajax ) {
1208          echo $loading;
1209          return false;
1210      }
1211  
1212      if ( $callback && is_callable( $callback ) ) {
1213          array_unshift( $args, $widget_id, $check_urls );
1214          ob_start();
1215          call_user_func_array( $callback, $args );
1216          // Default lifetime in cache of 12 hours (same as the feeds).
1217          set_transient( $cache_key, ob_get_flush(), 12 * HOUR_IN_SECONDS );
1218      }
1219  
1220      return true;
1221  }
1222  
1223  //
1224  // Dashboard Widgets Controls.
1225  //
1226  
1227  /**
1228   * Calls widget control callback.
1229   *
1230   * @since 2.5.0
1231   *
1232   * @global callable[] $wp_dashboard_control_callbacks
1233   *
1234   * @param int|false $widget_control_id Optional. Registered widget ID. Default false.
1235   */
1236  function wp_dashboard_trigger_widget_control( $widget_control_id = false ) {
1237      global $wp_dashboard_control_callbacks;
1238  
1239      if ( is_scalar( $widget_control_id ) && $widget_control_id
1240          && isset( $wp_dashboard_control_callbacks[ $widget_control_id ] )
1241          && is_callable( $wp_dashboard_control_callbacks[ $widget_control_id ] )
1242      ) {
1243          call_user_func(
1244              $wp_dashboard_control_callbacks[ $widget_control_id ],
1245              '',
1246              array(
1247                  'id'       => $widget_control_id,
1248                  'callback' => $wp_dashboard_control_callbacks[ $widget_control_id ],
1249              )
1250          );
1251      }
1252  }
1253  
1254  /**
1255   * Sets up the RSS dashboard widget control and $args to be used as input to wp_widget_rss_form().
1256   *
1257   * Handles POST data from RSS-type widgets.
1258   *
1259   * @since 2.5.0
1260   *
1261   * @param string $widget_id
1262   * @param array  $form_inputs
1263   */
1264  function wp_dashboard_rss_control( $widget_id, $form_inputs = array() ) {
1265      $widget_options = get_option( 'dashboard_widget_options' );
1266  
1267      if ( ! $widget_options ) {
1268          $widget_options = array();
1269      }
1270  
1271      if ( ! isset( $widget_options[ $widget_id ] ) ) {
1272          $widget_options[ $widget_id ] = array();
1273      }
1274  
1275      $number = 1; // Hack to use wp_widget_rss_form().
1276  
1277      $widget_options[ $widget_id ]['number'] = $number;
1278  
1279      if ( 'POST' === $_SERVER['REQUEST_METHOD'] && isset( $_POST['widget-rss'][ $number ] ) ) {
1280          $_POST['widget-rss'][ $number ]         = wp_unslash( $_POST['widget-rss'][ $number ] );
1281          $widget_options[ $widget_id ]           = wp_widget_rss_process( $_POST['widget-rss'][ $number ] );
1282          $widget_options[ $widget_id ]['number'] = $number;
1283  
1284          // Title is optional. If black, fill it if possible.
1285          if ( ! $widget_options[ $widget_id ]['title'] && isset( $_POST['widget-rss'][ $number ]['title'] ) ) {
1286              $rss = fetch_feed( $widget_options[ $widget_id ]['url'] );
1287              if ( is_wp_error( $rss ) ) {
1288                  $widget_options[ $widget_id ]['title'] = htmlentities( __( 'Unknown Feed' ) );
1289              } else {
1290                  $widget_options[ $widget_id ]['title'] = htmlentities( strip_tags( $rss->get_title() ) );
1291                  $rss->__destruct();
1292                  unset( $rss );
1293              }
1294          }
1295  
1296          update_option( 'dashboard_widget_options', $widget_options, false );
1297  
1298          $locale    = get_user_locale();
1299          $cache_key = 'dash_v2_' . md5( $widget_id . '_' . $locale );
1300          delete_transient( $cache_key );
1301      }
1302  
1303      wp_widget_rss_form( $widget_options[ $widget_id ], $form_inputs );
1304  }
1305  
1306  
1307  /**
1308   * Renders the Events and News dashboard widget.
1309   *
1310   * @since 4.8.0
1311   */
1312  function wp_dashboard_events_news() {
1313      wp_print_community_events_markup();
1314  
1315      ?>
1316  
1317      <div class="wordpress-news hide-if-no-js">
1318          <?php wp_dashboard_primary(); ?>
1319      </div>
1320  
1321      <p class="community-events-footer">
1322          <?php
1323              printf(
1324                  '<a href="%1$s" target="_blank">%2$s <span class="screen-reader-text"> %3$s</span><span aria-hidden="true" class="dashicons dashicons-external"></span></a>',
1325                  'https://make.wordpress.org/community/meetups-landing-page',
1326                  __( 'Meetups' ),
1327                  /* translators: Hidden accessibility text. */
1328                  __( '(opens in a new tab)' )
1329              );
1330          ?>
1331  
1332          |
1333  
1334          <?php
1335              printf(
1336                  '<a href="%1$s" target="_blank">%2$s <span class="screen-reader-text"> %3$s</span><span aria-hidden="true" class="dashicons dashicons-external"></span></a>',
1337                  'https://central.wordcamp.org/schedule/',
1338                  __( 'WordCamps' ),
1339                  /* translators: Hidden accessibility text. */
1340                  __( '(opens in a new tab)' )
1341              );
1342          ?>
1343  
1344          |
1345  
1346          <?php
1347              printf(
1348                  '<a href="%1$s" target="_blank">%2$s <span class="screen-reader-text"> %3$s</span><span aria-hidden="true" class="dashicons dashicons-external"></span></a>',
1349                  /* translators: If a Rosetta site exists (e.g. https://es.wordpress.org/news/), then use that. Otherwise, leave untranslated. */
1350                  esc_url( _x( 'https://wordpress.org/news/', 'Events and News dashboard widget' ) ),
1351                  __( 'News' ),
1352                  /* translators: Hidden accessibility text. */
1353                  __( '(opens in a new tab)' )
1354              );
1355          ?>
1356      </p>
1357  
1358      <?php
1359  }
1360  
1361  /**
1362   * Prints the markup for the Community Events section of the Events and News Dashboard widget.
1363   *
1364   * @since 4.8.0
1365   */
1366  function wp_print_community_events_markup() {
1367      $community_events_notice  = '<p class="hide-if-js">' . ( 'This widget requires JavaScript.' ) . '</p>';
1368      $community_events_notice .= '<p class="community-events-error-occurred" aria-hidden="true">' . __( 'An error occurred. Please try again.' ) . '</p>';
1369      $community_events_notice .= '<p class="community-events-could-not-locate" aria-hidden="true"></p>';
1370  
1371      wp_admin_notice(
1372          $community_events_notice,
1373          array(
1374              'type'               => 'error',
1375              'additional_classes' => array( 'community-events-errors', 'inline', 'hide-if-js' ),
1376              'paragraph_wrap'     => false,
1377          )
1378      );
1379  
1380      /*
1381       * Hide the main element when the page first loads, because the content
1382       * won't be ready until wp.communityEvents.renderEventsTemplate() has run.
1383       */
1384      ?>
1385      <div id="community-events" class="community-events" aria-hidden="true">
1386          <div class="activity-block">
1387              <p>
1388                  <span id="community-events-location-message"></span>
1389  
1390                  <button class="button-link community-events-toggle-location" aria-expanded="false">
1391                      <span class="dashicons dashicons-location" aria-hidden="true"></span>
1392                      <span class="community-events-location-edit"><?php _e( 'Select location' ); ?></span>
1393                  </button>
1394              </p>
1395  
1396              <form class="community-events-form" aria-hidden="true" action="<?php echo esc_url( admin_url( 'admin-ajax.php' ) ); ?>" method="post">
1397                  <label for="community-events-location">
1398                      <?php _e( 'City:' ); ?>
1399                  </label>
1400                  <?php
1401                  /* translators: Replace with a city related to your locale.
1402                   * Test that it matches the expected location and has upcoming
1403                   * events before including it. If no cities related to your
1404                   * locale have events, then use a city related to your locale
1405                   * that would be recognizable to most users. Use only the city
1406                   * name itself, without any region or country. Use the endonym
1407                   * (native locale name) instead of the English name if possible.
1408                   */
1409                  ?>
1410                  <input id="community-events-location" class="regular-text" type="text" name="community-events-location" placeholder="<?php esc_attr_e( 'Cincinnati' ); ?>" />
1411  
1412                  <?php submit_button( __( 'Submit' ), 'secondary', 'community-events-submit', false ); ?>
1413  
1414                  <button class="community-events-cancel button-link" type="button" aria-expanded="false">
1415                      <?php _e( 'Cancel' ); ?>
1416                  </button>
1417  
1418                  <span class="spinner"></span>
1419              </form>
1420          </div>
1421  
1422          <ul class="community-events-results activity-block last"></ul>
1423      </div>
1424  
1425      <?php
1426  }
1427  
1428  /**
1429   * Renders the events templates for the Event and News widget.
1430   *
1431   * @since 4.8.0
1432   */
1433  function wp_print_community_events_templates() {
1434      ?>
1435  
1436      <script id="tmpl-community-events-attend-event-near" type="text/template">
1437          <?php
1438          printf(
1439              /* translators: %s: The name of a city. */
1440              __( 'Attend an upcoming event near %s.' ),
1441              '<strong>{{ data.location.description }}</strong>'
1442          );
1443          ?>
1444      </script>
1445  
1446      <script id="tmpl-community-events-could-not-locate" type="text/template">
1447          <?php
1448          printf(
1449              /* translators: %s is the name of the city we couldn't locate.
1450               * Replace the examples with cities in your locale, but test
1451               * that they match the expected location before including them.
1452               * Use endonyms (native locale names) whenever possible.
1453               */
1454              __( '%s could not be located. Please try another nearby city. For example: Kansas City; Springfield; Portland.' ),
1455              '<em>{{data.unknownCity}}</em>'
1456          );
1457          ?>
1458      </script>
1459  
1460      <script id="tmpl-community-events-event-list" type="text/template">
1461          <# _.each( data.events, function( event ) { #>
1462              <li class="event event-{{ event.type }} wp-clearfix">
1463                  <div class="event-info">
1464                      <div class="dashicons event-icon" aria-hidden="true"></div>
1465                      <div class="event-info-inner">
1466                          <a class="event-title" href="{{ event.url }}">{{ event.title }}</a>
1467                          <# if ( event.type ) {
1468                              const titleCaseEventType = event.type.replace(
1469                                  /\w\S*/g,
1470                                  function ( type ) { return type.charAt(0).toUpperCase() + type.substr(1).toLowerCase(); }
1471                              );
1472                          #>
1473                              {{ 'wordcamp' === event.type ? 'WordCamp' : titleCaseEventType }}
1474                              <span class="ce-separator" aria-hidden="true"></span>
1475                          <# } #>
1476                          <span class="event-city">{{ event.location.location }}</span>
1477                      </div>
1478                  </div>
1479  
1480                  <div class="event-date-time">
1481                      <span class="event-date">{{ event.user_formatted_date }}</span>
1482                      <# if ( 'meetup' === event.type ) { #>
1483                          <span class="event-time">
1484                              {{ event.user_formatted_time }} {{ event.timeZoneAbbreviation }}
1485                          </span>
1486                      <# } #>
1487                  </div>
1488              </li>
1489          <# } ) #>
1490  
1491          <# if ( data.events.length <= 2 ) { #>
1492              <li class="event-none">
1493                  <?php
1494                  printf(
1495                      /* translators: %s: Localized meetup organization documentation URL. */
1496                      __( 'Want more events? <a href="%s">Help organize the next one</a>!' ),
1497                      __( 'https://make.wordpress.org/community/organize-event-landing-page/' )
1498                  );
1499                  ?>
1500              </li>
1501          <# } #>
1502  
1503      </script>
1504  
1505      <script id="tmpl-community-events-no-upcoming-events" type="text/template">
1506          <li class="event-none">
1507              <# if ( data.location.description ) { #>
1508                  <?php
1509                  printf(
1510                      /* translators: 1: The city the user searched for, 2: Meetup organization documentation URL. */
1511                      __( 'There are no events scheduled near %1$s at the moment. Would you like to <a href="%2$s">organize a WordPress event</a>?' ),
1512                      '{{ data.location.description }}',
1513                      __( 'https://make.wordpress.org/community/handbook/meetup-organizer/welcome/' )
1514                  );
1515                  ?>
1516  
1517              <# } else { #>
1518                  <?php
1519                  printf(
1520                      /* translators: %s: Meetup organization documentation URL. */
1521                      __( 'There are no events scheduled near you at the moment. Would you like to <a href="%s">organize a WordPress event</a>?' ),
1522                      __( 'https://make.wordpress.org/community/handbook/meetup-organizer/welcome/' )
1523                  );
1524                  ?>
1525              <# } #>
1526          </li>
1527      </script>
1528      <?php
1529  }
1530  
1531  /**
1532   * 'WordPress Events and News' dashboard widget.
1533   *
1534   * @since 2.7.0
1535   * @since 4.8.0 Removed popular plugins feed.
1536   */
1537  function wp_dashboard_primary() {
1538      $feeds = array(
1539          'news'   => array(
1540  
1541              /**
1542               * Filters the primary link URL for the 'WordPress Events and News' dashboard widget.
1543               *
1544               * @since 2.5.0
1545               *
1546               * @param string $link The widget's primary link URL.
1547               */
1548              'link'         => apply_filters( 'dashboard_primary_link', __( 'https://wordpress.org/news/' ) ),
1549  
1550              /**
1551               * Filters the primary feed URL for the 'WordPress Events and News' dashboard widget.
1552               *
1553               * @since 2.3.0
1554               *
1555               * @param string $url The widget's primary feed URL.
1556               */
1557              'url'          => apply_filters( 'dashboard_primary_feed', __( 'https://wordpress.org/news/feed/' ) ),
1558  
1559              /**
1560               * Filters the primary link title for the 'WordPress Events and News' dashboard widget.
1561               *
1562               * @since 2.3.0
1563               *
1564               * @param string $title Title attribute for the widget's primary link.
1565               */
1566              'title'        => apply_filters( 'dashboard_primary_title', __( 'WordPress Blog' ) ),
1567              'items'        => 2,
1568              'show_summary' => 0,
1569              'show_author'  => 0,
1570              'show_date'    => 0,
1571          ),
1572          'planet' => array(
1573  
1574              /**
1575               * Filters the secondary link URL for the 'WordPress Events and News' dashboard widget.
1576               *
1577               * @since 2.3.0
1578               *
1579               * @param string $link The widget's secondary link URL.
1580               */
1581              'link'         => apply_filters(
1582                  'dashboard_secondary_link',
1583                  /* translators: Link to the Planet website of the locale. */
1584                  __( 'https://planet.wordpress.org/' )
1585              ),
1586  
1587              /**
1588               * Filters the secondary feed URL for the 'WordPress Events and News' dashboard widget.
1589               *
1590               * @since 2.3.0
1591               *
1592               * @param string $url The widget's secondary feed URL.
1593               */
1594              'url'          => apply_filters(
1595                  'dashboard_secondary_feed',
1596                  /* translators: Link to the Planet feed of the locale. */
1597                  __( 'https://planet.wordpress.org/feed/' )
1598              ),
1599  
1600              /**
1601               * Filters the secondary link title for the 'WordPress Events and News' dashboard widget.
1602               *
1603               * @since 2.3.0
1604               *
1605               * @param string $title Title attribute for the widget's secondary link.
1606               */
1607              'title'        => apply_filters( 'dashboard_secondary_title', __( 'Other WordPress News' ) ),
1608  
1609              /**
1610               * Filters the number of secondary link items for the 'WordPress Events and News' dashboard widget.
1611               *
1612               * @since 4.4.0
1613               *
1614               * @param string $items How many items to show in the secondary feed.
1615               */
1616              'items'        => apply_filters( 'dashboard_secondary_items', 3 ),
1617              'show_summary' => 0,
1618              'show_author'  => 0,
1619              'show_date'    => 0,
1620          ),
1621      );
1622  
1623      wp_dashboard_cached_rss_widget( 'dashboard_primary', 'wp_dashboard_primary_output', $feeds );
1624  }
1625  
1626  /**
1627   * Displays the WordPress events and news feeds.
1628   *
1629   * @since 3.8.0
1630   * @since 4.8.0 Removed popular plugins feed.
1631   *
1632   * @param string $widget_id Widget ID.
1633   * @param array  $feeds     Array of RSS feeds.
1634   */
1635  function wp_dashboard_primary_output( $widget_id, $feeds ) {
1636      foreach ( $feeds as $type => $args ) {
1637          $args['type'] = $type;
1638          echo '<div class="rss-widget">';
1639              wp_widget_rss_output( $args['url'], $args );
1640          echo '</div>';
1641      }
1642  }
1643  
1644  /**
1645   * Displays file upload quota on dashboard.
1646   *
1647   * Runs on the {@see 'activity_box_end'} hook in wp_dashboard_right_now().
1648   *
1649   * @since 3.0.0
1650   *
1651   * @return true|null True if not multisite, user can't upload files, or the space check option is disabled.
1652   */
1653  function wp_dashboard_quota() {
1654      if ( ! is_multisite() || ! current_user_can( 'upload_files' )
1655          || get_site_option( 'upload_space_check_disabled' )
1656      ) {
1657          return true;
1658      }
1659  
1660      $quota = get_space_allowed();
1661      $used  = get_space_used();
1662  
1663      if ( $used > $quota ) {
1664          $percentused = '100';
1665      } else {
1666          $percentused = ( $used / $quota ) * 100;
1667      }
1668  
1669      $used_class  = ( $percentused >= 70 ) ? ' warning' : '';
1670      $used        = round( $used, 2 );
1671      $percentused = number_format( $percentused );
1672  
1673      ?>
1674      <h3 class="mu-storage"><?php _e( 'Storage Space' ); ?></h3>
1675      <div class="mu-storage">
1676      <ul>
1677          <li class="storage-count">
1678              <?php
1679              $text = sprintf(
1680                  /* translators: %s: Number of megabytes. */
1681                  __( '%s MB Space Allowed' ),
1682                  number_format_i18n( $quota )
1683              );
1684              printf(
1685                  '<a href="%1$s">%2$s<span class="screen-reader-text"> (%3$s)</span></a>',
1686                  esc_url( admin_url( 'upload.php' ) ),
1687                  $text,
1688                  /* translators: Hidden accessibility text. */
1689                  __( 'Manage Uploads' )
1690              );
1691              ?>
1692          </li><li class="storage-count <?php echo $used_class; ?>">
1693              <?php
1694              $text = sprintf(
1695                  /* translators: 1: Number of megabytes, 2: Percentage. */
1696                  __( '%1$s MB (%2$s%%) Space Used' ),
1697                  number_format_i18n( $used, 2 ),
1698                  $percentused
1699              );
1700              printf(
1701                  '<a href="%1$s" class="musublink">%2$s<span class="screen-reader-text"> (%3$s)</span></a>',
1702                  esc_url( admin_url( 'upload.php' ) ),
1703                  $text,
1704                  /* translators: Hidden accessibility text. */
1705                  __( 'Manage Uploads' )
1706              );
1707              ?>
1708          </li>
1709      </ul>
1710      </div>
1711      <?php
1712      return null;
1713  }
1714  
1715  /**
1716   * Displays the browser update nag.
1717   *
1718   * @since 3.2.0
1719   * @since 5.8.0 Added a special message for Internet Explorer users.
1720   *
1721   * @global bool $is_IE
1722   */
1723  function wp_dashboard_browser_nag() {
1724      global $is_IE;
1725  
1726      $notice   = '';
1727      $response = wp_check_browser_version();
1728  
1729      if ( $response ) {
1730          if ( $is_IE ) {
1731              $msg = __( 'Internet Explorer does not give you the best WordPress experience. Switch to Microsoft Edge, or another more modern browser to get the most from your site.' );
1732          } elseif ( $response['insecure'] ) {
1733              $msg = sprintf(
1734                  /* translators: %s: Browser name and link. */
1735                  __( "It looks like you're using an insecure version of %s. Using an outdated browser makes your computer unsafe. For the best WordPress experience, please update your browser." ),
1736                  sprintf( '<a href="%s">%s</a>', esc_url( $response['update_url'] ), esc_html( $response['name'] ) )
1737              );
1738          } else {
1739              $msg = sprintf(
1740                  /* translators: %s: Browser name and link. */
1741                  __( "It looks like you're using an old version of %s. For the best WordPress experience, please update your browser." ),
1742                  sprintf( '<a href="%s">%s</a>', esc_url( $response['update_url'] ), esc_html( $response['name'] ) )
1743              );
1744          }
1745  
1746          $browser_nag_class = '';
1747          if ( ! empty( $response['img_src'] ) ) {
1748              $img_src = ( is_ssl() && ! empty( $response['img_src_ssl'] ) ) ? $response['img_src_ssl'] : $response['img_src'];
1749  
1750              $notice           .= '<div class="alignright browser-icon"><img src="' . esc_url( $img_src ) . '" alt="" /></div>';
1751              $browser_nag_class = ' has-browser-icon';
1752          }
1753          $notice .= "<p class='browser-update-nag{$browser_nag_class}'>{$msg}</p>";
1754  
1755          $browsehappy = 'https://browsehappy.com/';
1756          $locale      = get_user_locale();
1757          if ( 'en_US' !== $locale ) {
1758              $browsehappy = add_query_arg( 'locale', $locale, $browsehappy );
1759          }
1760  
1761          if ( $is_IE ) {
1762              $msg_browsehappy = sprintf(
1763                  /* translators: %s: Browse Happy URL. */
1764                  __( 'Learn how to <a href="%s" class="update-browser-link">browse happy</a>' ),
1765                  esc_url( $browsehappy )
1766              );
1767          } else {
1768              $msg_browsehappy = sprintf(
1769                  /* translators: 1: Browser update URL, 2: Browser name, 3: Browse Happy URL. */
1770                  __( '<a href="%1$s" class="update-browser-link">Update %2$s</a> or learn how to <a href="%3$s" class="browse-happy-link">browse happy</a>' ),
1771                  esc_attr( $response['update_url'] ),
1772                  esc_html( $response['name'] ),
1773                  esc_url( $browsehappy )
1774              );
1775          }
1776  
1777          $notice .= '<p>' . $msg_browsehappy . '</p>';
1778          $notice .= '<p class="hide-if-no-js"><a href="" class="dismiss" aria-label="' . esc_attr__( 'Dismiss the browser warning panel' ) . '">' . __( 'Dismiss' ) . '</a></p>';
1779          $notice .= '<div class="clear"></div>';
1780      }
1781  
1782      /**
1783       * Filters the notice output for the 'Browse Happy' nag meta box.
1784       *
1785       * @since 3.2.0
1786       *
1787       * @param string      $notice   The notice content.
1788       * @param array|false $response An array containing web browser information, or
1789       *                              false on failure. See wp_check_browser_version().
1790       */
1791      echo apply_filters( 'browse-happy-notice', $notice, $response ); // phpcs:ignore WordPress.NamingConventions.ValidHookName.UseUnderscores
1792  }
1793  
1794  /**
1795   * Adds an additional class to the browser nag if the current version is insecure.
1796   *
1797   * @since 3.2.0
1798   *
1799   * @param string[] $classes Array of meta box classes.
1800   * @return string[] Modified array of meta box classes.
1801   */
1802  function dashboard_browser_nag_class( $classes ) {
1803      $response = wp_check_browser_version();
1804  
1805      if ( $response && $response['insecure'] ) {
1806          $classes[] = 'browser-insecure';
1807      }
1808  
1809      return $classes;
1810  }
1811  
1812  /**
1813   * Checks if the user needs a browser update.
1814   *
1815   * @since 3.2.0
1816   *
1817   * @return array|false Array of browser data on success, false on failure.
1818   */
1819  function wp_check_browser_version() {
1820      if ( empty( $_SERVER['HTTP_USER_AGENT'] ) ) {
1821          return false;
1822      }
1823  
1824      $key = md5( $_SERVER['HTTP_USER_AGENT'] );
1825  
1826      $response = get_site_transient( 'browser_' . $key );
1827  
1828      if ( false === $response ) {
1829          $url     = 'http://api.wordpress.org/core/browse-happy/1.1/';
1830          $options = array(
1831              'body'       => array( 'useragent' => $_SERVER['HTTP_USER_AGENT'] ),
1832              'user-agent' => 'WordPress/' . wp_get_wp_version() . '; ' . home_url( '/' ),
1833          );
1834  
1835          if ( wp_http_supports( array( 'ssl' ) ) ) {
1836              $url = set_url_scheme( $url, 'https' );
1837          }
1838  
1839          $response = wp_remote_post( $url, $options );
1840  
1841          if ( is_wp_error( $response ) || 200 !== wp_remote_retrieve_response_code( $response ) ) {
1842              return false;
1843          }
1844  
1845          /**
1846           * Response should be an array with:
1847           *  'platform' - string - A user-friendly platform name, if it can be determined
1848           *  'name' - string - A user-friendly browser name
1849           *  'version' - string - The version of the browser the user is using
1850           *  'current_version' - string - The most recent version of the browser
1851           *  'upgrade' - boolean - Whether the browser needs an upgrade
1852           *  'insecure' - boolean - Whether the browser is deemed insecure
1853           *  'update_url' - string - The url to visit to upgrade
1854           *  'img_src' - string - An image representing the browser
1855           *  'img_src_ssl' - string - An image (over SSL) representing the browser
1856           */
1857          $response = json_decode( wp_remote_retrieve_body( $response ), true );
1858  
1859          if ( ! is_array( $response ) ) {
1860              return false;
1861          }
1862  
1863          set_site_transient( 'browser_' . $key, $response, WEEK_IN_SECONDS );
1864      }
1865  
1866      return $response;
1867  }
1868  
1869  /**
1870   * Displays the PHP update nag.
1871   *
1872   * @since 5.1.0
1873   */
1874  function wp_dashboard_php_nag() {
1875      $response = wp_check_php_version();
1876  
1877      if ( ! $response ) {
1878          return;
1879      }
1880  
1881      if ( isset( $response['is_secure'] ) && ! $response['is_secure'] ) {
1882          // The `is_secure` array key name doesn't actually imply this is a secure version of PHP. It only means it receives security updates.
1883  
1884          if ( $response['is_lower_than_future_minimum'] ) {
1885              $message = sprintf(
1886                  /* translators: %s: The server PHP version. */
1887                  __( 'Your site is running on an outdated version of PHP (%s), which does not receive security updates and soon will not be supported by WordPress. Ensure that PHP is updated on your server as soon as possible. Otherwise you will not be able to upgrade WordPress.' ),
1888                  PHP_VERSION
1889              );
1890          } else {
1891              $message = sprintf(
1892                  /* translators: %s: The server PHP version. */
1893                  __( 'Your site is running on an outdated version of PHP (%s), which does not receive security updates. It should be updated.' ),
1894                  PHP_VERSION
1895              );
1896          }
1897      } elseif ( $response['is_lower_than_future_minimum'] ) {
1898          $message = sprintf(
1899              /* translators: %s: The server PHP version. */
1900              __( 'Your site is running on an outdated version of PHP (%s), which soon will not be supported by WordPress. Ensure that PHP is updated on your server as soon as possible. Otherwise you will not be able to upgrade WordPress.' ),
1901              PHP_VERSION
1902          );
1903      } else {
1904          $message = sprintf(
1905              /* translators: %s: The server PHP version. */
1906              __( 'Your site is running on an outdated version of PHP (%s), which should be updated.' ),
1907              PHP_VERSION
1908          );
1909      }
1910      ?>
1911      <p class="bigger-bolder-text"><?php echo $message; ?></p>
1912  
1913      <p><?php _e( 'What is PHP and how does it affect my site?' ); ?></p>
1914      <p>
1915          <?php _e( 'PHP is one of the programming languages used to build WordPress. Newer versions of PHP receive regular security updates and may increase your site&#8217;s performance.' ); ?>
1916          <?php
1917          if ( ! empty( $response['recommended_version'] ) ) {
1918              printf(
1919                  /* translators: %s: The minimum recommended PHP version. */
1920                  __( 'The minimum recommended version of PHP is %s.' ),
1921                  $response['recommended_version']
1922              );
1923          }
1924          ?>
1925      </p>
1926  
1927      <p class="button-container">
1928          <?php
1929          printf(
1930              '<a class="button button-primary" href="%1$s" target="_blank">%2$s<span class="screen-reader-text"> %3$s</span><span aria-hidden="true" class="dashicons dashicons-external"></span></a>',
1931              esc_url( wp_get_update_php_url() ),
1932              __( 'Learn more about updating PHP' ),
1933              /* translators: Hidden accessibility text. */
1934              __( '(opens in a new tab)' )
1935          );
1936          ?>
1937      </p>
1938      <?php
1939  
1940      wp_update_php_annotation();
1941      wp_direct_php_update_button();
1942  }
1943  
1944  /**
1945   * Adds an additional class to the PHP nag if the current version is insecure.
1946   *
1947   * @since 5.1.0
1948   *
1949   * @param string[] $classes Array of meta box classes.
1950   * @return string[] Modified array of meta box classes.
1951   */
1952  function dashboard_php_nag_class( $classes ) {
1953      $response = wp_check_php_version();
1954  
1955      if ( ! $response ) {
1956          return $classes;
1957      }
1958  
1959      if ( isset( $response['is_secure'] ) && ! $response['is_secure'] ) {
1960          $classes[] = 'php-no-security-updates';
1961      } elseif ( $response['is_lower_than_future_minimum'] ) {
1962          $classes[] = 'php-version-lower-than-future-minimum';
1963      }
1964  
1965      return $classes;
1966  }
1967  
1968  /**
1969   * Displays the Site Health Status widget.
1970   *
1971   * @since 5.4.0
1972   */
1973  function wp_dashboard_site_health() {
1974      $get_issues = get_transient( 'health-check-site-status-result' );
1975  
1976      $issue_counts = array();
1977  
1978      if ( false !== $get_issues ) {
1979          $issue_counts = json_decode( $get_issues, true );
1980      }
1981  
1982      if ( ! is_array( $issue_counts ) || ! $issue_counts ) {
1983          $issue_counts = array(
1984              'good'        => 0,
1985              'recommended' => 0,
1986              'critical'    => 0,
1987          );
1988      }
1989  
1990      $issues_total = $issue_counts['recommended'] + $issue_counts['critical'];
1991      ?>
1992      <div class="health-check-widget">
1993          <div class="health-check-widget-title-section site-health-progress-wrapper loading hide-if-no-js">
1994              <div class="site-health-progress">
1995                  <svg aria-hidden="true" focusable="false" width="100%" height="100%" viewBox="0 0 200 200" version="1.1" xmlns="http://www.w3.org/2000/svg">
1996                      <circle r="90" cx="100" cy="100" fill="transparent" stroke-dasharray="565.48" stroke-dashoffset="0"></circle>
1997                      <circle id="bar" r="90" cx="100" cy="100" fill="transparent" stroke-dasharray="565.48" stroke-dashoffset="0"></circle>
1998                  </svg>
1999              </div>
2000              <div class="site-health-progress-label">
2001                  <?php if ( false === $get_issues ) : ?>
2002                      <?php _e( 'No information yet&hellip;' ); ?>
2003                  <?php else : ?>
2004                      <?php _e( 'Results are still loading&hellip;' ); ?>
2005                  <?php endif; ?>
2006              </div>
2007          </div>
2008  
2009          <div class="site-health-details">
2010              <?php if ( false === $get_issues ) : ?>
2011                  <p>
2012                      <?php
2013                      printf(
2014                          /* translators: %s: URL to Site Health screen. */
2015                          __( 'Site health checks will automatically run periodically to gather information about your site. You can also <a href="%s">visit the Site Health screen</a> to gather information about your site now.' ),
2016                          esc_url( admin_url( 'site-health.php' ) )
2017                      );
2018                      ?>
2019                  </p>
2020              <?php else : ?>
2021                  <p>
2022                      <?php if ( $issues_total <= 0 ) : ?>
2023                          <?php _e( 'Great job! Your site currently passes all site health checks.' ); ?>
2024                      <?php elseif ( 1 === (int) $issue_counts['critical'] ) : ?>
2025                          <?php _e( 'Your site has a critical issue that should be addressed as soon as possible to improve its performance and security.' ); ?>
2026                      <?php elseif ( $issue_counts['critical'] > 1 ) : ?>
2027                          <?php _e( 'Your site has critical issues that should be addressed as soon as possible to improve its performance and security.' ); ?>
2028                      <?php elseif ( 1 === (int) $issue_counts['recommended'] ) : ?>
2029                          <?php _e( 'Your site&#8217;s health is looking good, but there is still one thing you can do to improve its performance and security.' ); ?>
2030                      <?php else : ?>
2031                          <?php _e( 'Your site&#8217;s health is looking good, but there are still some things you can do to improve its performance and security.' ); ?>
2032                      <?php endif; ?>
2033                  </p>
2034              <?php endif; ?>
2035  
2036              <?php if ( $issues_total > 0 && false !== $get_issues ) : ?>
2037                  <p>
2038                      <?php
2039                      printf(
2040                          /* translators: 1: Number of issues. 2: URL to Site Health screen. */
2041                          _n(
2042                              'Take a look at the <strong>%1$d item</strong> on the <a href="%2$s">Site Health screen</a>.',
2043                              'Take a look at the <strong>%1$d items</strong> on the <a href="%2$s">Site Health screen</a>.',
2044                              $issues_total
2045                          ),
2046                          $issues_total,
2047                          esc_url( admin_url( 'site-health.php' ) )
2048                      );
2049                      ?>
2050                  </p>
2051              <?php endif; ?>
2052          </div>
2053      </div>
2054  
2055      <?php
2056  }
2057  
2058  /**
2059   * Outputs empty dashboard widget to be populated by JS later.
2060   *
2061   * Usable by plugins.
2062   *
2063   * @since 2.5.0
2064   */
2065  function wp_dashboard_empty() {}
2066  
2067  /**
2068   * Displays a welcome panel to introduce users to WordPress.
2069   *
2070   * @since 3.3.0
2071   * @since 5.9.0 Send users to the Site Editor if the active theme is block-based.
2072   */
2073  function wp_welcome_panel() {
2074      list( $display_version ) = explode( '-', wp_get_wp_version() );
2075      $can_customize           = current_user_can( 'customize' );
2076      $is_block_theme          = wp_is_block_theme();
2077      ?>
2078      <div class="welcome-panel-content">
2079      <div class="welcome-panel-header-wrap">
2080          <div class="welcome-panel-header">
2081              <div class="welcome-panel-header-image">
2082                  <?php echo file_get_contents( dirname( __DIR__ ) . '/images/dashboard-background.svg' ); ?>
2083              </div>
2084              <h2><?php _e( 'Welcome to WordPress!' ); ?></h2>
2085              <p>
2086                  <a href="<?php echo esc_url( admin_url( 'about.php' ) ); ?>">
2087                  <?php
2088                      /* translators: %s: Current WordPress version. */
2089                      printf( __( 'Learn more about the %s version.' ), esc_html( $display_version ) );
2090                  ?>
2091                  </a>
2092              </p>
2093          </div>
2094      </div>
2095      <div class="welcome-panel-column-container">
2096          <div class="welcome-panel-column">
2097              <svg width="48" height="48" viewBox="0 0 48 48" fill="none" xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false">
2098                  <rect width="48" height="48" rx="4" fill="#1E1E1E"/>
2099                  <path fill-rule="evenodd" clip-rule="evenodd" d="M32.0668 17.0854L28.8221 13.9454L18.2008 24.671L16.8983 29.0827L21.4257 27.8309L32.0668 17.0854ZM16 32.75H24V31.25H16V32.75Z" fill="white"/>
2100              </svg>
2101              <div class="welcome-panel-column-content">
2102                  <h3><?php _e( 'Author rich content with blocks and patterns' ); ?></h3>
2103                  <p><?php _e( 'Block patterns are pre-configured block layouts. Use them to get inspired or create new pages in a flash.' ); ?></p>
2104                  <a href="<?php echo esc_url( admin_url( 'post-new.php?post_type=page' ) ); ?>"><?php _e( 'Add a new page' ); ?></a>
2105              </div>
2106          </div>
2107          <div class="welcome-panel-column">
2108              <svg width="48" height="48" viewBox="0 0 48 48" fill="none" xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false">
2109                  <rect width="48" height="48" rx="4" fill="#1E1E1E"/>
2110                  <path fill-rule="evenodd" clip-rule="evenodd" d="M18 16h12a2 2 0 0 1 2 2v12a2 2 0 0 1-2 2H18a2 2 0 0 1-2-2V18a2 2 0 0 1 2-2zm12 1.5H18a.5.5 0 0 0-.5.5v3h13v-3a.5.5 0 0 0-.5-.5zm.5 5H22v8h8a.5.5 0 0 0 .5-.5v-7.5zm-10 0h-3V30a.5.5 0 0 0 .5.5h2.5v-8z" fill="#fff"/>
2111              </svg>
2112              <div class="welcome-panel-column-content">
2113              <?php if ( $is_block_theme ) : ?>
2114                  <h3><?php _e( 'Customize your entire site with block themes' ); ?></h3>
2115                  <p><?php _e( 'Design everything on your site &#8212; from the header down to the footer, all using blocks and patterns.' ); ?></p>
2116                  <a href="<?php echo esc_url( admin_url( 'site-editor.php' ) ); ?>"><?php _e( 'Open site editor' ); ?></a>
2117              <?php else : ?>
2118                  <h3><?php _e( 'Start Customizing' ); ?></h3>
2119                  <p><?php _e( 'Configure your site&#8217;s logo, header, menus, and more in the Customizer.' ); ?></p>
2120                  <?php if ( $can_customize ) : ?>
2121                      <a class="load-customize hide-if-no-customize" href="<?php echo wp_customize_url(); ?>"><?php _e( 'Open the Customizer' ); ?></a>
2122                  <?php endif; ?>
2123              <?php endif; ?>
2124              </div>
2125          </div>
2126          <div class="welcome-panel-column">
2127              <svg width="48" height="48" viewBox="0 0 48 48" fill="none" xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false">
2128                  <rect width="48" height="48" rx="4" fill="#1E1E1E"/>
2129                  <path fill-rule="evenodd" clip-rule="evenodd" d="M31 24a7 7 0 0 1-7 7V17a7 7 0 0 1 7 7zm-7-8a8 8 0 1 1 0 16 8 8 0 0 1 0-16z" fill="#fff"/>
2130              </svg>
2131              <div class="welcome-panel-column-content">
2132              <?php if ( $is_block_theme ) : ?>
2133                  <h3><?php _e( 'Switch up your site&#8217;s look & feel with Styles' ); ?></h3>
2134                  <p><?php _e( 'Tweak your site, or give it a whole new look! Get creative &#8212; how about a new color palette or font?' ); ?></p>
2135                  <a href="<?php echo esc_url( add_query_arg( 'p', rawurlencode( '/styles' ), admin_url( 'site-editor.php' ) ) ); ?>"><?php _e( 'Edit styles' ); ?></a>
2136              <?php else : ?>
2137                  <h3><?php _e( 'Discover a new way to build your site.' ); ?></h3>
2138                  <p><?php _e( 'There is a new kind of WordPress theme, called a block theme, that lets you build the site you&#8217;ve always wanted &#8212; with blocks and styles.' ); ?></p>
2139                  <a href="<?php echo esc_url( __( 'https://wordpress.org/documentation/article/block-themes/' ) ); ?>"><?php _e( 'Learn about block themes' ); ?></a>
2140              <?php endif; ?>
2141              </div>
2142          </div>
2143      </div>
2144      </div>
2145      <?php
2146  }


Generated : Mon May 4 08:20:14 2026 Cross-referenced by PHPXref