2023-08-09 14:43:30 +08:00
< ? php
/*
* This file is part of the Symfony package .
*
* ( c ) Fabien Potencier < fabien @ symfony . com >
*
* For the full copyright and license information , please view the LICENSE
* file that was distributed with this source code .
*/
namespace Symfony\Component\Translation ;
use Symfony\Component\Config\Resource\ResourceInterface ;
use Symfony\Component\Translation\Exception\LogicException ;
/**
* @ author Fabien Potencier < fabien @ symfony . com >
*/
2023-08-09 15:01:26 +08:00
class MessageCatalogue implements MessageCatalogueInterface , MetadataAwareInterface
2023-08-09 14:43:30 +08:00
{
private array $messages = [];
private array $metadata = [];
private array $resources = [];
private string $locale ;
2023-08-09 15:01:26 +08:00
private $fallbackCatalogue = null ;
2023-08-09 14:43:30 +08:00
private ? self $parent = null ;
/**
* @ param array $messages An array of messages classified by domain
*/
public function __construct ( string $locale , array $messages = [])
{
$this -> locale = $locale ;
$this -> messages = $messages ;
}
2023-08-09 15:01:26 +08:00
/**
* { @ inheritdoc }
*/
2023-08-09 14:43:30 +08:00
public function getLocale () : string
{
return $this -> locale ;
}
2023-08-09 15:01:26 +08:00
/**
* { @ inheritdoc }
*/
2023-08-09 14:43:30 +08:00
public function getDomains () : array
{
$domains = [];
foreach ( $this -> messages as $domain => $messages ) {
if ( str_ends_with ( $domain , self :: INTL_DOMAIN_SUFFIX )) {
$domain = substr ( $domain , 0 , - \strlen ( self :: INTL_DOMAIN_SUFFIX ));
}
$domains [ $domain ] = $domain ;
}
return array_values ( $domains );
}
2023-08-09 15:01:26 +08:00
/**
* { @ inheritdoc }
*/
2023-08-09 14:43:30 +08:00
public function all ( string $domain = null ) : array
{
if ( null !== $domain ) {
// skip messages merge if intl-icu requested explicitly
if ( str_ends_with ( $domain , self :: INTL_DOMAIN_SUFFIX )) {
return $this -> messages [ $domain ] ? ? [];
}
return ( $this -> messages [ $domain . self :: INTL_DOMAIN_SUFFIX ] ? ? []) + ( $this -> messages [ $domain ] ? ? []);
}
$allMessages = [];
foreach ( $this -> messages as $domain => $messages ) {
if ( str_ends_with ( $domain , self :: INTL_DOMAIN_SUFFIX )) {
$domain = substr ( $domain , 0 , - \strlen ( self :: INTL_DOMAIN_SUFFIX ));
$allMessages [ $domain ] = $messages + ( $allMessages [ $domain ] ? ? []);
} else {
$allMessages [ $domain ] = ( $allMessages [ $domain ] ? ? []) + $messages ;
}
}
return $allMessages ;
}
/**
2023-08-09 15:01:26 +08:00
* { @ inheritdoc }
2023-08-09 14:43:30 +08:00
*/
public function set ( string $id , string $translation , string $domain = 'messages' )
{
$this -> add ([ $id => $translation ], $domain );
}
2023-08-09 15:01:26 +08:00
/**
* { @ inheritdoc }
*/
2023-08-09 14:43:30 +08:00
public function has ( string $id , string $domain = 'messages' ) : bool
{
if ( isset ( $this -> messages [ $domain ][ $id ]) || isset ( $this -> messages [ $domain . self :: INTL_DOMAIN_SUFFIX ][ $id ])) {
return true ;
}
if ( null !== $this -> fallbackCatalogue ) {
return $this -> fallbackCatalogue -> has ( $id , $domain );
}
return false ;
}
2023-08-09 15:01:26 +08:00
/**
* { @ inheritdoc }
*/
2023-08-09 14:43:30 +08:00
public function defines ( string $id , string $domain = 'messages' ) : bool
{
return isset ( $this -> messages [ $domain ][ $id ]) || isset ( $this -> messages [ $domain . self :: INTL_DOMAIN_SUFFIX ][ $id ]);
}
2023-08-09 15:01:26 +08:00
/**
* { @ inheritdoc }
*/
2023-08-09 14:43:30 +08:00
public function get ( string $id , string $domain = 'messages' ) : string
{
if ( isset ( $this -> messages [ $domain . self :: INTL_DOMAIN_SUFFIX ][ $id ])) {
return $this -> messages [ $domain . self :: INTL_DOMAIN_SUFFIX ][ $id ];
}
if ( isset ( $this -> messages [ $domain ][ $id ])) {
return $this -> messages [ $domain ][ $id ];
}
if ( null !== $this -> fallbackCatalogue ) {
return $this -> fallbackCatalogue -> get ( $id , $domain );
}
return $id ;
}
/**
2023-08-09 15:01:26 +08:00
* { @ inheritdoc }
2023-08-09 14:43:30 +08:00
*/
public function replace ( array $messages , string $domain = 'messages' )
{
unset ( $this -> messages [ $domain ], $this -> messages [ $domain . self :: INTL_DOMAIN_SUFFIX ]);
$this -> add ( $messages , $domain );
}
/**
2023-08-09 15:01:26 +08:00
* { @ inheritdoc }
2023-08-09 14:43:30 +08:00
*/
public function add ( array $messages , string $domain = 'messages' )
{
$altDomain = str_ends_with ( $domain , self :: INTL_DOMAIN_SUFFIX ) ? substr ( $domain , 0 , - \strlen ( self :: INTL_DOMAIN_SUFFIX )) : $domain . self :: INTL_DOMAIN_SUFFIX ;
foreach ( $messages as $id => $message ) {
unset ( $this -> messages [ $altDomain ][ $id ]);
$this -> messages [ $domain ][ $id ] = $message ;
}
if ([] === ( $this -> messages [ $altDomain ] ? ? null )) {
unset ( $this -> messages [ $altDomain ]);
}
}
/**
2023-08-09 15:01:26 +08:00
* { @ inheritdoc }
2023-08-09 14:43:30 +08:00
*/
public function addCatalogue ( MessageCatalogueInterface $catalogue )
{
if ( $catalogue -> getLocale () !== $this -> locale ) {
throw new LogicException ( sprintf ( 'Cannot add a catalogue for locale "%s" as the current locale for this catalogue is "%s".' , $catalogue -> getLocale (), $this -> locale ));
}
foreach ( $catalogue -> all () as $domain => $messages ) {
if ( $intlMessages = $catalogue -> all ( $domain . self :: INTL_DOMAIN_SUFFIX )) {
$this -> add ( $intlMessages , $domain . self :: INTL_DOMAIN_SUFFIX );
$messages = array_diff_key ( $messages , $intlMessages );
}
$this -> add ( $messages , $domain );
}
foreach ( $catalogue -> getResources () as $resource ) {
$this -> addResource ( $resource );
}
if ( $catalogue instanceof MetadataAwareInterface ) {
$metadata = $catalogue -> getMetadata ( '' , '' );
$this -> addMetadata ( $metadata );
}
}
/**
2023-08-09 15:01:26 +08:00
* { @ inheritdoc }
2023-08-09 14:43:30 +08:00
*/
public function addFallbackCatalogue ( MessageCatalogueInterface $catalogue )
{
// detect circular references
$c = $catalogue ;
while ( $c = $c -> getFallbackCatalogue ()) {
if ( $c -> getLocale () === $this -> getLocale ()) {
throw new LogicException ( sprintf ( 'Circular reference detected when adding a fallback catalogue for locale "%s".' , $catalogue -> getLocale ()));
}
}
$c = $this ;
do {
if ( $c -> getLocale () === $catalogue -> getLocale ()) {
throw new LogicException ( sprintf ( 'Circular reference detected when adding a fallback catalogue for locale "%s".' , $catalogue -> getLocale ()));
}
foreach ( $catalogue -> getResources () as $resource ) {
$c -> addResource ( $resource );
}
} while ( $c = $c -> parent );
$catalogue -> parent = $this ;
$this -> fallbackCatalogue = $catalogue ;
foreach ( $catalogue -> getResources () as $resource ) {
$this -> addResource ( $resource );
}
}
2023-08-09 15:01:26 +08:00
/**
* { @ inheritdoc }
*/
2023-08-09 14:43:30 +08:00
public function getFallbackCatalogue () : ? MessageCatalogueInterface
{
return $this -> fallbackCatalogue ;
}
2023-08-09 15:01:26 +08:00
/**
* { @ inheritdoc }
*/
2023-08-09 14:43:30 +08:00
public function getResources () : array
{
return array_values ( $this -> resources );
}
/**
2023-08-09 15:01:26 +08:00
* { @ inheritdoc }
2023-08-09 14:43:30 +08:00
*/
public function addResource ( ResourceInterface $resource )
{
$this -> resources [ $resource -> __toString ()] = $resource ;
}
2023-08-09 15:01:26 +08:00
/**
* { @ inheritdoc }
*/
2023-08-09 14:43:30 +08:00
public function getMetadata ( string $key = '' , string $domain = 'messages' ) : mixed
{
if ( '' == $domain ) {
return $this -> metadata ;
}
if ( isset ( $this -> metadata [ $domain ])) {
if ( '' == $key ) {
return $this -> metadata [ $domain ];
}
if ( isset ( $this -> metadata [ $domain ][ $key ])) {
return $this -> metadata [ $domain ][ $key ];
}
}
return null ;
}
/**
2023-08-09 15:01:26 +08:00
* { @ inheritdoc }
2023-08-09 14:43:30 +08:00
*/
public function setMetadata ( string $key , mixed $value , string $domain = 'messages' )
{
$this -> metadata [ $domain ][ $key ] = $value ;
}
/**
2023-08-09 15:01:26 +08:00
* { @ inheritdoc }
2023-08-09 14:43:30 +08:00
*/
public function deleteMetadata ( string $key = '' , string $domain = 'messages' )
{
if ( '' == $domain ) {
$this -> metadata = [];
} elseif ( '' == $key ) {
unset ( $this -> metadata [ $domain ]);
} else {
unset ( $this -> metadata [ $domain ][ $key ]);
}
}
/**
* Adds current values with the new values .
*
* @ param array $values Values to add
*/
2023-08-09 15:01:26 +08:00
private function addMetadata ( array $values )
2023-08-09 14:43:30 +08:00
{
foreach ( $values as $domain => $keys ) {
foreach ( $keys as $key => $value ) {
$this -> setMetadata ( $key , $value , $domain );
}
}
}
}