[ Index ]

PHP Cross Reference of WordPress Trunk (Updated Daily)

Search

title

Body

[close]

/wp-includes/rest-api/endpoints/ -> class-wp-rest-widget-types-controller.php (source)

   1  <?php
   2  /**
   3   * REST API: WP_REST_Widget_Types_Controller class
   4   *
   5   * @package WordPress
   6   * @subpackage REST_API
   7   * @since 5.8.0
   8   */
   9  
  10  /**
  11   * Core class to access widget types via the REST API.
  12   *
  13   * @since 5.8.0
  14   *
  15   * @see WP_REST_Controller
  16   */
  17  class WP_REST_Widget_Types_Controller extends WP_REST_Controller {
  18  
  19      /**
  20       * Constructor.
  21       *
  22       * @since 5.8.0
  23       */
  24  	public function __construct() {
  25          $this->namespace = 'wp/v2';
  26          $this->rest_base = 'widget-types';
  27      }
  28  
  29      /**
  30       * Registers the widget type routes.
  31       *
  32       * @since 5.8.0
  33       *
  34       * @see register_rest_route()
  35       */
  36  	public function register_routes() {
  37          register_rest_route(
  38              $this->namespace,
  39              '/' . $this->rest_base,
  40              array(
  41                  array(
  42                      'methods'             => WP_REST_Server::READABLE,
  43                      'callback'            => array( $this, 'get_items' ),
  44                      'permission_callback' => array( $this, 'get_items_permissions_check' ),
  45                      'args'                => $this->get_collection_params(),
  46                  ),
  47                  'schema' => array( $this, 'get_public_item_schema' ),
  48              )
  49          );
  50  
  51          register_rest_route(
  52              $this->namespace,
  53              '/' . $this->rest_base . '/(?P<id>[a-zA-Z0-9_-]+)',
  54              array(
  55                  'args'   => array(
  56                      'id' => array(
  57                          'description' => __( 'The widget type id.' ),
  58                          'type'        => 'string',
  59                      ),
  60                  ),
  61                  array(
  62                      'methods'             => WP_REST_Server::READABLE,
  63                      'callback'            => array( $this, 'get_item' ),
  64                      'permission_callback' => array( $this, 'get_item_permissions_check' ),
  65                      'args'                => $this->get_collection_params(),
  66                  ),
  67                  'schema' => array( $this, 'get_public_item_schema' ),
  68              )
  69          );
  70  
  71          register_rest_route(
  72              $this->namespace,
  73              '/' . $this->rest_base . '/(?P<id>[a-zA-Z0-9_-]+)/encode',
  74              array(
  75                  'args' => array(
  76                      'id'        => array(
  77                          'description' => __( 'The widget type id.' ),
  78                          'type'        => 'string',
  79                          'required'    => true,
  80                      ),
  81                      'instance'  => array(
  82                          'description' => __( 'Current instance settings of the widget.' ),
  83                          'type'        => 'object',
  84                      ),
  85                      'form_data' => array(
  86                          'description'       => __( 'Serialized widget form data to encode into instance settings.' ),
  87                          'type'              => 'string',
  88                          'sanitize_callback' => static function ( $form_data ) {
  89                              $array = array();
  90                              wp_parse_str( $form_data, $array );
  91                              return $array;
  92                          },
  93                      ),
  94                  ),
  95                  array(
  96                      'methods'             => WP_REST_Server::CREATABLE,
  97                      'permission_callback' => array( $this, 'get_item_permissions_check' ),
  98                      'callback'            => array( $this, 'encode_form_data' ),
  99                  ),
 100              )
 101          );
 102  
 103          register_rest_route(
 104              $this->namespace,
 105              '/' . $this->rest_base . '/(?P<id>[a-zA-Z0-9_-]+)/render',
 106              array(
 107                  array(
 108                      'methods'             => WP_REST_Server::CREATABLE,
 109                      'permission_callback' => array( $this, 'get_item_permissions_check' ),
 110                      'callback'            => array( $this, 'render' ),
 111                      'args'                => array(
 112                          'id'       => array(
 113                              'description' => __( 'The widget type id.' ),
 114                              'type'        => 'string',
 115                              'required'    => true,
 116                          ),
 117                          'instance' => array(
 118                              'description' => __( 'Current instance settings of the widget.' ),
 119                              'type'        => 'object',
 120                          ),
 121                      ),
 122                  ),
 123              )
 124          );
 125      }
 126  
 127      /**
 128       * Checks whether a given request has permission to read widget types.
 129       *
 130       * @since 5.8.0
 131       *
 132       * @param WP_REST_Request $request Full details about the request.
 133       * @return true|WP_Error True if the request has read access, WP_Error object otherwise.
 134       */
 135  	public function get_items_permissions_check( $request ) {
 136          return $this->check_read_permission();
 137      }
 138  
 139      /**
 140       * Retrieves the list of all widget types.
 141       *
 142       * @since 5.8.0
 143       *
 144       * @param WP_REST_Request $request Full details about the request.
 145       * @return WP_REST_Response|WP_Error Response object on success, or WP_Error object on failure.
 146       */
 147  	public function get_items( $request ) {
 148          if ( $request->is_method( 'HEAD' ) ) {
 149              // Return early as this handler doesn't add any response headers.
 150              return new WP_REST_Response( array() );
 151          }
 152  
 153          $data = array();
 154          foreach ( $this->get_widgets() as $widget ) {
 155              $widget_type = $this->prepare_item_for_response( $widget, $request );
 156              $data[]      = $this->prepare_response_for_collection( $widget_type );
 157          }
 158  
 159          return rest_ensure_response( $data );
 160      }
 161  
 162      /**
 163       * Checks if a given request has access to read a widget type.
 164       *
 165       * @since 5.8.0
 166       *
 167       * @param WP_REST_Request $request Full details about the request.
 168       * @return true|WP_Error True if the request has read access for the item, WP_Error object otherwise.
 169       */
 170  	public function get_item_permissions_check( $request ) {
 171          $check = $this->check_read_permission();
 172          if ( is_wp_error( $check ) ) {
 173              return $check;
 174          }
 175          $widget_id   = $request['id'];
 176          $widget_type = $this->get_widget( $widget_id );
 177          if ( is_wp_error( $widget_type ) ) {
 178              return $widget_type;
 179          }
 180  
 181          return true;
 182      }
 183  
 184      /**
 185       * Checks whether the user can read widget types.
 186       *
 187       * @since 5.8.0
 188       *
 189       * @return true|WP_Error True if the widget type is visible, WP_Error otherwise.
 190       */
 191  	protected function check_read_permission() {
 192          if ( ! current_user_can( 'edit_theme_options' ) ) {
 193              return new WP_Error(
 194                  'rest_cannot_manage_widgets',
 195                  __( 'Sorry, you are not allowed to manage widgets on this site.' ),
 196                  array(
 197                      'status' => rest_authorization_required_code(),
 198                  )
 199              );
 200          }
 201  
 202          return true;
 203      }
 204  
 205      /**
 206       * Gets the details about the requested widget.
 207       *
 208       * @since 5.8.0
 209       *
 210       * @param string $id The widget type id.
 211       * @return array|WP_Error The array of widget data if the name is valid, WP_Error otherwise.
 212       */
 213  	public function get_widget( $id ) {
 214          foreach ( $this->get_widgets() as $widget ) {
 215              if ( $id === $widget['id'] ) {
 216                  return $widget;
 217              }
 218          }
 219  
 220          return new WP_Error( 'rest_widget_type_invalid', __( 'Invalid widget type.' ), array( 'status' => 404 ) );
 221      }
 222  
 223      /**
 224       * Normalize array of widgets.
 225       *
 226       * @since 5.8.0
 227       *
 228       * @global WP_Widget_Factory $wp_widget_factory
 229       * @global array             $wp_registered_widgets The list of registered widgets.
 230       *
 231       * @return array Array of widgets.
 232       */
 233  	protected function get_widgets() {
 234          global $wp_widget_factory, $wp_registered_widgets;
 235  
 236          $widgets = array();
 237  
 238          foreach ( $wp_registered_widgets as $widget ) {
 239              $parsed_id     = wp_parse_widget_id( $widget['id'] );
 240              $widget_object = $wp_widget_factory->get_widget_object( $parsed_id['id_base'] );
 241  
 242              $widget['id']       = $parsed_id['id_base'];
 243              $widget['is_multi'] = (bool) $widget_object;
 244  
 245              if ( isset( $widget['name'] ) ) {
 246                  $widget['name'] = html_entity_decode( $widget['name'], ENT_QUOTES, get_bloginfo( 'charset' ) );
 247              }
 248  
 249              if ( isset( $widget['description'] ) ) {
 250                  $widget['description'] = html_entity_decode( $widget['description'], ENT_QUOTES, get_bloginfo( 'charset' ) );
 251              }
 252  
 253              unset( $widget['callback'] );
 254  
 255              $classname = '';
 256              foreach ( (array) $widget['classname'] as $cn ) {
 257                  if ( is_string( $cn ) ) {
 258                      $classname .= '_' . $cn;
 259                  } elseif ( is_object( $cn ) ) {
 260                      $classname .= '_' . get_class( $cn );
 261                  }
 262              }
 263              $widget['classname'] = ltrim( $classname, '_' );
 264  
 265              $widgets[ $widget['id'] ] = $widget;
 266          }
 267  
 268          ksort( $widgets );
 269  
 270          return $widgets;
 271      }
 272  
 273      /**
 274       * Retrieves a single widget type from the collection.
 275       *
 276       * @since 5.8.0
 277       *
 278       * @param WP_REST_Request $request Full details about the request.
 279       * @return WP_REST_Response|WP_Error Response object on success, or WP_Error object on failure.
 280       */
 281  	public function get_item( $request ) {
 282          $widget_id   = $request['id'];
 283          $widget_type = $this->get_widget( $widget_id );
 284          if ( is_wp_error( $widget_type ) ) {
 285              return $widget_type;
 286          }
 287          $data = $this->prepare_item_for_response( $widget_type, $request );
 288  
 289          return rest_ensure_response( $data );
 290      }
 291  
 292      /**
 293       * Prepares a widget type object for serialization.
 294       *
 295       * @since 5.8.0
 296       * @since 5.9.0 Renamed `$widget_type` to `$item` to match parent class for PHP 8 named parameter support.
 297       *
 298       * @param array           $item    Widget type data.
 299       * @param WP_REST_Request $request Full details about the request.
 300       * @return WP_REST_Response Widget type data.
 301       */
 302  	public function prepare_item_for_response( $item, $request ) {
 303          // Restores the more descriptive, specific name for use within this method.
 304          $widget_type = $item;
 305  
 306          // Don't prepare the response body for HEAD requests.
 307          if ( $request->is_method( 'HEAD' ) ) {
 308              /** This filter is documented in wp-includes/rest-api/endpoints/class-wp-rest-widget-types-controller.php */
 309              return apply_filters( 'rest_prepare_widget_type', new WP_REST_Response( array() ), $widget_type, $request );
 310          }
 311  
 312          $fields = $this->get_fields_for_response( $request );
 313          $data   = array(
 314              'id' => $widget_type['id'],
 315          );
 316  
 317          $schema       = $this->get_item_schema();
 318          $extra_fields = array(
 319              'name',
 320              'description',
 321              'is_multi',
 322              'classname',
 323              'widget_class',
 324              'option_name',
 325              'customize_selective_refresh',
 326          );
 327  
 328          foreach ( $extra_fields as $extra_field ) {
 329              if ( ! rest_is_field_included( $extra_field, $fields ) ) {
 330                  continue;
 331              }
 332  
 333              if ( isset( $widget_type[ $extra_field ] ) ) {
 334                  $field = $widget_type[ $extra_field ];
 335              } elseif ( array_key_exists( 'default', $schema['properties'][ $extra_field ] ) ) {
 336                  $field = $schema['properties'][ $extra_field ]['default'];
 337              } else {
 338                  $field = '';
 339              }
 340  
 341              $data[ $extra_field ] = rest_sanitize_value_from_schema( $field, $schema['properties'][ $extra_field ] );
 342          }
 343  
 344          $context = ! empty( $request['context'] ) ? $request['context'] : 'view';
 345          $data    = $this->add_additional_fields_to_object( $data, $request );
 346          $data    = $this->filter_response_by_context( $data, $context );
 347  
 348          $response = rest_ensure_response( $data );
 349  
 350          if ( rest_is_field_included( '_links', $fields ) || rest_is_field_included( '_embedded', $fields ) ) {
 351              $response->add_links( $this->prepare_links( $widget_type ) );
 352          }
 353  
 354          /**
 355           * Filters the REST API response for a widget type.
 356           *
 357           * @since 5.8.0
 358           *
 359           * @param WP_REST_Response $response    The response object.
 360           * @param array            $widget_type The array of widget data.
 361           * @param WP_REST_Request  $request     The request object.
 362           */
 363          return apply_filters( 'rest_prepare_widget_type', $response, $widget_type, $request );
 364      }
 365  
 366      /**
 367       * Prepares links for the widget type.
 368       *
 369       * @since 5.8.0
 370       *
 371       * @param array $widget_type Widget type data.
 372       * @return array Links for the given widget type.
 373       */
 374  	protected function prepare_links( $widget_type ) {
 375          return array(
 376              'collection' => array(
 377                  'href' => rest_url( sprintf( '%s/%s', $this->namespace, $this->rest_base ) ),
 378              ),
 379              'self'       => array(
 380                  'href' => rest_url( sprintf( '%s/%s/%s', $this->namespace, $this->rest_base, $widget_type['id'] ) ),
 381              ),
 382          );
 383      }
 384  
 385      /**
 386       * Retrieves the widget type's schema, conforming to JSON Schema.
 387       *
 388       * @since 5.8.0
 389       *
 390       * @return array Item schema data.
 391       */
 392  	public function get_item_schema() {
 393          if ( $this->schema ) {
 394              return $this->add_additional_fields_schema( $this->schema );
 395          }
 396  
 397          $schema = array(
 398              '$schema'    => 'http://json-schema.org/draft-04/schema#',
 399              'title'      => 'widget-type',
 400              'type'       => 'object',
 401              'properties' => array(
 402                  'id'          => array(
 403                      'description' => __( 'Unique slug identifying the widget type.' ),
 404                      'type'        => 'string',
 405                      'context'     => array( 'embed', 'view', 'edit' ),
 406                      'readonly'    => true,
 407                  ),
 408                  'name'        => array(
 409                      'description' => __( 'Human-readable name identifying the widget type.' ),
 410                      'type'        => 'string',
 411                      'default'     => '',
 412                      'context'     => array( 'embed', 'view', 'edit' ),
 413                      'readonly'    => true,
 414                  ),
 415                  'description' => array(
 416                      'description' => __( 'Description of the widget.' ),
 417                      'type'        => 'string',
 418                      'default'     => '',
 419                      'context'     => array( 'view', 'edit', 'embed' ),
 420                  ),
 421                  'is_multi'    => array(
 422                      'description' => __( 'Whether the widget supports multiple instances' ),
 423                      'type'        => 'boolean',
 424                      'context'     => array( 'view', 'edit', 'embed' ),
 425                      'readonly'    => true,
 426                  ),
 427                  'classname'   => array(
 428                      'description' => __( 'Class name' ),
 429                      'type'        => 'string',
 430                      'default'     => '',
 431                      'context'     => array( 'embed', 'view', 'edit' ),
 432                      'readonly'    => true,
 433                  ),
 434              ),
 435          );
 436  
 437          $this->schema = $schema;
 438  
 439          return $this->add_additional_fields_schema( $this->schema );
 440      }
 441  
 442      /**
 443       * An RPC-style endpoint which can be used by clients to turn user input in
 444       * a widget admin form into an encoded instance object.
 445       *
 446       * Accepts:
 447       *
 448       * - id:        A widget type ID.
 449       * - instance:  A widget's encoded instance object. Optional.
 450       * - form_data: Form data from submitting a widget's admin form. Optional.
 451       *
 452       * Returns:
 453       * - instance: The encoded instance object after updating the widget with
 454       *             the given form data.
 455       * - form:     The widget's admin form after updating the widget with the
 456       *             given form data.
 457       *
 458       * @since 5.8.0
 459       *
 460       * @global WP_Widget_Factory $wp_widget_factory
 461       *
 462       * @param WP_REST_Request $request Full details about the request.
 463       * @return WP_REST_Response|WP_Error Response object on success, or WP_Error object on failure.
 464       */
 465  	public function encode_form_data( $request ) {
 466          global $wp_widget_factory;
 467  
 468          $id            = $request['id'];
 469          $widget_object = $wp_widget_factory->get_widget_object( $id );
 470  
 471          if ( ! $widget_object ) {
 472              return new WP_Error(
 473                  'rest_invalid_widget',
 474                  __( 'Cannot preview a widget that does not extend WP_Widget.' ),
 475                  array( 'status' => 400 )
 476              );
 477          }
 478  
 479          /*
 480           * Set the widget's number so that the id attributes in the HTML that we
 481           * return are predictable.
 482           */
 483          if ( isset( $request['number'] ) && is_numeric( $request['number'] ) ) {
 484              $widget_object->_set( (int) $request['number'] );
 485          } else {
 486              $widget_object->_set( -1 );
 487          }
 488  
 489          if ( isset( $request['instance']['encoded'], $request['instance']['hash'] ) ) {
 490              $serialized_instance = base64_decode( $request['instance']['encoded'] );
 491              if ( ! hash_equals( wp_hash( $serialized_instance ), $request['instance']['hash'] ) ) {
 492                  return new WP_Error(
 493                      'rest_invalid_widget',
 494                      __( 'The provided instance is malformed.' ),
 495                      array( 'status' => 400 )
 496                  );
 497              }
 498              $instance = unserialize( $serialized_instance );
 499          } else {
 500              $instance = array();
 501          }
 502  
 503          if (
 504              isset( $request['form_data'][ "widget-$id" ] ) &&
 505              is_array( $request['form_data'][ "widget-$id" ] )
 506          ) {
 507              $new_instance = array_values( $request['form_data'][ "widget-$id" ] )[0];
 508              $old_instance = $instance;
 509  
 510              $instance = $widget_object->update( $new_instance, $old_instance );
 511  
 512              /** This filter is documented in wp-includes/class-wp-widget.php */
 513              $instance = apply_filters(
 514                  'widget_update_callback',
 515                  $instance,
 516                  $new_instance,
 517                  $old_instance,
 518                  $widget_object
 519              );
 520          }
 521  
 522          $serialized_instance = serialize( $instance );
 523          $widget_key          = $wp_widget_factory->get_widget_key( $id );
 524  
 525          $response = array(
 526              'form'     => trim(
 527                  $this->get_widget_form(
 528                      $widget_object,
 529                      $instance
 530                  )
 531              ),
 532              'preview'  => trim(
 533                  $this->get_widget_preview(
 534                      $widget_key,
 535                      $instance
 536                  )
 537              ),
 538              'instance' => array(
 539                  'encoded' => base64_encode( $serialized_instance ),
 540                  'hash'    => wp_hash( $serialized_instance ),
 541              ),
 542          );
 543  
 544          if ( ! empty( $widget_object->widget_options['show_instance_in_rest'] ) ) {
 545              // Use new stdClass so that JSON result is {} and not [].
 546              $response['instance']['raw'] = empty( $instance ) ? new stdClass() : $instance;
 547          }
 548  
 549          return rest_ensure_response( $response );
 550      }
 551  
 552      /**
 553       * Returns the output of WP_Widget::widget() when called with the provided
 554       * instance. Used by encode_form_data() to preview a widget.
 555  
 556       * @since 5.8.0
 557       *
 558       * @param string    $widget   The widget's PHP class name (see class-wp-widget.php).
 559       * @param array     $instance Widget instance settings.
 560       * @return string
 561       */
 562  	private function get_widget_preview( $widget, $instance ) {
 563          ob_start();
 564          the_widget( $widget, $instance );
 565          return ob_get_clean();
 566      }
 567  
 568      /**
 569       * Returns the output of WP_Widget::form() when called with the provided
 570       * instance. Used by encode_form_data() to preview a widget's form.
 571       *
 572       * @since 5.8.0
 573       *
 574       * @param WP_Widget $widget_object Widget object to call widget() on.
 575       * @param array     $instance Widget instance settings.
 576       * @return string
 577       */
 578  	private function get_widget_form( $widget_object, $instance ) {
 579          ob_start();
 580  
 581          /** This filter is documented in wp-includes/class-wp-widget.php */
 582          $instance = apply_filters(
 583              'widget_form_callback',
 584              $instance,
 585              $widget_object
 586          );
 587  
 588          if ( false !== $instance ) {
 589              $return = $widget_object->form( $instance );
 590  
 591              /** This filter is documented in wp-includes/class-wp-widget.php */
 592              do_action_ref_array(
 593                  'in_widget_form',
 594                  array( &$widget_object, &$return, $instance )
 595              );
 596          }
 597  
 598          return ob_get_clean();
 599      }
 600  
 601      /**
 602       * Renders a single Legacy Widget and wraps it in a JSON-encodable array.
 603       *
 604       * @since 5.9.0
 605       *
 606       * @param WP_REST_Request $request Full details about the request.
 607       *
 608       * @return array An array with rendered Legacy Widget HTML.
 609       */
 610  	public function render( $request ) {
 611          return array(
 612              'preview' => $this->render_legacy_widget_preview_iframe(
 613                  $request['id'],
 614                  isset( $request['instance'] ) ? $request['instance'] : null
 615              ),
 616          );
 617      }
 618  
 619      /**
 620       * Renders a page containing a preview of the requested Legacy Widget block.
 621       *
 622       * @since 5.9.0
 623       *
 624       * @param string $id_base The id base of the requested widget.
 625       * @param array  $instance The widget instance attributes.
 626       *
 627       * @return string Rendered Legacy Widget block preview.
 628       */
 629  	private function render_legacy_widget_preview_iframe( $id_base, $instance ) {
 630          if ( ! defined( 'IFRAME_REQUEST' ) ) {
 631              define( 'IFRAME_REQUEST', true );
 632          }
 633  
 634          ob_start();
 635          ?>
 636          <!doctype html>
 637          <html <?php language_attributes(); ?>>
 638          <head>
 639              <meta charset="<?php bloginfo( 'charset' ); ?>" />
 640              <meta name="viewport" content="width=device-width, initial-scale=1" />
 641              <link rel="profile" href="https://gmpg.org/xfn/11" />
 642              <?php wp_head(); ?>
 643              <style>
 644                  /* Reset theme styles */
 645                  html, body, #page, #content {
 646                      padding: 0 !important;
 647                      margin: 0 !important;
 648                  }
 649              </style>
 650          </head>
 651          <body <?php body_class(); ?>>
 652          <div id="page" class="site">
 653              <div id="content" class="site-content">
 654                  <?php
 655                  $registry = WP_Block_Type_Registry::get_instance();
 656                  $block    = $registry->get_registered( 'core/legacy-widget' );
 657                  echo $block->render(
 658                      array(
 659                          'idBase'   => $id_base,
 660                          'instance' => $instance,
 661                      )
 662                  );
 663                  ?>
 664              </div><!-- #content -->
 665          </div><!-- #page -->
 666          <?php wp_footer(); ?>
 667          </body>
 668          </html>
 669          <?php
 670          return ob_get_clean();
 671      }
 672  
 673      /**
 674       * Retrieves the query params for collections.
 675       *
 676       * @since 5.8.0
 677       *
 678       * @return array Collection parameters.
 679       */
 680  	public function get_collection_params() {
 681          return array(
 682              'context' => $this->get_context_param( array( 'default' => 'view' ) ),
 683          );
 684      }
 685  }


Generated : Thu Apr 3 08:20:01 2025 Cross-referenced by PHPXref