privilegedemarque/controllers/admin/AdminTranslationsController.php

2983 lines
106 KiB
PHP
Raw Normal View History

2016-04-14 16:14:31 +02:00
<?php
/*
* 2007-2014 PrestaShop
*
* NOTICE OF LICENSE
*
* This source file is subject to the Open Software License (OSL 3.0)
* that is bundled with this package in the file LICENSE.txt.
* It is also available through the world-wide-web at this URL:
* http://opensource.org/licenses/osl-3.0.php
* If you did not receive a copy of the license and are unable to
* obtain it through the world-wide-web, please send an email
* to license@prestashop.com so we can send you a copy immediately.
*
* DISCLAIMER
*
* Do not edit or add to this file if you wish to upgrade PrestaShop to newer
* versions in the future. If you wish to customize PrestaShop for your
* needs please refer to http://www.prestashop.com for more information.
*
* @author PrestaShop SA <contact@prestashop.com>
* @copyright 2007-2014 PrestaShop SA
* @license http://opensource.org/licenses/osl-3.0.php Open Software License (OSL 3.0)
* International Registered Trademark & Property of PrestaShop SA
*/
class AdminTranslationsControllerCore extends AdminController
{
/** Name of theme by default */
const DEFAULT_THEME_NAME = _PS_DEFAULT_THEME_NAME_;
const TEXTAREA_SIZED = 70;
/** @var string : Link which list all pack of language */
protected $link_lang_pack = 'http://www.prestashop.com/download/lang_packs/get_each_language_pack.php';
/** @var int : number of sentence which can be translated */
protected $total_expression = 0;
/** @var int : number of sentence which aren't translated */
protected $missing_translations = 0;
/** @var array : List of ISO code for all languages */
protected $all_iso_lang = array();
/** @var array */
protected $modules_translations = array();
/** @var array : List of folder which must be ignored */
protected static $ignore_folder = array('.', '..', '.svn', '.git', '.htaccess', 'index.php');
/** @var array : List of theme by translation type : FRONT, BACK, ERRORS... */
protected $translations_informations = array();
/** @var array : List of all languages */
protected $languages;
/** @var array : List of all themes */
protected $themes;
/** @var string : Directory of selected theme */
protected $theme_selected;
/** @var string : Name of translations type */
protected $type_selected;
/** @var Language object : Language for the selected language */
protected $lang_selected;
/** @var boolean : Is true if number of var exceed the suhosin request or post limit */
protected $post_limit_exceed = false;
public function __construct()
{
$this->bootstrap = true;
$this->multishop_context = Shop::CONTEXT_ALL;
$this->table = 'translations';
parent::__construct();
}
/*
* Set the type which is selected
*/
public function setTypeSelected($type_selected)
{
$this->type_selected = $type_selected;
}
/**
* AdminController::initContent() override
* @see AdminController::initContent()
*/
public function initContent()
{
$this->initTabModuleList();
$this->initPageHeaderToolbar();
if (!is_null($this->type_selected))
{
$method_name = 'initForm'.$this->type_selected;
if (method_exists($this, $method_name))
$this->content = $this->initForm($method_name);
else
{
$this->errors[] = sprintf(Tools::displayError('"%s" does not exist.'), $this->type_selected);
$this->content = $this->initMain();
}
}
else
$this->content = $this->initMain();
$this->context->smarty->assign(array(
'content' => $this->content,
'show_page_header_toolbar' => $this->show_page_header_toolbar,
'page_header_toolbar_title' => $this->page_header_toolbar_title,
'page_header_toolbar_btn' => $this->page_header_toolbar_btn));
}
/**
* This function create vars by default and call the good method for generate form
*
* @param $method_name
* @return call the method $this->method_name()
*/
public function initForm($method_name)
{
// Create a title for each translation page
$title = sprintf(
$this->l('%1$s (Language: %2$s, Theme: %3$s)'),
$this->translations_informations[$this->type_selected]['name'],
$this->lang_selected->name,
$this->theme_selected ? $this->theme_selected : $this->l('none')
);
// Set vars for all forms
$this->tpl_view_vars = array(
'lang' => $this->lang_selected->iso_code,
'title' => $title,
'type' => $this->type_selected,
'theme' => $this->theme_selected,
'post_limit_exceeded' => $this->post_limit_exceed,
'url_submit' => self::$currentIndex.'&submitTranslations'.ucfirst($this->type_selected).'=1&token='.$this->token,
'toggle_button' => $this->displayToggleButton(),
'textarea_sized' => AdminTranslationsControllerCore::TEXTAREA_SIZED
);
// Call method initForm for a type
return $this->{$method_name}();
}
/**
* AdminController::initToolbar() override
* @see AdminController::initToolbar()
*/
public function initToolbar()
{
$this->toolbar_btn['save-and-stay'] = array(
'short' => 'SaveAndStay',
'href' => '#',
'desc' => $this->l('Save and stay'),
);
$this->toolbar_btn['save'] = array(
'href' => '#',
'desc' => $this->l('Update translations')
);
$this->toolbar_btn['cancel'] = array(
'href' => self::$currentIndex.'&token='.$this->token,
'desc' => $this->l('Cancel')
);
}
/**
* Generate the Main page
*/
public function initMain()
{
// Block add/update a language
$packs_to_install = array();
$packs_to_update = array();
$token = Tools::getAdminToken('AdminLanguages'.(int)Tab::getIdFromClassName('AdminLanguages').(int)$this->context->employee->id);
$file_name = $this->link_lang_pack.'?version='._PS_VERSION_;
$array_stream_context = @stream_context_create(array('http' => array('method' => 'GET', 'timeout' => 8)));
if ($lang_packs = Tools::file_get_contents($file_name, false, $array_stream_context))
// Notice : for php < 5.2 compatibility, Tools::jsonDecode. The second parameter to true will set us
if ($lang_packs != '' && $lang_packs = Tools::jsonDecode($lang_packs, true))
foreach ($lang_packs as $key => $lang_pack)
{
if (!Language::isInstalled($lang_pack['iso_code']))
$packs_to_install[$key] = $lang_pack;
else
$packs_to_update[$key] = $lang_pack;
}
$this->tpl_view_vars = array(
'theme_default' => self::DEFAULT_THEME_NAME,
'theme_lang_dir' =>_THEME_LANG_DIR_,
'token' => $this->token,
'languages' => $this->languages,
'translations_type' => $this->translations_informations,
'packs_to_install' => $packs_to_install,
'packs_to_update' => $packs_to_update,
'url_submit' => self::$currentIndex.'&token='.$this->token,
'themes' => $this->themes,
'id_theme_current' => $this->context->shop->id_theme,
'url_create_language' => 'index.php?controller=AdminLanguages&addlang&token='.$token,
);
$this->toolbar_scroll = false;
$this->base_tpl_view = 'main.tpl';
$this->content .= $this->renderKpis();
$this->content .= parent::renderView();
return $this->content;
}
/**
* This method merge each arrays of modules translation in the array of modules translations
*/
protected function getModuleTranslations()
{
global $_MODULE;
$name_var = $this->translations_informations[$this->type_selected]['var'];
if (!isset($_MODULE) && !isset($GLOBALS[$name_var]))
$GLOBALS[$name_var] = array();
else if (isset($_MODULE))
if (is_array($GLOBALS[$name_var]) && is_array($_MODULE))
$GLOBALS[$name_var] = array_merge($GLOBALS[$name_var], $_MODULE);
else
$GLOBALS[$name_var] = $_MODULE;
}
/**
* This method is only used by AdminTranslations::submitCopyLang().
*
* It try to create folder in new theme.
*
* When a translation file is copied for a module, its translation key is wrong.
* We have to change the translation key and rewrite the file.
*
* @param string $dest file name
* @return bool
*/
protected function checkDirAndCreate($dest)
{
$bool = true;
// To get only folder path
$path = dirname($dest);
// If folder wasn't already added
// Do not use Tools::file_exists_cache because it changes over time!
if (!file_exists($path))
if (!mkdir($path, 0777, true))
{
$bool &= false;
$this->errors[] = sprintf($this->l('Cannot create the folder "%s". Please check your directory writing permissions.'), $path);
}
return $bool;
}
/**
* Read the Post var and write the translation file.
* This method overwrites the old translation file.
*
* @param bool $override_file : set true if this file is a override
*/
protected function writeTranslationFile($override_file = false)
{
$type = Tools::toCamelCase($this->type_selected, true);
$translation_informations = $this->translations_informations[$this->type_selected];
if ($override_file)
$file_path = $translation_informations['override']['dir'].$translation_informations['override']['file'];
else
$file_path = $translation_informations['dir'].$translation_informations['file'];
if (!file_exists($file_path))
{
if (!file_exists(dirname($file_path)) && !mkdir(dirname($file_path), 0777, true))
throw new PrestaShopException(sprintf(Tools::displayError('Directory "%s" cannot be created'), dirname($file_path)));
elseif (!touch($file_path))
throw new PrestaShopException(sprintf(Tools::displayError('File "%s" cannot be created'), $file_path));
}
$thm_name = str_replace('.', '', Tools::getValue('theme'));
$kpi_key = substr(strtoupper($thm_name.'_'.Tools::getValue('lang')), 0, 16);
if ($fd = fopen($file_path, 'w'))
{
// Get value of button save and stay
$save_and_stay = Tools::isSubmit('submitTranslations'.$type.'AndStay');
// Get language
$lang = strtolower(Tools::getValue('lang'));
// Unset all POST which are not translations
unset(
$_POST['submitTranslations'.$type],
$_POST['submitTranslations'.$type.'AndStay'],
$_POST['lang'],
$_POST['token'],
$_POST['theme'],
$_POST['type']
);
// Get all POST which aren't empty
$to_insert = array();
foreach ($_POST as $key => $value)
if (!empty($value))
$to_insert[$key] = $value;
ConfigurationKPI::updateValue('FRONTOFFICE_TRANSLATIONS_EXPIRE', time());
ConfigurationKPI::updateValue('TRANSLATE_TOTAL_'.$kpi_key, count($_POST));
ConfigurationKPI::updateValue('TRANSLATE_DONE_'.$kpi_key, count($to_insert));
// translations array is ordered by key (easy merge)
ksort($to_insert);
$tab = $translation_informations['var'];
fwrite($fd, "<?php\n\nglobal \$".$tab.";\n\$".$tab." = array();\n");
foreach ($to_insert as $key => $value)
fwrite($fd, '$'.$tab.'[\''.pSQL($key, true).'\'] = \''.pSQL($value, true).'\';'."\n");
fwrite($fd, "\n?>");
fclose($fd);
// Redirect
if ($save_and_stay)
$this->redirect(true);
else
$this->redirect();
}
else
throw new PrestaShopException(sprintf(Tools::displayError('Cannot write this file: "%s"'), $file_path));
}
public function submitCopyLang()
{
if (!($from_lang = Tools::getValue('fromLang')) || !($to_lang = Tools::getValue('toLang')))
$this->errors[] = $this->l('You must select two languages in order to copy data from one to another.');
else if (!($from_theme = Tools::getValue('fromTheme')) || !($to_theme = Tools::getValue('toTheme')))
$this->errors[] = $this->l('You must select two themes in order to copy data from one to another.');
else if (!Language::copyLanguageData(Language::getIdByIso($from_lang), Language::getIdByIso($to_lang)))
$this->errors[] = $this->l('An error occurred while copying data.');
else if ($from_lang == $to_lang && $from_theme == $to_theme)
$this->errors[] = $this->l('There is nothing to copy (same language and theme).');
else
{
$theme_exists = array('from_theme' => false, 'to_theme' => false);
foreach ($this->themes as $theme)
{
if ($theme->directory == $from_theme)
$theme_exists['from_theme'] = true;
if ($theme->directory == $to_theme)
$theme_exists['to_theme'] = true;
}
if ($theme_exists['from_theme'] == false || $theme_exists['to_theme'] == false)
$this->errors[] = $this->l('Theme(s) not found');
}
if (count($this->errors))
return;
$bool = true;
$items = Language::getFilesList($from_lang, $from_theme, $to_lang, $to_theme, false, false, true);
foreach ($items as $source => $dest)
{
if (!$this->checkDirAndCreate($dest))
$this->errors[] = sprintf($this->l('Impossible to create the directory "%s".'), $dest);
elseif (!copy($source, $dest))
$this->errors[] = sprintf($this->l('Impossible to copy "%s" to "%s".'), $source, $dest);
elseif (strpos($dest, 'modules') && basename($source) === $from_lang.'.php' && $bool !== false)
if (!$this->changeModulesKeyTranslation($dest, $from_theme, $to_theme))
$this->errors[] = sprintf($this->l('Impossible to translate "$dest".'), $dest);
}
if (!count($this->errors))
$this->redirect(false, 14);
$this->errors[] = $this->l('A part of the data has been copied but some of the language files could not be found.');
}
/**
* Change the key translation to according it to theme name.
*
* @param string $path
* @param string $theme_from
* @param string $theme_to
* @return boolean
*/
public function changeModulesKeyTranslation($path, $theme_from, $theme_to)
{
$content = file_get_contents($path);
$arr_replace = array();
$bool_flag = true;
if (preg_match_all('#\$_MODULE\[\'([^\']+)\'\]#Ui', $content, $matches))
{
foreach ($matches[1] as $key => $value)
$arr_replace[$value] = str_replace($theme_from, $theme_to, $value);
$content = str_replace(array_keys($arr_replace), array_values($arr_replace), $content);
$bool_flag = (file_put_contents($path, $content) === false) ? false : true;
}
return $bool_flag;
}
public function exportTabs()
{
// Get name tabs by iso code
$tabs = Tab::getTabs($this->lang_selected->id);
// Get name of the default tabs
$tabs_default_lang = Tab::getTabs(1);
$tabs_default = array();
foreach ($tabs_default_lang as $tab)
$tabs_default[$tab['class_name']] = pSQL($tab['name']);
// Create content
$content = "<?php\n\n\$tabs = array();";
if (!empty($tabs))
foreach ($tabs as $tab)
if ($tabs_default[$tab['class_name']] != pSQL($tab['name']))
$content .= "\n\$tabs['".$tab['class_name']."'] = '".pSQL($tab['name'])."';";
$content .= "\n\nreturn \$tabs;";
$dir = _PS_TRANSLATIONS_DIR_.$this->lang_selected->iso_code.DIRECTORY_SEPARATOR;
$path = $dir.'tabs.php';
// Check if tabs.php exists for the selected Iso Code
if (!Tools::file_exists_cache($dir))
if (!mkdir($dir, 0777, true))
throw new PrestaShopException('The file '.$dir.' cannot be created.');
if (!file_put_contents($path, $content))
throw new PrestaShopException('File "'.$path.'" doesn\'t exists and cannot be created in '.$dir);
if (!is_writable($path))
$this->displayWarning(sprintf(Tools::displayError('This file must be writable: %s'), $path));
}
public function submitExportLang()
{
if ($this->lang_selected->iso_code && $this->theme_selected)
{
$this->exportTabs();
$items = array_flip(Language::getFilesList($this->lang_selected->iso_code, $this->theme_selected, false, false, false, false, true));
$file_name = _PS_TRANSLATIONS_DIR_.'/export/'.$this->lang_selected->iso_code.'.gzip';
require_once(_PS_TOOL_DIR_.'tar/Archive_Tar.php');
$gz = new Archive_Tar($file_name, true);
if ($gz->createModify($items, null, _PS_ROOT_DIR_))
{
ob_start();
header('Pragma: public');
header('Expires: 0');
header('Cache-Control: must-revalidate, post-check=0, pre-check=0');
header('Cache-Control: public');
header('Content-Description: File Transfer');
header('Content-type: application/octet-stream');
header('Content-Disposition: attachment; filename="'.$this->lang_selected->iso_code.'.gzip'.'"');
header('Content-Transfer-Encoding: binary');
ob_end_flush();
readfile($file_name);
@unlink($file_name);
exit;
}
$this->errors[] = Tools::displayError('An error occurred while creating archive.');
}
$this->errors[] = Tools::displayError('Please select a language and a theme.');
}
public static function checkAndAddMailsFiles($iso_code, $files_list)
{
if (Language::getIdByIso('en'))
$default_language = 'en';
else
$default_language = Language::getIsoById((int)Configuration::get('PS_LANG_DEFAULT'));
if (!$default_language || !Validate::isLanguageIsoCode($default_language))
return false;
// 1 - Scan mails files
$mails = array();
if (Tools::file_exists_cache(_PS_MAIL_DIR_.$default_language.'/'))
$mails = scandir(_PS_MAIL_DIR_.$default_language.'/');
$mails_new_lang = array();
// Get all email files
foreach ($files_list as $file)
{
if (preg_match('#^mails\/([a-z0-9]+)\/#Ui', $file['filename'], $matches))
{
$slash_pos = strrpos($file['filename'], '/');
$mails_new_lang[] = substr($file['filename'], -(strlen($file['filename']) - $slash_pos - 1));
}
}
// Get the difference
$arr_mails_needed = array_diff($mails, $mails_new_lang);
// Add mails files
foreach ($arr_mails_needed as $mail_to_add)
if (!in_array($mail_to_add, self::$ignore_folder))
@copy(_PS_MAIL_DIR_.$default_language.'/'.$mail_to_add, _PS_MAIL_DIR_.$iso_code.'/'.$mail_to_add);
// 2 - Scan modules files
$modules = scandir(_PS_MODULE_DIR_);
$module_mail_en = array();
$module_mail_iso_code = array();
foreach ($modules as $module)
{
if (!in_array($module, self::$ignore_folder) && Tools::file_exists_cache(_PS_MODULE_DIR_.$module.'/mails/'.$default_language.'/'))
{
$arr_files = scandir(_PS_MODULE_DIR_.$module.'/mails/'.$default_language.'/');
foreach ($arr_files as $file)
{
if (!in_array($file, self::$ignore_folder))
{
if (Tools::file_exists_cache(_PS_MODULE_DIR_.$module.'/mails/'.$default_language.'/'.$file))
$module_mail_en[] = _PS_MODULE_DIR_.$module.'/mails/ISO_CODE/'.$file;
if (Tools::file_exists_cache(_PS_MODULE_DIR_.$module.'/mails/'.$iso_code.'/'.$file))
$module_mail_iso_code[] = _PS_MODULE_DIR_.$module.'/mails/ISO_CODE/'.$file;
}
}
}
}
// Get the difference in this modules
$arr_modules_mails_needed = array_diff($module_mail_en, $module_mail_iso_code);
// Add mails files for this modules
foreach ($arr_modules_mails_needed as $file)
{
$file_en = str_replace('ISO_CODE', $default_language, $file);
$file_iso_code = str_replace('ISO_CODE', $iso_code, $file);
$dir_iso_code = substr($file_iso_code, 0, -(strlen($file_iso_code) - strrpos($file_iso_code, '/') - 1));
if (!file_exists($dir_iso_code))
{
mkdir($dir_iso_code);
file_put_contents($dir_iso_code.'/index.php', Tools::getDefaultIndexContent());
}
if (Tools::file_exists_cache($file_en))
copy($file_en, $file_iso_code);
}
}
/**
* Move theme translations in selected themes
*
* @param array $files
* @param array $themes_selected
*/
public function checkAndAddThemesFiles($files, $themes_selected)
{
foreach ($files as $file)
{
// Check if file is a file theme
if (preg_match('#^themes\/([a-z0-9]+)\/lang\/#Ui', $file['filename'], $matches))
{
$slash_pos = strrpos($file['filename'], '/');
$name_file = substr($file['filename'], -(strlen($file['filename']) - $slash_pos - 1));
$name_default_theme = $matches[1];
$deleted_old_theme = false;
// Get the old file theme
if (file_exists(_PS_THEME_DIR_.'lang/'.$name_file))
$theme_file_old = _PS_THEME_DIR_.'lang/'.$name_file;
else
{
$deleted_old_theme = true;
$theme_file_old = str_replace(self::DEFAULT_THEME_NAME, $name_default_theme, _PS_THEME_DIR_.'lang/'.$name_file);
}
// Move the old file theme in the new folder
foreach ($themes_selected as $theme_name)
if (file_exists($theme_file_old))
copy($theme_file_old, str_replace($name_default_theme, $theme_name, $theme_file_old));
if ($deleted_old_theme)
@unlink($theme_file_old);
}
}
}
/**
* Add new translations tabs by code ISO
*
* @param array $iso_code
* @param array $files
*/
public static function addNewTabs($iso_code, $files)
{
$errors = array();
foreach ($files as $file)
{
// Check if file is a file theme
if (preg_match('#^translations\/'.$iso_code.'\/tabs.php#Ui', $file['filename'], $matches) && Validate::isLanguageIsoCode($iso_code))
{
// Include array width new translations tabs
$_TABS = array();
clearstatcache();
if (file_exists(_PS_ROOT_DIR_.DIRECTORY_SEPARATOR.$file['filename']))
include_once(_PS_ROOT_DIR_.DIRECTORY_SEPARATOR.$file['filename']);
if (count($_TABS))
{
foreach ($_TABS as $class_name => $translations)
{
// Get instance of this tab by class name
$tab = Tab::getInstanceFromClassName($class_name);
//Check if class name exists
if (isset($tab->class_name) && !empty($tab->class_name))
{
$id_lang = Language::getIdByIso($iso_code);
$tab->name[(int)$id_lang] = $translations;
// Do not crash at intall
if (!isset($tab->name[Configuration::get('PS_LANG_DEFAULT')]))
$tab->name[(int)Configuration::get('PS_LANG_DEFAULT')] = $translations;
if (!Validate::isGenericName($tab->name[(int)$id_lang]))
$errors[] = sprintf(Tools::displayError('Tab "%s" is not valid'), $tab->name[(int)$id_lang]);
else
$tab->update();
}
}
}
}
}
return $errors;
}
public static function checkTranslationFile($content)
{
$lines = array_map('trim', explode("\n", $content));
$global = false;
foreach ($lines as $line)
{
// PHP tags
if (in_array($line, array('<?php', '?>', '')))
continue;
// Global variable declaration
if (!$global && preg_match('/^global\s+\$([a-z0-9-_]+)\s*;$/i', $line, $matches))
{
$global = $matches[1];
continue;
}
// Global variable initialization
if ($global != false && preg_match('/^\$'.preg_quote($global, '/').'\s*=\s*array\(\s*\)\s*;$/i', $line))
continue;
// Global variable initialization without declaration
if (!$global && preg_match('/^\$([a-z0-9-_]+)\s*=\s*array\(\s*\)\s*;$/i', $line, $matches))
{
$global = $matches[1];
continue;
}
// Assignation
if (preg_match('/^\$'.preg_quote($global, '/').'\[\''._PS_TRANS_PATTERN_.'\'\]\s*=\s*\''._PS_TRANS_PATTERN_.'\'\s*;$/i', $line))
continue;
// Sometimes the global variable is returned...
if (preg_match('/^return\s+\$'.preg_quote($global, '/').'\s*;$/i', $line, $matches))
continue;
return false;
}
return true;
}
public function submitImportLang()
{
if (!isset($_FILES['file']['tmp_name']) || !$_FILES['file']['tmp_name'])
$this->errors[] = Tools::displayError('No file has been selected.');
else
{
require_once(_PS_TOOL_DIR_.'tar/Archive_Tar.php');
$gz = new Archive_Tar($_FILES['file']['tmp_name'], true);
$filename = $_FILES['file']['name'];
$iso_code = str_replace(array('.tar.gz', '.gzip'), '', $filename);
if (Validate::isLangIsoCode($iso_code))
{
$themes_selected = Tools::getValue('theme', array(self::DEFAULT_THEME_NAME));
$files_list = AdminTranslationsController::filterTranslationFiles($gz->listContent());
$files_paths = AdminTranslationsController::filesListToPaths($files_list);
$uniqid = uniqid();
$sandbox = _PS_CACHE_DIR_.'sandbox'.DIRECTORY_SEPARATOR.$uniqid.DIRECTORY_SEPARATOR;
if ($gz->extractList($files_paths, $sandbox))
{
foreach ($files_list as $file2check)
{
//don't validate index.php, will be overwrite when extract in translation directory
if (pathinfo($file2check['filename'], PATHINFO_BASENAME) == 'index.php')
continue;
if (preg_match('@^[0-9a-z-_/\\\\]+\.php$@i', $file2check['filename']))
{
if (!AdminTranslationsController::checkTranslationFile(file_get_contents($sandbox.$file2check['filename'])))
$this->errors[] = sprintf(Tools::displayError('Validation failed for: %s'), $file2check['filename']);
}
elseif (!preg_match('@^[0-9a-z-_/\\\\]+\.(html|tpl|txt)$@i', $file2check['filename']))
$this->errors[] = sprintf(Tools::displayError('Unidentified file found: %s'), $file2check['filename']);
}
Tools::deleteDirectory($sandbox, true);
}
$i = 0;
$tmp_array = array();
foreach ($files_paths as $files_path)
{
$path = dirname($files_path);
if (is_dir(_PS_TRANSLATIONS_DIR_.'../'.$path) && !is_writable(_PS_TRANSLATIONS_DIR_.'../'.$path) && !in_array($path, $tmp_array))
{
$this->errors[] = (!$i++? Tools::displayError('The archive cannot be extracted.').' ' : '').Tools::displayError('The server does not have permissions for writing.').' '.sprintf(Tools::displayError('Please check rights for %s'), $path);
$tmp_array[] = $path;
}
}
if (count($this->errors))
return false;
if ($error = $gz->extractList($files_paths, _PS_TRANSLATIONS_DIR_.'../'))
{
if (is_object($error) && !empty($error->message))
$this->errors[] = Tools::displayError('The archive cannot be extracted.'). ' '.$error->message;
else
{
foreach ($files_list as $file2check)
if (pathinfo($file2check['filename'], PATHINFO_BASENAME) == 'index.php' && file_put_contents(_PS_TRANSLATIONS_DIR_.'../'.$file2check['filename'], Tools::getDefaultIndexContent()))
continue;
// Clear smarty modules cache
Tools::clearCache();
if (Validate::isLanguageFileName($filename))
{
if (!Language::checkAndAddLanguage($iso_code))
$conf = 20;
else
{
// Reset cache
Language::loadLanguages();
AdminTranslationsController::checkAndAddMailsFiles($iso_code, $files_list);
$this->checkAndAddThemesFiles($files_list, $themes_selected);
$tab_errors = AdminTranslationsController::addNewTabs($iso_code, $files_list);
if (count($tab_errors))
{
$this->errors += $tab_errors;
return false;
}
}
}
$this->redirect(false, (isset($conf) ? $conf : '15'));
}
}
$this->errors[] = Tools::displayError('The archive cannot be extracted.');
}
else
$this->errors[] = sprintf(Tools::displayError('ISO CODE invalid "%1$s" for the following file: "%2$s"'), $iso_code, $filename);
}
}
/**
* Filter the translation files contained in a .gzip pack
* and return only the ones that we want.
*
* Right now the function only needs to check that
* the modules for which we want to add translations
* are present on the shop (installed or not).
*
* $list is the output of Archive_Tar::listContent()
*/
public static function filterTranslationFiles($list)
{
$kept = array();
foreach ($list as $file)
{
if ('index.php' == basename($file['filename']))
continue;
if (preg_match('#^modules/([^/]+)/#', $file['filename'], $m))
{
if (is_dir(_PS_MODULE_DIR_.$m[1]))
$kept[] = $file;
}
else
$kept[] = $file;
}
return $kept;
}
/**
* Turn the list returned by
* AdminTranslationsController::filterTranslationFiles()
* into a list of paths that can be passed to
* Archive_Tar::extractList()
*/
public static function filesListToPaths($list)
{
$paths = array();
foreach ($list as $item)
$paths[] = $item['filename'];
return $paths;
}
public function submitAddLang()
{
$arr_import_lang = explode('|', Tools::getValue('params_import_language')); /* 0 = Language ISO code, 1 = PS version */
if (Validate::isLangIsoCode($arr_import_lang[0]))
{
$array_stream_context = @stream_context_create(array('http' => array('method' => 'GET', 'timeout' => 10)));
$content = Tools::file_get_contents('http://www.prestashop.com/download/lang_packs/gzip/'.$arr_import_lang[1].'/'.Tools::strtolower($arr_import_lang[0]).'.gzip', false, $array_stream_context);
if ($content)
{
$file = _PS_TRANSLATIONS_DIR_.$arr_import_lang[0].'.gzip';
if ((bool)@file_put_contents($file, $content))
{
require_once(_PS_TOOL_DIR_.'/tar/Archive_Tar.php');
$gz = new Archive_Tar($file, true);
$files_list = AdminTranslationsController::filterTranslationFiles($gz->listContent());
if ($error = $gz->extractList(AdminTranslationsController::filesListToPaths($files_list), _PS_TRANSLATIONS_DIR_.'../'))
{
if (is_object($error) && !empty($error->message))
$this->errors[] = Tools::displayError('The archive cannot be extracted.'). ' '.$error->message;
else
{
if (!Language::checkAndAddLanguage($arr_import_lang[0]))
$conf = 20;
else
{
// Reset cache
Language::loadLanguages();
// Clear smarty modules cache
Tools::clearCache();
AdminTranslationsController::checkAndAddMailsFiles($arr_import_lang[0], $files_list);
if ($tab_errors = AdminTranslationsController::addNewTabs($arr_import_lang[0], $files_list))
$this->errors += $tab_errors;
}
if (!unlink($file))
$this->errors[] = sprintf(Tools::displayError('Cannot delete the archive %s.'), $file);
$this->redirect(false, (isset($conf) ? $conf : '15'));
}
}
elseif (!unlink($file))
$this->errors[] = sprintf(Tools::displayError('Cannot delete the archive %s.'), $file);
}
else
$this->errors[] = Tools::displayError('The server does not have permissions for writing.').' '.sprintf(Tools::displayError('Please check rights for %s'), dirname($file));
}
else
$this->errors[] = Tools::displayError('Language not found.');
}
else
$this->errors[] = Tools::displayError('Invalid parameter.');
}
/**
* This method check each file (tpl or php file), get its sentences to translate,
* compare with posted values and write in iso code translation file.
*
* @param string $file_name
* @param array $files
* @param string $theme_name
* @param string $module_name
* @param string|boolean $dir
* @return void
*/
protected function findAndWriteTranslationsIntoFile($file_name, $files, $theme_name, $module_name, $dir = false)
{
// These static vars allow to use file to write just one time.
static $cache_file = array();
static $str_write = '';
static $array_check_duplicate = array();
// Set file_name in static var, this allow to open and wright the file just one time
if (!isset($cache_file[$theme_name.'-'.$file_name]))
{
$str_write = '';
$cache_file[$theme_name.'-'.$file_name] = true;
if (!Tools::file_exists_cache(dirname($file_name)))
mkdir(dirname($file_name), 0777, true);
if (!Tools::file_exists_cache($file_name))
file_put_contents($file_name, '');
if (!is_writable($file_name))
throw new PrestaShopException(sprintf(
Tools::displayError('Cannot write to the theme\'s language file (%s). Please check write permissions.'),
$file_name
));
// this string is initialized one time for a file
$str_write .= "<?php\n\nglobal \$_MODULE;\n\$_MODULE = array();\n";
$array_check_duplicate = array();
}
foreach ($files as $file)
{
if (preg_match('/^(.*)\.(tpl|php)$/', $file) && Tools::file_exists_cache($dir.$file) && !in_array($file, self::$ignore_folder))
{
// Get content for this file
$content = file_get_contents($dir.$file);
// Get file type
$type_file = substr($file, -4) == '.tpl' ? 'tpl' : 'php';
// Parse this content
$matches = $this->userParseFile($content, $this->type_selected, $type_file, $module_name);
// Write each translation on its module file
$template_name = substr(basename($file), 0, -4);
foreach ($matches as $key)
{
if ($theme_name)
{
$post_key = md5(strtolower($module_name).'_'.strtolower($theme_name).'_'.strtolower($template_name).'_'.md5($key));
$pattern = '\'<{'.strtolower($module_name).'}'.strtolower($theme_name).'>'.strtolower($template_name).'_'.md5($key).'\'';
}
else
{
$post_key = md5(strtolower($module_name).'_'.strtolower($template_name).'_'.md5($key));
$pattern = '\'<{'.strtolower($module_name).'}prestashop>'.strtolower($template_name).'_'.md5($key).'\'';
}
if (array_key_exists($post_key, $_POST) && !empty($_POST[$post_key]) && !in_array($pattern, $array_check_duplicate))
{
$array_check_duplicate[] = $pattern;
$str_write .= '$_MODULE['.$pattern.'] = \''.pSQL(str_replace(array("\r\n", "\r", "\n"), ' ', $_POST[$post_key])).'\';'."\n";
$this->total_expression++;
}
}
}
}
if (isset($cache_file[$theme_name.'-'.$file_name]) && $str_write != "<?php\n\nglobal \$_MODULE;\n\$_MODULE = array();\n")
file_put_contents($file_name, $str_write);
}
/**
* Clear the list of module file by type (file or directory)
*
* @param $files : list of files
* @param string $type_clear (file|directory)
* @param string $path
* @return array : list of a good files
*/
public function clearModuleFiles($files, $type_clear = 'file', $path = '')
{
// List of directory which not must be parsed
$arr_exclude = array('img', 'js', 'mails','override');
// List of good extention files
$arr_good_ext = array('.tpl', '.php');
foreach ($files as $key => $file)
{
if ($file{0} === '.' || in_array(substr($file, 0, strrpos($file, '.')), $this->all_iso_lang))
unset($files[$key]);
else if ($type_clear === 'file' && !in_array(substr($file, strrpos($file, '.')), $arr_good_ext))
unset($files[$key]);
else if ($type_clear === 'directory' && (!is_dir($path.$file) || in_array($file, $arr_exclude)))
unset($files[$key]);
}
return $files;
}
/**
* This method get translation for each files of a module,
* compare with global $_MODULES array and fill AdminTranslations::modules_translations array
* With key as English sentences and values as their iso code translations.
*
* @param array $files
* @param string $theme_name
* @param string $module_name
* @param string|boolean $dir
* @param string $iso_code
* @return void
*/
protected function findAndFillTranslations($files, $theme_name, $module_name, $dir = false)
{
$name_var = $this->translations_informations[$this->type_selected]['var'];
// added for compatibility
$GLOBALS[$name_var] = array_change_key_case($GLOBALS[$name_var]);
// Thank to this var similar keys are not duplicate
// in AndminTranslation::modules_translations array
// see below
$array_check_duplicate = array();
foreach ($files as $file)
{
if ((preg_match('/^(.*).tpl$/', $file) || preg_match('/^(.*).php$/', $file)) && Tools::file_exists_cache($file_path = $dir.$file))
{
// Get content for this file
$content = file_get_contents($file_path);
// Module files can now be ignored by adding this string in a file
if (strpos($content, 'IGNORE_THIS_FILE_FOR_TRANSLATION') !== false)
continue;
// Get file type
$type_file = substr($file, -4) == '.tpl' ? 'tpl' : 'php';
// Parse this content
$matches = $this->userParseFile($content, $this->type_selected, $type_file, $module_name);
// Write each translation on its module file
$template_name = substr(basename($file), 0, -4);
foreach ($matches as $key)
{
$md5_key = md5($key);
$module_key = '<{'.Tools::strtolower($module_name).'}'.strtolower($theme_name).'>'.Tools::strtolower($template_name).'_'.$md5_key;
$default_key = '<{'.Tools::strtolower($module_name).'}prestashop>'.Tools::strtolower($template_name).'_'.$md5_key;
// to avoid duplicate entry
if (!in_array($module_key, $array_check_duplicate))
{
$array_check_duplicate[] = $module_key;
if (!isset($this->modules_translations[$theme_name][$module_name][$template_name][$key]['trad']))
$this->total_expression++;
if ($theme_name && array_key_exists($module_key, $GLOBALS[$name_var]))
$this->modules_translations[$theme_name][$module_name][$template_name][$key]['trad'] = html_entity_decode($GLOBALS[$name_var][$module_key], ENT_COMPAT, 'UTF-8');
elseif (array_key_exists($default_key, $GLOBALS[$name_var]))
$this->modules_translations[$theme_name][$module_name][$template_name][$key]['trad'] = html_entity_decode($GLOBALS[$name_var][$default_key], ENT_COMPAT, 'UTF-8');
else
{
$this->modules_translations[$theme_name][$module_name][$template_name][$key]['trad'] = '';
$this->missing_translations++;
}
$this->modules_translations[$theme_name][$module_name][$template_name][$key]['use_sprintf'] = $this->checkIfKeyUseSprintf($key);
}
}
}
}
}
/**
* Get list of files which must be parsed by directory and by type of translations
*
* @return array : list of files by directory
*/
public function getFileToParseByTypeTranslation()
{
$directories = array();
switch ($this->type_selected)
{
case 'front':
$directories['tpl'] = array(_PS_ALL_THEMES_DIR_ => scandir(_PS_ALL_THEMES_DIR_));
self::$ignore_folder[] = 'modules';
$directories['tpl'] = array_merge($directories['tpl'], $this->listFiles(_PS_THEME_SELECTED_DIR_));
if (isset($directories['tpl'][_PS_THEME_SELECTED_DIR_.'pdf/']))
unset($directories['tpl'][_PS_THEME_SELECTED_DIR_.'pdf/']);
if (Tools::file_exists_cache(_PS_THEME_OVERRIDE_DIR_))
$directories['tpl'] = array_merge($directories['tpl'], $this->listFiles(_PS_THEME_OVERRIDE_DIR_));
break;
case 'back':
$directories = array(
'php' => array(
_PS_ADMIN_CONTROLLER_DIR_ => scandir(_PS_ADMIN_CONTROLLER_DIR_),
_PS_OVERRIDE_DIR_.'controllers/admin/' => scandir(_PS_OVERRIDE_DIR_.'controllers/admin/'),
_PS_CLASS_DIR_.'helper/' => scandir(_PS_CLASS_DIR_.'helper/'),
_PS_CLASS_DIR_.'controller/' => array('AdminController.php'),
_PS_CLASS_DIR_ => array('PaymentModule.php')
),
'tpl' => $this->listFiles(_PS_ADMIN_DIR_.DIRECTORY_SEPARATOR.'themes/'),
'specific' => array(
_PS_ADMIN_DIR_.DIRECTORY_SEPARATOR => array(
'header.inc.php',
'footer.inc.php',
'index.php',
'functions.php'
)
)
);
// For translate the template which are overridden
if (file_exists(_PS_OVERRIDE_DIR_.'controllers'.DIRECTORY_SEPARATOR.'admin'.DIRECTORY_SEPARATOR.'templates'))
$directories['tpl'] = array_merge($directories['tpl'], $this->listFiles(_PS_OVERRIDE_DIR_.'controllers'.DIRECTORY_SEPARATOR.'admin'.DIRECTORY_SEPARATOR.'templates'));
break;
case 'errors':
$directories['php'] = array(
_PS_ROOT_DIR_ => scandir(_PS_ROOT_DIR_),
_PS_ADMIN_DIR_.DIRECTORY_SEPARATOR => scandir(_PS_ADMIN_DIR_.DIRECTORY_SEPARATOR),
_PS_FRONT_CONTROLLER_DIR_ => scandir(_PS_FRONT_CONTROLLER_DIR_),
_PS_ADMIN_CONTROLLER_DIR_ => scandir(_PS_ADMIN_CONTROLLER_DIR_),
_PS_OVERRIDE_DIR_.'controllers/front/' => scandir(_PS_OVERRIDE_DIR_.'controllers/front/'),
_PS_OVERRIDE_DIR_.'controllers/admin/' => scandir(_PS_OVERRIDE_DIR_.'controllers/admin/')
);
// Get all files for folders classes/ and override/classes/ recursively
$directories['php'] = array_merge($directories['php'], $this->listFiles(_PS_CLASS_DIR_, array(), 'php'));
$directories['php'] = array_merge($directories['php'], $this->listFiles(_PS_OVERRIDE_DIR_.'classes/', array(), 'php'));
break;
case 'fields':
$directories['php'] = $this->listFiles(_PS_CLASS_DIR_, array(), 'php');
break;
case 'pdf':
$tpl_theme = Tools::file_exists_cache(_PS_THEME_SELECTED_DIR_.'pdf/') ? scandir(_PS_THEME_SELECTED_DIR_.'pdf/') : array();
$directories = array(
'php' => array(
_PS_CLASS_DIR_.'pdf/' => scandir(_PS_CLASS_DIR_.'pdf/'),
_PS_OVERRIDE_DIR_.'classes/pdf/' => scandir(_PS_OVERRIDE_DIR_.'classes/pdf/')
),
'tpl' => array(
_PS_PDF_DIR_ => scandir(_PS_PDF_DIR_),
_PS_THEME_SELECTED_DIR_.'pdf/' => $tpl_theme
)
);
break;
case 'mails':
$directories['php'] = array(
_PS_FRONT_CONTROLLER_DIR_ => scandir(_PS_FRONT_CONTROLLER_DIR_),
_PS_ADMIN_CONTROLLER_DIR_ => scandir(_PS_ADMIN_CONTROLLER_DIR_),
_PS_OVERRIDE_DIR_.'controllers/front/' => scandir(_PS_OVERRIDE_DIR_.'controllers/front/'),
_PS_OVERRIDE_DIR_.'controllers/admin/' => scandir(_PS_OVERRIDE_DIR_.'controllers/admin/'),
_PS_ADMIN_DIR_.DIRECTORY_SEPARATOR => scandir(_PS_ADMIN_DIR_.DIRECTORY_SEPARATOR),
_PS_ADMIN_DIR_.DIRECTORY_SEPARATOR.'tabs/' => scandir(_PS_ADMIN_DIR_.DIRECTORY_SEPARATOR.'/tabs')
);
// Get all files for folders classes/ and override/classes/ recursively
$directories['php'] = array_merge($directories['php'], $this->listFiles(_PS_CLASS_DIR_, array(), 'php'));
$directories['php'] = array_merge($directories['php'], $this->listFiles(_PS_OVERRIDE_DIR_.'classes/', array(), 'php'));
$directories['php'] = array_merge($directories['php'], $this->getModulesHasMails());
break;
}
return $directories;
}
/**
* This method parse a file by type of translation and type file
*
* @param $content
* @param $type_translation : front, back, errors, modules...
* @param string|bool $type_file : (tpl|php)
* @param string $module_name : name of the module
* @return return $matches
*/
protected function userParseFile($content, $type_translation, $type_file = false, $module_name = '')
{
switch ($type_translation)
{
case 'front':
// Parsing file in Front office
$regex = '/\{l\s*s=([\'\"])'._PS_TRANS_PATTERN_.'\1(\s*sprintf=.*)?(\s*js=1)?\s*\}/U';
break;
case 'back':
// Parsing file in Back office
if ($type_file == 'php')
$regex = '/this->l\((\')'._PS_TRANS_PATTERN_.'\'[\)|\,]/U';
else if ($type_file == 'specific')
$regex = '/Translate::getAdminTranslation\((\')'._PS_TRANS_PATTERN_.'\'(?:,.*)*\)/U';
else
$regex = '/\{l\s*s\s*=([\'\"])'._PS_TRANS_PATTERN_.'\1(\s*sprintf=.*)?(\s*js=1)?(\s*slashes=1)?.*\}/U';
break;
case 'errors':
// Parsing file for all errors syntax
$regex = '/Tools::displayError\((\')'._PS_TRANS_PATTERN_.'\'(,\s*(.+))?\)/U';
break;
case 'modules':
// Parsing modules file
if ($type_file == 'php')
$regex = '/->l\((\')'._PS_TRANS_PATTERN_.'\'(, ?\'(.+)\')?(, ?(.+))?\)/U';
else
// In tpl file look for something that should contain mod='module_name' according to the documentation
$regex = '/\{l\s*s=([\'\"])'._PS_TRANS_PATTERN_.'\1.*\s+mod=\''.$module_name.'\'.*\}/U';
break;
case 'pdf':
// Parsing PDF file
if ($type_file == 'php')
$regex = '/HTMLTemplate.*::l\((\')'._PS_TRANS_PATTERN_.'\'[\)|\,]/U';
else
$regex = '/\{l\s*s=([\'\"])'._PS_TRANS_PATTERN_.'\1(\s*sprintf=.*)?(\s*js=1)?(\s*pdf=\'true\')?\s*\}/U';
break;
}
if (!is_array($regex))
$regex = array($regex);
$strings = array();
foreach ($regex as $regex_row)
{
$matches = array();
$n = preg_match_all($regex_row, $content, $matches);
for ($i = 0; $i < $n; $i += 1)
{
$quote = $matches[1][$i];
$string = $matches[2][$i];
if ($quote === '"')
{
// Escape single quotes because the core will do it when looking for the translation of this string
$string = str_replace('\'', '\\\'', $string);
// Unescape double quotes
$string = preg_replace('/\\\\+"/', '"', $string);
}
$strings[] = $string;
}
}
return array_unique($strings);
}
/**
* Get all translations informations for all type of translations
*
* array(
* 'type' => array(
* 'name' => string : title for the translation type,
* 'var' => string : name of var for the translation file,
* 'dir' => string : dir of translation file
* 'file' => string : file name of translation file
* )
* )
*/
public function getTranslationsInformations()
{
$this->translations_informations = array(
'front' => array(
'name' => $this->l('Front-office translations'),
'var' => '_LANG',
'dir' => defined('_PS_THEME_SELECTED_DIR_') ? _PS_THEME_SELECTED_DIR_.'lang/' : '',
'file' => $this->lang_selected->iso_code.'.php'
),
'back' => array(
'name' => $this->l('Back-office translations'),
'var' => '_LANGADM',
'dir' => _PS_TRANSLATIONS_DIR_.$this->lang_selected->iso_code.'/',
'file' => 'admin.php'
),
'errors' => array(
'name' => $this->l('Error message translations'),
'var' => '_ERRORS',
'dir' => _PS_TRANSLATIONS_DIR_.$this->lang_selected->iso_code.'/',
'file' => 'errors.php'
),
'fields' => array(
'name' => $this->l('Field name translations'),
'var' => '_FIELDS',
'dir' => _PS_TRANSLATIONS_DIR_.$this->lang_selected->iso_code.'/',
'file' => 'fields.php'
),
'modules' => array(
'name' => $this->l('Installed modules translations'),
'var' => '_MODULES',
'dir' => _PS_MODULE_DIR_,
'file' => ''
),
'pdf' => array(
'name' => $this->l('PDF translations'),
'var' => '_LANGPDF',
'dir' => _PS_TRANSLATIONS_DIR_.$this->lang_selected->iso_code.'/',
'file' => 'pdf.php'
),
'mails' => array(
'name' => $this->l('Email templates translations'),
'var' => '_LANGMAIL',
'dir' => _PS_MAIL_DIR_.$this->lang_selected->iso_code.'/',
'file' => 'lang.php'
)
);
if (defined('_PS_THEME_SELECTED_DIR_'))
{
$this->translations_informations['modules']['override'] = array('dir' => _PS_THEME_SELECTED_DIR_.'modules/', 'file' => '');
$this->translations_informations['pdf']['override'] = array('dir' => _PS_THEME_SELECTED_DIR_.'pdf/lang/', 'file' => $this->lang_selected->iso_code.'.php');
$this->translations_informations['mails']['override'] = array('dir' => _PS_THEME_SELECTED_DIR_.'mails/'.$this->lang_selected->iso_code.'/', 'file' => 'lang.php');
}
}
/**
* Get all informations on : languages, theme and the translation type.
*/
public function getInformations()
{
// Get all Languages
$this->languages = Language::getLanguages(false);
// Get all iso_code of languages
foreach ($this->languages as $language)
$this->all_iso_lang[] = $language['iso_code'];
// Get all themes
$this->themes = Theme::getThemes();
// Get folder name of theme
if (($theme = Tools::getValue('theme')) && !is_array($theme))
{
$theme_exists = $this->theme_exists($theme);
if (!$theme_exists)
throw new PrestaShopException(sprintf(Tools::displayError('Invalid theme "%s"'), Tools::safeOutput($theme)));
$this->theme_selected = Tools::safeOutput($theme);
}
// Set the path of selected theme
if ($this->theme_selected)
define('_PS_THEME_SELECTED_DIR_', _PS_ROOT_DIR_.'/themes/'.$this->theme_selected.'/');
else
define('_PS_THEME_SELECTED_DIR_', '');
// Get type of translation
if (($type = Tools::getValue('type')) && !is_array($type))
$this->type_selected = strtolower(Tools::safeOutput($type));
// Get selected language
if (Tools::getValue('lang') || Tools::getValue('iso_code'))
{
$iso_code = Tools::getValue('lang') ? Tools::getValue('lang') : Tools::getValue('iso_code');
if (!Validate::isLangIsoCode($iso_code) || !in_array($iso_code, $this->all_iso_lang))
throw new PrestaShopException(sprintf(Tools::displayError('Invalid iso code "%s"'), Tools::safeOutput($iso_code)));
$this->lang_selected = new Language((int)Language::getIdByIso($iso_code));
}
else
$this->lang_selected = new Language((int)Language::getIdByIso('en'));
// Get all information for translations
$this->getTranslationsInformations();
}
public function renderKpis()
{
$time = time();
$kpis = array();
/* The data generation is located in AdminStatsControllerCore */
$helper = new HelperKpi();
$helper->id = 'box-languages';
$helper->icon = 'icon-microphone';
$helper->color = 'color1';
$helper->href = $this->context->link->getAdminLink('AdminLanguages');
$helper->title = $this->l('Enabled Languages', null, null, false);
if (ConfigurationKPI::get('ENABLED_LANGUAGES') !== false)
$helper->value = ConfigurationKPI::get('ENABLED_LANGUAGES');
if (ConfigurationKPI::get('ENABLED_LANGUAGES_EXPIRE') < $time)
$helper->source = $this->context->link->getAdminLink('AdminStats').'&ajax=1&action=getKpi&kpi=enabled_languages';
$kpis[] = $helper->generate();
$helper = new HelperKpi();
$helper->id = 'box-country';
$helper->icon = 'icon-home';
$helper->color = 'color2';
$helper->title = $this->l('Main Country', null, null, false);
$helper->subtitle = $this->l('30 Days', null, null, false);
if (ConfigurationKPI::get('MAIN_COUNTRY', $this->context->language->id) !== false)
$helper->value = ConfigurationKPI::get('MAIN_COUNTRY', $this->context->language->id);
if (ConfigurationKPI::get('MAIN_COUNTRY_EXPIRE', $this->context->language->id) < $time)
$helper->source = $this->context->link->getAdminLink('AdminStats').'&ajax=1&action=getKpi&kpi=main_country';
$kpis[] = $helper->generate();
$helper = new HelperKpi();
$helper->id = 'box-translations';
$helper->icon = 'icon-list';
$helper->color = 'color3';
$helper->title = $this->l('Front Office Translations', null, null, false);
if (ConfigurationKPI::get('FRONTOFFICE_TRANSLATIONS') !== false)
$helper->value = ConfigurationKPI::get('FRONTOFFICE_TRANSLATIONS');
if (ConfigurationKPI::get('FRONTOFFICE_TRANSLATIONS_EXPIRE') < $time)
$helper->source = $this->context->link->getAdminLink('AdminStats').'&ajax=1&action=getKpi&kpi=frontoffice_translations';
$kpis[] = $helper->generate();
$helper = new HelperKpiRow();
$helper->kpis = $kpis;
return $helper->generate();
}
/**
* AdminController::postProcess() override
* @see AdminController::postProcess()
*/
public function postProcess()
{
$this->getInformations();
/* PrestaShop demo mode */
if (_PS_MODE_DEMO_)
{
$this->errors[] = Tools::displayError('This functionality has been disabled.');
return;
}
/* PrestaShop demo mode */
try {
if (Tools::isSubmit('submitCopyLang'))
{
if ($this->tabAccess['add'] === '1')
$this->submitCopyLang();
else
$this->errors[] = Tools::displayError('You do not have permission to add this.');
}
elseif (Tools::isSubmit('submitExport'))
{
if ($this->tabAccess['add'] === '1')
$this->submitExportLang();
else
$this->errors[] = Tools::displayError('You do not have permission to add this.');
}
elseif (Tools::isSubmit('submitImport'))
{
if ($this->tabAccess['add'] === '1')
$this->submitImportLang();
else
$this->errors[] = Tools::displayError('You do not have permission to add this.');
}
elseif (Tools::isSubmit('submitAddLanguage'))
{
if ($this->tabAccess['add'] === '1')
$this->submitAddLang();
else
$this->errors[] = Tools::displayError('You do not have permission to add this.');
}
elseif (Tools::isSubmit('submitTranslationsPdf'))
{
if ($this->tabAccess['edit'] === '1')
// Only the PrestaShop team should write the translations into the _PS_TRANSLATIONS_DIR_
if (!$this->theme_selected)
$this->writeTranslationFile();
else
$this->writeTranslationFile(true);
else
$this->errors[] = Tools::displayError('You do not have permission to edit this.');
}
elseif (Tools::isSubmit('submitTranslationsBack') || Tools::isSubmit('submitTranslationsErrors') || Tools::isSubmit('submitTranslationsFields') || Tools::isSubmit('submitTranslationsFront'))
{
if ($this->tabAccess['edit'] === '1')
$this->writeTranslationFile();
else
$this->errors[] = Tools::displayError('You do not have permission to edit this.');
}
elseif (Tools::isSubmit('submitTranslationsMails') || Tools::isSubmit('submitTranslationsMailsAndStay'))
{
if ($this->tabAccess['edit'] === '1')
$this->submitTranslationsMails();
else
$this->errors[] = Tools::displayError('You do not have permission to edit this.');
}
elseif (Tools::isSubmit('submitTranslationsModules'))
{
if ($this->tabAccess['edit'] === '1')
{
// Get list of modules
if ($modules = $this->getListModules())
{
// Get files of all modules
$arr_files = $this->getAllModuleFiles($modules, null, $this->lang_selected->iso_code, true);
// Find and write all translation modules files
foreach ($arr_files as $value)
$this->findAndWriteTranslationsIntoFile($value['file_name'], $value['files'], $value['theme'], $value['module'], $value['dir']);
// Clear modules cache
Tools::clearCache();
// Redirect
if (Tools::getIsset('submitTranslationsModulesAndStay'))
$this->redirect(true);
else
$this->redirect();
}
}
else
$this->errors[] = Tools::displayError('You do not have permission to edit this.');
}
} catch (PrestaShopException $e) {
$this->errors[] = $e->getMessage();
}
}
/**
* This method redirect in the translation main page or in the translation page
*
* @param bool $save_and_stay : true if the user has clicked on the button "save and stay"
* @param bool $conf : id of confirmation message
*/
protected function redirect($save_and_stay = false, $conf = false)
{
$conf = !$conf ? 4 : $conf;
$url_base = self::$currentIndex.'&token='.$this->token.'&conf='.$conf;
if ($save_and_stay)
Tools::redirectAdmin($url_base.'&lang='.$this->lang_selected->iso_code.'&type='.$this->type_selected.'&theme='.$this->theme_selected);
else
Tools::redirectAdmin($url_base);
}
protected function getMailPattern()
{
// Let the indentation like it.
return '<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01//EN" "http://www.w3.org/TR/1999/REC-html401-19991224/strict.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8">
<title>#title</title>
</head>
<body>
#content
</body>
</html>';
}
/**
* This method is used to write translation for mails.
* This writes subject translation files
* (in root/mails/lang_choosen/lang.php or root/_PS_THEMES_DIR_/mails/lang_choosen/lang.php)
* and mails files.
*/
protected function submitTranslationsMails()
{
$arr_mail_content = array();
$arr_mail_path = array();
if (Tools::getValue('core_mail'))
{
$arr_mail_content['core_mail'] = Tools::getValue('core_mail');
// Get path of directory for find a good path of translation file
if (!$this->theme_selected)
$arr_mail_path['core_mail'] = $this->translations_informations[$this->type_selected]['dir'];
else
$arr_mail_path['core_mail'] = $this->translations_informations[$this->type_selected]['override']['dir'];
}
if (Tools::getValue('module_mail'))
{
$arr_mail_content['module_mail'] = Tools::getValue('module_mail');
// Get path of directory for find a good path of translation file
if (!$this->theme_selected)
$arr_mail_path['module_mail'] = $this->translations_informations['modules']['dir'].'{module}/mails/'.$this->lang_selected->iso_code.'/';
else
$arr_mail_path['module_mail'] = $this->translations_informations['modules']['override']['dir'].'{module}/mails/'.$this->lang_selected->iso_code.'/';
}
// Save each mail content
foreach ($arr_mail_content as $group_name => $all_content)
{
foreach ($all_content as $type_content => $mails)
{
foreach ($mails as $mail_name => $content)
{
$module_name = false;
$module_name_pipe_pos = stripos($mail_name, '|');
if ($module_name_pipe_pos)
{
$module_name = substr($mail_name, 0, $module_name_pipe_pos);
if (!Validate::isModuleName($module_name))
throw new PrestaShopException(sprintf(Tools::displayError('Invalid module name "%s"'), Tools::safeOutput($module_name)));
$mail_name = substr($mail_name, $module_name_pipe_pos + 1);
if (!Validate::isTplName($mail_name))
throw new PrestaShopException(sprintf(Tools::displayError('Invalid mail name "%s"'), Tools::safeOutput($mail_name)));
}
if ($type_content == 'html')
{
$content = Tools::htmlentitiesUTF8($content);
$content = htmlspecialchars_decode($content);
// replace correct end of line
$content = str_replace("\r\n", PHP_EOL, $content);
$title = '';
if (Tools::getValue('title_'.$group_name.'_'.$mail_name))
$title = Tools::getValue('title_'.$group_name.'_'.$mail_name);
$string_mail = $this->getMailPattern();
$content = str_replace(array('#title', '#content'), array($title, $content), $string_mail);
// Magic Quotes shall... not.. PASS!
if (_PS_MAGIC_QUOTES_GPC_)
$content = stripslashes($content);
}
if (Validate::isCleanHTML($content))
{
$path = $arr_mail_path[$group_name];
if ($module_name)
$path = str_replace('{module}', $module_name, $path);
if (!file_exists($path) && !mkdir($path, 0777, true))
throw new PrestaShopException(sprintf(Tools::displayError('Directory "%s" cannot be created'), dirname($path)));
file_put_contents($path.$mail_name.'.'.$type_content, $content);
}
else
throw new PrestaShopException(Tools::displayError('Your HTML email templates cannot contain JavaScript code.'));
}
}
}
// Update subjects
$array_subjects = array();
if (($subjects = Tools::getValue('subject')) && is_array($subjects))
{
$array_subjects['core_and_modules'] = array('translations' => array(), 'path' => $arr_mail_path['core_mail'].'lang.php');
foreach ($subjects as $subject_translation)
$array_subjects['core_and_modules']['translations'] = array_merge($array_subjects['core_and_modules']['translations'], $subject_translation);
}
if (!empty($array_subjects))
foreach ($array_subjects as $infos)
$this->writeSubjectTranslationFile($infos['translations'], $infos['path']);
if (Tools::isSubmit('submitTranslationsMailsAndStay'))
$this->redirect(true);
else
$this->redirect();
}
/**
* Include file $dir/$file and return the var $var declared in it.
* This create the file if not exists
*
* return array : translations
*/
public function fileExists()
{
$var = $this->translations_informations[$this->type_selected]['var'];
$dir = $this->translations_informations[$this->type_selected]['dir'];
$file = $this->translations_informations[$this->type_selected]['file'];
$$var = array();
if (!Tools::file_exists_cache($dir))
if (!mkdir($dir, 0700))
throw new PrestaShopException('Directory '.$dir.' cannot be created.');
if (!Tools::file_exists_cache($dir.DIRECTORY_SEPARATOR.$file))
if (!file_put_contents($dir.'/'.$file, "<?php\n\nglobal \$".$var.";\n\$".$var." = array();\n\n?>"))
throw new PrestaShopException('File "'.$file.'" doesn\'t exists and cannot be created in '.$dir);
if (!is_writable($dir.DIRECTORY_SEPARATOR.$file))
$this->displayWarning(Tools::displayError('This file must be writable:').' '.$dir.'/'.$file);
include($dir.DIRECTORY_SEPARATOR.$file);
return $$var;
}
public function displayToggleButton($closed = false)
{
$str_output = '
<script type="text/javascript">';
if (Tools::getValue('type') == 'mails')
$str_output .= '$(document).ready(function(){
toggleDiv(\''.$this->type_selected.'_div\'); toggleButtonValue(this.id, openAll, closeAll);
});';
$str_output .= '
var openAll = \''.html_entity_decode($this->l('Expand all fieldsets'), ENT_NOQUOTES, 'UTF-8').'\';
var closeAll = \''.html_entity_decode($this->l('Close all fieldsets'), ENT_NOQUOTES, 'UTF-8').'\';
</script>
<button type="button" class="btn btn-default" id="buttonall" data-status="open" onclick="toggleDiv(\''.$this->type_selected.'_div\', $(this).data(\'status\')); toggleButtonValue(this.id, openAll, closeAll);"><i class="process-icon-compress"></i> <span>'.$this->l('Close all fieldsets').'</span></button>';
return $str_output;
}
public function displayLimitPostWarning($count)
{
$return = array();
if ((ini_get('suhosin.post.max_vars') && ini_get('suhosin.post.max_vars') < $count) || (ini_get('suhosin.request.max_vars') && ini_get('suhosin.request.max_vars') < $count))
{
$return['error_type'] = 'suhosin';
$return['post.max_vars'] = ini_get('suhosin.post.max_vars');
$return['request.max_vars'] = ini_get('suhosin.request.max_vars');
$return['needed_limit'] = $count + 100;
}
elseif (ini_get('max_input_vars') && ini_get('max_input_vars') < $count)
{
$return['error_type'] = 'conf';
$return['max_input_vars'] = ini_get('max_input_vars');
$return['needed_limit'] = $count + 100;
}
return $return;
}
/**
* Find sentence which use %d, %s, %%, %1$d, %1$s...
*
* @param $key : english sentence
* @return array|bool return list of matches
*/
public function checkIfKeyUseSprintf($key)
{
if (preg_match_all('#(?:%%|%(?:[0-9]+\$)?[+-]?(?:[ 0]|\'.)?-?[0-9]*(?:\.[0-9]+)?[bcdeufFosxX])#', $key, $matches))
return implode(', ', $matches[0]);
return false;
}
/**
* This method generate the form for front translations
*/
public function initFormFront()
{
if (!$this->theme_exists(Tools::getValue('theme')))
{
$this->errors[] = sprintf(Tools::displayError('Invalid theme "%s"'), Tools::getValue('theme'));
return;
}
$missing_translations_front = array();
$name_var = $this->translations_informations[$this->type_selected]['var'];
$GLOBALS[$name_var] = $this->fileExists();
/* List templates to parse */
$files_by_directory = $this->getFileToParseByTypeTranslation();
$count = 0;
$tabs_array = array();
foreach ($files_by_directory['tpl'] as $dir => $files)
{
$prefix = '';
if ($dir == _PS_THEME_OVERRIDE_DIR_)
$prefix = 'override_';
foreach ($files as $file)
{
if (preg_match('/^(.*).tpl$/', $file) && (Tools::file_exists_cache($file_path = $dir.$file)))
{
$prefix_key = $prefix.substr(basename($file), 0, -4);
$new_lang = array();
// Get content for this file
$content = file_get_contents($file_path);
// Parse this content
$matches = $this->userParseFile($content, $this->type_selected);
/* Get string translation */
foreach ($matches as $key)
{
if (empty($key))
{
$this->errors[] = sprintf($this->l('Empty string found, please edit: "%s"'), $file_path);
$new_lang[$key] = '';
}
else
{
// Caution ! front has underscore between prefix key and md5, back has not
if (isset($GLOBALS[$name_var][$prefix_key.'_'.md5($key)]))
$new_lang[$key]['trad'] = stripslashes(html_entity_decode($GLOBALS[$name_var][$prefix_key.'_'.md5($key)], ENT_COMPAT, 'UTF-8'));
else
{
if (!isset($new_lang[$key]['trad']))
{
$new_lang[$key]['trad'] = '';
if (!isset($missing_translations_front[$prefix_key]))
$missing_translations_front[$prefix_key] = 1;
else
$missing_translations_front[$prefix_key]++;
}
}
$new_lang[$key]['use_sprintf'] = $this->checkIfKeyUseSprintf($key);
}
}
if (isset($tabs_array[$prefix_key]))
$tabs_array[$prefix_key] = array_merge($tabs_array[$prefix_key], $new_lang);
else
$tabs_array[$prefix_key] = $new_lang;
$count += count($new_lang);
}
}
}
$this->tpl_view_vars = array_merge($this->tpl_view_vars, array(
'missing_translations' => $missing_translations_front,
'count' => $count,
'cancel_url' => $this->context->link->getAdminLink('AdminTranslations'),
'limit_warning' => $this->displayLimitPostWarning($count),
'mod_security_warning' => Tools::apacheModExists('mod_security'),
'tabsArray' => $tabs_array,
));
$this->initToolbar();
$this->base_tpl_view = 'translation_form.tpl';
return parent::renderView();
}
/**
* This method generate the form for back translations
*/
public function initFormBack()
{
$name_var = $this->translations_informations[$this->type_selected]['var'];
$GLOBALS[$name_var] = $this->fileExists();
$missing_translations_back = array();
// Get all types of file (PHP, TPL...) and a list of files to parse by folder
$files_per_directory = $this->getFileToParseByTypeTranslation();
foreach ($files_per_directory['php'] as $dir => $files)
foreach ($files as $file)
// Check if is a PHP file and if the override file exists
if (preg_match('/^(.*)\.php$/', $file) && Tools::file_exists_cache($file_path = $dir.$file) && !in_array($file, self::$ignore_folder))
{
$prefix_key = basename($file);
// -4 becomes -14 to remove the ending "Controller.php" from the filename
if (strpos($file, 'Controller.php') !== false)
$prefix_key = basename(substr($file, 0, -14));
else if (strpos($file, 'Helper') !== false)
$prefix_key = 'Helper';
if ($prefix_key == 'Admin')
$prefix_key = 'AdminController';
if ($prefix_key == 'PaymentModule.php')
$prefix_key = 'PaymentModule';
// Get content for this file
$content = file_get_contents($file_path);
// Parse this content
$matches = $this->userParseFile($content, $this->type_selected, 'php');
foreach ($matches as $key)
{
// Caution ! front has underscore between prefix key and md5, back has not
if (isset($GLOBALS[$name_var][$prefix_key.md5($key)]))
$tabs_array[$prefix_key][$key]['trad'] = stripslashes(html_entity_decode($GLOBALS[$name_var][$prefix_key.md5($key)], ENT_COMPAT, 'UTF-8'));
else
{
if (!isset($tabs_array[$prefix_key][$key]['trad']))
{
$tabs_array[$prefix_key][$key]['trad'] = '';
if (!isset($missing_translations_back[$prefix_key]))
$missing_translations_back[$prefix_key] = 1;
else
$missing_translations_back[$prefix_key]++;
}
}
$tabs_array[$prefix_key][$key]['use_sprintf'] = $this->checkIfKeyUseSprintf($key);
}
}
foreach ($files_per_directory['specific'] as $dir => $files)
foreach ($files as $file)
if (Tools::file_exists_cache($file_path = $dir.$file) && !in_array($file, self::$ignore_folder))
{
$prefix_key = 'index';
// Get content for this file
$content = file_get_contents($file_path);
// Parse this content
$matches = $this->userParseFile($content, $this->type_selected, 'specific');
foreach ($matches as $key)
{
// Caution ! front has underscore between prefix key and md5, back has not
if (isset($GLOBALS[$name_var][$prefix_key.md5($key)]))
$tabs_array[$prefix_key][$key]['trad'] = stripslashes(html_entity_decode($GLOBALS[$name_var][$prefix_key.md5($key)], ENT_COMPAT, 'UTF-8'));
else
{
if (!isset($tabs_array[$prefix_key][$key]['trad']))
{
$tabs_array[$prefix_key][$key]['trad'] = '';
if (!isset($missing_translations_back[$prefix_key]))
$missing_translations_back[$prefix_key] = 1;
else
$missing_translations_back[$prefix_key]++;
}
}
$tabs_array[$prefix_key][$key]['use_sprintf'] = $this->checkIfKeyUseSprintf($key);
}
}
foreach ($files_per_directory['tpl'] as $dir => $files)
foreach ($files as $file)
if (preg_match('/^(.*).tpl$/', $file) && Tools::file_exists_cache($file_path = $dir.$file))
{
// get controller name instead of file name
$prefix_key = Tools::toCamelCase(str_replace(_PS_ADMIN_DIR_.DIRECTORY_SEPARATOR.'themes', '', $file_path), true);
$pos = strrpos($prefix_key, DIRECTORY_SEPARATOR);
$tmp = substr($prefix_key, 0, $pos);
if (preg_match('#controllers#', $tmp))
{
$parent_class = explode(DIRECTORY_SEPARATOR, str_replace('/', DIRECTORY_SEPARATOR, $tmp));
$override = array_search('override', $parent_class);
if ($override !== false)
// case override/controllers/admin/templates/controller_name
$prefix_key = 'Admin'.ucfirst($parent_class[$override + 4]);
else
{
// case admin_name/themes/theme_name/template/controllers/controller_name
$key = array_search('controllers', $parent_class);
$prefix_key = 'Admin'.ucfirst($parent_class[$key + 1]);
}
}
else
$prefix_key = 'Admin'.ucfirst(substr($tmp, strrpos($tmp, DIRECTORY_SEPARATOR) + 1, $pos));
// Adding list, form, option in Helper Translations
$list_prefix_key = array('AdminHelpers', 'AdminList', 'AdminView', 'AdminOptions', 'AdminForm',
'AdminCalendar', 'AdminTree', 'AdminUploader', 'AdminDataviz', 'AdminKpi', 'AdminModule_list', 'AdminModulesList');
if (in_array($prefix_key, $list_prefix_key))
$prefix_key = 'Helper';
// Adding the folder backup/download/ in AdminBackup Translations
if ($prefix_key == 'AdminDownload')
$prefix_key = 'AdminBackup';
// use the prefix "AdminController" (like old php files 'header', 'footer.inc', 'index', 'login', 'password', 'functions'
if ($prefix_key == 'Admin' || $prefix_key == 'AdminTemplate')
$prefix_key = 'AdminController';
$new_lang = array();
// Get content for this file
$content = file_get_contents($file_path);
// Parse this content
$matches = $this->userParseFile($content, $this->type_selected, 'tpl');
/* Get string translation for each tpl file */
foreach ($matches as $english_string)
{
if (empty($english_string))
{
$this->errors[] = sprintf($this->l('There is an error in template, an empty string has been found. Please edit: "%s"'), $file_path);
$new_lang[$english_string] = '';
}
else
{
$trans_key = $prefix_key.md5($english_string);
if (isset($GLOBALS[$name_var][$trans_key]))
$new_lang[$english_string]['trad'] = html_entity_decode($GLOBALS[$name_var][$trans_key], ENT_COMPAT, 'UTF-8');
else
{
if (!isset($new_lang[$english_string]['trad']))
{
$new_lang[$english_string]['trad'] = '';
if (!isset($missing_translations_back[$prefix_key]))
$missing_translations_back[$prefix_key] = 1;
else
$missing_translations_back[$prefix_key]++;
}
}
$new_lang[$english_string]['use_sprintf'] = $this->checkIfKeyUseSprintf($key);
}
}
if (isset($tabs_array[$prefix_key]))
$tabs_array[$prefix_key] = array_merge($tabs_array[$prefix_key], $new_lang);
else
$tabs_array[$prefix_key] = $new_lang;
}
// count will contain the number of expressions of the page
$count = 0;
foreach ($tabs_array as $array)
$count += count($array);
$this->tpl_view_vars = array_merge($this->tpl_view_vars, array(
'count' => $count,
'cancel_url' => $this->context->link->getAdminLink('AdminTranslations'),
'limit_warning' => $this->displayLimitPostWarning($count),
'mod_security_warning' => Tools::apacheModExists('mod_security'),
'tabsArray' => $tabs_array,
'missing_translations' => $missing_translations_back
));
$this->initToolbar();
$this->base_tpl_view = 'translation_form.tpl';
return parent::renderView();
}
/**
* Check if directory and file exist and return an list of modules
*
* @return array : list of modules
*/
public function getListModules()
{
if (!Tools::file_exists_cache($this->translations_informations['modules']['dir']))
throw new PrestaShopException(Tools::displayError('Fatal error: The module directory does not exist.').'('.$this->translations_informations['modules']['dir'].')');
if (!is_writable($this->translations_informations['modules']['dir']))
throw new PrestaShopException(Tools::displayError('The module directory must be writable.'));
$modules = array();
// Get all module which are installed for to have a minimum of POST
$modules = Module::getModulesInstalled();
foreach ($modules as &$module)
$module = $module['name'];
return $modules;
}
/**
* This method generate the form for errors translations
*/
public function initFormErrors()
{
$name_var = $this->translations_informations[$this->type_selected]['var'];
$GLOBALS[$name_var] = $this->fileExists();
$count_empty = array();
/* List files to parse */
$string_to_translate = array();
$file_by_directory = $this->getFileToParseByTypeTranslation();
if ($modules = $this->getListModules())
{
foreach ($modules as $module)
if (is_dir(_PS_MODULE_DIR_.$module) && !in_array($module, self::$ignore_folder))
$file_by_directory['php'] = array_merge($file_by_directory['php'], $this->listFiles(_PS_MODULE_DIR_.$module.'/', array(), 'php'));
}
foreach ($file_by_directory['php'] as $dir => $files)
foreach ($files as $file)
if (preg_match('/\.php$/', $file) && Tools::file_exists_cache($file_path = $dir.$file) && !in_array($file, self::$ignore_folder))
{
if (!filesize($file_path))
continue;
// Get content for this file
$content = file_get_contents($file_path);
// Parse this content
$matches = $this->userParseFile($content, $this->type_selected);
foreach ($matches as $key)
{
if (array_key_exists(md5($key), $GLOBALS[$name_var]))
$string_to_translate[$key]['trad'] = html_entity_decode($GLOBALS[$name_var][md5($key)], ENT_COMPAT, 'UTF-8');
else
{
$string_to_translate[$key]['trad'] = '';
if (!isset($count_empty[$key]))
$count_empty[$key] = 1;
}
$string_to_translate[$key]['use_sprintf'] = $this->checkIfKeyUseSprintf($key);
}
}
$this->tpl_view_vars = array_merge($this->tpl_view_vars, array(
'count' => count($string_to_translate),
'cancel_url' => $this->context->link->getAdminLink('AdminTranslations'),
'limit_warning' => $this->displayLimitPostWarning(count($string_to_translate)),
'mod_security_warning' => Tools::apacheModExists('mod_security'),
'errorsArray' => $string_to_translate,
'missing_translations' => $count_empty
));
$this->initToolbar();
$this->base_tpl_view = 'translation_errors.tpl';
return parent::renderView();
}
/**
* This method generate the form for fields translations
*/
public function initFormFields()
{
$name_var = $this->translations_informations[$this->type_selected]['var'];
$GLOBALS[$name_var] = $this->fileExists();
$missing_translations_fields = array();
$class_array = array();
$tabs_array = array();
$count = 0;
$files_by_directory = $this->getFileToParseByTypeTranslation();
foreach ($files_by_directory['php'] as $dir => $files)
foreach ($files as $file)
{
$exclude_files = array('index.php', 'PrestaShopAutoload.php', 'StockManagerInterface.php',
'TaxManagerInterface.php', 'WebserviceOutputInterface.php', 'WebserviceSpecificManagementInterface.php');
if (!preg_match('/\.php$/', $file) || in_array($file, $exclude_files))
continue;
$class_name = substr($file, 0, -4);
if (!class_exists($class_name, false) && !class_exists($class_name.'Core', false))
PrestaShopAutoload::getInstance()->load($class_name);
if (!is_subclass_of($class_name.'Core', 'ObjectModel'))
continue;
$class_array[$class_name] = call_user_func(array($class_name, 'getValidationRules'), $class_name);
}
foreach ($class_array as $prefix_key => $rules)
{
if (isset($rules['validate']))
foreach ($rules['validate'] as $key => $value)
{
if (isset($GLOBALS[$name_var][$prefix_key.'_'.md5($key)]))
{
$tabs_array[$prefix_key][$key]['trad'] = html_entity_decode($GLOBALS[$name_var][$prefix_key.'_'.md5($key)], ENT_COMPAT, 'UTF-8');
$count++;
}
else
{
if (!isset($tabs_array[$prefix_key][$key]['trad']))
{
$tabs_array[$prefix_key][$key]['trad'] = '';
if (!isset($missing_translations_fields[$prefix_key]))
$missing_translations_fields[$prefix_key] = 1;
else
$missing_translations_fields[$prefix_key]++;
$count++;
}
}
}
if (isset($rules['validateLang']))
foreach ($rules['validateLang'] as $key => $value)
{
if (isset($GLOBALS[$name_var][$prefix_key.'_'.md5($key)]))
{
$tabs_array[$prefix_key][$key]['trad'] = '';
if (array_key_exists($prefix_key.'_'.md5(addslashes($key)), $GLOBALS[$name_var]))
$tabs_array[$prefix_key][$key]['trad'] = html_entity_decode($GLOBALS[$name_var][$prefix_key.'_'.md5(addslashes($key))], ENT_COMPAT, 'UTF-8');
$count++;
}
else
{
if (!isset($tabs_array[$prefix_key][$key]['trad']))
{
$tabs_array[$prefix_key][$key]['trad'] = '';
if (!isset($missing_translations_fields[$prefix_key]))
$missing_translations_fields[$prefix_key] = 1;
else
$missing_translations_fields[$prefix_key]++;
$count++;
}
}
}
}
$this->tpl_view_vars = array_merge($this->tpl_view_vars, array(
'count' => $count,
'limit_warning' => $this->displayLimitPostWarning($count),
'mod_security_warning' => Tools::apacheModExists('mod_security'),
'tabsArray' => $tabs_array,
'cancel_url' => $this->context->link->getAdminLink('AdminTranslations'),
'missing_translations' => $missing_translations_fields
));
$this->initToolbar();
$this->base_tpl_view = 'translation_form.tpl';
return parent::renderView();
}
/**
* Get each informations for each mails found in the folder $dir.
*
* @since 1.4.0.14
* @param string $dir
* @param string $group_name
* @return array : list of mails
*/
public function getMailFiles($dir, $group_name = 'mail')
{
$arr_return = array();
if (Language::getIdByIso('en'))
$default_language = 'en';
else
$default_language = Language::getIsoById((int)Configuration::get('PS_LANG_DEFAULT'));
if (!$default_language || !Validate::isLanguageIsoCode($default_language))
return false;
// Very usefull to name input and textarea fields
$arr_return['group_name'] = $group_name;
$arr_return['empty_values'] = 0;
$arr_return['total_filled'] = 0;
$arr_return['directory'] = $dir;
// Get path for english mail directory
$dir_en = str_replace('/'.$this->lang_selected->iso_code.'/', '/'.$default_language.'/', $dir);
if (Tools::file_exists_cache($dir_en))
{
// Get all english files to compare with the language to translate
foreach (scandir($dir_en) as $email_file)
{
if (strripos($email_file, '.html') > 0 || strripos($email_file, '.txt') > 0)
{
$email_name = substr($email_file, 0, strripos($email_file, '.'));
$type = substr($email_file, strripos($email_file, '.') + 1);
if (!isset($arr_return['files'][$email_name]))
$arr_return['files'][$email_name] = array();
// $email_file is from scandir ($dir), so we already know that file exists
$arr_return['files'][$email_name][$type]['en'] = $this->getMailContent($dir_en, $email_file);
// check if the file exists in the language to translate
if (Tools::file_exists_cache($dir.'/'.$email_file))
{
$arr_return['files'][$email_name][$type][$this->lang_selected->iso_code] = $this->getMailContent($dir, $email_file);
$this->total_expression++;
}
else
$arr_return['files'][$email_name][$type][$this->lang_selected->iso_code] = '';
if ($arr_return['files'][$email_name][$type][$this->lang_selected->iso_code] == '')
$arr_return['empty_values']++;
else
$arr_return['total_filled']++;
}
}
}
else
$this->warnings[] = sprintf(Tools::displayError('A mail directory exists for the "%1$s" language, but not for the default language in %2$s'),
$this->lang_selected->iso_code, str_replace(_PS_ROOT_DIR_, '', $dir));
return $arr_return;
}
/**
* Get content of the mail file.
*
* @since 1.4.0.14
* @param string $dir
* @param string $file
* @return array : content of file
*/
protected function getMailContent($dir, $file)
{
$content = file_get_contents($dir.'/'.$file);
if (Tools::strlen($content) === 0)
$content = '';
return $content;
}
/**
* Display mails in html format.
* This was create for factorize the html displaying
*
* @since 1.4.0.14
* @param array $mails
* @param array $all_subject_mail
* @param Language $obj_lang
* @param string $id_html use for set html id attribute for the block
* @param string $title Set the title for the block
* @param string|boolean $name_for_module is not false define add a name for disntiguish mails module
*/
protected function displayMailContent($mails, $all_subject_mail, $obj_lang, $id_html, $title, $name_for_module = false)
{
$str_return = '';
$group_name = 'mail';
if (array_key_exists('group_name', $mails))
$group_name = $mails['group_name'];
if($mails['empty_values'] == 0) {
$translation_missing_badge_type = 'badge-success';
} else {
$translation_missing_badge_type = 'badge-danger';
}
$str_return .= '<div class="mails_field">
<h4>
<span class="badge">'.((int)$mails['empty_values'] + (int)$mails['total_filled']).' <i class="icon-envelope-o"></i></span>
<a href="javascript:void(0);" onclick="$(\'#'.$id_html.'\').slideToggle();">'.$title.'</a>
<span class="pull-right badge '.$translation_missing_badge_type.'">'.$mails['empty_values'].' '.$this->l('missing translation(s)').'</span>
</h4>
<div name="mails_div" id="'.$id_html.'" class="panel-group">';
if (!empty($mails['files']))
{
$topic_already_displayed = array();
foreach ($mails['files'] as $mail_name => $mail_files)
{
$str_return .= '<div class="panel translations-email-panel">';
$str_return .= '<a href="#email-'.$mail_name.'" class="panel-title" data-toggle="collapse" data-parent="#'.$id_html.'" >'.$mail_name.' <i class="icon-caret-down"></i> </a>';
$str_return .= '<div id="email-'.$mail_name.'" class="email-collapse panel-collapse collapse">';
if (array_key_exists('html', $mail_files) || array_key_exists('txt', $mail_files))
{
if (array_key_exists($mail_name, $all_subject_mail))
{
foreach ($all_subject_mail[$mail_name] as $subject_mail)
{
$subject_key = 'subject['.Tools::htmlentitiesUTF8($group_name).']['.Tools::htmlentitiesUTF8($subject_mail).']';
if (in_array($subject_key, $topic_already_displayed))
continue;
$topic_already_displayed[] = $subject_key;
$value_subject_mail = isset($mails['subject'][$subject_mail]) ? $mails['subject'][$subject_mail] : '';
$str_return .= '
<div class="label-subject row">
<label class="control-label col-lg-3">'.sprintf($this->l('Email subject'));
if (isset($value_subject_mail['use_sprintf']) && $value_subject_mail['use_sprintf'])
$str_return .= '<span class="useSpecialSyntax" title="'.$this->l('This expression uses a special syntax:').' '.$value_subject_mail['use_sprintf'].'">
<i class="icon-exclamation-triangle"></i>
</span>';
$str_return .= '</label><div class="col-lg-9">';
if (isset($value_subject_mail['trad']) && $value_subject_mail['trad'])
$str_return .= '<input class="form-control" type="text" name="subject['.Tools::htmlentitiesUTF8($group_name).']['.Tools::htmlentitiesUTF8($subject_mail).']" value="'.$value_subject_mail['trad'].'" />';
else
$str_return .= '<input class="form-control" type="text" name="subject['.Tools::htmlentitiesUTF8($group_name).']['.Tools::htmlentitiesUTF8($subject_mail).']" value="" />';
$str_return .= '<p class="help-block">'.$subject_mail.'</p>';
$str_return .= '</div></div>';
}
}
else
{
$str_return .= '
<hr><div class="alert alert-info">'
.sprintf($this->l('No Subject was found for %s in the database.'), $mail_name)
.'</div>';
}
// tab menu
$str_return .= '<hr><ul class="nav nav-pills">
<li class="active"><a href="#'.$mail_name.'-html" data-toggle="tab">'.$this->l('View HTML version').'</a></li>
<li><a href="#'.$mail_name.'-editor" data-toggle="tab">'.$this->l('Edit HTML version').'</a></li>
<li><a href="#'.$mail_name.'-text" data-toggle="tab">'.$this->l('View/Edit TXT version').'</a></li>
</ul>';
// tab-content
$str_return .= '<div class="tab-content">';
if (array_key_exists('html', $mail_files))
{
$str_return .= '<div class="tab-pane active" id="'.$mail_name.'-html">';
$base_uri = str_replace(_PS_ROOT_DIR_, __PS_BASE_URI__, $mails['directory']);
$base_uri = str_replace('//', '/', $base_uri);
$url_mail = $base_uri.$mail_name.'.html';
$str_return .= $this->displayMailBlockHtml($mail_files['html'], $obj_lang->iso_code, $url_mail, $mail_name, $group_name, $name_for_module);
$str_return .= '</div>';
}
if (array_key_exists('txt', $mail_files))
{
$str_return .= '<div class="tab-pane" id="'.$mail_name.'-text">';
$str_return .= $this->displayMailBlockTxt($mail_files['txt'], $obj_lang->iso_code, $mail_name, $group_name, $name_for_module);
$str_return .= '</div>';
}
$str_return .= '<div class="tab-pane" id="'.$mail_name.'-editor">';
if (isset($mail_files['html']))
$str_return .= $this->displayMailEditor($mail_files['html'], $obj_lang->iso_code, $url_mail, $mail_name, $group_name, $name_for_module);
$str_return .= '</div>';
$str_return .= '</div>';
$str_return .= '</div><!--end .panel-collapse -->';
$str_return .= '</div><!--end .panel -->';
}
}
}
else
$str_return .= '<p class="error">
'.$this->l('There was a problem getting the mail files.').'<br>
'.sprintf($this->l('English language files must exist in %s folder'), '<em>'.preg_replace('@/[a-z]{2}(/?)$@', '/en$1', $mails['directory']).'</em>').'
</p>';
$str_return .= '</div><!-- #'.$id_html.' --></div><!-- end .mails_field -->';
return $str_return;
}
/**
* Just build the html structure for display txt mails
*
* @since 1.4.0.14
* @param array $content with english and language needed contents
* @param string $lang iso code of the needed language
* @param string $mail_name name of the file to translate (same for txt and html files)
* @param string $group_name group name allow to distinguish each block of mail.
* @param string|boolean $name_for_module is not false define add a name for disntiguish mails module
*/
protected function displayMailBlockTxt($content, $lang, $mail_name, $group_name, $name_for_module = false)
{
return '<div class="block-mail" >
<div class="mail-form">
<div><textarea class="rte noEditor" name="'.$group_name.'[txt]['.($name_for_module ? $name_for_module.'|' : '' ).$mail_name.']">'.Tools::htmlentitiesUTF8(stripslashes(strip_tags($content[$lang]))).'</textarea></div>
</div>
</div>';
}
/**
* Just build the html structure for display html mails.
*
* @since 1.4.0.14
* @param array $content with english and language needed contents
* @param string $lang iso code of the needed language
* @param string $url for the html page and displaying an outline
* @param string $mail_name name of the file to translate (same for txt and html files)
* @param string $group_name group name allow to distinguish each block of mail.
* @param string|boolean $name_for_module is not false define add a name for disntiguish mails module
*/
protected function displayMailBlockHtml($content, $lang, $url, $mail_name, $group_name, $name_for_module = false)
{
$title = array();
$this->cleanMailContent($content, $lang, $title);
$name_for_module = $name_for_module ? $name_for_module.'|' : '';
return '<div class="block-mail" >
<div class="mail-form">
<div class="form-group">
<label class="control-label col-lg-3">'.$this->l('HTML "title" tag').'</label>
<div class="col-lg-9">
<input class="form-control" type="text" name="title_'.$group_name.'_'.$mail_name.'" value="'.(isset($title[$lang]) ? $title[$lang] : '').'" />
<p class="help-block">'.(isset($title['en']) ? $title['en'] : '').'</p>
</div>
</div>
<div class="thumbnail email-html-frame" data-email-src="'.$url.'?'.(rand(0, 1000000000)).'"></div>
</div>
</div>';
}
protected function displayMailEditor($content, $lang, $url, $mail_name, $group_name, $name_for_module = false)
{
$title = array();
$this->cleanMailContent($content, $lang, $title);
$name_for_module = $name_for_module ? $name_for_module.'|' : '';
return '<textarea class="rte-mail rte-mail-'.$mail_name.' form-control" data-rte="'.$mail_name.'" name="'.$group_name.'[html]['.$name_for_module.$mail_name.']">'.$content[$lang].'</textarea>';
}
protected function cleanMailContent(&$content, $lang, &$title)
{
// Because TinyMCE don't work correctly with <DOCTYPE>, <html> and <body> tags
if (stripos($content[$lang], '<body'))
{
$array_lang = $lang != 'en' ? array('en', $lang) : array($lang);
foreach ($array_lang as $language)
{
$title[$language] = substr($content[$language], 0, stripos($content[$language], '<body'));
preg_match('#<title>([^<]+)</title>#Ui', $title[$language], $matches);
$title[$language] = empty($matches[1])?'':$matches[1];
// The 2 lines below allow to exlude <body> tag from the content.
// This allow to exclude body tag even if attributs are setted.
$content[$language] = substr($content[$language], stripos($content[$language], '<body') + 5);
$content[$language] = substr($content[$language], stripos($content[$language], '>') + 1);
$content[$language] = substr($content[$language], 0, stripos($content[$language], '</body>'));
}
}
$content[$lang] = (isset($content[$lang]) ? Tools::htmlentitiesUTF8(stripslashes($content[$lang])) : '');
}
/**
* Check in each module if contains mails folder.
*
* @return array of module which has mails
*/
public function getModulesHasMails($with_module_name = false)
{
$arr_modules = array();
foreach (scandir($this->translations_informations['modules']['dir']) as $module_dir)
{
if (!in_array($module_dir, self::$ignore_folder))
{
$dir = false;
if ($this->theme_selected && Tools::file_exists_cache($this->translations_informations['modules']['override']['dir'].$module_dir.'/mails/'))
$dir = $this->translations_informations['modules']['override']['dir'].$module_dir.'/';
elseif (Tools::file_exists_cache($this->translations_informations['modules']['dir'].$module_dir.'/mails/'))
$dir = $this->translations_informations['modules']['dir'].$module_dir.'/';
if ($dir !== false)
{
if ($with_module_name)
$arr_modules[$module_dir] = $dir;
else
$arr_modules[$dir] = scandir($dir);
}
}
}
return $arr_modules;
}
protected function getTinyMCEForMails($iso_lang)
{
// TinyMCE
$iso_tiny_mce = (Tools::file_exists_cache(_PS_ROOT_DIR_.'/js/tiny_mce/langs/'.$iso_lang.'.js') ? $iso_lang : 'en');
$ad = __PS_BASE_URI__.basename(_PS_ADMIN_DIR_);
//return false;
return '
<script type="text/javascript">
var iso = \''.$iso_tiny_mce.'\' ;
var pathCSS = \''._THEME_CSS_DIR_.'\' ;
var ad = \''.$ad.'\' ;
</script>
<script type="text/javascript" src="'.__PS_BASE_URI__.'js/tiny_mce/tiny_mce.js"></script>
<script type="text/javascript" src="'.__PS_BASE_URI__.'js/tinymce.inc.js"></script>';
}
/**
* This method generate the form for mails translations
*/
public function initFormMails($no_display = false)
{
$module_mails = array();
// get all mail subjects, this method parse each files in Prestashop !!
$subject_mail = array();
$modules_has_mails = $this->getModulesHasMails(true);
$files_by_directiories = $this->getFileToParseByTypeTranslation();
if (!$this->theme_selected || !@filemtime($this->translations_informations[$this->type_selected]['override']['dir']))
$this->copyMailFilesForAllLanguages();
foreach ($files_by_directiories['php'] as $dir => $files)
foreach ($files as $file)
// If file exist and is not in ignore_folder, in the next step we check if a folder or mail
if (Tools::file_exists_cache($dir.$file) && !in_array($file, self::$ignore_folder))
$subject_mail = $this->getSubjectMail($dir, $file, $subject_mail);
// Get path of directory for find a good path of translation file
if ($this->theme_selected && @filemtime($this->translations_informations[$this->type_selected]['override']['dir']))
$i18n_dir = $this->translations_informations[$this->type_selected]['override']['dir'];
else
$i18n_dir = $this->translations_informations[$this->type_selected]['dir'];
$core_mails = $this->getMailFiles($i18n_dir, 'core_mail');
$core_mails['subject'] = $this->getSubjectMailContent($i18n_dir);
foreach ($modules_has_mails as $module_name => $module_path)
{
$module_mails[$module_name] = $this->getMailFiles($module_path.'mails/'.$this->lang_selected->iso_code.'/', 'module_mail');
$module_mails[$module_name]['subject'] = $core_mails['subject'];
$module_mails[$module_name]['display'] = $this->displayMailContent($module_mails[$module_name], $subject_mail, $this->lang_selected, Tools::strtolower($module_name), $module_name, $module_name);
}
if ($no_display)
{
$empty = 0;
$total = 0;
$total += (int)$core_mails['total_filled'];
$empty += (int)$core_mails['empty_values'];
foreach ($module_mails as $mod_infos)
{
$total += (int)$mod_infos['total_filled'];
$empty += (int)$mod_infos['empty_values'];
}
return array('total' => $total, 'empty' => $empty);
}
$this->tpl_view_vars = array_merge($this->tpl_view_vars, array(
'limit_warning' => $this->displayLimitPostWarning($this->total_expression),
'mod_security_warning' => Tools::apacheModExists('mod_security'),
'tinyMCE' => $this->getTinyMCEForMails($this->lang_selected->iso_code),
'mail_content' => $this->displayMailContent($core_mails, $subject_mail, $this->lang_selected, 'core', $this->l('Core emails')),
'cancel_url' => $this->context->link->getAdminLink('AdminTranslations'),
'module_mails' => $module_mails,
'theme_name' => $this->theme_selected
));
$this->initToolbar();
$this->base_tpl_view = 'translation_mails.tpl';
return parent::renderView();
}
public function copyMailFilesForAllLanguages()
{
$current_theme = Tools::safeOutput($this->context->theme->name);
$languages = Language::getLanguages();
foreach ($languages as $key => $lang) {
$dir_to_copy_iso = array();
$files_to_copy_iso = array();
$current_iso_code = $lang['iso_code'];
$dir_to_copy_iso[] = _PS_MAIL_DIR_.$current_iso_code.'/';
$modules_has_mails = $this->getModulesHasMails(true);
foreach ($modules_has_mails as $module_name => $module_path)
{
if ($pos = strpos($module_path, '/modules'))
$dir_to_copy_iso[] = _PS_ROOT_DIR_.substr($module_path, $pos).'mails/'.$current_iso_code.'/';
}
foreach ($dir_to_copy_iso as $dir)
foreach (scandir($dir) as $file)
if (!in_array($file, self::$ignore_folder))
$files_to_copy_iso[] = array(
"from" => $dir.$file,
"to" => str_replace(_PS_ROOT_DIR_, _PS_ROOT_DIR_.'/themes/'.$current_theme, $dir).$file
);
foreach ($files_to_copy_iso as $file)
{
if (!file_exists($file['to']))
{
$content = file_get_contents($file['from']);
$stack = array();
$folder = dirname($file['to']);
while (!is_dir($folder))
{
array_push($stack, $folder);
$folder = dirname($folder);
}
while ($folder = array_pop($stack))
mkdir($folder);
$success = file_put_contents($file['to'], $content);
if ($success === false)
Tools::dieOrLog(sprintf("%s cannot be copied to %s", $file['from'], $file['to']), false);
}
}
}
return true;
}
/**
* Get list of subjects of mails
*
* @param $dir
* @param $file
* @param $subject_mail
* @return array : list of subjects of mails
*/
protected function getSubjectMail($dir, $file, $subject_mail)
{
// If is file and is not in ignore_folder
if (is_file($dir.'/'.$file) && !in_array($file, self::$ignore_folder) && preg_match('/\.php$/', $file))
{
$content = file_get_contents($dir.'/'.$file);
$content = str_replace("\n", ' ', $content);
// Subject must match with a template, therefor we first grep the Mail::Send() function then the Mail::l() inside.
if (preg_match_all('/Mail::Send([^;]*);/si', $content, $tab))
{
for ($i = 0; isset($tab[1][$i]); $i++)
{
$tab2 = explode(',', $tab[1][$i]);
if (is_array($tab2) && isset($tab2[1]))
{
$template = trim(str_replace('\'', '', $tab2[1]));
foreach ($tab2 as $tab3)
if (preg_match('/Mail::l\(\''._PS_TRANS_PATTERN_.'\'\)/Us', $tab3.')', $matches))
{
if (!isset($subject_mail[$template]))
$subject_mail[$template] = array();
if (!in_array($matches[1], $subject_mail[$template]))
$subject_mail[$template][] = $matches[1];
}
}
}
}
}
// Or if is colder, we scan colder for check if find in folder and subfolder
else if (!in_array($file, self::$ignore_folder) && is_dir($dir.'/'.$file))
foreach( scandir($dir.'/'.$file ) as $temp )
$subject_mail = $this->getSubjectMail($dir.'/'.$file, $temp, $subject_mail);
return $subject_mail;
}
/**
* @param $directory : name of directory
* @return array
*/
protected function getSubjectMailContent($directory)
{
$subject_mail_content = array();
if (Tools::file_exists_cache($directory.'/lang.php'))
{
// we need to include this even if already included (no include once)
include($directory.'/lang.php');
foreach ($GLOBALS[$this->translations_informations[$this->type_selected]['var']] as $key => $subject)
{
$this->total_expression++;
$subject = str_replace('\n', ' ', $subject);
$subject = str_replace("\\'", "\'", $subject);
$subject_mail_content[$key]['trad'] = htmlentities($subject, ENT_QUOTES, 'UTF-8');
$subject_mail_content[$key]['use_sprintf'] = $this->checkIfKeyUseSprintf($key);
}
}
else
$this->errors[] = sprintf($this->l('Email subject translation file not found in "%s".'), $directory);
return $subject_mail_content;
}
protected function writeSubjectTranslationFile($sub, $path)
{
if (!Tools::file_exists_cache(dirname($path)))
if (!mkdir(dirname(path), 0700))
throw new PrestaShopException('Directory '.dirname(path).' cannot be created.');
if ($fd = @fopen($path, 'w'))
{
$tab = 'LANGMAIL';
fwrite($fd, "<?php\n\nglobal \$_".$tab.";\n\$_".$tab." = array();\n");
foreach ($sub as $key => $value)
{
// Magic Quotes shall... not.. PASS!
if (_PS_MAGIC_QUOTES_GPC_)
$value = stripslashes($value);
fwrite($fd, '$_'.$tab.'[\''.pSQL($key).'\'] = \''.pSQL($value).'\';'."\n");
}
fwrite($fd, "\n?>");
fclose($fd);
}
else
throw new PrestaShopException(sprintf(Tools::displayError('Cannot write language file for email subjects. Path is: %s'), $path));
}
/**
* This get files to translate in module directory.
* Recursive method allow to get each files for a module no matter his depth.
*
* @param string $path directory path to scan
* @param array $array_files by reference - array which saved files to parse.
* @param string $module_name module name
* @param string $lang_file full path of translation file
* @param boolean $is_default
*/
protected function recursiveGetModuleFiles($path, &$array_files, $module_name, $lang_file, $is_default = false)
{
$files_module = array();
if (Tools::file_exists_cache($path))
$files_module = scandir($path);
$files_for_module = $this->clearModuleFiles($files_module, 'file');
if (!empty($files_for_module))
$array_files[] = array(
'file_name' => $lang_file,
'dir' => $path,
'files' => $files_for_module,
'module' => $module_name,
'is_default' => $is_default,
'theme' => $this->theme_selected,
);
$dir_module = $this->clearModuleFiles($files_module, 'directory', $path);
if (!empty($dir_module))
foreach ($dir_module as $folder)
$this->recursiveGetModuleFiles($path.$folder.'/', $array_files, $module_name, $lang_file, $is_default);
}
/**
* This method get translation in each translations file.
* The file depend on $lang param.
*
* @param array $modules list of modules
* @param string $root_dir path where it get each modules
* @param string $lang iso code of choosen language to translate
* @param boolean $is_default set it if modules are located in root/prestashop/modules folder
* This allow to distinguish overrided prestashop theme and original module
*/
protected function getAllModuleFiles($modules, $root_dir = null, $lang, $is_default = false)
{
$array_files = array();
$initial_root_dir = $root_dir;
foreach ($modules as $module)
{
$root_dir = $initial_root_dir;
if ($module{0} == '.')
continue;
// First we load the default translation file
if ($root_dir == null)
{
$i18n_dir = $this->translations_informations[$this->type_selected]['dir'];
if (is_dir($i18n_dir.$module))
$root_dir = $i18n_dir;
$lang_file = $root_dir.$module.'/translations/'.$lang.'.php';
if (!Tools::file_exists_cache($root_dir.$module.'/translations/'.$lang.'.php') && Tools::file_exists_cache($root_dir.$module.'/'.$lang.'.php'))
$lang_file = $root_dir.$module.'/'.$lang.'.php';
@include($lang_file);
$this->getModuleTranslations();
// If a theme is selected, then the destination translation file must be in the theme
if ($this->theme_selected)
$lang_file = $this->translations_informations[$this->type_selected]['override']['dir'].$module.'/translations/'.$lang.'.php';
$this->recursiveGetModuleFiles($root_dir.$module.'/', $array_files, $module, $lang_file, $is_default);
}
$root_dir = $initial_root_dir;
// Then we load the overriden translation file
if ($this->theme_selected && isset($this->translations_informations[$this->type_selected]['override']))
{
$i18n_dir = $this->translations_informations[$this->type_selected]['override']['dir'];
if (is_dir($i18n_dir.$module))
$root_dir = $i18n_dir;
if (Tools::file_exists_cache($root_dir.$module.'/translations/'.$lang.'.php'))
$lang_file = $root_dir.$module.'/translations/'.$lang.'.php';
elseif (Tools::file_exists_cache($root_dir.$module.'/'.$lang.'.php'))
$lang_file = $root_dir.$module.'/'.$lang.'.php';
@include($lang_file);
$this->getModuleTranslations();
$this->recursiveGetModuleFiles($root_dir.$module.'/', $array_files, $module, $lang_file, $is_default);
}
}
return $array_files;
}
/**
* This method generate the form for modules translations
*/
public function initFormModules()
{
// Get list of modules
$modules = $this->getListModules();
if (!empty($modules))
{
// Get all modules files and include all translation files
$arr_files = $this->getAllModuleFiles($modules, null, $this->lang_selected->iso_code, true);
foreach ($arr_files as $value)
$this->findAndFillTranslations($value['files'], $value['theme'], $value['module'], $value['dir']);
$this->tpl_view_vars = array_merge($this->tpl_view_vars, array(
'default_theme_name' => self::DEFAULT_THEME_NAME,
'count' => $this->total_expression,
'limit_warning' => $this->displayLimitPostWarning($this->total_expression),
'mod_security_warning' => Tools::apacheModExists('mod_security'),
'textarea_sized' => AdminTranslationsControllerCore::TEXTAREA_SIZED,
'cancel_url' => $this->context->link->getAdminLink('AdminTranslations'),
'modules_translations' => isset($this->modules_translations) ? $this->modules_translations : array(),
'missing_translations' => $this->missing_translations
));
$this->initToolbar();
$this->base_tpl_view = 'translation_modules.tpl';
return parent::renderView();
}
}
/** Parse PDF class
*
* @param string $file_path file to parse
* @param string $file_type type of file
* @param array $langArray contains expression in the chosen language
* @param string $tab name to use with the md5 key
* @param array $tabs_array
* @return array containing all datas needed for building the translation form
* @since 1.4.5.0
*/
protected function parsePdfClass($file_path, $file_type, $lang_array, $tab, $tabs_array, &$count_missing)
{
// Get content for this file
$content = file_get_contents($file_path);
// Parse this content
$matches = $this->userParseFile($content, $this->type_selected, $file_type);
foreach ($matches as $key)
{
if (stripslashes(array_key_exists($tab.md5(addslashes($key)), $lang_array)))
$tabs_array[$tab][$key]['trad'] = html_entity_decode($lang_array[$tab.md5(addslashes($key))], ENT_COMPAT, 'UTF-8');
else
{
$tabs_array[$tab][$key]['trad'] = '';
if (!isset($count_missing[$tab]))
$count_missing[$tab] = 1;
else
$count_missing[$tab]++;
}
$tabs_array[$tab][$key]['use_sprintf'] = $this->checkIfKeyUseSprintf($key);
}
return $tabs_array;
}
/**
* This method generate the form for PDF translations
*/
public function initFormPDF()
{
$name_var = $this->translations_informations[$this->type_selected]['var'];
$GLOBALS[$name_var] = array();
$missing_translations_pdf = array();
$i18n_dir = $this->translations_informations[$this->type_selected]['dir'];
$default_i18n_file = $i18n_dir.$this->translations_informations[$this->type_selected]['file'];
if (!$this->theme_selected)
$i18n_file = $default_i18n_file;
else
{
$i18n_dir = $this->translations_informations[$this->type_selected]['override']['dir'];
$i18n_file = $i18n_dir.$this->translations_informations[$this->type_selected]['override']['file'];
}
$this->checkDirAndCreate($i18n_file);
if ((!file_exists($i18n_file) && !is_writable($i18n_dir)) && !is_writable($i18n_file))
$this->errors[] = sprintf(Tools::displayError('Cannot write into the "%s"'), $i18n_file);
@include($i18n_file);
// if the override's translation file is empty load the default file
if (!isset($GLOBALS[$name_var]) || count($GLOBALS[$name_var]) == 0)
@include($default_i18n_file);
$prefix_key = 'PDF';
$tabs_array = array($prefix_key => array());
$files_by_directory = $this->getFileToParseByTypeTranslation();
foreach ($files_by_directory as $type => $directories)
foreach ($directories as $dir => $files)
foreach ($files as $file)
if (!in_array($file, self::$ignore_folder) && Tools::file_exists_cache($file_path = $dir.$file))
{
if ($type == 'tpl')
{
if (Tools::file_exists_cache($file_path) && is_file($file_path))
{
// Get content for this file
$content = file_get_contents($file_path);
// Parse this content
$matches = $this->userParseFile($content, $this->type_selected, 'tpl');
foreach ($matches as $key)
{
if (isset($GLOBALS[$name_var][$prefix_key.md5($key)]))
$tabs_array[$prefix_key][$key]['trad'] = (html_entity_decode($GLOBALS[$name_var][$prefix_key.md5($key)], ENT_COMPAT, 'UTF-8'));
else
{
if (!isset($tabs_array[$prefix_key][$key]['trad']))
{
$tabs_array[$prefix_key][$key]['trad'] = '';
if (!isset($missing_translations_pdf[$prefix_key]))
$missing_translations_pdf[$prefix_key] = 1;
else
$missing_translations_pdf[$prefix_key]++;
}
}
$tabs_array[$prefix_key][$key]['use_sprintf'] = $this->checkIfKeyUseSprintf($key);
}
}
}
else
if (Tools::file_exists_cache($file_path))
$tabs_array = $this->parsePdfClass($file_path, 'php', $GLOBALS[$name_var], $prefix_key, $tabs_array, $missing_translations_pdf);
}
$this->tpl_view_vars = array_merge($this->tpl_view_vars, array(
'count' => count($tabs_array['PDF']),
'limit_warning' => $this->displayLimitPostWarning(count($tabs_array['PDF'])),
'mod_security_warning' => Tools::apacheModExists('mod_security'),
'tabsArray' => $tabs_array,
'cancel_url' => $this->context->link->getAdminLink('AdminTranslations'),
'missing_translations' => $missing_translations_pdf
));
$this->initToolbar();
$this->base_tpl_view = 'translation_form.tpl';
return parent::renderView();
}
/**
* recursively list files in directory $dir
*/
public function listFiles($dir, $list = array(), $file_ext = 'tpl')
{
$dir = rtrim($dir, '/').DIRECTORY_SEPARATOR;
$to_parse = scandir($dir);
// copied (and kind of) adapted from AdminImages.php
foreach ($to_parse as $file)
{
if (!in_array($file, self::$ignore_folder))
{
if (preg_match('#'.preg_quote($file_ext, '#').'$#i', $file))
$list[$dir][] = $file;
else if (is_dir($dir.$file))
$list = $this->listFiles($dir.$file, $list, $file_ext);
}
}
return $list;
}
protected function theme_exists($theme)
{
if (!is_array($this->themes))
$this->themes = Theme::getThemes();
$theme_exists = false;
foreach ($this->themes as $existing_theme)
if ($existing_theme->directory == $theme)
return true;
return false;
}
}