Automattic\WooCommerce\Internal\RestApi\Routes\V4\Settings\Email\Schema

EmailSettingsSchema{}WC 1.0└─ AbstractSchema

EmailSettingsSchema class.

No Hooks.

Usage

$EmailSettingsSchema = new EmailSettingsSchema();
// use class methods

Methods

  1. public get_item_response( $item, WP_REST_Request $request, array $include_fields = array() )
  2. public get_item_schema_properties()
  3. private get_field_schema()
  4. private map_setting_type_to_api_type( string $setting_type )

EmailSettingsSchema{} code WC 10.4.3

class EmailSettingsSchema extends AbstractSchema {
	/**
	 * The schema item identifier.
	 *
	 * @var string
	 */
	const IDENTIFIER = 'email_settings';

	/**
	 * List of non-editable field types.
	 *
	 * @var string[]
	 */
	const NON_EDITABLE_TYPES = array( 'title', 'sectionend', 'email_color_palette', 'previewing_new_templates', 'email_improvements_button', 'email_notification', 'email_notification_block_emails', 'hidden' );

	/**
	 * Return all properties for the item schema.
	 *
	 * @return array
	 */
	public function get_item_schema_properties(): array {
		return array(
			'id'          => array(
				'description' => __( 'Unique identifier for the settings group.', 'woocommerce' ),
				'type'        => 'string',
				'context'     => self::VIEW_EDIT_CONTEXT,
				'readonly'    => true,
			),
			'title'       => array(
				'description' => __( 'Settings title.', 'woocommerce' ),
				'type'        => 'string',
				'context'     => self::VIEW_EDIT_CONTEXT,
				'readonly'    => true,
			),
			'description' => array(
				'description' => __( 'Settings description.', 'woocommerce' ),
				'type'        => 'string',
				'context'     => self::VIEW_EDIT_CONTEXT,
				'readonly'    => true,
			),
			'values'      => array(
				'description' => __( 'Flat key-value mapping of all setting field values.', 'woocommerce' ),
				'type'        => 'object',
				'context'     => self::VIEW_EDIT_CONTEXT,
			),
			'groups'      => array(
				'description'          => __( 'Collection of setting groups.', 'woocommerce' ),
				'type'                 => 'object',
				'context'              => self::VIEW_EDIT_CONTEXT,
				'additionalProperties' => array(
					'type'        => 'object',
					'description' => __( 'Settings group.', 'woocommerce' ),
					'properties'  => array(
						'title'       => array(
							'description' => __( 'Group title.', 'woocommerce' ),
							'type'        => 'string',
							'context'     => self::VIEW_EDIT_CONTEXT,
						),
						'description' => array(
							'description' => __( 'Group description.', 'woocommerce' ),
							'type'        => 'string',
							'context'     => self::VIEW_EDIT_CONTEXT,
						),
						'order'       => array(
							'description' => __( 'Display order for the group.', 'woocommerce' ),
							'type'        => 'integer',
							'context'     => self::VIEW_EDIT_CONTEXT,
							'readonly'    => true,
						),
						'fields'      => array(
							'description' => __( 'Settings fields.', 'woocommerce' ),
							'type'        => 'array',
							'context'     => self::VIEW_EDIT_CONTEXT,
							'items'       => $this->get_field_schema(),
						),
					),
				),
			),
		);
	}

	/**
	 * Get the schema for individual setting fields.
	 *
	 * @return array
	 */
	private function get_field_schema(): array {
		return array(
			'type'       => 'object',
			'properties' => array(
				'id'      => array(
					'description' => __( 'Setting field ID.', 'woocommerce' ),
					'type'        => 'string',
					'context'     => self::VIEW_EDIT_CONTEXT,
				),
				'label'   => array(
					'description' => __( 'Setting field label.', 'woocommerce' ),
					'type'        => 'string',
					'context'     => self::VIEW_EDIT_CONTEXT,
				),
				'type'    => array(
					'description' => __( 'Setting field type.', 'woocommerce' ),
					'type'        => 'string',
					'enum'        => array( 'text', 'email', 'checkbox', 'number', 'color', 'select' ),
					'context'     => self::VIEW_EDIT_CONTEXT,
				),
				'desc'    => array(
					'description' => __( 'Setting field description.', 'woocommerce' ),
					'type'        => 'string',
					'context'     => self::VIEW_EDIT_CONTEXT,
				),
				'options' => array(
					'description' => __( 'Available options for selectable fields.', 'woocommerce' ),
					'type'        => 'object',
					'context'     => self::VIEW_EDIT_CONTEXT,
				),
			),
		);
	}

	/**
	 * Get email settings data by transforming email settings into REST API format.
	 *
	 * @param mixed           $item             Settings array from WC_Settings_Emails.
	 * @param WP_REST_Request $request          Request object.
	 * @param array           $include_fields   Fields to include.
	 * @return array
	 */
	public function get_item_response( $item, WP_REST_Request $request, array $include_fields = array() ): array {
		$settings = is_array( $item ) ? $item : array();

		// Transform settings into grouped format based on title/sectionend markers.
		$groups           = array();
		$values           = array();
		$current_group    = null;
		$current_group_id = null;

		foreach ( $settings as $setting ) {
			$setting_type = $setting['type'] ?? '';

			// Handle section titles and email_color_palette - start of a new group.
			if ( 'title' === $setting_type || 'email_color_palette' === $setting_type ) {
				$current_group_id = $setting['id'] ?? '';
				$current_group    = array(
					'title'       => $setting['title'] ?? '',
					'description' => $setting['desc'] ?? '',
					'order'       => isset( $setting['order'] ) ? (int) $setting['order'] : 999,
					'fields'      => array(),
				);
				continue;
			}

			// Handle section ends - save the current group.
			if ( 'sectionend' === $setting_type ) {
				if ( $current_group && $current_group_id ) {
					$groups[ $current_group_id ] = $current_group;
				}
				$current_group    = null;
				$current_group_id = null;
				continue;
			}

			// Skip non-editable field types.
			if ( in_array( $setting_type, self::NON_EDITABLE_TYPES, true ) ) {
				continue;
			}

			// Process field if we have a current group and the setting has an ID.
			if ( isset( $setting['id'] ) && $current_group ) {
				$setting_id   = $setting['id'];
				$setting_type = $setting['type'] ?? 'text';

				// Map WooCommerce field types to REST API types.
				$api_type = $this->map_setting_type_to_api_type( $setting_type );

				// Build field definition.
				$field = array(
					'id'    => $setting_id,
					'label' => $setting['title'] ?? $setting_id,
					'type'  => $api_type,
				);

				// Add description if available.
				if ( ! empty( $setting['desc'] ) ) {
					$field['desc'] = $setting['desc'];
				}

				// Add options if available.
				if ( isset( $setting['options'] ) && is_array( $setting['options'] ) ) {
					$field['options'] = $setting['options'];
				}

				$current_group['fields'][] = $field;

				// Get current value.
				$default_value = $setting['default'] ?? '';
				$current_value = get_option( $setting_id, $default_value );

				// Convert checkbox values to boolean for API.
				if ( 'checkbox' === $setting_type ) {
					$current_value = 'yes' === $current_value;
				}

				$values[ $setting_id ] = $current_value;
			}
		}

		// Filter groups without fields.
		$groups = array_filter(
			$groups,
			function ( $group ) {
				return ! empty( $group['fields'] );
			}
		);

		$response = array(
			'id'          => 'email',
			'title'       => __( 'Email design', 'woocommerce' ),
			'description' => __( 'Customize the look and feel of all you notification emails.', 'woocommerce' ),
			'values'      => $values,
			'groups'      => $groups,
		);

		if ( ! empty( $include_fields ) ) {
			$response = array_intersect_key( $response, array_flip( $include_fields ) );
		}

		return $response;
	}

	/**
	 * Map WooCommerce setting type to REST API type.
	 *
	 * @param string $setting_type WooCommerce setting type.
	 * @return string REST API type.
	 */
	private function map_setting_type_to_api_type( string $setting_type ): string {
		$type_map = array(
			'text'     => 'text',
			'email'    => 'email',
			'checkbox' => 'checkbox',
			'number'   => 'number',
			'color'    => 'color',
			'select'   => 'select',
		);

		return $type_map[ $setting_type ] ?? 'text';
	}
}