[ Index ]

PHP Cross Reference of WordPress Trunk (Updated Daily)

Search

title

Body

[close]

/wp-includes/rest-api/endpoints/ -> class-wp-rest-global-styles-revisions-controller.php (source)

   1  <?php
   2  /**
   3   * REST API: WP_REST_Global_Styles_Revisions_Controller class
   4   *
   5   * @package WordPress
   6   * @subpackage REST_API
   7   * @since 6.3.0
   8   */
   9  
  10  /**
  11   * Core class used to access global styles revisions via the REST API.
  12   *
  13   * @since 6.3.0
  14   *
  15   * @see WP_REST_Controller
  16   */
  17  class WP_REST_Global_Styles_Revisions_Controller extends WP_REST_Revisions_Controller {
  18      /**
  19       * Parent controller.
  20       *
  21       * @since 6.6.0
  22       * @var WP_REST_Controller
  23       */
  24      private $parent_controller;
  25  
  26      /**
  27       * The base of the parent controller's route.
  28       *
  29       * @since 6.3.0
  30       * @var string
  31       */
  32      protected $parent_base;
  33  
  34      /**
  35       * Parent post type.
  36       *
  37       * @since 6.6.0
  38       * @var string
  39       */
  40      protected $parent_post_type;
  41  
  42      /**
  43       * Constructor.
  44       *
  45       * @since 6.3.0
  46       * @since 6.6.0 Extends class from WP_REST_Revisions_Controller.
  47       *
  48       * @param string $parent_post_type Post type of the parent.
  49       */
  50  	public function __construct( $parent_post_type = 'wp_global_styles' ) {
  51          parent::__construct( $parent_post_type );
  52          $post_type_object  = get_post_type_object( $parent_post_type );
  53          $parent_controller = $post_type_object->get_rest_controller();
  54  
  55          if ( ! $parent_controller ) {
  56              $parent_controller = new WP_REST_Global_Styles_Controller( $parent_post_type );
  57          }
  58  
  59          $this->parent_controller = $parent_controller;
  60          $this->rest_base         = 'revisions';
  61          $this->parent_base       = ! empty( $post_type_object->rest_base ) ? $post_type_object->rest_base : $post_type_object->name;
  62          $this->namespace         = ! empty( $post_type_object->rest_namespace ) ? $post_type_object->rest_namespace : 'wp/v2';
  63      }
  64  
  65      /**
  66       * Registers the controller's routes.
  67       *
  68       * @since 6.3.0
  69       * @since 6.6.0 Added route to fetch individual global styles revisions.
  70       */
  71  	public function register_routes() {
  72          register_rest_route(
  73              $this->namespace,
  74              '/' . $this->parent_base . '/(?P<parent>[\d]+)/' . $this->rest_base,
  75              array(
  76                  'args'   => array(
  77                      'parent' => array(
  78                          'description' => __( 'The ID for the parent of the revision.' ),
  79                          'type'        => 'integer',
  80                      ),
  81                  ),
  82                  array(
  83                      'methods'             => WP_REST_Server::READABLE,
  84                      'callback'            => array( $this, 'get_items' ),
  85                      'permission_callback' => array( $this, 'get_items_permissions_check' ),
  86                      'args'                => $this->get_collection_params(),
  87                  ),
  88                  'schema' => array( $this, 'get_public_item_schema' ),
  89              )
  90          );
  91  
  92          register_rest_route(
  93              $this->namespace,
  94              '/' . $this->parent_base . '/(?P<parent>[\d]+)/' . $this->rest_base . '/(?P<id>[\d]+)',
  95              array(
  96                  'args'   => array(
  97                      'parent' => array(
  98                          'description' => __( 'The ID for the parent of the global styles revision.' ),
  99                          'type'        => 'integer',
 100                      ),
 101                      'id'     => array(
 102                          'description' => __( 'Unique identifier for the global styles revision.' ),
 103                          'type'        => 'integer',
 104                      ),
 105                  ),
 106                  array(
 107                      'methods'             => WP_REST_Server::READABLE,
 108                      'callback'            => array( $this, 'get_item' ),
 109                      'permission_callback' => array( $this, 'get_item_permissions_check' ),
 110                      'args'                => array(
 111                          'context' => $this->get_context_param( array( 'default' => 'view' ) ),
 112                      ),
 113                  ),
 114                  'schema' => array( $this, 'get_public_item_schema' ),
 115              )
 116          );
 117      }
 118  
 119      /**
 120       * Returns decoded JSON from post content string,
 121       * or a 404 if not found.
 122       *
 123       * @since 6.3.0
 124       *
 125       * @param string $raw_json Encoded JSON from global styles custom post content.
 126       * @return Array|WP_Error
 127       */
 128  	protected function get_decoded_global_styles_json( $raw_json ) {
 129          $decoded_json = json_decode( $raw_json, true );
 130  
 131          if ( is_array( $decoded_json ) && isset( $decoded_json['isGlobalStylesUserThemeJSON'] ) && true === $decoded_json['isGlobalStylesUserThemeJSON'] ) {
 132              return $decoded_json;
 133          }
 134  
 135          return new WP_Error(
 136              'rest_global_styles_not_found',
 137              __( 'Cannot find user global styles revisions.' ),
 138              array( 'status' => 404 )
 139          );
 140      }
 141  
 142      /**
 143       * Returns paginated revisions of the given global styles config custom post type.
 144       *
 145       * The bulk of the body is taken from WP_REST_Revisions_Controller->get_items,
 146       * but global styles does not require as many parameters.
 147       *
 148       * @since 6.3.0
 149       *
 150       * @param WP_REST_Request $request The request instance.
 151       * @return WP_REST_Response|WP_Error
 152       */
 153  	public function get_items( $request ) {
 154          $parent = $this->get_parent( $request['parent'] );
 155  
 156          if ( is_wp_error( $parent ) ) {
 157              return $parent;
 158          }
 159  
 160          $global_styles_config = $this->get_decoded_global_styles_json( $parent->post_content );
 161  
 162          if ( is_wp_error( $global_styles_config ) ) {
 163              return $global_styles_config;
 164          }
 165  
 166          $is_head_request = $request->is_method( 'HEAD' );
 167  
 168          if ( wp_revisions_enabled( $parent ) ) {
 169              $registered = $this->get_collection_params();
 170              $query_args = array(
 171                  'post_parent'    => $parent->ID,
 172                  'post_type'      => 'revision',
 173                  'post_status'    => 'inherit',
 174                  'posts_per_page' => -1,
 175                  'orderby'        => 'date ID',
 176                  'order'          => 'DESC',
 177              );
 178  
 179              $parameter_mappings = array(
 180                  'offset'   => 'offset',
 181                  'page'     => 'paged',
 182                  'per_page' => 'posts_per_page',
 183              );
 184  
 185              foreach ( $parameter_mappings as $api_param => $wp_param ) {
 186                  if ( isset( $registered[ $api_param ], $request[ $api_param ] ) ) {
 187                      $query_args[ $wp_param ] = $request[ $api_param ];
 188                  }
 189              }
 190  
 191              if ( $is_head_request ) {
 192                  // Force the 'fields' argument. For HEAD requests, only post IDs are required to calculate pagination.
 193                  $query_args['fields'] = 'ids';
 194                  // Disable priming post meta for HEAD requests to improve performance.
 195                  $query_args['update_post_term_cache'] = false;
 196                  $query_args['update_post_meta_cache'] = false;
 197              }
 198  
 199              $revisions_query = new WP_Query();
 200              $revisions       = $revisions_query->query( $query_args );
 201              $offset          = isset( $query_args['offset'] ) ? (int) $query_args['offset'] : 0;
 202              $page            = isset( $query_args['paged'] ) ? (int) $query_args['paged'] : 0;
 203              $total_revisions = $revisions_query->found_posts;
 204  
 205              if ( $total_revisions < 1 ) {
 206                  // Out-of-bounds, run the query again without LIMIT for total count.
 207                  unset( $query_args['paged'], $query_args['offset'] );
 208                  $count_query = new WP_Query();
 209                  $count_query->query( $query_args );
 210  
 211                  $total_revisions = $count_query->found_posts;
 212              }
 213  
 214              if ( $revisions_query->query_vars['posts_per_page'] > 0 ) {
 215                  $max_pages = (int) ceil( $total_revisions / (int) $revisions_query->query_vars['posts_per_page'] );
 216              } else {
 217                  $max_pages = $total_revisions > 0 ? 1 : 0;
 218              }
 219              if ( $total_revisions > 0 ) {
 220                  if ( $offset >= $total_revisions ) {
 221                      return new WP_Error(
 222                          'rest_revision_invalid_offset_number',
 223                          __( 'The offset number requested is larger than or equal to the number of available revisions.' ),
 224                          array( 'status' => 400 )
 225                      );
 226                  } elseif ( ! $offset && $page > $max_pages ) {
 227                      return new WP_Error(
 228                          'rest_revision_invalid_page_number',
 229                          __( 'The page number requested is larger than the number of pages available.' ),
 230                          array( 'status' => 400 )
 231                      );
 232                  }
 233              }
 234          } else {
 235              $revisions       = array();
 236              $total_revisions = 0;
 237              $max_pages       = 0;
 238              $page            = (int) $request['page'];
 239          }
 240  
 241          if ( ! $is_head_request ) {
 242              $response = array();
 243  
 244              foreach ( $revisions as $revision ) {
 245                  $data       = $this->prepare_item_for_response( $revision, $request );
 246                  $response[] = $this->prepare_response_for_collection( $data );
 247              }
 248  
 249              $response = rest_ensure_response( $response );
 250          } else {
 251              $response = new WP_REST_Response( array() );
 252          }
 253  
 254          $response->header( 'X-WP-Total', (int) $total_revisions );
 255          $response->header( 'X-WP-TotalPages', (int) $max_pages );
 256  
 257          $request_params = $request->get_query_params();
 258          $base_path      = rest_url( sprintf( '%s/%s/%d/%s', $this->namespace, $this->parent_base, $request['parent'], $this->rest_base ) );
 259          $base           = add_query_arg( urlencode_deep( $request_params ), $base_path );
 260  
 261          if ( $page > 1 ) {
 262              $prev_page = $page - 1;
 263  
 264              if ( $prev_page > $max_pages ) {
 265                  $prev_page = $max_pages;
 266              }
 267  
 268              $prev_link = add_query_arg( 'page', $prev_page, $base );
 269              $response->link_header( 'prev', $prev_link );
 270          }
 271          if ( $max_pages > $page ) {
 272              $next_page = $page + 1;
 273              $next_link = add_query_arg( 'page', $next_page, $base );
 274  
 275              $response->link_header( 'next', $next_link );
 276          }
 277  
 278          return $response;
 279      }
 280  
 281      /**
 282       * Prepares the revision for the REST response.
 283       *
 284       * @since 6.3.0
 285       * @since 6.6.0 Added resolved URI links to the response.
 286       *
 287       * @param WP_Post         $post    Post revision object.
 288       * @param WP_REST_Request $request Request object.
 289       * @return WP_REST_Response|WP_Error Response object.
 290       */
 291  	public function prepare_item_for_response( $post, $request ) {
 292          // Don't prepare the response body for HEAD requests.
 293          if ( $request->is_method( 'HEAD' ) ) {
 294              return new WP_REST_Response( array() );
 295          }
 296  
 297          $parent               = $this->get_parent( $request['parent'] );
 298          $global_styles_config = $this->get_decoded_global_styles_json( $post->post_content );
 299  
 300          if ( is_wp_error( $global_styles_config ) ) {
 301              return $global_styles_config;
 302          }
 303  
 304          $fields     = $this->get_fields_for_response( $request );
 305          $data       = array();
 306          $theme_json = null;
 307  
 308          if ( ! empty( $global_styles_config['styles'] ) || ! empty( $global_styles_config['settings'] ) ) {
 309              $theme_json           = new WP_Theme_JSON( $global_styles_config, 'custom' );
 310              $global_styles_config = $theme_json->get_raw_data();
 311              if ( rest_is_field_included( 'settings', $fields ) ) {
 312                  $data['settings'] = ! empty( $global_styles_config['settings'] ) ? $global_styles_config['settings'] : new stdClass();
 313              }
 314              if ( rest_is_field_included( 'styles', $fields ) ) {
 315                  $data['styles'] = ! empty( $global_styles_config['styles'] ) ? $global_styles_config['styles'] : new stdClass();
 316              }
 317          }
 318  
 319          if ( rest_is_field_included( 'author', $fields ) ) {
 320              $data['author'] = (int) $post->post_author;
 321          }
 322  
 323          if ( rest_is_field_included( 'date', $fields ) ) {
 324              $data['date'] = $this->prepare_date_response( $post->post_date_gmt, $post->post_date );
 325          }
 326  
 327          if ( rest_is_field_included( 'date_gmt', $fields ) ) {
 328              $data['date_gmt'] = $this->prepare_date_response( $post->post_date_gmt );
 329          }
 330  
 331          if ( rest_is_field_included( 'id', $fields ) ) {
 332              $data['id'] = (int) $post->ID;
 333          }
 334  
 335          if ( rest_is_field_included( 'modified', $fields ) ) {
 336              $data['modified'] = $this->prepare_date_response( $post->post_modified_gmt, $post->post_modified );
 337          }
 338  
 339          if ( rest_is_field_included( 'modified_gmt', $fields ) ) {
 340              $data['modified_gmt'] = $this->prepare_date_response( $post->post_modified_gmt );
 341          }
 342  
 343          if ( rest_is_field_included( 'parent', $fields ) ) {
 344              $data['parent'] = (int) $parent->ID;
 345          }
 346  
 347          $context             = ! empty( $request['context'] ) ? $request['context'] : 'view';
 348          $data                = $this->add_additional_fields_to_object( $data, $request );
 349          $data                = $this->filter_response_by_context( $data, $context );
 350          $response            = rest_ensure_response( $data );
 351          $resolved_theme_uris = WP_Theme_JSON_Resolver::get_resolved_theme_uris( $theme_json );
 352  
 353          if ( ! empty( $resolved_theme_uris ) ) {
 354              $response->add_links(
 355                  array(
 356                      'https://api.w.org/theme-file' => $resolved_theme_uris,
 357                  )
 358              );
 359          }
 360  
 361          return $response;
 362      }
 363  
 364      /**
 365       * Retrieves the revision's schema, conforming to JSON Schema.
 366       *
 367       * @since 6.3.0
 368       * @since 6.6.0 Merged parent and parent controller schema data.
 369       *
 370       * @return array Item schema data.
 371       */
 372  	public function get_item_schema() {
 373          if ( $this->schema ) {
 374              return $this->add_additional_fields_schema( $this->schema );
 375          }
 376  
 377          $schema               = parent::get_item_schema();
 378          $parent_schema        = $this->parent_controller->get_item_schema();
 379          $schema['properties'] = array_merge( $schema['properties'], $parent_schema['properties'] );
 380  
 381          unset(
 382              $schema['properties']['guid'],
 383              $schema['properties']['slug'],
 384              $schema['properties']['meta'],
 385              $schema['properties']['content'],
 386              $schema['properties']['title']
 387          );
 388  
 389              $this->schema = $schema;
 390  
 391          return $this->add_additional_fields_schema( $this->schema );
 392      }
 393  
 394      /**
 395       * Retrieves the query params for collections.
 396       * Removes params that are not supported by global styles revisions.
 397       *
 398       * @since 6.6.0
 399       *
 400       * @return array Collection parameters.
 401       */
 402  	public function get_collection_params() {
 403          $query_params = parent::get_collection_params();
 404          unset(
 405              $query_params['exclude'],
 406              $query_params['include'],
 407              $query_params['search'],
 408              $query_params['order'],
 409              $query_params['orderby']
 410          );
 411          return $query_params;
 412      }
 413  }


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