*/ private static array $instances = []; private Transformers_Registry $transformers; private function __construct( Transformers_Registry $transformers ) { $this->transformers = $transformers; } public static function for_styles(): self { return self::instance( self::CONTEXT_STYLES ); } public static function for_settings(): self { return self::instance( self::CONTEXT_SETTINGS ); } private static function instance( string $context ): self { if ( ! isset( self::$instances[ $context ] ) ) { $registry = new Transformers_Registry(); do_action( "elementor/atomic-widgets/$context/transformers/register", $registry ); self::$instances[ $context ] = new self( $registry ); } return self::$instances[ $context ]; } public static function reset(): void { self::$instances = []; } public function resolve( array $schema, array $props ): array { $resolved = []; foreach ( $schema as $key => $prop_type ) { if ( ! ( $prop_type instanceof Prop_Type ) ) { continue; } $resolved[ $key ] = $props[ $key ] ?? $prop_type->get_default(); } return $this->assign_values( $resolved, $schema ); } private function transform( $value, $key, Prop_Type $prop_type, int $depth = 0 ) { if ( ! $value || ! $this->is_transformable( $value ) ) { return $value; } if ( $depth >= self::TRANSFORM_DEPTH_LIMIT ) { return null; } if ( isset( $value['disabled'] ) && true === $value['disabled'] ) { return null; } if ( $prop_type instanceof Union_Prop_Type ) { $prop_type = $prop_type->get_prop_type( $value['$$type'] ); if ( ! $prop_type ) { return null; } } if ( $prop_type instanceof Object_Prop_Type ) { if ( ! is_array( $value['value'] ) ) { return null; } $value['value'] = $this->resolve( $prop_type->get_shape(), $value['value'] ); } if ( $prop_type instanceof Array_Prop_Type ) { if ( ! is_array( $value['value'] ) ) { return null; } $value['value'] = $this->assign_values( $value['value'], $prop_type->get_item_type() ); } $transformer = $this->transformers->get( $value['$$type'] ); if ( ! ( $transformer instanceof Transformer_Base ) ) { return null; } try { $transformed_value = $transformer->transform( $value['value'], $key ); return $this->transform( $transformed_value, $key, $prop_type, $depth + 1 ); } catch ( Exception $e ) { return null; } } private function is_transformable( $value ): bool { return ( ! empty( $value['$$type'] ) && array_key_exists( 'value', $value ) ); } private function assign_values( $values, $schema ) { $assigned = []; foreach ( $values as $key => $value ) { $prop_type = $schema instanceof Prop_Type ? $schema : $schema[ $key ]; $transformed = $this->transform( $value, $key, $prop_type ); if ( Multi_Props::is( $transformed ) ) { $assigned = array_merge( $assigned, Multi_Props::get_value( $transformed ) ); continue; } $assigned[ $key ] = $transformed; } return $assigned; } }