* @copyright Copyright (C) 2008-2018, Rémi Jean * @author Frédéric Tempez * @copyright Copyright (C) 2018-2024, Frédéric Tempez * @license CC Attribution-NonCommercial-NoDerivatives 4.0 International * @link http://zwiicms.fr/ */ class course extends common { public static $actions = [ 'swap' => self::GROUP_VISITOR, 'suscribe' => self::GROUP_VISITOR, 'unsuscribe' => self::GROUP_MEMBER, 'index' => self::GROUP_EDITOR, // Fait 'edit' => self::GROUP_EDITOR, // Fait 'manage' => self::GROUP_EDITOR, // Fait 'users' => self::GROUP_EDITOR, // fait 'usersAdd' => self::GROUP_EDITOR, //Fait 'usersDelete' => self::GROUP_EDITOR, //Fait 'usersHistoryExport' => self::GROUP_EDITOR, //fait 'userDelete' => self::GROUP_EDITOR, //Fait 'userHistory' => self::GROUP_EDITOR, //Fait 'userHistoryExport' => self::GROUP_EDITOR, //Fait 'backup' => self::GROUP_EDITOR, // Fait 'restore' => self::GROUP_EDITOR, //Fait 'clone' => self::GROUP_ADMIN, 'add' => self::GROUP_ADMIN, 'delete' => self::GROUP_ADMIN, 'category' => self::GROUP_ADMIN, 'categoryAdd' => self::GROUP_ADMIN, 'categoryEdit' => self::GROUP_ADMIN, 'categoryDelete' => self::GROUP_ADMIN, ]; public static $courseAccess = [ self::COURSE_ACCESS_OPEN => 'Ouvert', self::COURSE_ACCESS_DATE => 'Période d\'ouverture', self::COURSE_ACCESS_CLOSE => 'Fermé', ]; public static $courseEnrolment = [ self::COURSE_ENROLMENT_GUEST => 'Anonyme', self::COURSE_ENROLMENT_SELF => 'Inscription libre', self::COURSE_ENROLMENT_SELF_KEY => 'Inscription avec clé', self::COURSE_ENROLMENT_MANDATORY => 'Imposée' ]; public static $courseTeachers = []; public static $courseCategories = []; public static $courseUsers = []; public static $alphabet = []; public static $courseGroups = [ 'all' => 'Tout' ]; public static $courses = []; public static $swapMessage = []; public static $pagesList = ['accueil' => 'Accueil']; public static $userHistory = []; public static $userGraph = []; public static $userStat = []; public function index() { // Tableau à transmettre à la fvue self::$courses = array(); if ( $this->getUser('id') && $this->getUser('group') ) { foreach ($this->getData(['course']) as $courseId => $courseValue) { /** * Filtres : * Groupes acceptés : * admin : tous les espaces * editor : gère son espace son espace dans lequel il est inscrit */ if ( $this->permissionControl(__FUNCTION__, $courseId) ) { $author = $this->getData(['course', $courseId, 'author']) ? sprintf('%s %s', $this->getData(['user', $this->getData(['course', $courseId, 'author']), 'firstname']), $this->getData(['user', $this->getData(['course', $courseId, 'author']), 'lastname'])) : ''; $categorieUrl = helper::baseUrl() . 'course/swap/' . $courseId; $info = sprintf('%s
Auteur : %s
Id : %s
', $this->getData(['course', $courseId, 'title']), $author, $categorieUrl, $courseId); $enrolment = sprintf( 'Accès : %s
Inscription : %s
', self::$courseAccess[$this->getData(['course', $courseId, 'access'])], self::$courseEnrolment[$this->getData(['course', $courseId, 'enrolment'])] ); self::$courses[] = [ $info, $this->getData(['course', $courseId, 'description']), $enrolment, template::button('categoryUser' . $courseId, [ 'href' => helper::baseUrl() . 'course/manage/' . $courseId, 'value' => template::ico('eye'), 'help' => 'Gérer' ]) ]; } } } // Valeurs en sortie $this->addOutput([ 'title' => helper::translate('Gestionnaire d\'espaces'), 'view' => 'index', 'vendor' => [ 'datatables' ] ]); } /** * Ajoute un nouveau contenu */ public function add() { // Accès limité aux admins if ( $this->getUser('group') !== self::GROUP_ADMIN ) { // Valeurs en sortie $this->addOutput([ 'access' => false ]); } // Soumission du formulaire if ( $this->isPost() ) { $courseId = uniqid(); // Créer la structure de données mkdir(self::DATA_DIR . $courseId); $this->initDB('page', $courseId); $this->initDB('module', $courseId); $this->initDB('theme', $courseId); $this->initData('page', $courseId); $this->initData('module', $courseId); $this->initData('theme', $courseId); // BDD des inscrits $this->setData([ 'enrolment', $courseId, [] ]); $this->setData([ 'course', $courseId, [ 'title' => $this->getInput('courseAddTitle', helper::FILTER_STRING_SHORT, true), 'author' => $this->getInput('courseAddAuthor'), 'homePageId' => 'accueil', 'category' => $this->getInput('courseAddCategorie'), 'description' => $this->getInput('courseAddDescription', helper::FILTER_STRING_SHORT, true), 'access' => $this->getInput('courseAddAccess', helper::FILTER_INT), 'openingDate' => $this->getInput('courseAddOpeningDate', helper::FILTER_DATETIME), 'closingDate' => $this->getInput('courseAddClosingDate', helper::FILTER_DATETIME), 'enrolment' => $this->getInput('courseAddEnrolment', helper::FILTER_INT), 'enrolmentKey' => $this->getInput('courseAddEnrolmentKey'), 'limitEnrolment' => $this->getInput('courseAddEnrolmentLimit', helper::FILTER_BOOLEAN), 'limitEnrolmentDate' => $this->getInput('courseAddEnrolmentLimitDate', helper::FILTER_DATETIME), ] ]); // Dossier du gestionnaire de fichier mkdir(self::FILE_DIR . 'source/' . $courseId); // Copie du thème $sourceId = $this->getInput('courseAddTheme'); copy(self::DATA_DIR . $sourceId . '/theme.json', self::DATA_DIR . $courseId . '/theme.json'); copy(self::DATA_DIR . $sourceId . '/theme.css', self::DATA_DIR . $courseId . '/theme.css'); // Valeurs en sortie $this->addOutput([ 'redirect' => helper::baseUrl() . 'course', 'notification' => helper::translate('Espace créé'), 'state' => true ]); } // Liste des enseignants pour le sélecteur d'auteurs $teachers = $this->getData(['user']); foreach ($teachers as $teacherId => $teacherInfo) { if ($teacherInfo["group"] >= 2) { self::$courseTeachers[$teacherId] = $teacherInfo["firstname"] . ' ' . $teacherInfo["lastname"]; } } // Liste des catégories de contenu self::$courseCategories = $this->getData(['category']); // Liste des contenus disponibles pour la copie du thème self::$courses = $this->getData(['course']); self::$courses = helper::arrayColumn(self::$courses, 'title', 'SORT_ASC'); self::$courses = array_merge(['home' => 'Accueil de la plate-forme'], self::$courses); // Valeurs en sortie $this->addOutput([ 'title' => helper::translate('Ajouter un espace'), 'view' => 'add' ]); } /** * Edite un espace */ public function edit() { // Espace sélectionné $courseId = $this->getUrl(2); // Accès limité aux admins, à l'auteur ou éditeurs inscrits if ( $this->permissionControl(__FUNCTION__, $courseId) === false ) { // Valeurs en sortie $this->addOutput([ 'access' => false ]); } // Soumission du formulaire if ( //$this->getUser('permission', __CLASS__, __FUNCTION__) === true && $this->isPost() ) { $this->setData([ 'course', $courseId, [ 'title' => $this->getInput('courseEditShortTitle', helper::FILTER_STRING_SHORT, true), 'author' => $this->getInput('courseEditAuthor'), 'homePageId' => $this->getInput('courseEditHomePageId'), 'category' => $this->getInput('courseEditCategorie'), 'description' => $this->getInput('courseEditDescription', helper::FILTER_STRING_LONG, true), 'access' => $this->getInput('courseEditAccess', helper::FILTER_INT), 'openingDate' => $this->getInput('courseOpeningDate', helper::FILTER_DATETIME), 'closingDate' => $this->getInput('courseClosingDate', helper::FILTER_DATETIME), 'enrolment' => $this->getInput('courseEditEnrolment', helper::FILTER_INT), 'enrolmentKey' => $this->getInput('courseEditEnrolmentKey'), 'limitEnrolment' => $this->getInput('courseEditEnrolmentLimit', helper::FILTER_BOOLEAN), 'limitEnrolmentDate' => $this->getInput('courseEditEnrolmentLimitDate', helper::FILTER_DATETIME), ] ]); // Valeurs en sortie $this->addOutput([ 'redirect' => helper::baseUrl() . 'course/manage/' . $this->getUrl(2), 'notification' => helper::translate('Espace modifié'), 'state' => true ]); } // Liste des enseignants pour le sélecteur d'auteurs $teachers = $this->getData(['user']); foreach ($teachers as $teacherId => $teacherInfo) { if ($teacherInfo["group"] >= 2) { self::$courseTeachers[$teacherId] = $teacherInfo["firstname"] . ' ' . $teacherInfo["lastname"]; } } // Liste des catégories de contenu self::$courseCategories = $this->getData(['category']); // Liste des pages disponibles $this->initDB('page', $this->getUrl(2)); 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]); } } // Valeurs en sortie $this->addOutput([ 'title' => sprintf('%s id : %s', helper::translate('Éditer l\'espace'), $this->getUrl(2)), 'view' => 'edit' ]); } /** * Affiche un contenu et pointe vers les utilitaires */ public function manage() { // Espace sélectionné $courseId = $this->getUrl(2); // Accès limité aux admins, à l'auteur ou éditeurs inscrits if ( $this->permissionControl(__FUNCTION__, $courseId) === false ) { // Valeurs en sortie $this->addOutput([ 'access' => false ]); } // Liste des enseignants pour le sélecteur d'auteurs $teachers = $this->getData(['user']); foreach ($teachers as $teacherId => $teacherInfo) { if ($teacherInfo["group"] >= 2) { self::$courseTeachers[$teacherId] = $teacherInfo["firstname"] . ' ' . $teacherInfo["lastname"]; } } // Liste des catégories de contenu self::$courseCategories = $this->getData(['category']); // Liste des pages disponibles $this->initDB('page', $this->getUrl(2)); 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]); } } // Valeurs en sortie $this->addOutput([ 'title' => sprintf('%s id : %s', helper::translate('Gérer l\'espace'), $this->getUrl(2)), 'view' => 'manage' ]); } /** * Duplique un cours et l'affiche dans l'éditeur */ public function clone () { // Cours à dupliquer $courseId = $this->getUrl(2); // Accès limité aux admins, à l'auteur ou éditeurs inscrits if ( $this->getUser('group') !== self::$actions[__FUNCTION__] ) { // Valeurs en sortie $this->addOutput([ 'access' => false ]); } else { // Id du nouveau cours $target = uniqid(); // Créer la structure de données mkdir(self::DATA_DIR . $target); $this->copyDir(self::DATA_DIR . $courseId, self::DATA_DIR . $target); $this->setData(['course', $target, $this->getData(['course', $courseId])]); // Valeurs en sortie $this->addOutput([ 'redirect' => helper::baseUrl() . 'course', 'notification' => helper::translate('Espace dupliqué'), 'state' => true ]); } } public function delete() { // Espace sélectionné $courseId = $this->getUrl(2); if ( // Accès limité aux admins $this->getUser('group') !== self::$actions[__FUNCTION__] // Le contenu n'existe pas || $this->getData(['course', $courseId]) === null ) { // Valeurs en sortie $this->addOutput([ 'access' => false ]); // Suppression } else { // Active l'accueil $_SESSION['ZWII_SITE_CONTENT'] = 'home'; // ET efface la structure if (is_dir(self::DATA_DIR . $courseId)) { $success = $this->deleteDir(self::DATA_DIR . $courseId); $this->deleteData(['course', $courseId]); $this->deleteData(['enrolment', $courseId]); } // Dossier du gestionnaire de fichier if (is_dir(self::FILE_DIR . 'source/' . $courseId)) { $this->deleteDir(self::FILE_DIR . 'source/' . $courseId); } // Valeurs en sortie $this->addOutput([ 'redirect' => helper::baseUrl() . 'course', 'notification' => $success ? helper::translate('Espace supprimé') : helper::translate('Erreur de suppression'), 'state' => $success ]); } } /** * Liste les catégories d'un contenu */ public function category() { if ( // Accès limité aux admins $this->getUser('group') !== self::$actions[__FUNCTION__] ) { // Valeurs en sortie $this->addOutput([ 'access' => false ]); } else { $categories = $this->getData(['category']); ksort($categories); foreach ($categories as $categoryId => $categoryTitle) { self::$courseCategories[] = [ $categoryId, $categoryTitle, template::button('categoryEdit' . $categoryId, [ 'href' => helper::baseUrl() . 'course/categoryEdit/' . $categoryId, 'value' => template::ico('pencil'), 'help' => 'Éditer' ]), template::button('courseDelete' . $categoryId, [ 'class' => 'categoryDelete buttonRed', 'href' => helper::baseUrl() . 'course/categoryDelete/' . $categoryId, 'value' => template::ico('trash'), 'help' => 'Supprimer' ]) ]; } // Valeurs en sortie $this->addOutput([ 'title' => helper::translate('Catégories'), 'view' => 'category' ]); } } public function categoryAdd() { if ( // Accès limité aux admins $this->getUser('group') !== self::$actions[__FUNCTION__] ) { // Valeurs en sortie $this->addOutput([ 'access' => false ]); // Soumission du formulaire } elseif ($this->isPost()) { $categoryId = $this->getInput('categoryAddTitle', helper::FILTER_ID, true); $this->setData([ 'category', $categoryId, $this->getInput('categoryAddTitle', helper::FILTER_STRING_SHORT, true) ]); // Valeurs en sortie $this->addOutput([ 'redirect' => helper::baseUrl() . 'course/category', 'notification' => helper::translate('Catégorie créée'), 'state' => true ]); // Valeurs en sortie } $this->addOutput([ 'title' => helper::translate('Ajouter une catégorie'), 'view' => 'categoryAdd' ]); } public function categoryEdit() { if ( // Accès limité aux admins $this->getUser('group') !== self::$actions[__FUNCTION__] ) { // Valeurs en sortie $this->addOutput([ 'access' => false ]); // Soumission du formulaire } elseif ( $this->isPost() ) { $categoryId = $this->getUrl(2); $this->setData([ 'category', $categoryId, $this->getInput('categoryEditTitle', helper::FILTER_STRING_SHORT, true) ]); // Valeurs en sortie $this->addOutput([ 'redirect' => helper::baseUrl() . 'course/category', 'notification' => helper::translate('Catégorie éditée'), 'state' => true ]); } // Valeurs en sortie $this->addOutput([ 'title' => helper::translate('Éditer une catégorie'), 'view' => 'categoryEdit' ]); } public function categoryDelete() { if ( // Accès limité aux admins $this->getUser('group') !== self::$actions[__FUNCTION__] ) { // Valeurs en sortie $this->addOutput([ 'access' => false ]); } else { $categories = helper::arrayColumn($this->getData(['course']), 'category', 'SORT_ASC'); $courseId = $this->getUrl(2); $message = helper::translate('Une catégorie affectée ne peut pas être effacée'); $state = false; if (in_array($courseId, $categories) === false) { $this->deleteData(['category', $this->getUrl(2)]); // Valeurs en sortie $message = helper::translate('Catégorie effacée'); $state = true; } // Valeurs en sortie $this->addOutput([ 'redirect' => helper::baseUrl() . 'course/category', 'notification' => $message, 'state' => $state ]); } } public function users() { $courseId = $this->getUrl(2); // Accès limité au propriétaire, admin ou éditeurs isncrits if ( $this->permissionControl(__FUNCTION__, $courseId) === false ) { // Valeurs en sortie $this->addOutput([ 'access' => false ]); } // Contenu sélectionné $courseId = $this->getUrl(2); // Liste des groupes et des profils $courseGroups = $this->getData(['profil']); foreach ($courseGroups as $groupId => $groupValue) { switch ($groupId) { case "-1": case "0": break; case "3": self::$courseGroups['30'] = 'Administrateur'; $profils['30'] = 0; break; case "1": case "2": foreach ($groupValue as $profilId => $profilValue) { if ($profilId) { self::$courseGroups[$groupId . $profilId] = sprintf(helper::translate('Groupe %s - Profil %s'), self::$groupPublics[$groupId], $profilValue['name']); $profils[$groupId . $profilId] = 0; } } } } // Liste alphabétique self::$alphabet = range('A', 'Z'); $alphabet = range('A', 'Z'); self::$alphabet = array_combine($alphabet, self::$alphabet); self::$alphabet = array_merge(['all' => 'Tout'], self::$alphabet); // Liste des pages contenues dans cet espace et exclure les barres et les pages masquées $sumPages = 0; $pages = json_decode(file_get_contents(self::DATA_DIR . $courseId . '/page.json'), true); $pages = $pages['page']; foreach ($pages as $pageId => $pageData) { if ($pageData['position'] > 0) { $sumPages++; } } // Liste des inscrits dans le contenu sélectionné. $users = $this->getData(['enrolment', $courseId]); if (is_array($users)) { // Tri du tableau par défaut par $userId ksort($users); foreach ($users as $userId => $userValue) { // Date et heure de la dernière page vue // Compatibilité anciennes versions if ( $this->getData(['enrolment', $courseId, $userId, 'lastPageView']) === null or $this->getData(['enrolment', $courseId, $userId, 'datePageView']) === null ) { if (!empty($userValue['history'])) { $maxTime = max($userValue['history']); $lastPageId = array_search($maxTime, $userValue['history']); $this->setData(['enrolment', $courseId, $userId, 'lastPageView', $lastPageId]); $this->setData(['enrolment', $courseId, $userId, 'datePageView', $maxTime]); } } // Compte les rôles valides if (isset($profils[$this->getData(['user', $userId, 'group']) . $this->getData(['user', $userId, 'profil'])])) { $profils[$this->getData(['user', $userId, 'group']) . $this->getData(['user', $userId, 'profil'])]++; } // Filtres if ($this->isPost()) { // Groupe et profils $group = (string) $this->getData(['user', $userId, 'group']); $profil = (string) $this->getData(['user', $userId, 'profil']); $firstName = $this->getData(['user', $userId, 'firstname']); $lastName = $this->getData(['user', $userId, 'lastname']); if ( $this->getInput('courseFilterGroup', helper::FILTER_INT) > 0 && $this->getInput('courseFilterGroup', helper::FILTER_STRING_SHORT) !== $group . $profil ) continue; // Première lettre du prénom if ( $this->getInput('courseFilterFirstName', helper::FILTER_STRING_SHORT) !== 'all' && $this->getInput('courseFilterFirstName', helper::FILTER_STRING_SHORT) !== strtoupper(substr($firstName, 0, 1)) ) continue; // Première lettre du nom if ( $this->getInput('courseFilterLastName', helper::FILTER_STRING_SHORT) !== 'all' && $this->getInput('courseFilterLastName', helper::FILTER_STRING_SHORT) !== strtoupper(substr($lastName, 0, 1)) ) continue; } // Progression $viewPages = $this->getData(['enrolment', $courseId, $userId, 'history']) !== null ? count(array_keys($this->getData(['enrolment', $courseId, $userId, 'history']))) : 0; // Construction du tableau self::$courseUsers[] = [ //$userId, $this->getData(['user', $userId, 'firstname']) . ' ' . $this->getData(['user', $userId, 'lastname']), isset($pages[$this->getData(['enrolment', $courseId, $userId, 'lastPageView'])]['title']) ? $pages[$this->getData(['enrolment', $courseId, $userId, 'lastPageView'])]['title'] : '', $this->getData(['enrolment', $courseId, $userId, 'datePageView']) ? helper::dateUTF8('%d/%m/%Y', $this->getData(['enrolment', $courseId, $userId, 'datePageView'])) : '', $this->getData(['enrolment', $courseId, $userId, 'datePageView']) ? helper::dateUTF8('%H:%M', $this->getData(['enrolment', $courseId, $userId, 'datePageView'])) : '', $this->getData(['user', $userId, 'tags']), template::button('userHistory' . $userId, [ 'href' => helper::baseUrl() . 'course/userHistory/' . $courseId . '/' . $userId, 'value' => !empty($userValue['history']) ? min(round(($viewPages * 100) / $sumPages, 1), 100) . ' %' : '0%', 'disable' => empty($userValue['history']) ]), template::button('userDelete' . $userId, [ 'class' => 'userDelete buttonRed', 'href' => helper::baseUrl() . 'course/userDelete/' . $courseId . '/' . $userId, 'value' => template::ico('user'), 'help' => 'Désinscrire' ]) ]; } } // Ajoute les effectifs aux profils du sélecteur foreach (self::$courseGroups as $groupId => $groupValue) { if ($groupId === 'all') { self::$courseGroups['all'] = self::$courseGroups['all'] . ' (' . array_sum($profils) . ')'; } else { self::$courseGroups[$groupId] = self::$courseGroups[$groupId] . ' (' . $profils[$groupId] . ')'; } } // Valeurs en sortie $this->addOutput([ 'title' => sprintf(helper::translate('Participants %s'), $this->getData(['course', $courseId, 'title'])), 'view' => 'users', 'vendor' => [ 'datatables' ] ]); } public function usersAdd() { // Contenu sélectionné $courseId = $this->getUrl(2); // Accès limité au propriétaire ou éditeurs inscrits ou admin if ( $this->permissionControl(__FUNCTION__, $courseId) === false ) { // Valeurs en sortie $this->addOutput([ 'access' => false ]); } // Inscription des utilisateurs cochés if ( isset($_POST['courseUsersAddSubmit']) ) { foreach ($_POST as $keyPost => $valuePost) { // Exclure les variables post qui ne sont pas des userId et ne traiter que les non inscrits if ( $this->getData(['user', $keyPost]) !== null && $this->getData(['enrolment', $courseId, $keyPost]) === null ) { $this->setData(['enrolment', $courseId, $keyPost, 'history', array()]); } } } // Liste des groupes et des profils $courseGroups = $this->getData(['profil']); foreach ($courseGroups as $groupId => $groupValue) { switch ($groupId) { case "-1": case "0": break; case "3": self::$courseGroups['30'] = 'Administrateur'; $profils['30'] = 0; break; case "1": case "2": foreach ($groupValue as $profilId => $profilValue) { if ($profilId) { self::$courseGroups[$groupId . $profilId] = sprintf(helper::translate('Groupe %s - Profil %s'), self::$groupPublics[$groupId], $profilValue['name']); $profils[$groupId . $profilId] = 0; } } } } // Liste alphabétique self::$alphabet = range('A', 'Z'); $alphabet = range('A', 'Z'); self::$alphabet = array_combine($alphabet, self::$alphabet); self::$alphabet = array_merge(['all' => 'Tout'], self::$alphabet); // Liste des inscrits dans l'espace sélectionné afin de les supprimer de la liste des candidats $users = $this->getData(['user']); $suscribers = $this->getData(['enrolment', $courseId]); if (is_array($suscribers)) { $suscribers = array_keys($suscribers); $users = array_diff_key($users, array_flip($suscribers)); } // Tri du tableau par défaut par $userId ksort($users); foreach ($users as $userId => $userValue) { // Compte les rôles if (isset($profils[$this->getData(['user', $userId, 'group']) . $this->getData(['user', $userId, 'profil'])])) { $profils[$this->getData(['user', $userId, 'group']) . $this->getData(['user', $userId, 'profil'])]++; } // Filtres if ( isset($_POST['courseFilterGroup']) || isset($_POST['courseFilterFirstName']) || isset($_POST['courseFilterLastName']) ) { // Groupe et profils $group = (string) $this->getData(['user', $userId, 'group']); $profil = (string) $this->getData(['user', $userId, 'profil']); $firstName = $this->getData(['user', $userId, 'firstname']); $lastName = $this->getData(['user', $userId, 'lastname']); if ( $this->getInput('courseFilterGroup', helper::FILTER_INT) > 0 && $this->getInput('courseFilterGroup', helper::FILTER_STRING_SHORT) !== $group . $profil ) continue; // Première lettre du prénom if ( $this->getInput('courseFilterFirstName', helper::FILTER_STRING_SHORT) !== 'all' && $this->getInput('courseFilterFirstName', helper::FILTER_STRING_SHORT) !== strtoupper(substr($firstName, 0, 1)) ) continue; // Première lettre du nom if ( $this->getInput('courseFilterLastName', helper::FILTER_STRING_SHORT) !== 'all' && $this->getInput('courseFilterLastName', helper::FILTER_STRING_SHORT) !== strtoupper(substr($lastName, 0, 1)) ) continue; } // Construction du tableau self::$courseUsers[] = [ template::checkbox($userId, true, '', ['class' => 'checkboxSelect']), $userId, $this->getData(['user', $userId, 'firstname']), $this->getData(['user', $userId, 'lastname']), $this->getData(['user', $userId, 'tags']), ]; } // Ajoute les effectifs aux profils du sélecteur foreach (self::$courseGroups as $groupId => $groupValue) { if ($groupId === 'all') { self::$courseGroups['all'] = self::$courseGroups['all'] . ' (' . array_sum($profils) . ')'; } else { self::$courseGroups[$groupId] = self::$courseGroups[$groupId] . ' (' . $profils[$groupId] . ')'; } } // Valeurs en sortie $this->addOutput([ 'title' => helper::translate('Inscription en masse'), 'view' => 'usersAdd', 'vendor' => [ 'datatables' ] ]); } /** * Désinscription d'un utilisateur */ public function userDelete() { // Contenu sélectionné $courseId = $this->getUrl(2); // Accès limité au propriétaire ou admin ou éditeurs inscrits if ( $this->permissionControl(__FUNCTION__, $courseId) === false ) { // Valeurs en sortie $this->addOutput([ 'access' => false ]); } else { $this->deleteData(['enrolment', $this->getUrl(2), $this->getUrl(3)]); // Valeurs en sortie $this->addOutput([ 'redirect' => helper::baseUrl() . 'course/users/' . $this->getUrl(2), 'notification' => sprintf(helper::translate('%s est désinscrit'), $this->getUrl(3)), 'state' => true ]); } } /** * Désinscription de tous les utilisateurs */ public function usersDelete() { // Contenu sélectionné $courseId = $this->getUrl(2); // Accès limité aux admins, à l'auteur ou éditeurs inscrits if ( $this->permissionControl(__FUNCTION__, $courseId) === false ) { // Valeurs en sortie $this->addOutput([ 'access' => false ]); } // Inscription des utilisateurs cochés if ( isset($_POST['courseUsersDeleteSubmit']) ) { foreach ($_POST as $keyPost => $valuePost) { // Exclure les variables post qui ne sont pas des userId et ne traiter que les non inscrits if ( $this->getData(['user', $keyPost]) !== null && $this->getData(['enrolment', $courseId, $keyPost]) !== null ) { $this->deleteData(['enrolment', $courseId, $keyPost]); } } } // Liste des groupes et des profils $courseGroups = $this->getData(['profil']); foreach ($courseGroups as $groupId => $groupValue) { switch ($groupId) { case "-1": case "0": break; case "3": self::$courseGroups['30'] = 'Administrateur'; $profils['30'] = 0; break; case "1": case "2": foreach ($groupValue as $profilId => $profilValue) { if ($profilId) { self::$courseGroups[$groupId . $profilId] = sprintf(helper::translate('Groupe %s - Profil %s'), self::$groupPublics[$groupId], $profilValue['name']); $profils[$groupId . $profilId] = 0; } } } } // Liste alphabétique self::$alphabet = range('A', 'Z'); $alphabet = range('A', 'Z'); self::$alphabet = array_combine($alphabet, self::$alphabet); self::$alphabet = array_merge(['all' => 'Tout'], self::$alphabet); // Liste des inscrits dans le contenu sélectionné. $users = $this->getData(['enrolment', $courseId]); if (is_array($users)) { // Tri du tableau par défaut par $userId ksort($users); foreach ($users as $userId => $userValue) { // Compte les rôles if (isset($profils[$this->getData(['user', $userId, 'group']) . $this->getData(['user', $userId, 'profil'])])) { $profils[$this->getData(['user', $userId, 'group']) . $this->getData(['user', $userId, 'profil'])]++; } // Filtres if ( isset($_POST['courseFilterGroup']) || isset($_POST['courseFilterFirstName']) || isset($_POST['courseFilterLastName']) ) { // Groupe et profils $group = (string) $this->getData(['user', $userId, 'group']); $profil = (string) $this->getData(['user', $userId, 'profil']); $firstName = $this->getData(['user', $userId, 'firstname']); $lastName = $this->getData(['user', $userId, 'lastname']); if ( $this->getInput('courseFilterGroup', helper::FILTER_INT) > 0 && $this->getInput('courseFilterGroup', helper::FILTER_STRING_SHORT) !== $group . $profil ) continue; // Première lettre du prénom if ( $this->getInput('courseFilterFirstName', helper::FILTER_STRING_SHORT) !== 'all' && $this->getInput('courseFilterFirstName', helper::FILTER_STRING_SHORT) !== strtoupper(substr($firstName, 0, 1)) ) continue; // Première lettre du nom if ( $this->getInput('courseFilterLastName', helper::FILTER_STRING_SHORT) !== 'all' && $this->getInput('courseFilterLastName', helper::FILTER_STRING_SHORT) !== strtoupper(substr($lastName, 0, 1)) ) continue; } // Construction du tableau self::$courseUsers[] = [ template::checkbox($userId, true, '', ['class' => 'checkboxSelect']), $userId, $this->getData(['user', $userId, 'firstname']), $this->getData(['user', $userId, 'lastname']), $this->getData(['user', $userId, 'tags']), ]; } } // Ajoute les effectifs aux profils du sélecteur foreach (self::$courseGroups as $groupId => $groupValue) { if ($groupId === 'all') { self::$courseGroups['all'] = self::$courseGroups['all'] . ' (' . array_sum($profils) . ')'; } else { self::$courseGroups[$groupId] = self::$courseGroups[$groupId] . ' (' . $profils[$groupId] . ')'; } } // Valeurs en sortie $this->addOutput([ 'title' => helper::translate('Désincription en masse'), 'view' => 'usersDelete', 'vendor' => [ 'datatables' ] ]); } /* * Traitement du changement de langue */ public function swap() { $courseId = $this->getUrl(2); // pageIfd est transmis lors de l'appel de la page depuis un lien direct alors que l'espace n'est pas sélectionné. $pageId = $this->getUrl(3); $userId = $this->getuser('id'); $message = ''; $redirect = helper::baseUrl(); $state = true; // Récupérer les pahes pour contrôler la dernière page vue $p = json_decode(file_get_contents(self::DATA_DIR . $courseId . '/page.json'), true); $pages = $p['page']; // Routage if ( // Sortir du contenu et afficher l'accueil $courseId === 'home' ) { $_SESSION['ZWII_SITE_CONTENT'] = $courseId; } // l'étudiant est inscrit dans le contenu ET le contenu est ouvert // ou un admin ou le prof du contenu sont connectés elseif ( $this->courseIsUserEnroled($courseId) && $this->courseIsAvailable($courseId) ) { // Récupérer la dernière page visitée par cet utilisateur si elle existe $redirect = ( $this->getData(['enrolment', $courseId, $userId, 'lastPageView']) && array_key_exists($this->getData(['enrolment', $courseId, $userId, 'lastPageView']), $pages) ) ? helper::baseUrl() . $this->getData(['enrolment', $courseId, $userId, 'lastPageView']) : helper::baseUrl() . $pageId; /* $essage = $this->getData(['enrolment', $courseId, $userId, 'datePageView']) ? $this->getData(['enrolment', $courseId, $userId, 'datePageView']) : ''; */ if ($this->getData(['course', $courseId, 'access']) === self::COURSE_ACCESS_DATE) { $to = helper::dateUTF8('%d %B %Y', $this->getData(['course', $courseId, 'closingDate']), self::$i18nUI) . helper::translate(' à ') . helper::dateUTF8('%H:%M', $this->getData(['course', $courseId, 'closingDate']), self::$i18nUI); $message .= sprintf(helper::translate('Ce contenu ferme le %s'), $to); } else { $message .= sprintf(helper::translate('Bienvenue dans l\'espace %s'), $this->getData(['course', $courseId, 'title'])); } $_SESSION['ZWII_SITE_CONTENT'] = $courseId; } // Le contenu est fermé elseif ($this->courseIsAvailable($courseId) === false) { // Génération du message $message = helper::translate('Cet espace est fermé'); $state = false; if ($this->getData(['course', $courseId, 'access']) === self::COURSE_ACCESS_DATE) { $from = helper::dateUTF8('%d %B %Y', $this->getData(['course', $courseId, 'openingDate']), self::$i18nUI) . helper::translate(' à ') . helper::dateUTF8('%H:%M', $this->getData(['course', $courseId, 'openingDate']), self::$i18nUI); $to = helper::dateUTF8('%d %B %Y', $this->getData(['course', $courseId, 'closingDate']), self::$i18nUI) . helper::translate(' à ') . helper::dateUTF8('%H:%M', $this->getData(['course', $courseId, 'closingDate']), self::$i18nUI); $message = sprintf(helper::translate('Cet espace ouvre le
%s
et ferme le %s'), $from, $to); } } // le contenu est ouvert, l'étudiant n'est pas inscrit, l'accès au contenu est anonyme elseif ( $this->courseIsAvailable($courseId) && $this->courseIsUserEnroled($courseId) === false ) { // Gérer les modalités d'inscription switch ($this->getData(['course', $courseId, 'enrolment'])) { // Anonyme case self::COURSE_ENROLMENT_GUEST: $_SESSION['ZWII_SITE_CONTENT'] = $courseId; // Accès direct à la page $redirect = helper::baseUrl() . $pageId; break; // Auto avec ou sans clé case self::COURSE_ENROLMENT_SELF: //L'étudiant doit disposer d'un compte if ($this->getUser('id')) { $redirect = helper::baseUrl() . 'course/suscribe/' . $courseId; } else { $message = helper::translate('Vous devez disposer d\'un compte pour accéder à cet espace'); $state = false; } break; case self::COURSE_ENROLMENT_SELF_KEY: //L'étudiant doit disposer d'un compte if ($this->getUser('id')) { $redirect = helper::baseUrl() . 'course/suscribe/' . $courseId; } else { $message = helper::translate('Vous devez disposer d\'un compte et d\'une clé pour accéder à cet espace'); $state = false; } break; // Par le prof case self::COURSE_ENROLMENT_MANDATORY: $message = helper::translate('L\'enseignant doit vous inscrire'); $state = false; break; default: } } // Valeurs en sortie $this->addOutput([ 'redirect' => $redirect, 'notification' => helper::translate($message), 'state' => $state, ]); } /** * Liste les pages consultées par un utilisateur */ public function userHistory() { // Espace sélectionné $courseId = $this->getUrl(2); // Accès limité au propriétaire ou éditeurs inscrits ou admin if ( $this->permissionControl(__FUNCTION__, $courseId) === false ) { // Valeurs en sortie $this->addOutput([ 'access' => false ]); } $userId = $this->getUrl(3); $h = $this->getData(['enrolment', $courseId, $userId, 'history']); // Inversion des clés et des valeurs $history = array(); foreach ($h as $key => $values) { foreach ($values as $value) { $history[$value] = $key; } } ksort($history); // Liste des pages contenues dans cet espace et exclure les barres et les pages masquées $p = json_decode(file_get_contents(self::DATA_DIR . $courseId . '/page.json'), true); foreach ($p['page'] as $pageId => $pageData) { if ($pageData['position'] > 0) { $pages[$pageId] = [ 'title' => $pageData['title'], ]; } } $floorTime = 99999999999; $topTime = 0; $lastView = 0; foreach ($history as $time => $pageId) { if (isset($pages[$pageId]['title'])) { $lastView = ($lastView === 0) ? $time : $lastView; $diff = $time - $lastView; self::$userHistory[] = [ html_entity_decode($pages[$pageId]['title']), $time, ($diff < 1800) ? sprintf("%d' %d''", floor($diff / 60), $diff % 60) : "Non significatif", ]; if ($diff < 1800) { self::$userGraph[] = [ helper::dateUTF8('%Y-%m-%d %H:%M:%S', $time), $diff, html_entity_decode($pages[$pageId]['title']) . ' (' . helper::dateUTF8('%M\'%S"', $diff) . ')' ]; } $lastView = $time; $floorTime = isset($floorTime) && $floorTime < $time ? $floorTime : $time; $topTime = isset($topTime) && $topTime > $time ? $topTime : $time; } } // Décale les temps de consultation for ($i = 0; $i < count(self::$userHistory) - 1; $i++) { self::$userHistory[$i][2] = self::$userHistory[$i + 1][2]; } // Décale les temps de consultation for ($i = 0; $i < count(self::$userGraph) - 1; $i++) { self::$userHistory[$i][1] = self::$userHistory[$i + 1][1]; } // Formate le timestamp array_walk(self::$userHistory, function (&$item) { $item[1] = helper::dateUTF8('%d/%m/%Y %H:%M:%S', $item[1]); }); self::$userStat['floor'] = helper::dateUTF8('%d %B %Y %H:%M', $floorTime); self::$userStat['top'] = helper::dateUTF8('%d %B %Y %H:%M', $topTime); $d = $topTime - $floorTime; // Conversion de la différence en jours, heures et minutes $d_days = floor($d / 86400); // 1 jour = 86400 secondes (24 heures * 60 minutes * 60 secondes) $d_hours = floor(($d % 86400) / 3600); $d_minutes = floor(($d % 3600) / 60); // Affichage du résultat self::$userStat['time'] = $d_days . ' jours, ' . $d_hours . ' heures, ' . $d_minutes . ' minutes '; // Valeurs en sortie $this->addOutput([ 'title' => helper::translate('Historique ') . $this->getData(['user', $userId, 'firstname']) . ' ' . $this->getData(['user', $userId, 'lastname']), 'view' => 'userHistory', 'vendor' => [ "plotly" ] ]); } public function usersHistoryExport() { $courseId = $this->getUrl(2); // Accès limité au propriétaire ou éditeurs inscrits ou admin if ( $this->permissionControl(__FUNCTION__, $courseId) === false ) { // Valeurs en sortie $this->addOutput([ 'access' => false ]); } self::$courseUsers = [ 0 => ['UserId', 'Prénom', 'Nom', 'Page Titre', 'Consultation Date', 'Consultation Heure', 'Progression'] ]; // Statistiques du contenu sélectionné calcul du nombre de pages $sumPages = 0; $data = json_decode(file_get_contents(self::DATA_DIR . $courseId . '/page.json'), true); // Exclure les barres et les pages masquées foreach ($data['page'] as $pageId => $pageData) { if ($pageData['position'] > 0) { $sumPages++; $pages[$pageId] = $pageData['title']; } } // Liste des inscrits dans le contenu sélectionné. $users = $this->getData(['enrolment', $courseId]); if (is_array($users)) { // Tri du tableau par défaut par $userId ksort($users); // Dossier d'export if (is_dir(self::FILE_DIR . 'source/' . $courseId) === false) { mkdir(self::FILE_DIR . 'source/' . $courseId); } if (is_dir(self::FILE_DIR . 'source/' . $courseId . '/export/') === false) { mkdir(self::FILE_DIR . 'source/' . $courseId . '/export/'); } $filename = self::FILE_DIR . 'source/' . $courseId . '/export/' . '/synthèse' . helper::dateUTF8('%Y%m%d', time()) . '.csv'; foreach ($users as $userId => $userValue) { // Date et heure de la dernière page vue // Compatibilité anciennes versions if ( $this->getData(['enrolment', $courseId, $userId, 'lastPageView']) === null or $this->getData(['enrolment', $courseId, $userId, 'datePageView']) === null ) { if (!empty($userValue['history'])) { $maxTime = max($userValue['history']); $lastPageId = array_search($maxTime, $userValue['history']); $this->setData(['enrolment', $courseId, $userId, 'lastPageView', $lastPageId]); $this->setData(['enrolment', $courseId, $userId, 'datePageView', $maxTime]); } } // Progression $viewPages = $this->getData(['enrolment', $courseId, $userId, 'history']) !== null ? count(array_keys($this->getData(['enrolment', $courseId, $userId, 'history']))) : 0; // Construction du tableau self::$courseUsers[] = [ $userId, $this->getData(['user', $userId, 'firstname']), $this->getData(['user', $userId, 'lastname']), isset($pages[$this->getData(['enrolment', $courseId, $userId, 'lastPageView'])]) ? $pages[$this->getData(['enrolment', $courseId, $userId, 'lastPageView'])] : $this->getData(['enrolment', $courseId, $userId, 'lastPageView']) . ' (supprimée)', helper::dateUTF8('%d/%d/%Y', $this->getData(['enrolment', $courseId, $userId, 'datePageView'])), helper::dateUTF8('%H:%M', $this->getData(['enrolment', $courseId, $userId, 'datePageView'])), number_format(min(round(($viewPages * 100) / $sumPages, 1) / 100, 1), 2, ',') ]; // Synthèse des historiques // ------------------------ // Ouverture du fichier en écriture $file = fopen($filename, 'w'); foreach (self::$courseUsers as $user) { // Décode les entités HTML dans chaque élément du tableau $decodedUser = array_map('html_entity_decode', $user); // Écrire la ligne dans le fichier CSV fputcsv($file, $decodedUser, ';'); } // Fermeture du fichier fclose($file); // Valeurs en sortie $this->addOutput([ 'redirect' => helper::baseUrl() . 'course/users/' . $courseId, 'notification' => 'Création ' . basename($filename) . ' dans le dossier "Export"', 'state' => true, ]); } } } public function userHistoryExport() { $courseId = $this->getUrl(2); $userId = $this->getUrl(3); // Accès limité au propriétaire ou éditeur inscrit ou admin if ( $this->permissionControl(__FUNCTION__, $courseId) === false ) { // Valeurs en sortie $this->addOutput([ 'access' => false ]); } // Traitement de l'historique $h = $this->getData(['enrolment', $courseId, $userId, 'history']); // Inversion des clés et des valeurs $history = array(); foreach ($h as $key => $values) { foreach ($values as $value) { $history[$value] = $key; } } ksort($history); // Liste des pages contenues dans cet espace et exclure les barres et les pages masquées $p = json_decode(file_get_contents(self::DATA_DIR . $courseId . '/page.json'), true); foreach ($p['page'] as $pageId => $pageData) { if ($pageData['position'] > 0) { $pages[$pageId] = [ 'title' => $pageData['title'], ]; } } $lastView = 0; foreach ($history as $time => $pageId) { if (isset($pages[$pageId]['title'])) { $lastView = ($lastView === 0) ? $time : $lastView; $diff = $time - $lastView; self::$userHistory[] = [ $pageId, html_entity_decode($pages[$pageId]['title']), $time, ($diff < 1800) ? sprintf("%d' %d''", floor($diff / 60), $diff % 60) : "Non significatif", ]; $lastView = $time; $floorTime = isset($floorTime) && $floorTime < $time ? $floorTime : $time; $topTime = isset($topTime) && $topTime > $time ? $topTime : $time; } } // Décale les temps de consultation for ($i = 0; $i < count(self::$userHistory) - 1; $i++) { self::$userHistory[$i][3] = self::$userHistory[$i + 1][3]; } // Formate le timestamp array_walk(self::$userHistory, function (&$item) { $item[2] = helper::dateUTF8('%d/%m/%Y %H:%M:%S', $item[2]); }); // Ajoute les entêtes self::$userHistory = array_merge([0 => ['PageId', 'Page Titre', 'Consultation Date', 'Temps Consultation']], self::$userHistory); // Dossier d'export if (is_dir(self::FILE_DIR . 'source/' . $courseId) === false) { mkdir(self::FILE_DIR . 'source/' . $courseId); } if (is_dir(self::FILE_DIR . 'source/' . $courseId . '/export/') === false) { mkdir(self::FILE_DIR . 'source/' . $courseId . '/export/'); } $filename = self::FILE_DIR . 'source/' . $courseId . '/export/' . $userId . '.csv'; $file = fopen($filename, 'w'); foreach (self::$userHistory as $keys => $values) { $data = $values; // Écrire la ligne dans le fichier CSV fputcsv($file, $data, ';'); } // Fermeture du fichier fclose($file); // Valeurs en sortie $this->addOutput([ 'redirect' => helper::baseUrl() . 'course/userHistory/' . $courseId . '/' . $userId, 'notification' => 'Création ' . basename($filename) . ' dans le dossier "Export"', 'state' => true, ]); } // Génération du message d'inscription public function suscribe() { $courseId = $this->getUrl(2); $userId = $this->getUser('id'); // Soumission du formulaire if ( $this->isPost() ) { if ( $this->courseIsAvailable($courseId) ) { // Inscrit l'étudiant switch ($this->getData(['course', $courseId, 'enrolment'])) { case self::COURSE_ENROLMENT_SELF: $this->courseEnrolUser($courseId, $userId); // Stocker la sélection $_SESSION['ZWII_SITE_CONTENT'] = $courseId; // Valeurs en sortie $this->addOutput([ 'redirect' => helper::baseUrl() ]); break; case self::COURSE_ENROLMENT_SELF_KEY: if ($this->getInput('courseSwapEnrolmentKey', helper::FILTER_STRING_SHORT, true) === $this->getData(['course', $courseId, 'enrolmentKey'])) { $this->courseEnrolUser($courseId, $userId); // Stocker la sélection $_SESSION['ZWII_SITE_CONTENT'] = $courseId; // Valeurs en sortie $this->addOutput([ 'redirect' => helper::baseUrl() ]); } else { // Valeurs en sortie $this->addOutput([ 'redirect' => helper::baseUrl() . 'course/suscribe/' . $courseId, 'state' => false, 'notification' => 'La clé est incorrecte' ]); } break; } } } // L'étudiant est-il inscrit self::$swapMessage['submitLabel'] = helper::translate('M\'inscrire'); self::$swapMessage['enrolmentMessage'] = ''; self::$swapMessage['enrolmentKey'] = ''; if ($this->courseIsUserEnroled($courseId) === false) { switch ($this->getData(['course', $courseId, 'enrolment'])) { case self::COURSE_ENROLMENT_SELF: if ($userId == '') { self::$swapMessage['enrolmentMessage'] = helper::translate('Connectez-vous pour accéder à ce espace.'); self::$swapMessage['submitLabel'] = helper::translate('Connexion'); } break; case self::COURSE_ENROLMENT_SELF_KEY: if ($userId == '') { self::$swapMessage['enrolmentMessage'] = helper::translate('Connectez-vous pour accéder à cet espace.'); self::$swapMessage['submitLabel'] = helper::translate('Connexion'); } else { self::$swapMessage['enrolmentKey'] = template::text('courseSwapEnrolmentKey', [ 'label' => helper::translate('Clé d\'inscription'), ]); } break; case self::COURSE_ENROLMENT_MANDATORY: self::$swapMessage['enrolmentMessage'] = helper::translate('Vous ne pouvez pas vous inscrire par vous-même.'); break; } // Valeurs en sortie $this->addOutput([ 'title' => sprintf(helper::translate('Accéder à l\'espace %s'), $this->getData(['course', $this->getUrl(2), 'title'])), 'view' => 'suscribe', 'display' => self::DISPLAY_LAYOUT_LIGHT, ]); } } /** * Désinscription d'un participant */ public function unsuscribe() { // Désincription du contenu ouvert ou du contenu sélectionné $courseId = $this->getUrl(2) ? $this->getUrl(2) : self::$siteContent; // home n'est pas un contenu dans lequel on peut se désincrire if ( $courseId !== 'home' && array_key_exists($courseId, $this->getData(['course'])) ) { $userId = $this->getUser('id'); $this->deleteData(['enrolment', $courseId, $userId]); $_SESSION['ZWII_SITE_CONTENT'] = 'home'; // Valeurs en sortie $this->addOutput([ 'redirect' => helper::baseUrl(), 'notification' => helper::translate('Désinscription'), 'state' => true, ]); } } /** * Sauvegarde d'un cours sans option */ public function backup() { // Espace sélectionné $courseId = $this->getUrl(2); // Accès limité aux admins, à l'auteur ou éditeurs inscrits if ( $this->permissionControl(__FUNCTION__, $courseId) === false ) { // Valeurs en sortie $this->addOutput([ 'access' => false ]); } else { // Participants avec historiques $enrolment = $this->getData(['enrolment', $courseId]); // Générer un fichier dans le dossier de l'espace file_put_contents(self::DATA_DIR . $courseId . '/enrolment.json', json_encode([$courseId => $enrolment], JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES | JSON_PRETTY_PRINT)); // Idem pour les données du cours $course = $this->getData(['course', $courseId]); // Générer un fichier dans le dossier de l'espace file_put_contents(self::DATA_DIR . $courseId . '/course.json', json_encode([$courseId => $course], JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES | JSON_PRETTY_PRINT)); // Idem pour la catégorie $category = $this->getData(['category', $this->getData(['course', $courseId, 'category'])]); // Générer un fichier dans le dossier de l'espace file_put_contents(self::DATA_DIR . $courseId . '/category.json', json_encode([$this->getData(['course', $courseId, 'category']) => $category], JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES | JSON_PRETTY_PRINT)); // Génère une archive ZIP $this->makeZip(self::TEMP_DIR . $courseId . '-' . date('Y-m-d-H-i-s', time()) . '.zip', self::DATA_DIR . $courseId); $success = false; $message = helper::translate('Erreur : sauvegarde non générée !'); // Transférer dans RFM if (file_exists(self::TEMP_DIR . $courseId . '-' . date('Y-m-d-H-i-s', time()) . '.zip')) { if (!is_dir(self::FILE_DIR . 'source/' . $courseId)) { mkdir(self::FILE_DIR . 'source/' . $courseId); } if (!is_dir(self::FILE_DIR . 'source/' . $courseId . '/backup/')) { mkdir(self::FILE_DIR . 'source/' . $courseId . '/backup/'); } copy(self::TEMP_DIR . $courseId . '-' . date('Y-m-d-H-i-s', time()) . '.zip', self::FILE_DIR . 'source/' . $courseId . '/backup/' . $courseId . '-' . date('Y-m-d-H-i-s', time()) . '.zip'); unlink(self::TEMP_DIR . $courseId . '-' . date('Y-m-d-H-i-s', time()) . '.zip'); $success = true; $message = helper::translate('Sauvegarde générée avec succès'); } // Valeurs en sortie $this->addOutput([ 'redirect' => helper::baseUrl() . 'course/manage/' . $this->getUrl(2), 'state' => $success, 'notification' => $message, ]); } } /** * Sauvegarde d'un cours sans option */ public function restore() { // Espace sélectionné $courseId = $this->getUrl(2); // Accès limité aux admins, à l'auteur ou éditeurs inscrits if ( $this->permissionControl(__FUNCTION__, $courseId) === false ) { // Valeurs en sortie $this->addOutput([ 'access' => false ]); } // Soumission du formulaire if ( $this->isPost() ) { // Récupérer le dossier du profil $userPath = $this->getData(['profil', $this->getuser('group'), $this->getuser('profil'), 'folder', 'path']); $userPath = $userPath === '' ? self::$siteContent : $userPath; // Fichier avec le bon chemin selon le profil $zipName = self::FILE_DIR . 'source/' . $userPath . '/' . $this->getInput('courseRestoreFile', null, true); // Existence de l'archive 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); // Drapeaux de gestion des erreurs $success = false; $notification = ''; // Récupérer les données de base à intégrer $courseData = array(); if (file_exists(self::TEMP_DIR . $tempFolder . '/course.json')) { $courseData = json_decode(file_get_contents(self::TEMP_DIR . $tempFolder . '/course.json'), true); // Lire l'id du cours $courseIds = array_keys($courseData); ; $courseId = $courseIds[0]; $success = true; } else { // Pas une archive d'espace $notification = helper::translate('Archive invalide'); } if ($success && $courseId) { // récupérer les inscriptions disponibles $enrolmentData = array(); if (file_exists(self::TEMP_DIR . $tempFolder . '/enrolment.json')) { $enrolmentData = json_decode(file_get_contents(self::TEMP_DIR . $tempFolder . '/enrolment.json'), true); } // Créer le dossier absent if (!is_dir(self::DATA_DIR . $courseId)) { mkdir(self::DATA_DIR . $courseId); $notification = sprintf(helper::translate('Importation terminée : l\'espace %s a été créé'), $courseId); } else { $notification = sprintf(helper::translate('Importation terminée : l\'espace %s a été actualisé'), $courseId); } // traiter l'archive $success = $zip->extractTo(self::DATA_DIR . $courseId); $zip->close(); // Effacer les données de transport unlink(self::DATA_DIR . $courseId . '/course.json'); unlink(self::DATA_DIR . $courseId . '/enrolment.json'); // Fusionne les deux tableaux $c = $this->getData(['course']); $courseData = array_merge($c, $courseData); $e = $this->getData(['enrolment']); $enrolmentData = array_merge($e, $enrolmentData); // Sauvegarde les bases $this->setData(['course', $courseData]); $this->setData(['enrolment', $enrolmentData]); // traitement d'erreur en cas de problème de désachivage $notification = $success ? $notification : helper::translate('Erreur lors de l\'extraction, vérifiez les permissions'); } // Supprimer le dossier temporaire même si le thème est invalide $this->deleteDir(self::TEMP_DIR . $tempFolder); } else { // erreur à l'ouverture $success = false; $notification = helper::translate('Impossible d\'ouvrir l\'archive'); } // Valeurs en sortie $this->addOutput([ 'redirect' => helper::baseUrl() . 'course', 'state' => $success, 'notification' => $notification, ]); } // Valeurs en sortie $this->addOutput([ 'redirect' => helper::baseUrl() . 'course', 'state' => $success, 'notification' => $notification, ]); } // Valeurs en sortie $this->addOutput([ 'title' => helper::translate('Restaurer un espace'), 'view' => 'restore' ]); } /** * Retourne false quand l'utilisateur ne dispose pas des droits d'accès à la fonction * Règles d'accès à un espace : * Admin : tous les droits * Editor : Inscrits dans le cours ou propriétaire */ public function permissionControl($function, $courseId) { switch ($this->getUser('group')) { case self::GROUP_ADMIN: return true; case self::GROUP_EDITOR: return ( $this->getUser('permission', __CLASS__, $function) && $this->getUser('group') === self::$actions[$function] && // Permission d'accèder aux espaces dans lesquels le membre auteur ( $this->getData(['enrolment', $courseId]) && ($this->getUser('id') === $this->getData(['course', $courseId, 'author'])) ) || ( // Permission d'accèder aux esapces dans lesquels le membre est inscrits avec les $this->getData(['enrolment', $courseId]) && $this->getUser('permission', __CLASS__, 'tutor') === true && array_key_exists($this->getUser('id'), $this->getData(['enrolment', $courseId])) ) ); default: return false; } } /** * Autorise l'accès à un contenu * @param @return bool le user a le droit d'entrée dans le contenu * @param string $courseId identifiant du contenu sollicité */ public function courseIsAvailable($courseId) { // L'accès à l'accueil est toujours autorisé if ($courseId === 'home') { return true; } // Si un utilisateur connecté est admin ou auteur, c'est autorisé if ( $this->getUser('password') === $this->getInput('ZWII_USER_PASSWORD') && ($this->getUser('group') === self::GROUP_ADMIN || $this->getUser('id') === $this->getData(['course', $courseId, 'author'])) ) { return true; } // Retourne le statut du contenu dans les autres cas $access = $this->getData(['course', $courseId, 'access']); switch ($access) { case self::COURSE_ACCESS_OPEN: return true; case self::COURSE_ACCESS_DATE: return ( time() >= $this->getData(['course', $courseId, 'openingDate']) && time() <= $this->getData(['course', $courseId, 'closingDate']) ); case self::COURSE_ACCESS_CLOSE: return false; } } private function countPages($array) { $count = 0; foreach ($array as $key => $value) { $count++; // Incrémente le compteur pour chaque clé associative trouvée if (is_array($value)) { $count += $this->countPages($value); // Appelle récursivement la fonction si la valeur est un tableau } } return $count; } private function courseEnrolUser($courseId, $userId) { $this->setData([ 'enrolment', $courseId, $userId, [ 'history' => [], ] ]); } /** * Autorise l'accès à un contenu * @param @return bool le user a le droit d'entrée dans le contenu * @param string $userId identifiant de l'utilisateur * @param string $courseId identifiant du contenu sollicité */ private function courseIsUserEnroled($courseId) { $userId = $this->getUser('id'); $group = $userId ? $this->getData(['user', $userId, 'group']) : null; switch ($group) { case self::GROUP_ADMIN: $r = true; break; case self::GROUP_EDITOR: $r = in_array($userId, array_keys($this->getData(['enrolment', $courseId]))); break; case self::GROUP_MEMBER: $r = in_array($userId, array_keys($this->getData(['enrolment', $courseId]))); break; // Visiteur non connecté case self::GROUP_VISITOR: case null: $r = $this->getData(['course', $courseId, 'enrolment']) === self::COURSE_ENROLMENT_GUEST; break; default: $r = false; } return $r; } }