2023-09-05 15:21:01 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This file is part of Zwii.
|
|
|
|
* For full copyright and license information, please see the LICENSE
|
|
|
|
* file that was distributed with this source code.
|
|
|
|
*
|
|
|
|
* @author Rémi Jean <remi.jean@outlook.com>
|
|
|
|
* @copyright Copyright (C) 2008-2018, Rémi Jean
|
|
|
|
* @author Frédéric Tempez <frederic.tempez@outlook.com>
|
|
|
|
* @copyright Copyright (C) 2018-2023, Frédéric Tempez
|
|
|
|
* @license CC Attribution-NonCommercial-NoDerivatives 4.0 International
|
|
|
|
* @link http://zwiicms.fr/
|
|
|
|
*/
|
|
|
|
|
|
|
|
class language extends common
|
|
|
|
{
|
|
|
|
|
|
|
|
// URL langues de l'UI en ligne
|
|
|
|
const ZWII_UI_URL = 'https://forge.chapril.org/ZwiiCMS-Team/zwiicms-translations/raw/branch/master/v13/';
|
|
|
|
|
|
|
|
public static $actions = [
|
|
|
|
'index' => self::GROUP_ADMIN,
|
|
|
|
'copy' => self::GROUP_ADMIN,
|
|
|
|
'add' => self::GROUP_ADMIN,
|
|
|
|
// Ajouter une langue de contenu
|
|
|
|
'edit' => self::GROUP_ADMIN,
|
|
|
|
// Éditer une langue de l'UI
|
2023-09-05 15:31:35 +02:00
|
|
|
'config' => self::GROUP_ADMIN,
|
2023-09-05 15:21:01 +02:00
|
|
|
// Éditer une langue de contenu
|
|
|
|
'delete' => self::GROUP_ADMIN,
|
|
|
|
// Effacer une langue de contenu ou de l'interface
|
|
|
|
'update' => self::GROUP_ADMIN,
|
|
|
|
'default' => self::GROUP_ADMIN
|
|
|
|
];
|
|
|
|
|
|
|
|
const PAGINATION = '20';
|
|
|
|
|
|
|
|
// Language contents
|
|
|
|
public static $translateOptions = [];
|
|
|
|
|
|
|
|
// Page pour la configuration dans la langue
|
|
|
|
public static $pages = '';
|
|
|
|
|
|
|
|
// Liste des langues installées
|
|
|
|
public static $languagesUiInstalled = [];
|
|
|
|
public static $languagesInstalled = [];
|
|
|
|
public static $languagesStore = [];
|
|
|
|
public static $dialogues = [];
|
|
|
|
|
|
|
|
// Liste des langues cibles
|
|
|
|
public static $languagesTarget = [];
|
|
|
|
|
|
|
|
// Activation du bouton de copie
|
|
|
|
public static $siteCopy = true;
|
|
|
|
|
|
|
|
// Localisation en cours d'édition
|
|
|
|
public static $locales = [];
|
|
|
|
|
|
|
|
//UI
|
|
|
|
// Fichiers des langues de l'interface
|
|
|
|
public static $i18nFiles = [];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Met à jour les traduction du site depuis le store
|
|
|
|
*/
|
|
|
|
public function update()
|
|
|
|
{
|
|
|
|
if (
|
|
|
|
$this->getUser('permission', __CLASS__, __FUNCTION__) !== true
|
|
|
|
) {
|
|
|
|
// Valeurs en sortie
|
|
|
|
$this->addOutput([
|
|
|
|
'access' => false
|
|
|
|
]);
|
|
|
|
} else {
|
|
|
|
$lang = $this->getUrl(2);
|
|
|
|
// Action interdite ou URl avec le code langue incorrecte
|
|
|
|
if (
|
|
|
|
array_key_exists($lang, self::$languages) === false
|
|
|
|
) {
|
|
|
|
// Valeurs en sortie
|
|
|
|
$this->addOutput([
|
|
|
|
'redirect' => helper::baseUrl() . 'language',
|
|
|
|
'state' => false,
|
|
|
|
'notification' => helper::translate('Action interdite')
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Télécharger le descripteur en ligne
|
|
|
|
$languageData = json_decode(helper::getUrlContents(self::ZWII_UI_URL . $lang . '.json'), true);
|
|
|
|
$descripteur = json_decode(helper::getUrlContents(self::ZWII_UI_URL . 'language.json'), true);
|
|
|
|
$success = false;
|
|
|
|
if (
|
|
|
|
is_array($languageData) &&
|
|
|
|
is_array($descripteur['language'][$lang])
|
|
|
|
) {
|
|
|
|
if ($this->setData(['language', $lang, $descripteur['language'][$lang]])) {
|
|
|
|
$success = file_put_contents(self::I18N_DIR . $lang . '.json', json_encode($languageData, JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES | JSON_PRETTY_PRINT));
|
|
|
|
$success = is_int($success) ? true : false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Valeurs en sortie
|
|
|
|
$this->addOutput([
|
|
|
|
'redirect' => helper::baseUrl() . 'language',
|
|
|
|
'notification' => $success ? helper::translate('Copie terminée avec succès') : 'Copie terminée avec des erreurs',
|
|
|
|
'state' => $success
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Configuration avancée des langues
|
|
|
|
*/
|
|
|
|
public function copy()
|
|
|
|
{
|
|
|
|
// Soumission du formulaire
|
|
|
|
if (
|
|
|
|
$this->getUser('permission', __CLASS__, __FUNCTION__) === true &&
|
|
|
|
$this->isPost()
|
|
|
|
) {
|
|
|
|
// Initialisation
|
|
|
|
$success = false;
|
|
|
|
$copyFrom = $this->getInput('translateFormCopySource');
|
|
|
|
$toCreate = $this->getInput('translateFormCopyTarget');
|
|
|
|
if ($copyFrom !== $toCreate) {
|
|
|
|
// Création du dossier
|
|
|
|
if (is_dir(self::DATA_DIR . $toCreate) === false) { // Si le dossier est déjà créé
|
|
|
|
$success = mkdir(self::DATA_DIR . $toCreate, 0755);
|
|
|
|
$success = mkdir(self::DATA_DIR . $toCreate . '/content', 0755);
|
|
|
|
} else {
|
|
|
|
$success = true;
|
|
|
|
}
|
|
|
|
// Copier les données par défaut
|
|
|
|
$success = (copy(self::DATA_DIR . $copyFrom . '/locale.json', self::DATA_DIR . $toCreate . '/locale.json') === true && $success === true) ? true : false;
|
|
|
|
$success = (copy(self::DATA_DIR . $copyFrom . '/module.json', self::DATA_DIR . $toCreate . '/module.json') === true && $success === true) ? true : false;
|
|
|
|
$success = (copy(self::DATA_DIR . $copyFrom . '/page.json', self::DATA_DIR . $toCreate . '/page.json') === true && $success === true) ? true : false;
|
|
|
|
$success = ($this->copyDir(self::DATA_DIR . $copyFrom . '/content', self::DATA_DIR . $toCreate . '/content') === true && $success === true) ? true : false;
|
|
|
|
// Enregistrer la langue
|
|
|
|
if ($success) {
|
|
|
|
$notification = sprintf(helper::translate('Données %s copiées vers %s'), self::$languages[$copyFrom], self::$languages[$toCreate]);
|
|
|
|
} else {
|
|
|
|
$notification = helper::translate('Erreur de copie, vérifiez les permissions');
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$success = false;
|
|
|
|
$notification = helper::translate('Les langues sélectionnées sont identiques');
|
|
|
|
}
|
|
|
|
// Valeurs en sortie
|
|
|
|
$this->addOutput([
|
|
|
|
'notification' => $notification,
|
|
|
|
'title' => 'Utilitaire de copie',
|
|
|
|
'view' => 'index',
|
|
|
|
'state' => $success
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Tableau des langues installées
|
|
|
|
foreach (self::$languages as $key => $value) {
|
|
|
|
// tableau des langues installées
|
|
|
|
if (is_dir(self::DATA_DIR . $key)) {
|
|
|
|
self::$languagesTarget[$key] = self::$languages[$key];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Langues cibles fr en plus
|
|
|
|
self::$languagesInstalled = self::$languagesTarget;
|
|
|
|
|
|
|
|
// Valeurs en sortie
|
|
|
|
$this->addOutput([
|
|
|
|
'title' => helper::translate('Copie de contenus localisés'),
|
|
|
|
'view' => 'copy'
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Configuration
|
|
|
|
*/
|
|
|
|
public function index()
|
|
|
|
{
|
|
|
|
|
|
|
|
// Langues attachées à des utilisateurs non effaçables
|
|
|
|
$usersUI = [];
|
|
|
|
$users = $this->getData(['user']);
|
|
|
|
foreach ($users as $key => $value) {
|
|
|
|
array_push($usersUI, $this->getData(['user', $key, 'language']));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Langues installées
|
|
|
|
$installedUI = $this->getData(['language']);
|
|
|
|
|
|
|
|
if (array_key_exists('language', $installedUI)) {
|
|
|
|
$installedUI = $installedUI['language'];
|
|
|
|
}
|
|
|
|
|
|
|
|
// Langues disponibles en ligne
|
|
|
|
$storeUI = json_decode(helper::getUrlContents(self::ZWII_UI_URL . 'language.json'), true);
|
|
|
|
$storeUI = $storeUI['language'];
|
|
|
|
|
|
|
|
// Construction du tableau à partir des langues disponibles dans le store
|
|
|
|
foreach ($installedUI as $file => $value) {
|
|
|
|
// La langue est-elle référencée ?
|
|
|
|
if (array_key_exists(basename($file, '.json'), $installedUI)) {
|
|
|
|
// La langue est déjà installée
|
|
|
|
self::$languagesUiInstalled[$file] = [
|
|
|
|
template::flag($file, '20 %') . ' ' . self::$languages[$file],
|
|
|
|
$value['version'],
|
|
|
|
helper::dateUTF8('%d/%m/%Y', $value['date']),
|
|
|
|
//self::$i18nUI === $file ? helper::translate('Interface') : '',
|
|
|
|
'',
|
|
|
|
/*
|
|
|
|
template::button('translateContentLanguageUIEdit' . $file, [
|
|
|
|
'href' => helper::baseUrl() . $this->getUrl(0) . '/edit/' . $file,
|
|
|
|
'value' => template::ico('pencil'),
|
|
|
|
'help' => 'Éditer',
|
|
|
|
'disabled' => 'fr_FR' === $file
|
|
|
|
]),
|
|
|
|
*/
|
|
|
|
|
|
|
|
template::button('translateContentLanguageUIDownload' . $file, [
|
|
|
|
'class' => version_compare($installedUI[$file]['version'], $storeUI[$file]['version']) < 0 ? 'buttonGreen' : '',
|
|
|
|
'href' => helper::baseUrl() . $this->getUrl(0) . '/update/' . $file,
|
|
|
|
'value' => template::ico('update'),
|
|
|
|
'help' => 'Mettre à jour',
|
|
|
|
]),
|
|
|
|
template::button('translateContentLanguageUIDelete' . $file, [
|
|
|
|
'class' => 'translateDelete buttonRed' . (in_array($file, $usersUI) ? ' disabled' : ''),
|
|
|
|
'href' => helper::baseUrl() . $this->getUrl(0) . '/delete/ui/' . $file,
|
|
|
|
'value' => template::ico('trash'),
|
|
|
|
'help' => 'Supprimer',
|
|
|
|
]),
|
|
|
|
];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Construction du tableau à partir des langues disponibles dans le store
|
|
|
|
foreach ($storeUI as $file => $value) {
|
|
|
|
|
|
|
|
// La langue est-elle installée ?
|
|
|
|
if (array_key_exists($file, $installedUI) === false) {
|
|
|
|
self::$languagesStore[$file] = [
|
|
|
|
template::flag($file, '20 %') . ' ' . self::$languages[$file],
|
|
|
|
$value['version'],
|
|
|
|
helper::dateUTF8('%d/%m/%Y', $value['date']),
|
|
|
|
'',
|
|
|
|
template::button('translateContentLanguageUIDownload' . $file, [
|
|
|
|
'class' => 'buttonGreen',
|
|
|
|
'href' => helper::baseUrl() . $this->getUrl(0) . '/update/' . $file,
|
|
|
|
'value' => template::ico('shopping-basket'),
|
|
|
|
'help' => 'Installer',
|
|
|
|
])
|
|
|
|
];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Valeurs en sortie
|
|
|
|
$this->addOutput([
|
2023-09-05 21:14:04 +02:00
|
|
|
'title' => helper::translate('Langues de l\'interface'),
|
2023-09-05 15:21:01 +02:00
|
|
|
'view' => 'index'
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/***
|
|
|
|
* Ajouter une langue de contenu
|
|
|
|
*/
|
|
|
|
|
|
|
|
public function add()
|
|
|
|
{
|
|
|
|
|
|
|
|
// Soumission du formulaire
|
|
|
|
if (
|
|
|
|
$this->getUser('permission', __CLASS__, __FUNCTION__) === true &&
|
|
|
|
$this->isPost()
|
|
|
|
) {
|
|
|
|
|
|
|
|
$lang = $this->getInput('translateAddContent');
|
|
|
|
|
|
|
|
// Constructeur pour cette langue
|
|
|
|
$this->jsonDB($lang);
|
|
|
|
|
|
|
|
// Création du contenu
|
|
|
|
$this->initData('page', $lang);
|
|
|
|
$this->initData('module', $lang);
|
2023-09-05 15:31:35 +02:00
|
|
|
$this->initData('config', $lang);
|
2023-09-05 15:21:01 +02:00
|
|
|
|
|
|
|
|
|
|
|
// Valeurs en sortie
|
|
|
|
$this->addOutput([
|
|
|
|
'redirect' => helper::baseUrl() . 'language',
|
|
|
|
'notification' => helper::translate('Modifications enregistrées'),
|
|
|
|
'state' => true
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Préparation de l'affichage du formulaire
|
|
|
|
//-----------------------------------------
|
|
|
|
|
|
|
|
// Tableau des langues non installées
|
|
|
|
foreach (self::$languages as $key => $value) {
|
|
|
|
if (!is_dir(self::DATA_DIR . $key))
|
|
|
|
self::$i18nFiles[$key] = $value;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Valeurs en sortie
|
|
|
|
$this->addOutput([
|
|
|
|
'title' => helper::translate('Nouveau contenu localisé'),
|
|
|
|
'view' => 'add'
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Edition de la langue de l'interface
|
|
|
|
*/
|
|
|
|
public function edit()
|
|
|
|
{
|
|
|
|
$lang = $this->getUrl(2);
|
|
|
|
// Action interdite ou URl avec le code langue incorrecte
|
|
|
|
if (
|
|
|
|
array_key_exists($lang, self::$languages) === false
|
|
|
|
) {
|
|
|
|
// Valeurs en sortie
|
|
|
|
$this->addOutput([
|
|
|
|
'redirect' => helper::baseUrl() . 'language',
|
|
|
|
'state' => false,
|
|
|
|
'notification' => helper::translate('Action interdite')
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
// Soumission du formulaire
|
|
|
|
if (
|
|
|
|
$this->getUser('permission', __CLASS__, __FUNCTION__) === true &&
|
|
|
|
$this->isPost()
|
|
|
|
) {
|
|
|
|
|
|
|
|
// Sauvegarder les champs de la langue
|
|
|
|
$data = json_decode(file_get_contents(self::I18N_DIR . $lang . '.json'), true);
|
|
|
|
foreach ($data as $key => $value) {
|
|
|
|
$target = $this->getInput('translateString' . array_search($key, array_keys($data)));
|
|
|
|
if (empty($target) === false) {
|
|
|
|
$data[$key] = $target;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
file_put_contents(self::I18N_DIR . $lang . '.json', json_encode($data, JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES | JSON_PRETTY_PRINT), LOCK_EX);
|
|
|
|
|
|
|
|
// Mettre à jour le descripteur
|
|
|
|
$this->setData([
|
|
|
|
'language',
|
|
|
|
$lang,
|
|
|
|
[
|
|
|
|
'version' => $this->getInput('translateEditVersion'),
|
|
|
|
'date' => $this->getInput('translateEditDate', helper::FILTER_DATETIME),
|
|
|
|
]
|
|
|
|
]);
|
|
|
|
|
|
|
|
// Valeurs en sortie
|
|
|
|
$this->addOutput([
|
|
|
|
'notification' => helper::translate('Modifications enregistrées'),
|
|
|
|
'redirect' => helper::baseUrl() . 'language',
|
|
|
|
'state' => true,
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Construction du formulaire
|
|
|
|
|
|
|
|
// Chargement des dialogue de la langue cible
|
|
|
|
if (!isset($data)) {
|
|
|
|
$data = json_decode(file_get_contents(self::I18N_DIR . $this->getUrl(2) . '.json'), true);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ajout des champs absents selon la langue de référence
|
|
|
|
$dataFr = json_decode(file_get_contents(self::I18N_DIR . 'fr_FR.json'), true);
|
|
|
|
foreach ($dataFr as $key => $value) {
|
|
|
|
if (!array_key_exists($key, $data)) {
|
|
|
|
$data[$key] = '';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
file_put_contents(self::I18N_DIR . $lang . '.json', json_encode($data, JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES | JSON_PRETTY_PRINT), LOCK_EX);
|
|
|
|
|
|
|
|
// Tableau des chaines à traduire dans la langue sélectionnée
|
|
|
|
foreach ($data as $key => $value) {
|
|
|
|
$dialogues[] = ['source' => $key, 'target' => $value];
|
|
|
|
}
|
|
|
|
|
|
|
|
// Pagination
|
|
|
|
$pagination = helper::pagination($dialogues, $this->getUrl(), self::PAGINATION);
|
|
|
|
|
|
|
|
// Liste des pages
|
|
|
|
self::$pages = $pagination['pages'];
|
|
|
|
|
|
|
|
// Articles en fonction de la pagination
|
|
|
|
for ($i = $pagination['first']; $i < $pagination['last']; $i++) {
|
|
|
|
self::$dialogues[$i] = $dialogues[$i];
|
|
|
|
}
|
|
|
|
|
|
|
|
// Valeurs en sortie
|
|
|
|
$this->addOutput([
|
|
|
|
'title' => helper::translate('Éditer les dialogues') . ' ' . template::flag($lang, '20 %'),
|
|
|
|
'view' => 'edit',
|
|
|
|
'vendor' => [
|
|
|
|
'flatpickr',
|
|
|
|
],
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/***
|
|
|
|
* Effacer une langue de contenu
|
|
|
|
*/
|
|
|
|
public function delete()
|
|
|
|
{
|
|
|
|
if (
|
|
|
|
$this->getUser('permission', __CLASS__, __FUNCTION__) !== true
|
|
|
|
) {
|
|
|
|
// Valeurs en sortie
|
|
|
|
$this->addOutput([
|
|
|
|
'access' => false
|
|
|
|
]);
|
|
|
|
} else {
|
|
|
|
// Action interdite ou URl avec le code langue incorrecte
|
|
|
|
$target = $this->getUrl(2);
|
|
|
|
$lang = $this->getUrl(3);
|
|
|
|
if (
|
|
|
|
array_key_exists($lang, self::$languages) === false
|
|
|
|
) {
|
|
|
|
// Valeurs en sortie
|
|
|
|
$this->addOutput([
|
|
|
|
'redirect' => helper::baseUrl() . 'language',
|
|
|
|
'state' => false,
|
|
|
|
'notification' => helper::translate('Action interdite')
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
switch ($target) {
|
2023-09-05 15:31:35 +02:00
|
|
|
case 'config':
|
2023-09-05 15:21:01 +02:00
|
|
|
$success = false;
|
|
|
|
// Effacement d'une site dans une langue
|
|
|
|
if (is_dir(self::DATA_DIR . $lang) === true) {
|
|
|
|
$success = $this->deleteDir(self::DATA_DIR . $lang);
|
|
|
|
}
|
|
|
|
// Valeurs en sortie
|
|
|
|
$this->addOutput([
|
|
|
|
'redirect' => helper::baseUrl() . 'language',
|
|
|
|
'notification' => $success ? helper::translate('Traduction supprimée') : helper::translate('Erreur inconnue'),
|
|
|
|
'state' => $success
|
|
|
|
]);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'ui':
|
|
|
|
$success = false;
|
|
|
|
// Effacement d'une langue de l'interface
|
|
|
|
if (file_exists(self::I18N_DIR . $lang . '.json') === true) {
|
|
|
|
$this->deleteData(['language', $lang]);
|
|
|
|
$success = unlink(self::I18N_DIR . $lang . '.json');
|
|
|
|
}
|
|
|
|
// Valeurs en sortie
|
|
|
|
$this->addOutput([
|
|
|
|
'redirect' => helper::baseUrl() . 'language',
|
|
|
|
'notification' => $success ? helper::translate('Traduction supprimée') : helper::translate('Erreur inconnue'),
|
|
|
|
'state' => $success
|
|
|
|
]);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
# Do nothing
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Modifie la langue du site par défaut
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
public function default()
|
|
|
|
{
|
|
|
|
if (
|
|
|
|
$this->getUser('permission', __CLASS__, __FUNCTION__) !== true
|
|
|
|
) {
|
|
|
|
// Valeurs en sortie
|
|
|
|
$this->addOutput([
|
|
|
|
'access' => false
|
|
|
|
]);
|
|
|
|
} else {
|
|
|
|
// Action interdite ou URl avec le code langue incorrecte
|
|
|
|
$lang = $this->getUrl(2);
|
|
|
|
if (
|
|
|
|
array_key_exists($lang, self::$languages) === false
|
|
|
|
) {
|
|
|
|
// Valeurs en sortie
|
|
|
|
$this->addOutput([
|
|
|
|
'redirect' => helper::baseUrl() . 'language',
|
|
|
|
'state' => false,
|
|
|
|
'notification' => helper::translate('Action interdite')
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach (self::$languages as $key => $value) {
|
|
|
|
if (file_exists(self::DATA_DIR . $key . '/.default')) {
|
|
|
|
unlink(self::DATA_DIR . $key . '/.default');
|
|
|
|
touch(self::DATA_DIR . $lang . '/.default');
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Valeurs en sortie
|
|
|
|
$this->addOutput([
|
|
|
|
'notification' => helper::translate('Modifications enregistrées'),
|
|
|
|
'redirect' => helper::baseUrl() . 'language',
|
|
|
|
'state' => true,
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
}
|