spip_nursit/ecrire/inc/traduire.php

353 lines
11 KiB
PHP
Raw Permalink Normal View History

2023-06-01 15:30:12 +00:00
<?php
/***************************************************************************\
* SPIP, Systeme de publication pour l'internet *
* *
* Copyright (c) 2001-2019 *
* Arnaud Martin, Antoine Pitrou, Philippe Riviere, Emmanuel Saint-James *
* *
* Ce programme est un logiciel libre distribue sous licence GNU/GPL. *
* Pour plus de details voir le fichier COPYING.txt ou l'aide en ligne. *
\***************************************************************************/
/**
* Outils pour la traduction et recherche de traductions
*
* @package SPIP\Core\Traductions
**/
if (!defined('_ECRIRE_INC_VERSION')) {
return;
}
/**
* Rechercher tous les lang/file dans le path
* qui seront ensuite chargés dans l'ordre du path
*
* Version dédiée et optimisée pour cet usage de find_in_path
*
* @see find_in_path()
*
* @staticvar array $dirs
*
* @param string $file
* Nom du fichier cherché, tel que `mots_fr.php`
* @param string $dirname
* Nom du répertoire de recherche
* @return array
* Liste des fichiers de langue trouvés, dans l'ordre des chemins
*/
function find_langs_in_path($file, $dirname = 'lang') {
static $dirs = array();
$liste = array();
foreach (creer_chemin() as $dir) {
if (!isset($dirs[$a = $dir . $dirname])) {
$dirs[$a] = (is_dir($a) || !$a);
}
if ($dirs[$a]) {
if (is_readable($a .= $file)) {
$liste[] = $a;
}
}
}
return array_reverse($liste);
}
/**
* Recherche le ou les fichiers de langue d'un module de langue
*
* @param string $module
* Nom du module de langue, tel que `mots` ou `ecrire`
* @param string $lang
* Langue dont on veut obtenir les traductions.
* Paramètre optionnel uniquement si le module est `local`
* @return array
* Liste des fichiers touvés pour ce module et cette langue.
**/
function chercher_module_lang($module, $lang = '') {
if ($lang) {
$lang = '_' . $lang;
}
// 1) dans un repertoire nomme lang/ se trouvant sur le chemin
if ($f = ($module == 'local'
? find_in_path($module . $lang . '.php', 'lang/')
: find_langs_in_path($module . $lang . '.php', 'lang/'))
) {
return is_array($f) ? $f : array($f);
}
// 2) directement dans le chemin (old style, uniquement pour local)
return (($module == 'local') or strpos($module, '/'))
? (($f = find_in_path($module . $lang . '.php')) ? array($f) : false)
: false;
}
/**
* Charge en mémoire les couples cle/traduction d'un module de langue
* et une langue donnée
*
* Interprête un fichier de langue pour le module et la langue désignée
* s'il existe, et sinon se rabat soit sur la langue principale du site
* (définie par la meta `langue_site`), soit sur le français.
*
* Définit la globale `idx_lang` qui sert à la lecture du fichier de langue
* (include) et aux surcharges via `surcharger_langue()`
*
* @uses chercher_module_lang()
* @uses surcharger_langue()
*
* @param string $lang Code de langue
* @param string $module Nom du module de langue
* @return string Langue du module chargé, sinon chaîne vide.
**/
function charger_langue($lang, $module = 'spip') {
$var = 'i18n_' . $module . '_' . $lang;
if ($lang and $fichiers_lang = chercher_module_lang($module, $lang)) {
$GLOBALS['idx_lang'] = $var;
include(array_shift($fichiers_lang));
surcharger_langue($fichiers_lang);
$GLOBALS['lang_' . $var] = $lang;
} else {
// si le fichier de langue du module n'existe pas, on se rabat sur
// la langue par defaut du site -- et au pire sur le francais, qui
// *par definition* doit exister, et on copie le tableau dans la
// var liee a la langue
$l = $GLOBALS['meta']['langue_site'];
if (!$fichiers_lang = chercher_module_lang($module, $l)) {
$l = _LANGUE_PAR_DEFAUT;
$fichiers_lang = chercher_module_lang($module, $l);
}
if ($fichiers_lang) {
$GLOBALS['idx_lang'] = 'i18n_' . $module . '_' . $l;
include(array_shift($fichiers_lang));
surcharger_langue($fichiers_lang);
$GLOBALS[$var] = &$GLOBALS['i18n_' . $module . '_' . $l];
$GLOBALS['lang_' . $var] = $l;
#spip_log("module de langue : ${module}_$l.php");
}
}
}
/**
* Surcharger le fichier de langue courant avec un ou plusieurs autres
*
* Charge chaque fichier de langue dont les chemins sont transmis et
* surcharge les infos de cette langue/module déjà connues par les nouvelles
* données chargées. Seule les clés nouvelles ou modifiées par la
* surcharge sont impactées (les clés non présentes dans la surcharge
* ne sont pas supprimées !).
*
* La fonction suppose la présence de la globale `idx_lang` indiquant
* la destination des couples de traduction, de la forme
* `i18n_${module}_${lang}`
*
* @param array $fichiers
* Liste des chemins de fichiers de langue à surcharger.
**/
function surcharger_langue($fichiers) {
static $surcharges = array();
if (!isset($GLOBALS['idx_lang'])) {
return;
}
if (!is_array($fichiers)) {
$fichiers = array($fichiers);
}
if (!count($fichiers)) {
return;
}
foreach ($fichiers as $fichier) {
if (!isset($surcharges[$fichier])) {
$idx_lang_normal = $GLOBALS['idx_lang'];
$GLOBALS['idx_lang'] = $GLOBALS['idx_lang'] . '@temporaire';
include($fichier);
$surcharges[$fichier] = $GLOBALS[$GLOBALS['idx_lang']];
unset($GLOBALS[$GLOBALS['idx_lang']]);
$GLOBALS['idx_lang'] = $idx_lang_normal;
}
if (is_array($surcharges[$fichier])) {
$GLOBALS[$GLOBALS['idx_lang']] = array_merge(
(isset($GLOBALS[$GLOBALS['idx_lang']]) ? (array)$GLOBALS[$GLOBALS['idx_lang']] : array()),
$surcharges[$fichier]
);
}
}
}
class SPIP_Traductions_Description {
/** @var string code de langue (hors module) */
public $code;
/** @var string nom du module de langue */
public $module;
/** @var string langue de la traduction */
public $langue;
/** @var string traduction */
public $texte;
/** @var string var mode particulier appliqué ? */
public $mode;
/** @var bool Corrections des textes appliqué ? */
public $corrections = false;
}
/**
* Traduire une chaine internationalisée
*
* Lorsque la langue demandée n'a pas de traduction pour la clé de langue
* transmise, la fonction cherche alors la traduction dans la langue
* principale du site (défini par la meta `langue_site`), puis, à défaut
* dans la langue française.
*
* Les traductions sont cherchées dans les modules de langue indiqués.
* Par exemple le module `mots` dans la clé `mots:titre_mot`, pour une
* traduction `es` (espagnol) provoquera une recherche dans tous les fichiers
* `lang\mots_es.php`.
*
* Des surcharges locales peuvent être présentes également
* dans les fichiers `lang/local_es.php` ou `lang/local.php`
*
* @note
* Les couples clé/traductions déjà connus sont sauvés en interne
* dans les globales `i18n_${module}_${lang}` tel que `i18n_mots_es`
* et sont également sauvés dans la variable statique `deja_vu`
* de cette fonction.
*
* @uses charger_langue()
* @uses chercher_module_lang()
* @uses surcharger_langue()
*
* @param string $ori
* Clé de traduction, tel que `bouton_enregistrer` ou `mots:titre_mot`
* @param string $lang
* Code de langue, la traduction doit se faire si possible dans cette langue
* @param bool $raw
* - false : retourne le texte (par défaut)
* - true : retourne une description de la chaine de langue (module, texte, langue)
* @return string|array
* - string : Traduction demandée. Chaîne vide si aucune traduction trouvée.
* - SPIP_Traductions_Description : traduction et description (texte, module, langue)
**/
function inc_traduire_dist($ori, $lang, $raw = false) {
static $deja_vu = array();
static $local = array();
if (isset($deja_vu[$lang][$ori]) and (_request('var_mode') != 'traduction')) {
return $raw ? $deja_vu[$lang][$ori] : $deja_vu[$lang][$ori]->texte;
}
// modules demandes explicitement <xxx|yyy|zzz:code> cf MODULES_IDIOMES
if (strpos($ori, ':')) {
list($modules, $code) = explode(':', $ori, 2);
$modules = explode('|', $modules);
$ori_complet = $ori;
} else {
$modules = array('spip', 'ecrire');
$code = $ori;
$ori_complet = implode('|', $modules) . ':' . $ori;
}
$desc = new SPIP_Traductions_Description();
// parcourir tous les modules jusqu'a ce qu'on trouve
foreach ($modules as $module) {
$var = "i18n_" . $module . "_" . $lang;
if (empty($GLOBALS[$var])) {
charger_langue($lang, $module);
// surcharges persos -- on cherche
// (lang/)local_xx.php et/ou (lang/)local.php ...
if (!isset($local['local_' . $lang])) {
// redéfinir la langue en cours pour les surcharges (chercher_langue a pu le changer)
$GLOBALS['idx_lang'] = $var;
// ... (lang/)local_xx.php
$local['local_' . $lang] = chercher_module_lang('local', $lang);
}
if ($local['local_' . $lang]) {
surcharger_langue($local['local_' . $lang]);
}
// ... puis (lang/)local.php
if (!isset($local['local'])) {
$local['local'] = chercher_module_lang('local');
}
if ($local['local']) {
surcharger_langue($local['local']);
}
}
if (isset($GLOBALS[$var][$code])) {
$desc->code = $code;
$desc->module = $module;
$desc->langue = $GLOBALS['lang_' . $var];
$desc->texte = $GLOBALS[$var][$code];
break;
}
}
if (!$desc->corrections) {
$desc->corrections = true;
// Retour aux sources si la chaine est absente dans la langue cible ;
// on essaie d'abord la langue du site, puis a defaut la langue fr
if (!strlen($desc->texte) and $lang !== _LANGUE_PAR_DEFAUT) {
if ($lang !== $GLOBALS['meta']['langue_site']) {
$desc = inc_traduire_dist($ori, $GLOBALS['meta']['langue_site'], true);
} else {
$desc = inc_traduire_dist($ori, _LANGUE_PAR_DEFAUT, true);
}
}
// Supprimer la mention <NEW> ou <MODIF>
if (substr($desc->texte, 0, 1) === '<') {
$desc->texte = str_replace(array('<NEW>', '<MODIF>'), array(), $desc->texte);
}
// Si on n'est pas en utf-8, la chaine peut l'etre...
// le cas echeant on la convertit en entites html &#xxx;
if ((!isset($GLOBALS['meta']['charset']) or $GLOBALS['meta']['charset'] !== 'utf-8')
and preg_match(',[\x7f-\xff],S', $desc->texte)
) {
include_spip('inc/charsets');
$desc->texte = charset2unicode($desc->texte, 'utf-8');
}
}
if (_request('var_mode') == 'traduction') {
$desc = definir_details_traduction($desc, $ori_complet);
} else {
$deja_vu[$lang][$ori] = $desc;
}
return $raw ? $desc : $desc->texte;
}
/**
* Modifie le texte de traduction pour indiquer des éléments
* servant au debug de celles-ci. (pour var_mode=traduction)
*
* @param SPIP_Traductions_Description $desc
* @param string $modules Les modules qui étaient demandés
* @return SPIP_Traductions_Description
*/
function definir_details_traduction($desc, $modules) {
if (!$desc->mode and $desc->texte) {
// ne pas modifier 2 fois l'affichage
$desc->mode = 'traduction';
$classe = 'debug-traduction' . ($desc->module == 'ecrire' ? '-prive' : '');
$desc->texte = '<span '
. 'lang=' . $desc->langue
. ' class=' . $classe
. ' title=' . $modules . '(' . $desc->langue . ')>'
. $desc->texte
. '</span>';
$desc->texte = str_replace(
array("$desc->module:", "$desc->module|"),
array("*$desc->module*:", "*$desc->module*|"),
$desc->texte
);
}
return $desc;
}