* @copyright 2007-2011 PrestaShop SA * @version Release: $Revision: 7715 $ * @license http://opensource.org/licenses/afl-3.0.php Academic Free License (AFL 3.0) * International Registered Trademark & Property of PrestaShop SA */ if (!defined('_CAN_LOAD_FILES_')) exit; class ThemeInstallator extends Module { /* ** Modules */ private $to_install = array(); private $to_enable = array(); private $to_disable = array(); private $to_export = array(); /* ** index */ private $page; /* ** Config File */ private $xml; public function __construct() { set_time_limit(0); ini_set('memory_limit', '2G'); $this->name = 'themeinstallator'; $this->version = '1.4'; $this->author = 'PrestaShop'; $this->need_instance = 0; if (version_compare(_PS_VERSION_, 1.4) >= 0) $this->tab = 'administration'; else $this->tab = 'Theme'; parent::__construct(); $this->displayName = $this->l('Import/export a theme'); $this->description = $this->l('Export or Install a theme and its modules on your shop.'); } private function getTheNativeModules() { $xml = simplexml_load_string(Tools::file_get_contents('http://www.prestashop.com/xml/modules_list.xml')); $natives = array(); if ($xml) foreach ($xml->modules as $row) foreach ($row->module as $row2) $natives[] = (string)$row2['name']; if (count($natives > 0)) return $natives; // use this list if we can't contact the prestashop.com server $natives = array('bankwire', 'birthdaypresent', 'blockadvertising', 'blockbestsellers', 'blockcart', 'blockcategories', 'blockcms', 'blockcurrencies', 'blockinfos', 'blocklanguages', 'blocklink', 'blockmanufacturer', 'blockmyaccount', 'blocknewproducts', 'blocknewsletter', 'blockpaymentlogo', 'blockpermanentlinks', 'blockrss', 'blocksearch', 'blockspecials', 'blocksupplier', 'blocktags', 'blockuserinfo', 'blockvariouslinks', 'blockviewed', 'blockwishlist', 'canonicalurl', 'cashondelivery', 'cheque', 'crossselling', 'dejala', 'editorial', 'feeder', 'followup', 'gadsense', 'ganalytics', 'gcheckout', 'graphartichow', 'graphgooglechart', 'graphvisifire', 'graphxmlswfcharts', 'gridhtml', 'gsitemap', 'hipay', 'homefeatured', 'loyalty', 'mailalerts', 'moneybookers', 'newsletter', 'pagesnotfound', 'paypal', 'paypalapi', 'productcomments', 'productscategory', 'producttooltip', 'referralprogram', 'reverso', 'sekeywords', 'sendtoafriend', 'statsbestcategories', 'statsbestcustomers', 'statsbestproducts', 'statsbestsuppliers', 'statsbestvouchers', 'statscarrier', 'statscatalog', 'statscheckup', 'statsdata', 'statsequipment', 'statsgeolocation', 'statshome', 'statslive', 'statsnewsletter', 'statsorigin', 'statspersonalinfos', 'statsproduct', 'statsregistrations', 'statssales', 'statssearch', 'themeinstallator', 'statsvisits', 'tm4b', 'trackingfront', 'watermark'); return $natives; } private function deleteDirectory($dirname) { $files = scandir($dirname); foreach ($files as $file) if ($file != '.' AND $file != '..') { if (is_dir($dirname.'/'.$file)) self::deleteDirectory($dirname.'/'.$file); elseif (file_exists($dirname.'/'.$file)) unlink($dirname.'/'.$file); } rmdir($dirname); } private function recurseCopy($src, $dst) { if (!$dir = opendir($src)) return; if (!file_exists($dst)) mkdir($dst); while (($file = readdir($dir)) !== false) if (strncmp($file, '.', 1) != 0) { if (is_dir($src.'/'.$file)) self::recurseCopy($src.'/'.$file, $dst.'/'.$file); elseif (is_readable($src.'/'.$file) AND $file != 'Thumbs.db' AND $file != '.DS_Store' AND substr($file, -1) != '~') copy($src.'/'.$file, $dst.'/'.$file); } closedir($dir); } /* ** Checks if module is installed ** Returns true if module is active ** Also returns false if it's a payment or stat module */ private function checkParentClass($name) { if (!$obj = Module::getInstanceByName($name)) return false; if (is_callable(array($obj, 'validateOrder'))) return false; if (is_callable(array($obj, 'getDateBetween'))) return false; if (is_callable(array($obj, 'getGridEngines'))) return false; if (is_callable(array($obj, 'getGraphEngines'))) return false; if (is_callable(array($obj, 'hookAdminStatsModules'))) return false; else return true; return false; } private function deleteTmpFiles() { if (file_exists(_IMPORT_FOLDER_.'doc')) self::deleteDirectory(_IMPORT_FOLDER_.'doc'); if (file_exists(_IMPORT_FOLDER_.XMLFILENAME)) unlink(_IMPORT_FOLDER_.XMLFILENAME); if (file_exists(_IMPORT_FOLDER_.'modules')) self::deleteDirectory(_IMPORT_FOLDER_.'modules'); if (file_exists(_IMPORT_FOLDER_.'themes')) self::deleteDirectory(_IMPORT_FOLDER_.'themes'); if (file_exists(_EXPORT_FOLDER_.'archive.zip')) unlink(_EXPORT_FOLDER_.'archive.zip'); } private function init_defines() { global $currentIndex, $cookie; define('_EXPORT_FOLDER_', dirname(__FILE__).'/export/'); define('_IMPORT_FOLDER_', dirname(__FILE__).'/import/'); $this->page = 1; if (!file_exists(_EXPORT_FOLDER_) OR !is_dir(_EXPORT_FOLDER_)) mkdir(_EXPORT_FOLDER_, 0777); if (!file_exists(_IMPORT_FOLDER_) OR !is_dir(_IMPORT_FOLDER_)) mkdir(_IMPORT_FOLDER_, 0777); if (!Tools::isSubmit('cancelExport') AND (Tools::isSubmit('exportTheme') OR Tools::isSubmit('submitExport'))) $this->page = 'exportPage'; $this->_html = '
'; if (Tools::isSubmit('modulesToExport') OR Tools::isSubmit('submitModules')) $this->to_export = Tools::getValue('modulesToExport'); if (Tools::isSubmit('submitThemes')) $this->selectedVariations = Tools::getValue('variation'); $_POST = @array_map('trim', $_POST); define('DEFAULT_COMPATIBILITY_FROM', '1.0.0.0'); define('DEFAULT_COMPATIBILITY_TO', _PS_VERSION_); define('DEFAULT_T_VER', '1.0'); define('MAX_NAME_LENGTH', 32); define('MAX_EMAIL_LENGTH', 128); define('MAX_WEBSITE_LENGTH', 64); define('MAX_DESCRIPTION_LENGTH', 64); define('MAX_T_VER_LENGTH', 3); define('MAX_COMPATIBILITY_VER', 7); define('ARCHIVE_NAME', _IMPORT_FOLDER_.'uploaded.zip'); define('XMLFILENAME', 'Config.xml'); $this->_msg = ''; $this->to_enable = false; $this->to_disable = false; $this->to_install = false; $this->errors = false; if ($this->page == 'exportPage' AND Tools::isSubmit('exportTheme') AND ($theme = Tools::getValue('mainTheme'))) if (!(is_dir(_PS_ALL_THEMES_DIR_.$theme) AND file_exists(_PS_ALL_THEMES_DIR_.$theme.'/index.tpl') AND $theme != 'prestashop')) { $this->page = 1; echo parent::displayError(''.$theme.' is not a valid theme to export'); } } private function handleInformations() { if (Tools::isSubmit('submitImport1')) { if ($_FILES['themearchive']['error'] OR !file_exists($_FILES['themearchive']['tmp_name'])) $this->errors .= parent::displayError($this->l('An error has occurred during the file upload.')); elseif (substr($_FILES['themearchive']['name'], -4) != '.zip') $this->errors .= parent::displayError($this->l('Only zip files are allowed')); elseif (!rename($_FILES['themearchive']['tmp_name'], ARCHIVE_NAME)) $this->errors .= parent::displayError($this->l('An error has occurred during the file copy.')); elseif (Tools::ZipTest(ARCHIVE_NAME)) $this->page = 2; else $this->errors .= parent::displayError($this->l('Zip file seems to be broken')); } elseif (Tools::isSubmit('submitImport2')) { if (!Validate::isModuleUrl($url = Tools::getValue('linkurl'), $this->errors)) $this->errors .= parent::displayError($this->l('Only zip files are allowed')); elseif (!copy($url, ARCHIVE_NAME)) $this->errors .= parent::displayError($this->l('Error during the file download')); elseif (Tools::ZipTest(ARCHIVE_NAME)) $this->errors .= parent::displayError($this->l('Zip file seems to be broken')); else $this->page = 2; } elseif (Tools::isSubmit('submitImport3')) { $filename = _IMPORT_FOLDER_.Tools::getValue('ArchiveName'); if (substr($filename, -4) != '.zip') $this->errors .= parent::displayError($this->l('Only zip files are allowed')); elseif (!copy($filename, ARCHIVE_NAME)) $this->errors .= parent::displayError($this->l('An error has occurred during the file copy.')); elseif (Tools::ZipTest(ARCHIVE_NAME)) $this->page = 2; else $this->errors .= parent::displayError($this->l('Zip file seems to be broken')); } elseif (Tools::isSubmit('prevThemes')) $this->page = 2; elseif (Tools::isSubmit('submitThemes')) $this->page = 3; elseif (Tools::isSubmit('submitModules')) $this->page = 4; if ($this->page == 2 AND file_exists(ARCHIVE_NAME)) { if (!Tools::ZipExtract(ARCHIVE_NAME, _IMPORT_FOLDER_)) { $this->errors .= parent::displayError($this->l('Error during zip extraction')); $this->page = 1; } } if (file_exists(ARCHIVE_NAME)) unlink(ARCHIVE_NAME); if ($this->page != 1) { if (!self::checkXmlFields()) { $this->errors .= parent::displayError($this->l('Bad configuration file')); $this->page = 1; } else return ; } self::deleteTmpFiles(); } private function checkXmlFields() { if (!file_exists(_IMPORT_FOLDER_.XMLFILENAME) OR !$xml = simplexml_load_file(_IMPORT_FOLDER_.XMLFILENAME)) return false; if (!$xml['version'] OR !$xml['name']) return false; foreach ($xml->variations->variation as $val) if (!$val['name'] OR !$val['directory'] OR !$val['from'] OR !$val['to']) return false; foreach ($xml->modules->module as $val) if (!$val['action'] OR !$val['name']) return false; foreach ($xml->modules->hooks->hook as $val) if (!$val['module'] OR !$val['hook'] OR !$val['position']) return false; return true; } public function getContentExport() { $this->theme_list = scandir(_PS_ALL_THEMES_DIR_); $this->error = false; self::getModuleState(); self::displayInformations(); if (Tools::isSubmit('submitExport') AND $this->error === false AND $this->checkPostedDatas() == true) { self::getThemeVariations(); self::getDocumentation(); self::getHookState(); self::getImageState(); self::generateXML(); self::generateArchive(); } self::AuthorInformationForm(); self::ModulesInformationForm(); self::ThemeInformationForm(); self::DocInformationForm(); self::VariationInformationForm(); return $this->_html; } /* ** Main function */ public function getContent() { self::init_defines(); if (!Tools::isSubmit('cancelExport') AND $this->page == 'exportPage') return self::getContentExport(); self::handleInformations(); switch ($this->page) { case 1: self::_displayForm1(); break; case 2: self::_displayForm2(); break; case 3: self::_displayForm3(); break; case 4: self::_displayForm4(); break; } return $this->errors.$this->_msg.$this->_html; } /* ** Checker si le dossier doc existe : Si oui appeler la fonction ! */ private function _loadDocForm() { $docname = array(); $docpath = array(); foreach ($this->xml->docs->doc as $row) { $docname[] = strval($row['name']); $docpath[] = strval($row['path']); } $doc = '
'.$this->l('Documentation').'
    '; $i = 0; foreach ($docname as $row) $doc .= '
  • '.$row.''; $doc .= '

'.$this->l('Right click on the name and choose "save link as"').'

 

'; $this->_html .= $doc; } private function getModules() { $this->native_modules = self::getTheNativeModules(); foreach ($this->xml->modules->module as $row) { if (strval($row['action']) == 'install' AND !in_array(strval($row['name']), $this->native_modules)) $this->to_install[] = strval($row['name']); elseif (strval($row['action']) == 'enable') $this->to_enable[] = strval($row['name']); elseif (strval($row['action']) == 'disable') $this->to_disable[] = strval($row['name']); } } private function updateImages() { foreach ($this->xml->images->image as $row) { $foo = Db::getInstance()->getRow(' SELECT name FROM `'._DB_PREFIX_.'image_type` i WHERE i.name LIKE \''.pSQL($row['name']).'\''); if ((int)(Tools::getValue('imagesConfig')) == 1 AND $foo) continue ; if ($foo) Db::getInstance()->Execute(' UPDATE `'._DB_PREFIX_.'image_type` i SET `width` = '.(int)($row['width']).', `height` = '.(int)($row['height']).', `products` = '.($row['products'] == 'true' ? 1 : 0).', `categories` = '.($row['categories'] == 'true' ? 1 : 0).', `manufacturers` = '.($row['manufacturers'] == 'true' ? 1 : 0).', `suppliers` = '.($row['suppliers'] == 'true' ? 1 : 0).', `scenes` = '.($row['scenes'] == 'true' ? 1 : 0).' WHERE i.name LIKE \''.pSQL($row['name']).'\''); else Db::getInstance()->Execute(' INSERT INTO `'._DB_PREFIX_.'image_type` (`name`, `width`, `height`, `products`, `categories`, `manufacturers`, `suppliers`, `scenes`) VALUES ( \''.pSQL($row['name']).'\', '.(int)($row['width']).', '.(int)($row['height']).', '.($row['products'] == 'true' ? 1 : 0).', '.($row['categories'] == 'true' ? 1 : 0).', '.($row['manufacturers'] == 'true' ? 1 : 0).', '.($row['suppliers'] == 'true' ? 1 : 0).', '.($row['scenes'] == 'true' ? 1 : 0).')'); } return true; } private function _displayForm4() { $xml = simplexml_load_file(_IMPORT_FOLDER_.XMLFILENAME); $this->xml = $xml; self::getModules(); $hook = array(); $hookedModule = array(); $position = array(); $msg = ''; foreach ($this->xml->modules->hooks->hook as $row) { $hookedModule[] = strval($row['module']); $hook[] = strval($row['hook']); $position[] = strval($row['position']); $exceptions[] = (isset($row['exceptions']) ? explode(',', strval($row['exceptions'])) : array()); } if (file_exists(_IMPORT_FOLDER_.'doc') AND sizeof($xml->docs->doc) != 0) self::_loadDocForm(); // install selected modules $flag = 0; if (isset($this->to_export) AND $this->to_export) foreach ($this->to_export as $row) { if (in_array($row, $this->native_modules)) continue; if ($flag++ == 0) $msg .= ''.$this->l('The following modules have been installed').' :
'; self::recurseCopy(_IMPORT_FOLDER_.'modules/'.$row, _PS_ROOT_DIR_.'/modules/'.$row); $obj = Module::getInstanceByName($row); if (Validate::isLoadedObject($obj)) Db::getInstance()->Execute(' UPDATE `'._DB_PREFIX_.'module` SET `active`= 1 WHERE `name` = \''.pSQL($row).'\''); elseif (!$obj OR !$obj->install()) continue; $msg .= '- '.pSQL($row).'
'; Db::getInstance()->Execute(' DELETE FROM `'._DB_PREFIX_.'hook_module` WHERE `id_module` = '.pSQL($obj->id)); $count = -1; while (isset($hookedModule[++$count])) if ($hookedModule[$count] == $row) { Db::getInstance()->Execute(' INSERT INTO `'._DB_PREFIX_.'hook_module` (`id_module`, `id_hook`, `position`) VALUES ('.(int)$obj->id.', '.(int)Hook::get($hook[$count]).', '.(int)$position[$count].')'); if ($exceptions[$count]) foreach ($exceptions[$count] as $file_name) Db::getInstance()->Execute(' INSERT INTO `'._DB_PREFIX_.'hook_module_exceptions` (`id_module`, `id_hook`, `file_name`) VALUES ('.(int)$obj->id.', '.(int)Hook::get($hook[$count]).', "'.pSQL($file_name).'")'); } } if (($val = (int)(Tools::getValue('nativeModules'))) != 1) { $flag = 0; // Disable native modules if ($val == 2 AND $this->to_disable AND count($this->to_disable)) foreach ($this->to_disable as $row) { $obj = Module::getInstanceByName($row); if (Validate::isLoadedObject($obj)) { if ($flag++ == 0) $msg .= ''.$this->l('The following modules have been disabled').' :
'; $msg .= '- '.pSQL($row).'
'; Db::getInstance()->Execute(' UPDATE `'._DB_PREFIX_.'module` SET `active`= 0 WHERE `name` = \''.pSQL($row).'\''); } } $flag = 0; if ($this->to_enable AND count($this->to_enable)) foreach ($this->to_enable as $row) { $obj = Module::getInstanceByName($row); if (Validate::isLoadedObject($obj)) Db::getInstance()->Execute(' UPDATE `'._DB_PREFIX_.'module` SET `active`= 1 WHERE `name` = \''.pSQL($row).'\''); elseif (!is_object($obj) OR !$obj->install()) continue ; if ($flag++ == 0) $msg .= ''.$this->l('The following modules have been enabled').' :
'; $msg .= '- '.pSQL($row).'
'; Db::getInstance()->Execute(' DELETE FROM `'._DB_PREFIX_.'hook_module` WHERE `id_module` = '.pSQL($obj->id)); $count = -1; while (isset($hookedModule[++$count])) if ($hookedModule[$count] == $row) { Db::getInstance()->Execute(' INSERT INTO `'._DB_PREFIX_.'hook_module` (`id_module`, `id_hook`, `position`) VALUES ('.(int)$obj->id.', '.(int)Hook::get($hook[$count]).', '.(int)$position[$count].')'); foreach ($exceptions[$count] as $filename) if (!empty($filename)) Db::getInstance()->Execute(' INSERT INTO `'._DB_PREFIX_.'hook_module_exceptions` (`id_module`, `id_hook`, `file_name`) VALUES ('.(int)$obj->id.', '.(int)Hook::get($hook[$count]).', "'.pSQL($filename).'")'); } } } if ((int)(Tools::getValue('imagesConfig')) != 3 AND self::updateImages()) $msg .= '
'.$this->l('Images have been correctly updated in database').'
'; $this->_msg .= parent::displayConfirmation($msg); $this->_html .= '
'; } private function _displayForm3() { $xml = simplexml_load_file(_IMPORT_FOLDER_.XMLFILENAME); $this->xml = $xml; if ($this->selectedVariations AND count($this->selectedVariations) > 0) { $ok = array(); foreach ($this->selectedVariations as $count => $theme) { if ($theme == 'prestashop') continue; self::recurseCopy(_IMPORT_FOLDER_.'themes/'.$theme, _PS_ALL_THEMES_DIR_.$theme); if (file_exists(_PS_ALL_THEMES_DIR_.$theme)) $ok[] = $theme; } if (sizeof($ok) > 0) { $msg = $this->l('The following themes were successfully imported').':'; $this->_msg = parent::displayConfirmation($msg); } } self::getModules(); if (file_exists(_IMPORT_FOLDER_.'doc') AND sizeof($xml->docs->doc) != 0) self::_loadDocForm(); $this->_html .= '
'; if ($this->to_install AND sizeof($this->to_install) > 0) { $var = ''; foreach ($this->to_install as $row) if (file_exists(_IMPORT_FOLDER_.'modules/'.$row)) $var .= '
'; if ($var != '') $this->_html .= '
'.$this->l('Select the theme\'s modules you wish to install').'

'.$var.'

 

'; } $this->_html .= '
'.$this->l('Native modules configuration').'

 

'; $this->_html .= '
'.$this->l('Theme images configuration').'
'; $this->_html .= '

 

'; } private function _displayForm2() { global $cookie; $iso = Language::getIsoById((int)($cookie->id_lang)); $xml = simplexml_load_file(_IMPORT_FOLDER_.XMLFILENAME); $this->xml = $xml; $res = $xml->xpath('/theme/descriptions/description[@iso="'.$iso.'"]'); $description = (isset($res[0]) ? (string)$res[0] : ''); $this->_msg = parent::displayConfirmation( $this->l('You are going to install the following theme').' :
'. $xml['name'].' v'.$xml['version'].'
'. (strlen($description) ? ''.$description.'
' : ''). $this->l('This theme is for Prestashop').' v'. $xml->variations->variation[0]['from']. ' -> v'.$xml->variations->variation[0]['to'].'
'. (file_exists(_PS_ALL_THEMES_DIR_.strval($xml->variations->variation[0]['directory'])) ? $this->l('Warning : You already have a theme with the same folder\'s name') : '').'
'); if (file_exists(_IMPORT_FOLDER_.'doc') AND sizeof($xml->docs->doc) != 0) self::_loadDocForm(); if (sizeof($xml->variations->variation) > 1) { $count = 0; $var = ''; while ($xml->variations->variation[++$count]) { $foo = (file_exists(_PS_ALL_THEMES_DIR_.strval($xml->variations->variation[$count]['directory'])) ? 1 : 0); $var .= '
'; } $this->_html .= '
'.$this->l('Choose the variations that you wish to import').'

'.$this->l('Uncheck this field if you do not want to install the main theme.').'

'.$this->l('Select the variations you wish to import').'

'.$var.'

'; } else $this->_html .= ' '; $this->_html .= ''; } private function _displayForm1() { $tmp = scandir(_PS_ALL_THEMES_DIR_); $themeList = array(); foreach ($tmp as $row) if (is_dir(_PS_ALL_THEMES_DIR_.$row) AND file_exists(_PS_ALL_THEMES_DIR_.$row.'/index.tpl') AND $row != 'prestashop') $themeList[] = $row; if (count($themeList) > 0) { $tmp = ''; foreach ($themeList as $row) $tmp .= ''; $this->_html .= '
'.$this->l('Export a theme').'
 
'; } $this->_html .= '
'.$this->l('Import from your computer').'

'.$this->l('Where is your zip file?').'

 
'; $this->_html .= '
'.$this->l('Import from the web').'
 
'; $tmp = scandir(_IMPORT_FOLDER_); $list = array(); foreach ($tmp as $row) if (substr(_IMPORT_FOLDER_.$row, -4) == '.zip') $list[] = $row; $tmp = ''; foreach ($list as $row) $tmp .= ''; $this->_html .= '
'.$this->l('Import from FTP').'
 
'; } /* ** EXPORT FUNCTIONS ######################################## */ private function displayInformations() { $this->_html .= ''; if ($this->error === false AND class_exists('ZipArchive', false) AND ($zip = new ZipArchive())) { if (!($zip->open(_EXPORT_FOLDER_.'archive.zip', ZipArchive::OVERWRITE) === true) OR !$zip->addEmptyDir('test') === true) $this->_html .= parent::displayError('Permission denied. Please set permisssion to 666 on this folder: '._EXPORT_FOLDER_); $zip->close(); if ($this->error === false) $this->_html .= parent::displayConfirmation('Fill this formular to export this theme: '.Tools::getValue('mainTheme').' in a ZIP file'); } } private function archiveThisFile($obj, $file, $serverPath, $archivePath) { if (is_dir($serverPath.$file)) { $dir = scandir($serverPath.$file); foreach ($dir as $row) if ($row != '.' AND $row != '..') $this->archiveThisFile($obj, $row, $serverPath.$file.'/', $archivePath.$file.'/'); } elseif (!$obj->addFile($serverPath.$file, $archivePath.$file)) $this->error = true; } /* ** Generate Archive ! */ private function generateArchive() { $count = 0; $zip = new ZipArchive(); $zip_file_name = md5(time()).".zip"; if ($zip->open(_EXPORT_FOLDER_.$zip_file_name, ZipArchive::OVERWRITE) === true) { if (!$zip->addFromString('Config.xml', $this->xmlFile)) $this->error = true; while (isset($_FILES['mydoc_'.++$count])) { if (!$_FILES['mydoc_'.$count]['name']) continue; if (!$zip->addFile($_FILES['mydoc_'.$count]['tmp_name'], 'doc/'.$_FILES['mydoc_'.$count]['name'])) $this->error = true; } foreach ($this->variations as $row) { $array = explode('¤', $row); $this->archiveThisFile($zip, $array[1], _PS_ALL_THEMES_DIR_, 'themes/'); } foreach ($this->to_export as $row) if (!in_array($row, $this->native_modules)) $this->archiveThisFile($zip, $row, dirname(__FILE__).'/../../modules/', 'modules/'); $zip->close(); if ($this->error === false) { ob_end_clean(); header('Content-Type: multipart/x-zip'); header('Content-Disposition:attachment;filename="'.$zip_file_name.'"'); readfile(_EXPORT_FOLDER_.$zip_file_name); unlink(_EXPORT_FOLDER_.$zip_file_name); die ; } } $this->_html .= parent::displayError($this->l('An error occurred during the archive generation')); } /* ** XML Generation, all vars should be GOOD at this point */ private function generateXML() { $theme = new SimpleXMLElement(''); $theme->addAttribute('version', Tools::getValue('version')); $theme->addAttribute('name', Tools::htmlentitiesUTF8(Tools::getValue('theme_name'))); $author = $theme->addChild('author'); $author->addAttribute('name', Tools::htmlentitiesUTF8(Tools::getValue('author_name'))); $author->addAttribute('email', Tools::htmlentitiesUTF8(Tools::getValue('email'))); $author->addAttribute('url', Tools::htmlentitiesUTF8(Tools::getValue('website'))); $descriptions = $theme->addChild('descriptions'); $languages = Language::getLanguages(); foreach ($languages as $language) { $val = Tools::htmlentitiesUTF8(Tools::getValue('body_title_'.$language['id_lang'])); $description = $descriptions->addChild('description', Tools::htmlentitiesUTF8($val)); $description->addAttribute('iso', $language['iso_code']); } $variations = $theme->addChild('variations'); foreach ($this->variations as $row) { $array = explode('¤', $row); $variation = $variations->addChild('variation'); $variation->addAttribute('name', Tools::htmlentitiesUTF8($array[0])); $variation->addAttribute('directory', $array[1]); $variation->addAttribute('from', $array[2]); $variation->addAttribute('to', $array[3]); } $docs = $theme->addChild('docs'); if (isset($this->user_doc)) foreach ($this->user_doc as $row) { $array = explode('¤', $row); $doc = $docs->addChild('doc'); $doc->addAttribute('name', $array[0]); $doc->addAttribute('path', $array[1]); } $modules = $theme->addChild('modules'); if (isset($this->to_export)) foreach ($this->to_export as $row) if (!in_array($row, $this->native_modules)) { $module = $modules->addChild('module'); $module->addAttribute('action', 'install'); $module->addAttribute('name', $row); } foreach ($this->to_enable as $row) { $module = $modules->addChild('module'); $module->addAttribute('action', 'enable'); $module->addAttribute('name', $row); } foreach ($this->to_disable as $row) { $module = $modules->addChild('module'); $module->addAttribute('action', 'disable'); $module->addAttribute('name', $row); } $hooks = $modules->addChild('hooks'); foreach ($this->to_hook as $row) { $array = explode(';', $row); $hook = $hooks->addChild('hook'); $hook->addAttribute('module', $array[0]); $hook->addAttribute('hook', $array[1]); $hook->addAttribute('position', $array[2]); if (!empty($array[3])) $hook->addAttribute('exceptions', $array[3]); } $images = $theme->addChild('images'); foreach ($this->image_list as $row) { $array = explode(';', $row); $image = $images->addChild('image'); $image->addAttribute('name', Tools::htmlentitiesUTF8($array[0])); $image->addAttribute('width', $array[1]); $image->addAttribute('height', $array[2]); $image->addAttribute('products', $array[3]); $image->addAttribute('categories', $array[4]); $image->addAttribute('manufacturers', $array[5]); $image->addAttribute('suppliers', $array[6]); $image->addAttribute('scenes', $array[7]); } $this->xmlFile = $theme->asXML(); } /* ** Init modules and Hooks */ private function _initList() { $this->native_modules = self::getTheNativeModules(); $this->module_list = Db::getInstance()->ExecuteS('SELECT id_module, name, active FROM `'._DB_PREFIX_.'module`'); $this->hook_list = Db::getInstance()->ExecuteS(' SELECT a.id_hook, a.name as name_hook, c.position, c.id_module, d.name as name_module, GROUP_CONCAT(hme.file_name, ",") as exceptions FROM `'._DB_PREFIX_.'hook` a LEFT JOIN `'._DB_PREFIX_.'hook_module` c ON c.id_hook = a.id_hook LEFT JOIN `'._DB_PREFIX_.'module` d ON c.id_module = d.id_module LEFT OUTER JOIN `'._DB_PREFIX_.'hook_module_exceptions` hme ON (hme.id_module = c.id_module AND hme.id_hook = a.id_hook) GROUP BY id_module, id_hook ORDER BY name_module'); foreach ($this->hook_list as &$row) $row['exceptions'] = trim(preg_replace('/(,,+)/', ',', $row['exceptions']), ','); } /* ** Fill module's vars */ private function getModuleState() { self::_initList(); foreach ($this->module_list as $array) { if (!self::checkParentClass($array['name'])) continue ; if (in_array($array['name'], $this->native_modules)) { if ($array['active'] == 1) $this->to_enable[] = $array['name']; else $this->to_disable[] = $array['name']; } elseif ($array['active'] == 1) $this->to_install[] = $array['name']; } foreach ($this->native_modules as $str) { $flag = 0; if (!self::checkParentClass($str)) continue ; foreach ($this->module_list as $tmp) if (in_array($str, $tmp)) { $flag = 1; break; } if ($flag == 0) $this->to_disable[] = $str; } } /* ** Fill Hook Var */ private function getHookState() { foreach ($this->to_install as $string) foreach ($this->hook_list as $tmp) if ($tmp['name_module'] == $string) $this->to_hook[] = $string.';'.$tmp['name_hook'].';'.$tmp['position'].';'.$tmp['exceptions']; foreach ($this->to_enable as $string) foreach ($this->hook_list as $tmp) if ($tmp['name_module'] == $string) $this->to_hook[] = $string.';'.$tmp['name_hook'].';'.$tmp['position'].';'.$tmp['exceptions']; } /* ** Fill Image var */ private function getImageState() { $table = Db::getInstance()->ExecuteS('SELECT name, width, height, products, categories, manufacturers, suppliers, scenes FROM `'._DB_PREFIX_.'image_type`'); foreach ($table as $row) $this->image_list[] = $row['name'].';'.$row['width'].';'.$row['height'].';'. ($row['products'] == 1 ? 'true' : 'false').';'. ($row['categories'] == 1 ? 'true' : 'false').';'. ($row['manufacturers'] == 1 ? 'true' : 'false').';'. ($row['suppliers'] == 1 ? 'true' : 'false').';'. ($row['scenes'] == 1 ? 'true' : 'false'); } /* ** Takes current and submited theme's informations */ private function getThemeVariations() { $from = Tools::getValue('compa_from'); $to = Tools::getValue('compa_to'); $this->variations[] = Tools::getValue('theme_name').'¤'.Tools::getValue('mainTheme').'¤'.$from.'¤'.$to; $count = 0; while (Tools::isSubmit('myvar_'.++$count)) { if ((int)(Tools::getValue('myvar_'.$count)) == -1) continue ; $name = Tools::getValue('themevariationname_'.$count); $dir = Tools::getValue('myvar_'.$count); $from = Tools::getValue('compafrom_'.$count); $to = Tools::getValue('compato_'.$count); $this->variations[] = $name.'¤'.$dir.'¤'.$from.'¤'.$to; } } private function getDocumentation() { $count = 0; while (Tools::isSubmit('documentationName_'.++$count)) { if (!($filename = Tools::htmlentitiesUTF8($_FILES['mydoc_'.$count]['name']))) continue ; $name = Tools::htmlentitiesUTF8(Tools::getValue('documentationName_'.$count)); $this->user_doc[] = $name.'¤'.'doc/'.$filename; } } private function checkPostedDatas() { $mail = Tools::getValue('email'); $website = Tools::getValue('website'); if ($mail AND !preg_match('#^[\w.-]+@[\w.-]+\.[a-zA-Z]{2,6}$#', $mail)) $this->_html .= parent::displayError($this->l('There is an error in your e-mail syntax!')); elseif ($website AND (!Validate::isURL($website) OR !Validate::isAbsoluteUrl($website))) $this->_html .= parent::displayError($this->l('There is an error in your URL syntax!')); elseif (!$this->checkVersionsAndCompatibility() OR !$this->checkNames() OR !$this->checkDocumentation()) return false; else return true; return false; } /* ** Checks posted documentation */ private function checkDocumentation() { $count = 0; $extensions = array('.pdf', '.txt'); while ($this->error == false AND isset($_FILES['mydoc_'.++$count])) { if (!$_FILES['mydoc_'.$count]['name']) continue; $extension = strrchr($_FILES['mydoc_'.$count]['name'], '.'); $name = Tools::getValue('documentationName_'.$count); if (!in_array($extension, $extensions)) $this->_html .= parent::displayError($this->l('File extension must be .txt or .pdf')); elseif ($_FILES['mydoc_'.$count]['error'] > 0 OR $_FILES['mydoc_'.$count]['size'] > 1048576) $this->_html .= parent::displayError($this->l('An error occurred during documentation upload')); elseif (!$name OR !Validate::isGenericName($name) OR strlen($name) > MAX_NAME_LENGTH) $this->_html .= parent::displayError($this->l('Please enter a valid documentation name')); } if ($this->error == true) return false; return true; } /* ** Checks theme's and author's name syntax, existence and length */ private function checkNames() { $author = Tools::getValue('author_name'); $name = Tools::getValue('theme_name'); $count = 0; if (!$author OR !Validate::isGenericName($author) OR strlen($author) > MAX_NAME_LENGTH) $this->_html .= parent::displayError($this->l('Please enter a valid author name')); elseif (!$name OR !Validate::isGenericName($name) OR strlen($name) > MAX_NAME_LENGTH) $this->_html .= parent::displayError($this->l('Please enter a valid theme name')); while ($this->error === false AND Tools::isSubmit('myvar_'.++$count)) { if ((int)(Tools::getValue('myvar_'.$count)) == -1) continue ; $name = Tools::getValue('themevariationname_'.$count); if (!$name OR !Validate::isGenericName($name) OR strlen($name) > MAX_NAME_LENGTH) $this->_html .= parent::displayError($this->l('Please enter a valid theme variation name')); } if ($this->error == true) return false; return true; } private function checkVersionsAndCompatibility() { $count = 0; $exp = "#^[0-9]+[.]+[0-9.]*[0-9]$#"; if (!preg_match("#^[0-9][.][0-9]$#", Tools::getValue('version')) OR !preg_match($exp, Tools::getValue('compa_from')) OR !preg_match($exp, Tools::getValue('compa_to')) OR version_compare(Tools::getValue('compa_from'), Tools::getValue('compa_to')) == 1) $this->_html .= parent::displayError($this->l('Syntax error on version field. Only digits and points are allowed and the compatibility should be increasing or equal.')); while ($this->error === false AND Tools::isSubmit('myvar_'.++$count)) { if ((int)(Tools::getValue('myvar_'.$count)) == -1) continue ; $from = Tools::getValue('compafrom_'.$count); $to = Tools::getValue('compato_'.$count); if (!preg_match($exp, $from) OR !preg_match($exp, $to) OR version_compare($from, $to) == 1) $this->_html .= parent::displayError($this->l('Syntax error on version. Only digits and points are allowed and compatibility should be increasing or equal.')); } if ($this->error == true) return false; return true; } private function ModulesInformationForm() { if ($this->to_install AND count($this->to_install)) { $tmp = ''; foreach ($this->to_install as $key => $val) $tmp .= 'to_export) ? 'checked="checked"' : "").'/>
'; $this->_html .= '
'.$this->l('Modules').'
'.$tmp.'

 

'; } } private function AuthorInformationForm() { global $cookie; $employee = new Employee((int)$cookie->id_employee); $mail = Tools::getValue('email') ? Tools::htmlentitiesUTF8(Tools::getValue('email')) : Tools::htmlentitiesUTF8($cookie->email); $author = Tools::getValue('author_name') ? Tools::htmlentitiesUTF8(Tools::getValue('author_name')) : Tools::htmlentitiesUTF8(($employee->firstname).' '.$employee->lastname); $website = Tools::getValue('website') ? Tools::htmlentitiesUTF8(Tools::getValue('website')) : Tools::getHttpHost(true); $this->_html .= '
'.$this->l('Author').'
 
'; } private function ThemeInformationForm() { global $cookie; $defaultLanguage = (int)(Configuration::get('PS_LANG_DEFAULT')); $languages = Language::getLanguages(); $iso = Language::getIsoById((int)($cookie->id_lang)); $divLangName = 'title'; $val = Tools::getValue('theme_name') ? Tools::getValue('theme_name') : Tools::getValue('mainTheme'); $this->_html .= '
'.$this->l('Theme').'

'.$this->l('Your theme\'s name').'

'; foreach ($languages as $language) { $val = Tools::htmlentitiesUTF8(Tools::getValue('body_title_'.$language['id_lang'])); $this->_html .= '
'; } $this->_html .= $this->displayFlags($languages, $defaultLanguage, $divLangName, 'title', true); $this->_html .= '

'.$this->l('Enter a short description of your theme').'

'; $val = Tools::getValue('version') ? Tools::getValue('version') : DEFAULT_T_VER; $this->_html .= '

'.$this->l('Your theme\'s version').'

'; $val = Tools::getValue('compa_from') ? Tools::getValue('compa_from') : DEFAULT_COMPATIBILITY_FROM; $val2 = Tools::getValue('compa_to') ? Tools::getValue('compa_to') : DEFAULT_COMPATIBILITY_TO; $this->_html .= '

 

'; } private function DocInformationForm() { $val = Tools::htmlentitiesUTF8(Tools::getValue('documentation')); $this->_html .= '

'. $this->l('Give the user some help. Add a field by clicking here').' add.
'. $this->l('File extension must be .txt or .pdf').'

'; $this->_html .= '

 

'; } private function VariationInformationForm() { $script = _MODULE_DIR_.$this->name.'/script.js'; $this->_html .= '

'.$this->l('Select theme to include and its compatibility.').'

 
'; $this->_html .= '
 
  '; } }