* @copyright Copyright (C) 2008-2018, Rémi Jean * @license CC Attribution-NonCommercial-NoDerivatives 4.0 International * @link http://zwiicms.fr/ * @copyright : Frédéric Tempez * @copyright Copyright (C) 2018-2025, Frédéric Tempez */ class theme extends common { public static $actions = [ 'advanced' => self::GROUP_ADMIN, 'body' => self::GROUP_ADMIN, 'footer' => self::GROUP_ADMIN, 'header' => self::GROUP_ADMIN, 'index' => self::GROUP_ADMIN, 'menu' => self::GROUP_ADMIN, 'reset' => self::GROUP_ADMIN, 'site' => self::GROUP_ADMIN, 'admin' => self::GROUP_ADMIN, 'manage' => self::GROUP_ADMIN, 'export' => self::GROUP_ADMIN, 'import' => self::GROUP_ADMIN, 'save' => self::GROUP_ADMIN, 'font' => self::GROUP_ADMIN, 'fontAdd' => self::GROUP_ADMIN, 'fontEdit' => self::GROUP_ADMIN, 'fontDelete' => self::GROUP_ADMIN ]; public static $aligns = [ 'left' => 'À gauche', 'center' => 'Au centre', 'right' => 'À droite' ]; public static $attachments = [ 'scroll' => 'Standard', 'fixed' => 'Fixe' ]; public static $containerWides = [ 'container' => 'Limitée au site', 'none' => 'Étendu sur la page' ]; public static $footerblocks = [ 1 => [ 'hide' => 'Masqué', 'center' => 'Affiché' ], 2 => [ 'hide' => 'Masqué', 'left' => 'À gauche', 'right' => 'À droite' ], 3 => [ 'hide' => 'Masqué', 'left' => 'À gauche', 'center' => 'Au centre', 'right' => 'À droite' ], 4 => [ 'hide' => 'Masqué', 'left' => 'En haut', 'center' => 'Au milieu', 'right' => 'En bas' ] ]; public static $fontWeights = [ 'normal' => 'Maigre', 'bold' => 'Gras' ]; public static $footerHeights = [ '0px' => '0px', '5px' => '5px', '10px' => '10px', '15px' => '15px', '20px' => '20px' ]; public static $footerPositions = [ 'hide' => 'Caché', 'site' => 'Dans le site', 'body' => 'En dessous du site' ]; public static $footerFontSizes = [ '.8em' => '80%', '.9em' => '90%', '1em' => 'Standard (100%)', '1.1em' => '110%', '1.2em' => '120%', '1.3em' => '130%' ]; public static $headerFontSizes = [ '1.6em' => '160%', '1.8em' => '180%', '2em' => '200%', '2.2em' => '220%', '2.4vmax' => '240%' ]; public static $headerHeights = [ 'unset' => 'Libre', // texte dynamique cf header.js.php '100px' => '100px', '150px' => '150px', '200px' => '200px', '300px' => '300px', '400px' => '400px', ]; public static $headerPositions = [ 'body' => 'Au-dessus du site', 'site' => 'Dans le site', 'hide' => 'Cachée' ]; public static $headerFeatures = [ 'wallpaper' => 'Couleur unie ou papier-peint', 'feature' => 'Contenu HTML' ]; public static $imagePositions = [ 'top left' => 'En haut à gauche', 'top center' => 'En haut au centre', 'top right' => 'En haut à droite', 'center left' => 'Au milieu à gauche', 'center center' => 'Au milieu au centre', 'center right' => 'Au milieu à droite', 'bottom left' => 'En bas à gauche', 'bottom center' => 'En bas au centre', 'bottom right' => 'En bas à droite' ]; public static $menuFontSizes = [ '.8em' => '80%', '.9em' => '90%', '1em' => 'Standard (100%)', '1.1em' => '110%', '1.2em' => '120%', '1.3em' => '130%' ]; public static $menuHeights = [ '5px 10px' => 'Très petite', '10px' => 'Petite', '15px 10px' => 'Moyenne', '20px 15px' => 'Grande', '25px 15px' => 'Très grande' ]; public static $menuPositionsSite = [ 'top' => 'En-dehors du site', 'site-first' => 'Avant la bannière', 'site-second' => 'Après la bannière', 'hide' => 'Caché' ]; public static $menuPositionsBody = [ 'top' => 'En-dehors du site', 'body-first' => 'Avant la bannière', 'body-second' => 'Après la bannière', 'site' => 'Dans le site', 'hide' => 'Caché' ]; public static $menuRadius = [ '0px' => 'Aucun', '3px 3px 0px 0px' => 'Très léger', '6px 6px 0px 0px' => 'Léger', '9px 9px 0px 0px' => 'Moyen', '12px 12px 0px 0px' => 'Important', '15px 15px 0px 0px' => 'Très important' ]; public static $radius = [ '0px' => 'Aucun', '5px' => 'Très léger', '10px' => 'Léger', '15px' => 'Moyen', '25px' => 'Important', '50px' => 'Très important' ]; public static $repeats = [ 'no-repeat' => 'Ne pas répéter', 'repeat-x' => 'Sur l\'axe horizontal', 'repeat-y' => 'Sur l\'axe vertical', 'repeat' => 'Sur les deux axes' ]; public static $shadows = [ '0px' => 'Aucune', '1px 1px 5px' => 'Très légère', '1px 1px 10px' => 'Légère', '1px 1px 15px' => 'Moyenne', '1px 1px 25px' => 'Importante', '1px 1px 50px' => 'Très importante' ]; public static $siteFontSizes = [ '12px' => '12 pixels', '13px' => '13 pixels', '14px' => '14 pixels', '15px' => '15 pixels', '16px' => '16 pixels' ]; public static $bodySizes = [ 'auto' => 'Automatique', '100% 100%' => 'Image étirée (100% 100%)', 'cover' => 'Responsive (cover)', 'contain' => 'Responsive (contain)' ]; public static $textTransforms = [ 'none' => 'Standard', 'lowercase' => 'Minuscules', 'uppercase' => 'Majuscules', 'capitalize' => 'Majuscule à chaque mot' ]; public static $siteWidths = [ '750px' => '750 pixels', '960px' => '960 pixels', '1170px' => '1170 pixels', '100%' => '100%', ]; public static $headerWide = [ 'auto auto' => 'Automatique', '100% 100%' => 'Image étirée (100% 100%)', 'cover' => 'Responsive (cover)', 'contain' => 'Responsive (contain)' ]; public static $footerTemplate = [ '1' => 'Une seule colonne', '2' => 'Deux colonnes : 1/2 - 1/2', '3' => 'Trois colonnes : 1/3 - 1/3 - 1/3', '4' => 'Trois lignes superposées' ]; public static $burgerContent = [ 'none' => 'Aucun', 'title' => 'Titre', 'logo' => 'Logo du site' ]; // Variable pour construire la liste des pages du site public static $pagesList = []; // Variable pour construire la liste des fontes installées public static $fontsNames = []; public static $fonts = []; // Variable pour détailler les fontes installées public static $fontsDetail = []; /** * Thème des écrans d'administration */ public function admin() { // Soumission du formulaire if ( $this->getUser('permission', __CLASS__, __FUNCTION__) === true && $this->isPost() ) { $this->setData([ 'admin', [ 'backgroundColor' => $this->getInput('adminBackgroundColor'), 'colorTitle' => $this->getInput('adminColorTitle'), 'colorText' => $this->getInput('adminColorText'), 'backgroundColorButton' => $this->getInput('adminColorButton'), 'backgroundColorButtonGrey' => $this->getInput('adminColorGrey'), 'backgroundColorButtonRed' => $this->getInput('adminColorRed'), 'backgroundColorButtonGreen' => $this->getInput('adminColorGreen'), 'backgroundColorButtonHelp' => $this->getInput('adminColorHelp'), 'fontText' => $this->getInput('adminFontText'), 'fontSize' => $this->getInput('adminFontTextSize'), 'fontTitle' => $this->getInput('adminFontTitle'), 'backgroundBlockColor' => $this->getInput('adminBackGroundBlockColor'), 'borderBlockColor' => $this->getInput('adminBorderBlockColor'), 'width' => $this->getInput('adminSiteWidth'), ] ]); // Valeurs en sortie $this->addOutput([ 'notification' => helper::translate('Modifications enregistrées'), 'redirect' => helper::baseUrl() . 'theme/admin', 'state' => true ]); } // Lire les fontes installées $this->enumFonts(); // Toutes les fontes installées sont chargées $this->setFonts('all'); // Valeurs en sortie $this->addOutput([ 'title' => helper::translate('Thème de l\'administration'), 'view' => 'admin', 'vendor' => [ 'tinycolorpicker' ], ]); } /** * Mode avancé */ public function advanced() { // Soumission du formulaire if ( $this->getUser('permission', __CLASS__, __FUNCTION__) === true && $this->isPost() ) { // Enregistre le CSS file_put_contents(self::DATA_DIR . 'custom.css', $this->getInput('themeAdvancedCss', null)); // Valeurs en sortie $this->addOutput([ 'notification' => helper::translate('Modifications enregistrées'), 'redirect' => helper::baseUrl() . 'theme/advanced', 'state' => true ]); } // Valeurs en sortie $this->addOutput([ 'title' => helper::translate('Éditeur CSS'), 'vendor' => [ 'codemirror' ], 'view' => 'advanced' ]); } /** * Options de l'arrière plan */ public function body() { // Soumission du formulaire if ( $this->getUser('permission', __CLASS__, __FUNCTION__) === true && $this->isPost() ) { $this->setData([ 'theme', 'body', [ 'backgroundColor' => $this->getInput('themeBodyBackgroundColor'), 'image' => $this->getInput('themeBodyImage'), 'imageAttachment' => $this->getInput('themeBodyImageAttachment'), 'imagePosition' => $this->getInput('themeBodyImagePosition'), 'imageRepeat' => $this->getInput('themeBodyImageRepeat'), 'imageSize' => $this->getInput('themeBodyImageSize'), 'toTopbackgroundColor' => $this->getInput('themeBodyToTopBackground'), 'toTopColor' => $this->getInput('themeBodyToTopColor') ] ]); // Valeurs en sortie $this->addOutput([ 'notification' => helper::translate('Modifications enregistrées'), 'redirect' => helper::baseUrl() . 'theme', 'state' => true ]); } // Valeurs en sortie $this->addOutput([ 'title' => helper::translate('Arrière plan'), 'vendor' => [ 'tinycolorpicker' ], 'view' => 'body' ]); } /** * Options du pied de page */ public function footer() { // Soumission du formulaire if ( $this->getUser('permission', __CLASS__, __FUNCTION__) === true && $this->isPost() ) { if ( $this->getInput('themeFooterCopyrightPosition') === 'hide' && $this->getInput('themeFooterSocialsPosition') === 'hide' && $this->getInput('themeFooterTextPosition') === 'hide' ) { // Valeurs en sortie $this->addOutput([ 'notification' => helper::translate('Sélectionnez au moins un contenu à afficher'), 'redirect' => helper::baseUrl() . 'theme/footer', 'state' => false ]); } else { $this->setData([ 'theme', 'footer', [ 'backgroundColor' => $this->getInput('themeFooterBackgroundColor'), 'copyrightAlign' => $this->getInput('themeFooterCopyrightAlign'), 'height' => $this->getInput('themeFooterHeight'), 'loginLink' => $this->getInput('themeFooterLoginLink'), 'margin' => $this->getInput('themeFooterMargin', helper::FILTER_BOOLEAN), 'position' => $this->getInput('themeFooterPosition'), 'fixed' => $this->getInput('themeFooterFixed', helper::FILTER_BOOLEAN), 'socialsAlign' => $this->getInput('themeFooterSocialsAlign'), 'text' => $this->getInput('themeFooterText', null), 'textAlign' => $this->getInput('themeFooterTextAlign'), 'textColor' => $this->getInput('themeFooterTextColor'), 'copyrightPosition' => $this->getInput('themeFooterCopyrightPosition'), 'textPosition' => $this->getInput('themeFooterTextPosition'), 'socialsPosition' => $this->getInput('themeFooterSocialsPosition'), 'textTransform' => $this->getInput('themeFooterTextTransform'), 'font' => $this->getInput('themeFooterFont'), 'fontSize' => $this->getInput('themeFooterFontSize'), 'fontWeight' => $this->getInput('themeFooterFontWeight'), 'displayVersion' => $this->getInput('themefooterDisplayVersion', helper::FILTER_BOOLEAN), 'displaySiteMap' => $this->getInput('themefooterDisplaySiteMap', helper::FILTER_BOOLEAN), 'displayCopyright' => $this->getInput('themefooterDisplayCopyright', helper::FILTER_BOOLEAN), 'displayCookie' => $this->getInput('themefooterDisplayCookie', helper::FILTER_BOOLEAN), 'displayLegal' => $this->getInput('themeFooterDisplayLegal', helper::FILTER_BOOLEAN), 'displaySearch' => $this->getInput('themeFooterDisplaySearch', helper::FILTER_BOOLEAN), 'memberBar' => $this->getInput('themeFooterMemberBar', helper::FILTER_BOOLEAN), 'template' => $this->getInput('themeFooterTemplate') ] ]); // Sauvegarder la configuration localisée $this->setData(['config', 'legalPageId', $this->getInput('configLegalPageId')]); $this->setData(['config', 'searchPageId', $this->getInput('configSearchPageId')]); // Valeurs en sortie $this->addOutput([ 'notification' => helper::translate('Modifications enregistrées'), 'redirect' => helper::baseUrl() . 'theme', 'state' => true ]); } } // Liste des pages self::$pagesList = $this->getData(['page']); foreach (self::$pagesList as $page => $pageId) { if ( $this->getData(['page', $page, 'block']) === 'bar' || $this->getData(['page', $page, 'disable']) === true ) { unset(self::$pagesList[$page]); } } // Lire les fontes installées $this->enumFonts(); // Toutes les fontes installées sont chargées $this->setFonts('all'); // Valeurs en sortie $this->addOutput([ 'title' => helper::translate('Pied de page'), 'vendor' => [ 'tinycolorpicker', 'tinymce' ], 'view' => 'footer' ]); } /** * Options de la bannière */ public function header() { // Soumission du formulaire if ( $this->getUser('permission', __CLASS__, __FUNCTION__) === true && $this->isPost() ) { // Modification des URL des images dans la bannière perso $featureContent = $this->getInput('themeHeaderText', null); /** * Stocker les images incluses dans la bannière perso dans un tableau */ $files = []; preg_match_all('/]+>/i', $featureContent, $results); foreach ($results[0] as $value) { // Lire le contenu XML $sx = simplexml_load_string($value); // Élément à remplacer $files[] = str_replace('./site/file/source/', '', (string) $sx[0]['src']); } // Sauvegarder $this->setData([ 'theme', 'header', [ 'backgroundColor' => $this->getInput('themeHeaderBackgroundColor'), 'font' => $this->getInput('themeHeaderFont'), 'fontSize' => $this->getInput('themeHeaderFontSize'), 'fontWeight' => $this->getInput('themeHeaderFontWeight'), 'height' => $this->getInput('themeHeaderHeight'), 'wide' => $this->getInput('themeHeaderWide'), 'image' => $this->getInput('themeHeaderImage'), 'imagePosition' => $this->getInput('themeHeaderImagePosition'), 'imageRepeat' => $this->getInput('themeHeaderImageRepeat'), 'margin' => $this->getInput('themeHeaderMargin', helper::FILTER_BOOLEAN), 'position' => $this->getInput('themeHeaderPosition'), 'textAlign' => $this->getInput('themeHeaderTextAlign'), 'textColor' => $this->getInput('themeHeaderTextColor'), 'textHide' => $this->getInput('themeHeaderTextHide', helper::FILTER_BOOLEAN), 'textTransform' => $this->getInput('themeHeaderTextTransform'), 'linkHomePage' => $this->getInput('themeHeaderlinkHomePage', helper::FILTER_BOOLEAN), 'imageContainer' => $this->getInput('themeHeaderImageContainer'), 'tinyHidden' => $this->getInput('themeHeaderTinyHidden', helper::FILTER_BOOLEAN), 'feature' => $this->getInput('themeHeaderFeature'), 'featureContent' => $featureContent, 'featureFiles' => $files ] ], false); // Modification de la position du menu selon la position de la bannière if ($this->getData(['theme', 'header', 'position']) == 'site') { $this->setData(['theme', 'menu', 'position', str_replace('body-', 'site-', $this->getData(['theme', 'menu', 'position']))], false); } if ($this->getData(['theme', 'header', 'position']) == 'body') { $this->setData(['theme', 'menu', 'position', str_replace('site-', 'body-', $this->getData(['theme', 'menu', 'position']))], false); } // Menu accroché à la bannière qui devient cachée if ( $this->getData(['theme', 'header', 'position']) == 'hide' && in_array($this->getData(['theme', 'menu', 'position']), ['body-first', 'site-first', 'body-first', 'site-second']) ) { $this->setData(['theme', 'menu', 'position', 'site'], false); } // Sauvegarde la base manuellement $this->saveDB(module: 'theme'); // Valeurs en sortie $this->addOutput([ 'notification' => helper::translate('Modifications enregistrées'), 'redirect' => helper::baseUrl() . 'theme', 'state' => true ]); } // Lire les fontes installées $this->enumFonts(); // Toutes les fontes installées sont chargées $this->setFonts('all'); // Valeurs en sortie $this->addOutput([ 'title' => helper::translate('Bannière'), 'vendor' => [ 'tinycolorpicker', 'tinymce' ], 'view' => 'header' ]); } /** * Accueil de la personnalisation */ public function index() { // Restaurer les fontes utilisateurs $this->setFonts('user'); // Valeurs en sortie $this->addOutput([ 'title' => helper::translate('Thèmes'), 'view' => 'index' ]); } /** * Options du menu */ public function menu() { // Soumission du formulaire if ( $this->getUser('permission', __CLASS__, __FUNCTION__) === true && $this->isPost() ) { $this->setData([ 'theme', 'menu', [ 'backgroundColor' => $this->getInput('themeMenuBackgroundColor'), 'backgroundColorSub' => $this->getInput('themeMenuBackgroundColorSub'), 'font' => $this->getInput('themeMenuFont'), 'fontSize' => $this->getInput('themeMenuFontSize'), 'fontWeight' => $this->getInput('themeMenuFontWeight'), 'height' => $this->getInput('themeMenuHeight'), 'wide' => $this->getInput('themeMenuWide'), 'loginLink' => $this->getInput('themeMenuLoginLink', helper::FILTER_BOOLEAN), 'margin' => $this->getInput('themeMenuMargin', helper::FILTER_BOOLEAN), 'position' => $this->getInput('themeMenuPosition'), 'textAlign' => $this->getInput('themeMenuTextAlign'), 'textColor' => $this->getInput('themeMenuTextColor'), 'textTransform' => $this->getInput('themeMenuTextTransform'), 'fixed' => $this->getInput('themeMenuFixed', helper::FILTER_BOOLEAN), 'activeColorAuto' => $this->getInput('themeMenuActiveColorAuto', helper::FILTER_BOOLEAN), 'activeColor' => $this->getInput('themeMenuActiveColor'), 'activeTextColor' => $this->getInput('themeMenuActiveTextColor'), 'radius' => $this->getInput('themeMenuRadius'), 'burgerTitle' => $this->getInput('themeMenuBurgerTitle', helper::FILTER_BOOLEAN), 'memberBar' => $this->getInput('themeMenuMemberBar', helper::FILTER_BOOLEAN), 'selectSpace' => $this->getInput('themeMenuSelectSpace', helper::FILTER_BOOLEAN), 'hidePages' => $this->getInput('themeMenuHidePages', helper::FILTER_BOOLEAN), 'burgerLogo' => $this->getInput('themeMenuBurgerLogo'), 'burgerContent' => $this->getInput('themeMenuBurgerContent'), ] ]); // Valeurs en sortie $this->addOutput([ 'notification' => helper::translate('Modifications enregistrées'), 'redirect' => helper::baseUrl() . 'theme', 'state' => true ]); } // Lire les fontes installées $this->enumFonts(); // Toutes les fontes installées sont chargées $this->setFonts('all'); // Valeurs en sortie $this->addOutput([ 'title' => helper::translate('Menu'), 'vendor' => [ 'tinycolorpicker' ], 'view' => 'menu' ]); } /** * Options des fontes */ public function font() { // Toutes les fontes installées sont chargées $this->setFonts('all'); // Polices liées au thème admin $admin = $this->getData(['admin']); $fonts['Titre (admin)'] = $this->getData(['admin', 'fontTitle']); $fonts['Texte (admin)'] = $this->getData(['admin', 'fontText']); // Polices liées aux thèmes des espaces foreach ($this->getData(['course']) as $courseId => $courseValue) { $theme = json_decode(file_get_contents(self::DATA_DIR . $courseId . '/theme.json'), true); $fonts['Bannière (' . $courseId . ')'] = $theme['theme']['header']['font']; $fonts['Menu (' . $courseId . ')'] = $theme['theme']['menu']['font']; $fonts['Titre (' . $courseId . ')'] = $theme['theme']['title']['font']; $fonts['Texte (' . $courseId . ')'] = $theme['theme']['text']['font']; $fonts['Pied de page (' . $courseId . ')'] = $theme['theme']['footer']['font']; } // Récupérer le détail des fontes installées //$f = $this->getFonts(); $f['files'] = $this->getData(['font', 'files']); $f['imported'] = $this->getData(['font', 'imported']); $f['websafe'] = self::$fontsWebSafe; // Listes des espaces à parcourir pour bloquer la suppression d'une fonte utilisée $courses = array_merge(['admin'], array_keys($this->getData(['course']))); // Parcourir les fontes disponibles et construire le tableau pour le formulaire foreach ($f as $type => $typeValue) { if (is_array($typeValue)) { foreach ($typeValue as $fontId => $fontValue) { // Recherche les correspondances $result = array_filter($fonts, function ($value) use ($fontId) { return $value == $fontId; }); $keyResults = array_keys($result); // Préparation du tableau self::$fontsDetail[] = [ $fontId, '' . $f[$type][$fontId]['name'] . '', $f[$type][$fontId]['font-family'], empty($keyResults) ? '' : '' . implode('
', $keyResults) . '
', $type, $type !== 'websafe' ? template::button('themeFontEdit' . $fontId, [ 'class' => 'themeFontEdit', 'href' => helper::baseUrl() . $this->getUrl(0) . '/fontEdit/' . $type . '/' . $fontId, 'value' => template::ico('pencil'), //'disabled' => !empty($fontUsed[$fontId]) ]) : '', $type !== 'websafe' ? template::button('themeFontDelete' . $fontId, [ 'class' => 'themeFontDelete buttonRed', 'href' => helper::baseUrl() . $this->getUrl(0) . '/fontDelete/' . $type . '/' . $fontId, 'value' => template::ico('cancel'), 'disabled' => !empty($fontUsed[$fontId]) ]) : '' ]; } } } sort(self::$fontsDetail); // Valeurs en sortie $this->addOutput([ 'title' => helper::translate('Fontes'), 'view' => 'font', 'vendor' => [ 'datatables' ] ]); } /** * Ajouter une fonte */ public function fontAdd() { // Soumission du formulaire if ( $this->getUser('permission', __CLASS__, __FUNCTION__) === true && $this->isPost() ) { // Type d'import en ligne ou local $type = $this->getInput('fontAddUrl', helper::FILTER_BOOLEAN) ? 'imported' : 'files'; $type === 'files' ? 'imported' : 'files'; $resource = $type === 'imported' ? $this->getInput('fontAddUrl', null) : $this->getInput('fontAddFile', null); if (!empty($resource)) { $fontId = $this->getInput('fontAddFontId', null, true); $fontName = $this->getInput('fontAddFontName', null, true); $fontFamilyName = $this->getInput('fontAddFontFamilyName', null, true); // Remplace les doubles quotes par des simples quotes $fontFamilyName = str_replace('"', '\'', $fontFamilyName); // Supprime la fonte si elle existe dans le type inverse if (is_array($this->getData(['font', $type, $fontId]))) { $this->deleteData(['font', $type, $fontId]); } // Paramètres de la sortie vrai par défaut, c'est une fonte en ligne $success = true; // Copier la fonte si le nom du fichier est fourni $success = false; if (!is_dir(self::DATA_DIR . 'font/')) { mkdir(self::DATA_DIR . 'font/'); } if ( $type === 'files' && file_exists(self::FILE_DIR . 'source/' . $resource) ) { $success = copy(self::FILE_DIR . 'source/' . $resource, self::DATA_DIR . 'font/' . basename($resource)); } // Stocker la fonte $this->setData([ 'font', $type, $fontId, [ 'name' => $fontName, 'font-family' => $fontFamilyName, // Stocke l'URL our lien vers la fonte dans data 'resource' => $type === 'imported' ? $resource : self::DATA_DIR . 'font/' . basename($resource), ] ]); // Valeurs en sortie $this->addOutput([ 'notification' => $success ? helper::translate('Fonte actualisée') : helper::translate('Fonte non créée, ressource absente !'), 'redirect' => helper::baseUrl() . 'theme/font', 'state' => $success ]); } else { // Valeurs en sortie $this->addOutput([ 'notification' => helper::translate('Fonte non créée, ressource absente !'), 'redirect' => helper::baseUrl() . 'theme/fontAdd', 'state' => false ]); } } // Valeurs en sortie $this->addOutput([ 'title' => helper::translate('Ajouter une fonte'), 'view' => 'fontAdd' ]); } /** * Ajouter une fonte */ public function fontEdit() { // Soumission du formulaire if ( $this->getUser('permission', __CLASS__, __FUNCTION__) === true && $this->isPost() ) { // Type d'import en ligne ou local $type = $this->getInput('fontEditUrl', helper::FILTER_BOOLEAN) ? 'imported' : 'files'; $fontId = $this->getInput('fontEditFontId', null, true); $resource = $this->getData(['font', $type, $fontId, 'resource']); $fontName = $this->getInput('fontEditFontName', null, true); $fontFamilyName = $this->getInput('fontEditFontFamilyName', null, true); // Remplace les doubles quotes par des simples quotes $fontFamilyName = str_replace('"', '\'', $fontFamilyName); // Supprime la fonte si elle existe dans le type inverse if (is_array($this->getData(['font', $type, $fontId]))) { $this->deleteData(['font', $type, $fontId]); } // Stocker les fontes $this->setData([ 'font', $type, $fontId, [ 'name' => $fontName, 'font-family' => $fontFamilyName, 'resource' => $resource ] ]); // Valeurs en sortie $this->addOutput([ 'notification' => helper::translate('Fonte actualisée'), 'redirect' => helper::baseUrl() . 'theme/font', 'state' => true ]); } // Valeurs en sortie $this->addOutput([ 'title' => helper::translate('Éditer une fonte'), 'view' => 'fontEdit' ]); } /** * Effacer une fonte */ public function fontDelete() { // Action interdite if ( $this->getUser('permission', __CLASS__, __FUNCTION__) !== true ) { // Valeurs en sortie // Valeurs en sortie $this->addOutput([ 'access' => false ]); } // Suppression else { // Effacer la fonte de la base $this->deleteData(['font', $this->getUrl(2), $this->getUrl(3)]); // Effacer le fichier existant if ( $this->getUrl(2) === 'files' && file_exists($this->getData(['font', 'files', $this->getUrl(3), 'resource'])) ) { unlink($this->getData(['font', 'files', $this->getUrl(3), 'resource'])); } // Valeurs en sortie $this->addOutput([ 'redirect' => helper::baseUrl() . 'theme/font', 'notification' => helper::translate('Fonte supprimée'), 'state' => true ]); } } /** * Réinitialisation de la personnalisation avancée */ public function reset() { // Action interdite if ( $this->getUser('permission', __CLASS__, __FUNCTION__) !== true ) { // Valeurs en sortie $this->addOutput([ 'access' => false ]); } else { // Réinitialisation $redirect = ''; switch ($this->getUrl(2)) { case 'admin': $this->initData('admin', self::$i18nUI); $redirect = helper::baseUrl() . 'theme/admin'; break; case 'manage': $this->initData('theme', self::$i18nUI); $redirect = helper::baseUrl() . 'theme/manage'; break; case 'custom': unlink(self::DATA_DIR . 'custom.css'); $redirect = helper::baseUrl() . 'theme/advanced'; break; default: $redirect = helper::baseUrl() . 'theme'; } // Valeurs en sortie $this->addOutput([ 'notification' => helper::translate('Thème réinitialisé'), 'redirect' => $redirect, 'state' => true ]); } } /** * Options du site */ public function site() { // Soumission du formulaire if ( $this->getUser('permission', __CLASS__, __FUNCTION__) === true && $this->isPost() ) { $this->setData([ 'theme', 'title', [ 'font' => $this->getInput('themeTitleFont'), 'textColor' => $this->getInput('themeTitleTextColor'), 'fontWeight' => $this->getInput('themeTitleFontWeight'), 'textTransform' => $this->getInput('themeTitleTextTransform') ] ], false); $this->setData([ 'theme', 'text', [ 'font' => $this->getInput('themeTextFont'), 'fontSize' => $this->getInput('themeTextFontSize'), 'textColor' => $this->getInput('themeTextTextColor'), 'linkColor' => $this->getInput('themeTextLinkColor') ] ], false); $this->setData([ 'theme', 'site', [ 'backgroundColor' => $this->getInput('themeSiteBackgroundColor'), 'radius' => $this->getInput('themeSiteRadius'), 'shadow' => $this->getInput('themeSiteShadow'), 'width' => $this->getInput('themeSiteWidth'), 'margin' => $this->getInput('themeSiteMargin', helper::FILTER_BOOLEAN) ] ], false); $this->setData([ 'theme', 'button', [ 'backgroundColor' => $this->getInput('themeButtonBackgroundColor') ] ], false); $this->setData([ 'theme', 'block', [ 'backgroundColor' => $this->getInput('themeBlockBackgroundColor'), 'borderColor' => $this->getInput('themeBlockBorderColor') ] ]); // Valeurs en sortie $this->addOutput([ 'notification' => helper::translate('Modifications enregistrées'), 'redirect' => helper::baseUrl() . 'theme', 'state' => true ]); } // Lire les fontes installées $this->enumFonts(); // Toutes les fontes installées sont chargées $this->setFonts('all'); // Valeurs en sortie $this->addOutput([ 'title' => helper::translate('Site'), 'vendor' => [ 'tinycolorpicker', 'tinymce' ], 'view' => 'site' ]); } /** * Import du thème */ public function manage() { if ( $this->getUser('permission', __CLASS__, __FUNCTION__) === true && $this->isPost() ) { $zipFilename = $this->getInput('themeManageImport', helper::FILTER_STRING_SHORT, true); $data = $this->import(self::FILE_DIR . 'source/' . $zipFilename); if ($data['success']) { header("Refresh:0"); } else { // Valeurs en sortie $this->addOutput([ 'title' => helper::translate('Gestion des thèmes'), 'notification' => $data['notification'], 'state' => $data['success'], 'view' => 'manage' ]); ; } } // Valeurs en sortie $this->addOutput([ 'title' => helper::translate('Gestion des thèmes'), 'view' => 'manage' ]); } /** * Importe un thème * @param string Url du thème à télécharger * @param @return array contenant $success = true ou false ; $ notification string message à afficher */ public function import($zipName = '') { // Action interdite if ( $this->getUser('permission', __CLASS__, __FUNCTION__) !== true ) { // Valeurs en sortie $this->addOutput([ 'access' => false ]); } else { if ( $zipName !== '' && file_exists($zipName) ) { // Init variables de retour $success = false; $notification = ''; // Dossier temporaire $tempFolder = uniqid(); // Ouvrir le zip $zip = new ZipArchive(); if ($zip->open($zipName) === TRUE) { mkdir(self::TEMP_DIR . $tempFolder, 0755); $zip->extractTo(self::TEMP_DIR . $tempFolder); $modele = ''; // Archive de thème ? if ( file_exists(self::TEMP_DIR . $tempFolder . '/site/data/custom.css') and file_exists(self::TEMP_DIR . $tempFolder . '/site/data/theme.css') and file_exists(self::TEMP_DIR . $tempFolder . '/site/data/theme.json') ) { $modele = 'theme'; } if ( file_exists(self::TEMP_DIR . $tempFolder . '/site/data/admin.json') and file_exists(self::TEMP_DIR . $tempFolder . '/site/data/admin.css') ) { $modele = 'admin'; } if (!empty($modele)) { // traiter l'archive $success = $zip->extractTo('.'); // Substitution des fontes Google if ($modele = 'theme') { // Déplacement des deux fichiers de theme dans le siteContent copy(self::DATA_DIR . 'theme.css', self::DATA_DIR . self::$siteContent . '/theme.css'); copy(self::DATA_DIR . 'theme.json', self::DATA_DIR . self::$siteContent . '/theme.json'); unlink(self::DATA_DIR . 'theme.css'); unlink(self::DATA_DIR . 'theme.json'); $c = $this->subFont(self::DATA_DIR . self::$siteContent . '/theme.json'); // Un remplacement nécessite la régénération de la feuille de style if ( $c > 0 and file_exists(self::DATA_DIR . self::$siteContent . '/theme.css') ) { unlink(self::DATA_DIR . self::$siteContent . '/theme.css'); } } if ($modele = 'admin') { $c = $this->subFont(self::DATA_DIR . 'admin.json'); // Un remplacement nécessite la régénération de la feuille de style if ( $c > 0 and file_exists(self::DATA_DIR . 'admin.css') ) { unlink(self::DATA_DIR . 'admin.css'); } } // traitement d'erreur $notification = $success ? helper::translate('Thème importé') : helper::translate('Erreur lors de l\'extraction, vérifiez les permissions'); } else { // pas une archive de thème $success = false; $notification = helper::translate('Archive de thème invalide'); } // Supprimer le dossier temporaire même si le thème est invalide $this->deleteDir(self::TEMP_DIR . $tempFolder); $zip->close(); } else { // erreur à l'ouverture $success = false; $notification = helper::translate('Impossible d\'ouvrir l\'archive'); } return (['success' => $success, 'notification' => $notification]); } return (['success' => false, 'notification' => helper::translate('Archive non spécifiée ou introuvable')]); } } /** * Export du thème */ public function export() { // Action interdite if ( $this->getUser('permission', __CLASS__, __FUNCTION__) !== true ) { // Valeurs en sortie $this->addOutput([ 'access' => false ]); } else { // Make zip $zipFilename = $this->zipTheme($this->getUrl(2)); // Téléchargement du ZIP header('Content-Description: File Transfer'); header('Content-Type: application/octet-stream'); header('Content-Transfer-Encoding: binary'); header('Content-Disposition: attachment; filename="' . $zipFilename . '"'); header('Content-Length: ' . filesize(self::TEMP_DIR . $zipFilename)); readfile(self::TEMP_DIR . $zipFilename); // Nettoyage du dossier unlink(self::TEMP_DIR . $zipFilename); exit(); } } /** * Export du thème */ public function save() { // Action interdite if ( $this->getUser('permission', __CLASS__, __FUNCTION__) !== true ) { // Valeurs en sortie $this->addOutput([ 'access' => false ]); } else { // Make zip $zipFilename = $this->zipTheme($this->getUrl(2)); // Téléchargement du ZIP if (!is_dir(self::FILE_DIR . 'source/theme')) { mkdir(self::FILE_DIR . 'source/theme', 0755); } copy(self::TEMP_DIR . $zipFilename, self::FILE_DIR . 'source/theme/' . $zipFilename); // Nettoyage du dossier unlink(self::TEMP_DIR . $zipFilename); // Valeurs en sortie $this->addOutput([ 'notification' => '' . $zipFilename . '' . helper::translate('sauvegardé avec succès'), 'redirect' => helper::baseUrl() . 'theme/manage', 'state' => true ]); } } /** * construction du zip Fonction appelée par export() et save() * @param string $modele theme ou admin */ private function zipTheme($modele) { // Creation du dossier $zipFilename = $modele . date('Y-m-d-H-i-s', time()) . '.zip'; $zip = new ZipArchive(); if ($zip->open(self::TEMP_DIR . $zipFilename, ZipArchive::CREATE | ZipArchive::OVERWRITE) === TRUE) { switch ($modele) { case 'admin': $zip->addFile(self::DATA_DIR . 'admin.json', self::DATA_DIR . 'admin.json'); $zip->addFile(self::DATA_DIR . 'admin.css', self::DATA_DIR . 'admin.css'); // Ajoute les fontes $zip->addEmptyDir(self::DATA_DIR . 'font'); $fonts = $this->getData(['font', 'files']); foreach ($fonts as $fontId => $fontName) { $zip->addFile(self::DATA_DIR . 'font/' . $fontName, self::DATA_DIR . 'font/' . $fontName); } if (file_exists(self::DATA_DIR . 'font/font.html')) { $zip->addFile(self::DATA_DIR . 'font/font.html', self::DATA_DIR . 'font/font.html'); } break; case 'theme': $zip->addFile(self::DATA_DIR . self::$siteContent . '/theme.json', self::DATA_DIR . 'theme.json'); $zip->addFile(self::DATA_DIR . self::$siteContent . '/theme.css', self::DATA_DIR . 'theme.css'); $zip->addFile(self::DATA_DIR . 'custom.css', self::DATA_DIR . 'custom.css'); // Traite l'image dans le body if ($this->getData(['theme', 'body', 'image']) !== '') { $zip->addFile( self::FILE_DIR . 'source/' . $this->getData(['theme', 'body', 'image']), self::FILE_DIR . 'source/' . $this->getData(['theme', 'body', 'image']) ); } // Traite l'image dans le header if ($this->getData(['theme', 'header', 'image']) !== '') { $zip->addFile( self::FILE_DIR . 'source/' . $this->getData(['theme', 'header', 'image']), self::FILE_DIR . 'source/' . $this->getData(['theme', 'header', 'image']) ); } // Traite les images du header perso if (!empty($this->getData(['theme', 'header', 'featureFiles']))) { foreach ($this->getData(['theme', 'header', 'featureFiles']) as $value) { $zip->addFile( self::FILE_DIR . 'source/' . $value, self::FILE_DIR . 'source/' . $value ); } } // Ajoute les fontes $zip->addEmptyDir(self::DATA_DIR . 'font'); $fonts = $this->getData(['font', 'files']); foreach ($fonts as $fontId => $fontInfo) { $zip->addFile($fontInfo['resource'], $fontInfo['resource']); } if (file_exists(self::DATA_DIR . 'font/font.html')) { $zip->addFile(self::DATA_DIR . 'font/font.html', self::DATA_DIR . 'font/font.html'); } if (file_exists(self::DATA_DIR . 'font/font.css')) { $zip->addFile(self::DATA_DIR . 'font/font.css', self::DATA_DIR . 'font/font.css'); } break; } $ret = $zip->close(); } return ($zipFilename); } /** * Substitution des fontes de Google Fonts vers CdnFont grâce à un tableau de conversion * Cette fonction est utilisée par l'import. * @param string $file, nom du fichier json à convertir * @return int nombre de substitution effectuées */ private function subFont($file) { // Tableau de substitution des fontes $fonts = [ 'Abril+Fatface' => 'abril-fatface', 'Arimo' => 'arimo', 'Arvo' => 'arvo', 'Berkshire+Swash' => 'berkshire-swash', 'Cabin' => 'genera', 'Dancing+Script' => 'dancing-script', 'Droid+Sans' => 'droid-sans-2', 'Droid+Serif' => 'droid-serif-2', 'Fira+Sans' => 'fira-sans', 'Inconsolata' => 'inconsolata-2', 'Indie+Flower' => 'indie-flower', 'Josefin+Slab' => 'josefin-sans-std', 'Lobster' => 'lobster-2', 'Lora' => 'lora', 'Lato' => 'lato', 'Marvel' => 'montserrat-ace', 'Old+Standard+TT' => 'old-standard-tt-3', 'Open+Sans' => 'open-sans', // Corriger l'erreur de nom de police installée par défaut, il manquait un O en majuscule 'open+Sans' => 'open-sans', 'Oswald' => 'oswald-4', 'PT+Mono' => 'pt-mono', 'PT+Serif' => 'pt-serif', 'Raleway' => 'raleway-5', 'Rancho' => 'rancho', 'Roboto' => 'Roboto', 'Signika' => 'signika', 'Ubuntu' => 'ubuntu', 'Vollkorn' => 'vollkorn' ]; $data = file_get_contents($file); $count = 0; foreach ($fonts as $oldId => $newId) { $data = str_replace($oldId, $newId, $data, $c); $count = $count + (int) $c; } // Sauvegarder la chaîne modifiée if ($count > 0) { file_put_contents($file, $data); } // Retourner le nombre d'occurrences return ($count); } // Retourne un tableau simple des fonts installées idfont avec le nom // Cette fonction est utile aux sélecteurs de fonts dans les formulaires. public function enumFonts() { /** * Récupère la liste des fontes installées et construit deux tableaux * id - nom * id - font-family - resource */ $f['files'] = $this->getData(['font', 'files']); $f['imported'] = $this->getData(['font', 'imported']); $f['websafe'] = self::$fontsWebSafe; // Construit un tableau avec leur ID et leur famille foreach (['websafe', 'imported', 'files'] as $type) { if (is_array($f[$type])) { foreach ($f[$type] as $fontId => $fontValue) { self::$fonts['name'][$fontId] = $fontValue['name']; self::$fonts['family'][$fontId] = $fontValue['font-family']; } } } // Liste des fontes pour les sélecteurs ksort(self::$fonts['name']); ksort(self::$fonts['family']); } /** * Création d'un fichier de liens d'appel des fontes * @param string $scope vaut all pour toutes les fontes ; 'user' pour les fontes utilisateurs */ private function setFonts($scope = 'all') { // Filtrage par fontes installées $fontsInstalled = [ $this->getData(['theme', 'text', 'font']), $this->getData(['theme', 'title', 'font']), $this->getData(['theme', 'header', 'font']), $this->getData(['theme', 'menu', 'font']), $this->getData(['theme', 'footer', 'font']), $this->getData(['admin', 'fontText']), $this->getData(['admin', 'fontTitle']), ]; // Compression $fontsInstalled = array_unique($fontsInstalled); /** * Chargement des polices en ligne dans un fichier font.html inclus dans main.php */ $gf = false; $fileContent = ''; if (!empty($this->getData(['font', 'imported']))) { foreach ($this->getData(['font', 'imported']) as $fontId => $fontValue) { if ( ($scope === 'user' && in_array($fontId, $fontsInstalled)) || $scope === 'all' ) { //Pré chargement à revoir //$fileContent .= ''; $fileContent .= ''; // Pré connect pour api.google $gf = strpos($fontValue['resource'], 'fonts.googleapis.com') === false ? $gf || false : $gf || true; } } } // Ajoute le préconnect des fontes Googles. $fileContent = $gf ? '' . $fileContent : $fileContent; /** * Fontes installées localement */ $fileContentCss = ''; if (!empty($this->getData(['font', 'files']))) { foreach ($this->getData(['font', 'files']) as $fontId => $fontValue) { if ( ($scope === 'user' && in_array($fontId, $fontsInstalled)) || $scope === 'all' ) { if (file_exists($fontValue['resource'])) { // Extension $path_parts = pathinfo(helper::baseUrl(false) . self::DATA_DIR . 'font/' . $fontValue['resource']); // Chargement de la police $fileContentCss .= '@font-face {'; $fileContentCss .= 'font-family:"' . $fontValue['name'] . '";'; $fileContentCss .= 'src: url("' . helper::baseUrl(false) . $fontValue['resource'] . '") format("' . $path_parts['extension'] . '");'; $fileContentCss .= '}'; // Préchargement //$fileContent = '' . $fileContent; } } } } // Enregistre la personnalisation file_put_contents(self::DATA_DIR . 'font/font.html', $fileContent); // Enregistre la personnalisation file_put_contents(self::DATA_DIR . 'font/font.css', $fileContentCss); } }