Files
2023-09-12 21:41:04 +02:00

475 lines
13 KiB
PHP

<?php
use WPML\Collect\Support\Collection;
use WPML\FP\Either;
use \WPML\FP\Obj;
use function WPML\Container\make;
use function \WPML\FP\partial;
use function \WPML\FP\invoke;
use function \WPML\FP\flip;
class WPML_Admin_Texts extends WPML_Admin_Text_Functionality {
const DOMAIN_NAME_PREFIX = 'admin_texts_';
/** @var array $cache - A cache for each option translation */
private $cache = [];
/** @var array $option_names - The option names from Admin texts settings */
private $option_names = [];
/** @var TranslationManagement $tm_instance */
private $tm_instance;
/** @var WPML_String_Translation $st_instance */
private $st_instance;
/** @var bool $lock */
private $lock = false;
/**
* @param TranslationManagement $tm_instance
* @param WPML_String_Translation $st_instance
*/
function __construct( &$tm_instance, &$st_instance ) {
add_action( 'plugins_loaded', [ $this, 'icl_st_set_admin_options_filters' ], 10 );
add_filter( 'wpml_unfiltered_admin_string', flip( [ $this, 'get_option_without_filtering' ] ), 10, 2 );
add_action( 'wpml_st_force_translate_admin_options', [ $this, 'force_translate_admin_options' ] );
$this->tm_instance = &$tm_instance;
$this->st_instance = &$st_instance;
}
/**
* @param mixed $value
*
* @return array|mixed|object
*/
private static function object_to_array( $value ) {
return is_object( $value ) ? object_to_array( $value ) : $value;
}
function icl_register_admin_options( $array, $key = '', $option = array() ) {
$option = self::object_to_array( $option );
foreach ( $array as $k => $v ) {
$option = $key === '' ? array( $k => maybe_unserialize( $this->get_option_without_filtering( $k ) ) ) : $option;
if ( is_array( $v ) ) {
$this->icl_register_admin_options( $v, $key . '[' . $k . ']', $option[ $k ] );
} else {
$context = $this->get_context( $key, $k );
if ( $v === '' ) {
icl_unregister_string( $context, $key . $k );
} elseif ( isset( $option[ $k ] ) && ( $key === '' || $opt_keys = self::findKeys( (string) $key ) ) ) {
icl_register_string( $context, $key . $k, $option[ $k ], true );
$vals = array( $k => 1 );
$opt_keys = isset( $opt_keys ) ? array_reverse( $opt_keys ) : [];
foreach ( $opt_keys as $opt ) {
$vals = array( $opt => $vals );
}
update_option(
'_icl_admin_option_names',
array_merge_recursive( (array) get_option( '_icl_admin_option_names' ), $vals ),
'no'
);
$this->option_names = [];
}
}
}
}
public function getModelForRender() {
return $this->getModel( $this->getOptions() );
}
/**
* @param Collection $options
*
* @return Collection
*/
public function getModel( Collection $options ) {
$stringNamesPerContext = $this->getStringNamesPerContext();
$isRegistered = function ( $context, $name ) use ( $stringNamesPerContext ) {
return $stringNamesPerContext->has( $context ) &&
$stringNamesPerContext->get( $context )->contains( $name );
};
$getItems = partial( [ $this, 'getItemModel' ], $isRegistered );
return $options->map( $getItems )
->filter()
->values()
->reduce( [ $this, 'flattenModelItems' ], wpml_collect() );
}
/**
* @param Collection $flattened
* @param array $item
*
* @return Collection
*/
public function flattenModelItems( Collection $flattened, array $item ) {
if ( empty( $item ) ) {
return $flattened;
}
if ( isset( $item['items'] ) ) {
return $flattened->merge(
$item['items']->reduce( [ $this, 'flattenModelItems' ], wpml_collect() )
);
}
return $flattened->push( $item );
}
/**
* @param callable $isRegistered - string -> string -> bool
* @param mixed $value
* @param string $name
* @param string $key
* @param array $stack
*
* @return array
*/
public function getItemModel( callable $isRegistered, $value, $name, $key = '', $stack = [] ) {
$sub_key = $this->getSubKey( $key, $name );
$result = [];
if ( $this->isMultiValue( $value ) ) {
$stack[] = $value;
$getSubItem = function ( $v, $key ) use ( $isRegistered, $sub_key, $stack ) {
return $this->getItemModel( $isRegistered, $v, $key, $sub_key, $stack );
};
$result['items'] = wpml_collect( $value )
->reject( $this->isOnStack( $stack ) )
->map( $getSubItem );
} elseif ( is_string( $value ) || is_numeric( $value ) ) {
$context = $this->get_context( $key, $name );
$stringName = $this->getDBStringName( $key, $name );
$result['value'] = $value;
$result['fixed'] = $this->is_sub_key_fixed( $sub_key );
$result['name'] = $sub_key;
$result['registered'] = $isRegistered( $context, $stringName );
$result['hasTranslations'] = ! $result['fixed'] && $result['registered']
&& icl_st_string_has_translations( $context, $stringName );
}
return $result;
}
private function isOnStack( array $stack ) {
return function ( $item ) use ( $stack ) {
return \wpml_collect( $stack )->first(
function ( $currentItem ) use ( $item ) {
return $currentItem === $item;
}
) !== null;
};
}
private function is_sub_key_fixed( $sub_key ) {
$fixed = false;
if ( $keys = self::findKeys( $sub_key ) ) {
$fixed = true;
$fixed_settings = $this->tm_instance->admin_texts_to_translate;
foreach ( $keys as $m ) {
if ( $fixed = isset( $fixed_settings[ $m ] ) ) {
$fixed_settings = $fixed_settings[ $m ];
} else {
break;
}
}
}
return $fixed;
}
private function get_context( $option_key, $option_name ) {
$keys = self::findKeys( (string) $option_key );
return self::DOMAIN_NAME_PREFIX . ( $keys ? reset( $keys ) : $option_name );
}
public function getOptions() {
return wpml_collect( wp_load_alloptions() )
->reject( flip( [ $this, 'is_blacklisted' ] ) )
->map( 'maybe_unserialize' );
}
function icl_st_set_admin_options_filters() {
$option_names = $this->getOptionNames();
$isAdmin = is_admin() && ! wpml_is_ajax();
foreach ( $option_names as $option_key => $option ) {
if ( $this->is_blacklisted( $option_key ) ) {
unset( $option_names[ $option_key ] );
update_option( '_icl_admin_option_names', $option_names, 'no' );
} elseif ( $option_key != 'theme' && $option_key != 'plugin' ) { // theme and plugin are an obsolete format before 3.2
/**
* We don't want to translate admin strings in admin panel because it causes a lot of confusion
* when a value is displayed inside the form input.
*/
if ( ! $isAdmin ) {
$this->add_filter_for( $option_key );
}
add_action( 'update_option_' . $option_key, array( $this, 'on_update_original_value' ), 10, 3 );
}
}
}
/**
* @param array $options
*/
public function force_translate_admin_options( $options ) {
wpml_collect( $options )->each( [ $this, 'add_filter_for' ] );
}
/**
* @param string $option
*/
public function add_filter_for( $option ) {
add_filter( 'option_' . $option, [ $this, 'icl_st_translate_admin_string' ] );
}
function icl_st_translate_admin_string( $option_value, $key = '', $name = '', $root_level = true ) {
if ( $root_level && $this->lock ) {
return $option_value;
}
$this->lock = true;
$lang = $this->st_instance->get_current_string_language( $name );
$option_name = substr( current_filter(), 7 );
$name = $name === '' ? $option_name : $name;
$blog_id = get_current_blog_id();
if ( isset( $this->cache[ $blog_id ][ $lang ][ $name ] ) ) {
$this->lock = false;
return $this->cache[ $blog_id ][ $lang ][ $name ];
}
$is_serialized = is_serialized( $option_value );
$option_value = $is_serialized ? unserialize( $option_value ) : $option_value;
if ( is_array( $option_value ) || is_object( $option_value ) ) {
$option_value = $this->translate_multiple( $option_value, $key, $name );
} else {
$option_value = $this->translate_single( $option_value, $key, $name, $option_name );
}
$option_value = $is_serialized ? serialize( $option_value ) : $option_value;
if ( $root_level ) {
$this->lock = false;
$this->cache[ $blog_id ][ $lang ][ $name ] = $option_value;
}
return $option_value;
}
/**
* @param string $key - string like '[key1][key2]'
* @param string $name
*
* @return bool
*/
private function isAdminText( $key, $name ) {
return null !== Either::of( $this->getSubKey( $key, $name ) )
->map( [ self::class, 'getKeysParts' ] )
->tryCatch( invoke( 'reduce' )->with( flip( Obj::prop() ), $this->getOptionNames() ) )
->getOrElse( null );
}
/**
* getKeys :: string [key1][key2][name] => Collection [key1, key2, name]
*
* @param string $option
*
* @return Collection
*/
public static function getKeysParts( $option ) {
return wpml_collect( self::findKeys( $option ) );
}
/**
* @param string $string
*
* @return array
*/
private static function findKeys( $string ) {
return array_filter( explode( '][', preg_replace( '/^\[(.*)\]$/', '$1', $string ) ), 'strlen' );
}
function clear_cache_for_option( $option_name ) {
$blog_id = get_current_blog_id();
if ( isset( $this->cache[ $blog_id ] ) ) {
foreach ( $this->cache[ $blog_id ] as $lang_code => &$cache_data ) {
if ( array_key_exists( $option_name, $cache_data ) ) {
unset( $cache_data[ $option_name ] );
}
}
}
}
/**
* @param string|array $old_value
* @param string|array $value
* @param string $option_name
* @param string $name
* @param string $sub_key
*/
public function on_update_original_value( $old_value, $value, $option_name, $name = '', $sub_key = '' ) {
$name = $name ? $name : $option_name;
$value = self::object_to_array( $value );
$old_value = self::object_to_array( $old_value );
if ( is_array( $value ) ) {
foreach ( array_keys( $value ) as $key ) {
$this->on_update_original_value(
isset( $old_value[ $key ] ) ? $old_value[ $key ] : '',
$value[ $key ],
$option_name,
$key,
$this->getSubKey( $sub_key, $name )
);
}
} else {
if ( $this->isAdminText( $sub_key, $name ) ) {
icl_st_update_string_actions( self::DOMAIN_NAME_PREFIX . $option_name, $this->getDBStringName( $sub_key, $name ), $old_value, $value );
}
}
if ( $sub_key === '' ) {
$this->clear_cache_for_option( $option_name );
}
}
public function migrate_original_values() {
$migrate = function ( $option_name ) {
$option_value = maybe_unserialize( $this->get_option_without_filtering( $option_name ) );
$this->on_update_original_value( '', $option_value, $option_name );
};
wpml_collect( $this->getOptionNames() )
->keys()
->filter()
->each( $migrate );
}
/**
* Returns a function to lazy load the migration
*
* @return Closure
*/
public static function get_migrator() {
return function () {
wpml_st_load_admin_texts()->migrate_original_values();
};
}
/**
* @param mixed $option_value
* @param string $key
* @param string $name
*
* @return array|mixed
*/
private function translate_multiple( $option_value, $key, $name ) {
$subKey = $this->getSubKey( $key, $name );
foreach ( $option_value as $k => &$value ) {
$value = $this->icl_st_translate_admin_string(
$value,
$subKey,
$k,
false
);
}
return $option_value;
}
/**
* @param string $option_value
* @param string $key
* @param string $name
* @param string $option_name
*
* @return string
*/
private function translate_single( $option_value, $key, $name, $option_name ) {
if ( $option_value !== '' && $this->isAdminText( $key, $name ) ) {
$option_value = icl_translate( self::DOMAIN_NAME_PREFIX . $option_name, $key . $name, $option_value );
}
return $option_value;
}
/**
* @return array
*/
private function getOptionNames() {
if ( empty( $this->option_names ) ) {
$this->option_names = get_option( '_icl_admin_option_names' );
if ( ! is_array( $this->option_names ) ) {
$this->option_names = [];
}
}
return $this->option_names;
}
/**
* getSubKeys :: string [key1][key2] -> string name => string [key1][key2][name]
*
* @param string $key - [key1][key2]
* @param string $name
*
* @return string
*/
private function getSubKey( $key, $name ) {
return $key . '[' . $name . ']';
}
/**
* getSubKeys :: string [key1][key2] -> string name => string [key1][key2]name
*
* @param string $key
* @param string $name
*
* @return string
*/
private function getDBStringName( $key, $name ) {
return $key . $name;
}
/**
* @return Collection
* @throws \WPML\Auryn\InjectionException
*/
private function getStringNamesPerContext() {
$strings = make( WPML_ST_DB_Mappers_Strings::class )
->get_all_by_context( self::DOMAIN_NAME_PREFIX . '%' );
return wpml_collect( $strings )
->groupBy( 'context' )
->map( invoke( 'pluck' )->with( 'name' ) );
}
/**
* @param mixed $value
*
* @return bool
*/
private function isMultiValue( $value ) {
return is_array( $value ) ||
( is_object( $value ) && '__PHP_Incomplete_Class' !== get_class( $value ) );
}
}