- BeezNest * @package chamilo.auth */ class CoursesController { private $toolname; private $view; private $model; /** * Constructor */ public function __construct() { $this->toolname = 'auth'; $actived_theme_path = api_get_template(); $this->view = new View($this->toolname, $actived_theme_path); $this->model = new Auth(); } /** * It's used for listing courses, * render to courses_list view * @param string action * @param string confirmation message(optional) */ public function courses_list($action, $message = '') { $data = array(); $user_id = api_get_user_id(); $data['user_courses'] = $this->model->get_courses_of_user($user_id); $data['user_course_categories'] = $this->model->get_user_course_categories(); $data['courses_in_category'] = $this->model->get_courses_in_category(); $data['all_user_categories'] = $this->model->get_user_course_categories( ); $data['action'] = $action; $data['message'] = $message; // render to the view $this->view->set_data($data); $this->view->set_layout('catalog_layout'); $this->view->set_template('courses_list'); $this->view->render(); } /** * It's used for listing categories, * render to categories_list view * @param string $action * @param string $message confirmation message(optional) * @param string $error error message(optional) */ public function categories_list($action, $message='', $error='') { $data = array(); $data['user_course_categories'] = $this->model->get_user_course_categories(); $data['action'] = $action; $data['message'] = $message; $data['error'] = $error; // render to the view $this->view->set_data($data); $this->view->set_layout('catalog_layout'); $this->view->set_template('categories_list'); $this->view->render(); } /** * It's used for listing courses with categories, * render to courses_categories view * @param $action * @param string $category_code * @param string $message * @param string $error * @param string $content * @param array $limit will be used if $random_value is not set. * This array should contains 'start' and 'length' keys * @internal param \action $string * @internal param \Category $string code (optional) */ public function courses_categories($action, $category_code = null, $message = '', $error = '', $content = null, $limit = array()) { $data = array(); $browse_course_categories = $this->model->browse_course_categories(); $data['countCoursesInCategory'] = $this->model->count_courses_in_category($category_code); if ($action == 'display_random_courses') { // Random value is used instead limit filter $data['browse_courses_in_category'] = $this->model->browse_courses_in_category(null, 10); $data['countCoursesInCategory'] = count($data['browse_courses_in_category']); } else { if (!isset($category_code)) { $category_code = $browse_course_categories[0][1]['code']; // by default first category } $limit = isset($limit) ? $limit : getLimitArray(); $data['browse_courses_in_category'] = $this->model->browse_courses_in_category($category_code, null, $limit); } $data['browse_course_categories'] = $browse_course_categories; $data['code'] = Security::remove_XSS($category_code); // getting all the courses to which the user is subscribed to $curr_user_id = api_get_user_id(); $user_courses = $this->model->get_courses_of_user($curr_user_id); $user_coursecodes = array(); // we need only the course codes as these will be used to match against the courses of the category if ($user_courses != '') { foreach($user_courses as $key => $value) { $user_coursecodes[] = $value['code']; } } if (api_is_drh()) { $courses = CourseManager::get_courses_followed_by_drh(api_get_user_id()); foreach ($courses as $course) { $user_coursecodes[] = $course['code']; } } $data['user_coursecodes'] = $user_coursecodes; $data['action'] = $action; $data['message'] = $message; $data['content'] = $content; $data['error'] = $error; $data['catalogShowCoursesSessions'] = 0; $showCoursesSessions = intval('catalog_show_courses_sessions'); if ($showCoursesSessions > 0) { $data['catalogShowCoursesSessions'] = $showCoursesSessions; } // render to the view $this->view->set_data($data); $this->view->set_layout('catalog_layout'); $this->view->set_template('courses_categories'); $this->view->render(); } /** * @param string $search_term * @param string $message * @param string $error * @param string $content * @param $limit */ public function search_courses($search_term, $message = '', $error = '', $content = null, $limit = array()) { $data = array(); $limit = !empty($limit) ? $limit : getLimitArray(); $browse_course_categories = $this->model->browse_course_categories(); $data['countCoursesInCategory'] = $this->model->count_courses_in_category('ALL', $search_term); $data['browse_courses_in_category'] = $this->model->search_courses($search_term, $limit); $data['browse_course_categories'] = $browse_course_categories; $data['search_term'] = Security::remove_XSS($search_term); //filter before showing in template // getting all the courses to which the user is subscribed to $curr_user_id = api_get_user_id(); $user_courses = $this->model->get_courses_of_user($curr_user_id); $user_coursecodes = array(); // we need only the course codes as these will be used to match against the courses of the category if ($user_courses != '') { foreach ($user_courses as $value) { $user_coursecodes[] = $value['code']; } } $data['user_coursecodes'] = $user_coursecodes; $data['message'] = $message; $data['content'] = $content; $data['error'] = $error; $data['action'] = 'display_courses'; // render to the view $this->view->set_data($data); $this->view->set_layout('catalog_layout'); $this->view->set_template('courses_categories'); $this->view->render(); } /** * Auto user subscription to a course */ public function subscribe_user($course_code, $search_term, $category_code) { $courseInfo = api_get_course_info($course_code); // The course must be open in order to access the auto subscription if (in_array( $courseInfo['visibility'], array(COURSE_VISIBILITY_CLOSED, COURSE_VISIBILITY_REGISTERED, COURSE_VISIBILITY_HIDDEN)) ) { $error = get_lang('SubscribingNotAllowed'); } else { $result = $this->model->subscribe_user($course_code); if (!$result) { $error = get_lang('CourseRegistrationCodeIncorrect'); } else { // Redirect directly to the course after subscription $message = $result['message']; $content = $result['content']; } } if (!empty($search_term)) { $this->search_courses($search_term, $message, $error, $content); } else { $this->courses_categories('subscribe', $category_code, $message, $error, $content); } return $result; } /** * Create a category * render to listing view * @param string Category title */ public function add_course_category($category_title) { $result = $this->model->store_course_category($category_title); if ($result) { Display::addFlash(Display::return_message(get_lang('CourseCategoryStored'))); } else { Display::addFlash(Display::return_message(get_lang('ACourseCategoryWithThisNameAlreadyExists'), 'error')); } $action = 'sortmycourses'; $this->courses_list($action); } /** * Change course category * render to listing view * @param string Course code * @param int Category id */ public function change_course_category($course_code, $category_id) { $courseInfo = api_get_course_info($course_code); $courseId = $courseInfo['real_id']; $result = $this->model->updateCourseCategory($courseId, $category_id); if ($result) { Display::addFlash(Display::return_message(get_lang('EditCourseCategorySucces'))); } $action = 'sortmycourses'; $this->courses_list($action); } /** * Move up/down courses inside a category * render to listing view * @param string move to up or down * @param string Course code * @param int Category id */ public function move_course($move, $course_code, $category_id) { $result = $this->model->move_course($move, $course_code, $category_id); if ($result) { Display::addFlash(Display::return_message(get_lang('CourseSortingDone'))); } $action = 'sortmycourses'; $this->courses_list($action); } /** * Move up/down categories * render to listing view * @param string move to up or down * @param int Category id */ public function move_category($move, $category_id) { $result = $this->model->move_category($move, $category_id); if ($result) { Display::addFlash(Display::return_message(get_lang('CategorySortingDone'))); } $action = 'sortmycourses'; $this->courses_list($action); } /** * Edit course category * render to listing view * @param string Category title * @param int Category id */ public function edit_course_category($title, $category) { $result = $this->model->store_edit_course_category($title, $category); if ($result) { Display::addFlash(Display::return_message(get_lang('CourseCategoryEditStored'))); } $action = 'sortmycourses'; $this->courses_list($action); } /** * Delete a course category * render to listing view * @param int Category id */ public function delete_course_category($category_id) { $result = $this->model->delete_course_category($category_id); if ($result) { Display::addFlash(Display::return_message(get_lang('CourseCategoryDeleted'))); } $action = 'sortmycourses'; $this->courses_list($action); } /** * Unsubscribe user from a course * render to listing view * @param string Course code */ public function unsubscribe_user_from_course($course_code, $search_term = null, $category_code = null) { $result = $this->model->remove_user_from_course($course_code); $message = ''; $error = ''; if ($result) { Display::addFlash(Display::return_message(get_lang('YouAreNowUnsubscribed'))); } $action = 'sortmycourses'; if (!empty($search_term)) { $this->search_courses($search_term, $message, $error); } else { $this->courses_categories('subcribe', $category_code, $message, $error); } } /** * Get the html block for courses categories * @param string $code Current category code * @param boolean $hiddenLinks Whether hidden links * @param array $limit * @return string The HTML block */ public function getCoursesCategoriesBlock($code = null, $hiddenLinks = false, $limit = null) { $categories = $this->model->browse_course_categories(); $html = ''; if (!empty($categories)) { $action = 'display_courses'; foreach ($categories[0] as $category) { $categoryName = $category['name']; $categoryCode = $category['code']; $categoryCourses = $category['count_courses']; $html .= '
  • '; if ($code == $categoryCode) { $html .= ''; $html .= "$categoryName ($categoryCourses)"; $html .= ''; } else { if (!empty($categoryCourses)) { $html .= ''; $html .= "$categoryName ($categoryCourses)"; $html .= ''; } else { $html .= "$categoryName ($categoryCourses)"; } } if (!empty($categories[$categoryCode])) { $html .= ''; } $html .= '
  • '; } } return $html; } /** * Get a HTML button for subscribe to session * @param int $sessionId The session ID * @param string $sessionName The session name * @param boolean $checkRequirements Optional. * Whether the session has requirement. Default is false * @return string The button HTML */ public function getRegisteredInSessionButton( $sessionId, $sessionName, $checkRequirements = false ) { if ($checkRequirements) { $url = api_get_path(WEB_AJAX_PATH); $url .= 'sequence.ajax.php?'; $url .= http_build_query([ 'a' => 'get_requirements', 'id' => intval($sessionId), 'type' => SequenceResource::SESSION_TYPE, ]); return Display::toolbarButton( get_lang('CheckRequirements'), $url, 'check-circle', 'primary', [ 'class' => 'btn-lg btn-block ajax', 'data-title' => get_lang('CheckRequirements'), 'data-size' => 'md' ] ); } $catalogSessionAutoSubscriptionAllowed = false; if ( api_get_setting('catalog_allow_session_auto_subscription') === 'true' ) { $catalogSessionAutoSubscriptionAllowed = true; } $url = api_get_path(WEB_CODE_PATH); if ($catalogSessionAutoSubscriptionAllowed) { $url .= 'auth/courses.php?'; $url .= http_build_query([ 'action' => 'subscribe_to_session', 'session_id' => intval($sessionId) ]); $result = Display::toolbarButton( get_lang('Subscribe'), $url, 'check-circle', 'primary', [ 'class' => 'btn-lg btn-block ajax', 'data-title' => get_lang('AreYouSureToSubscribe'), 'data-size' => 'md' ] ); } else { $url .= 'inc/email_editor.php?'; $url .= http_build_query([ 'action' => 'subscribe_me_to_session', 'session' => Security::remove_XSS($sessionName), ]); $result = Display::toolbarButton( get_lang('Subscribe'), $url, 'check-circle', 'primary', ['class' => 'btn-lg btn-block'] ); } $hook = HookResubscribe::create(); if (!empty($hook)) { $hook->setEventData(array( 'session_id' => intval($sessionId), )); try { $hook->notifyResubscribe(HOOK_EVENT_TYPE_PRE); } catch (Exception $exception) { $result = $exception->getMessage(); } } return $result; } /** * Generate a label if the user has been registered in session * @return string The label */ public function getAlreadyRegisteredInSessionLabel() { $icon = ''; return Display::div( $icon . ' ' . get_lang("AlreadyRegisteredToSession"), array('class' => 'info-catalog') ); } /** * Get a icon for a session * @param string $sessionName The session name * @return string The icon */ public function getSessionIcon($sessionName) { return Display::return_icon('window_list.png', $sessionName, null,ICON_SIZE_MEDIUM); } /** * Return Session Catalogue rendered view * @param string $action * @param string $nameTools * @param array $limit */ public function sessionsList($action, $nameTools, $limit = array()) { $date = isset($_POST['date']) ? $_POST['date'] : date('Y-m-d'); $hiddenLinks = isset($_GET['hidden_links']) ? intval($_GET['hidden_links']) == 1 : false; $limit = isset($limit) ? $limit : getLimitArray(); $countSessions = $this->model->countSessions($date); $sessions = $this->model->browseSessions($date, $limit); $pageTotal = intval(ceil(intval($countSessions) / $limit['length'])); // Do NOT show pagination if only one page or less $cataloguePagination = $pageTotal > 1 ? getCataloguePagination($limit['current'], $limit['length'], $pageTotal) : ''; $sessionsBlocks = $this->getFormatedSessionsBlock($sessions); // Get session list catalogue URL //$sessionUrl = getCourseCategoryUrl(1, $limit['length'], null, 0, 'display_sessions'); // Get session search catalogue URL $courseUrl = getCourseCategoryUrl(1, $limit['length'], null, 0, 'subscribe'); $tpl = new Template(); $tpl->assign('show_courses', CoursesAndSessionsCatalog::showCourses()); $tpl->assign('show_sessions', CoursesAndSessionsCatalog::showSessions()); $tpl->assign('show_tutor', (api_get_setting('show_session_coach')==='true' ? true : false)); $tpl->assign('course_url', $courseUrl); $tpl->assign('catalog_pagination', $cataloguePagination); $tpl->assign('hidden_links', $hiddenLinks); $tpl->assign('search_token', Security::get_token()); $tpl->assign('search_date', $date); $tpl->assign('web_session_courses_ajax_url', api_get_path(WEB_AJAX_PATH) . 'course.ajax.php'); $tpl->assign('sessions', $sessionsBlocks); $tpl->assign('already_subscribed_label', $this->getAlreadyRegisteredInSessionLabel()); $contentTemplate = $tpl->get_template('auth/session_catalog.tpl'); $tpl->display($contentTemplate); } /** * Show the Session Catalogue with filtered session by course tags * @param array $limit Limit info */ public function sessionsListByCoursesTag(array $limit) { $searchTag = isset($_POST['search_tag']) ? $_POST['search_tag'] : null; $searchDate = isset($_POST['date']) ? $_POST['date'] : date('Y-m-d'); $hiddenLinks = isset($_GET['hidden_links']) ? intval($_GET['hidden_links']) == 1 : false; $courseUrl = getCourseCategoryUrl(1, $limit['length'], null, 0, 'subscribe'); $sessions = $this->model->browseSessionsByTags($searchTag, $limit); $sessionsBlocks = $this->getFormatedSessionsBlock($sessions); $tpl = new Template(); $tpl->assign('show_courses', CoursesAndSessionsCatalog::showCourses()); $tpl->assign('show_sessions', CoursesAndSessionsCatalog::showSessions()); $tpl->assign('show_tutor', (api_get_setting('show_session_coach')==='true' ? true : false)); $tpl->assign('course_url', $courseUrl); $tpl->assign('already_subscribed_label', $this->getAlreadyRegisteredInSessionLabel()); $tpl->assign('hidden_links', $hiddenLinks); $tpl->assign('search_token', Security::get_token()); $tpl->assign('search_date', Security::remove_XSS($searchDate)); $tpl->assign('search_tag', Security::remove_XSS($searchTag)); $tpl->assign('sessions', $sessionsBlocks); $contentTemplate = $tpl->get_template('auth/session_catalog.tpl'); $tpl->display($contentTemplate); } /** * Show the Session Catalogue with filtered session by a query term * @param array $limit */ public function sessionListBySearch(array $limit) { $q = isset($_REQUEST['q']) ? Security::remove_XSS($_REQUEST['q']) : null; $hiddenLinks = isset($_GET['hidden_links']) ? intval($_GET['hidden_links']) == 1 : false; $courseUrl = getCourseCategoryUrl(1, $limit['length'], null, 0, 'subscribe'); $searchDate = isset($_POST['date']) ? $_POST['date'] : date('Y-m-d'); $sessions = $this->model->browseSessionsBySearch($q, $limit); $sessionsBlocks = $this->getFormatedSessionsBlock($sessions); $tpl = new Template(); $tpl->assign('show_courses', CoursesAndSessionsCatalog::showCourses()); $tpl->assign('show_sessions', CoursesAndSessionsCatalog::showSessions()); $tpl->assign('show_tutor', (api_get_setting('show_session_coach')==='true' ? true : false)); $tpl->assign('course_url', $courseUrl); $tpl->assign('already_subscribed_label', $this->getAlreadyRegisteredInSessionLabel()); $tpl->assign('hidden_links', $hiddenLinks); $tpl->assign('search_token', Security::get_token()); $tpl->assign('search_date', Security::remove_XSS($searchDate)); $tpl->assign('search_tag', Security::remove_XSS($q)); $tpl->assign('sessions', $sessionsBlocks); $contentTemplate = $tpl->get_template('auth/session_catalog.tpl'); $tpl->display($contentTemplate); } /** * Get the formated data for sessions block to be displayed on Session Catalog page * @param array $sessions The session list * @return array */ private function getFormatedSessionsBlock(array $sessions) { $extraFieldValue = new ExtraFieldValue('session'); $userId = api_get_user_id(); $sessionsBlocks = []; $entityManager = Database::getManager(); $sessionRelCourseRepo = $entityManager->getRepository('ChamiloCoreBundle:SessionRelCourse'); $extraFieldRepo = $entityManager->getRepository('ChamiloCoreBundle:ExtraField'); $extraFieldRelTagRepo = $entityManager->getRepository('ChamiloCoreBundle:ExtraFieldRelTag'); $tagsField = $extraFieldRepo->findOneBy([ 'extraFieldType' => Chamilo\CoreBundle\Entity\ExtraField::COURSE_FIELD_TYPE, 'variable' => 'tags', ]); /** @var \Chamilo\CoreBundle\Entity\Session $session */ foreach ($sessions as $session) { $sessionDates = SessionManager::parseSessionDates([ 'display_start_date' => $session->getDisplayStartDate(), 'display_end_date' => $session->getDisplayEndDate(), 'access_start_date' => $session->getAccessStartDate(), 'access_end_date' => $session->getAccessEndDate(), 'coach_access_start_date' => $session->getCoachAccessStartDate(), 'coach_access_end_date' => $session->getCoachAccessEndDate(), ]); $imageField = $extraFieldValue->get_values_by_handler_and_field_variable($session->getId(), 'image'); $sessionCourseTags = []; if (!is_null($tagsField)) { $sessionRelCourses = $sessionRelCourseRepo->findBy([ 'session' => $session, ]); foreach ($sessionRelCourses as $sessionRelCourse) { $courseTags = $extraFieldRelTagRepo->getTags( $tagsField, $sessionRelCourse->getCourse()->getId() ); foreach ($courseTags as $tag) { $sessionCourseTags[] = $tag->getTag(); } } } if (!empty($sessionCourseTags)) { $sessionCourseTags = array_unique($sessionCourseTags); } $repo = $entityManager->getRepository('ChamiloCoreBundle:SequenceResource'); $sequences = $repo->getRequirementsAndDependenciesWithinSequences( $session->getId(), SequenceResource::SESSION_TYPE ); $hasRequirements = false; foreach ($sequences['sequences'] as $sequence) { if (count($sequence['requirements']) === 0) { continue; } $hasRequirements = true; break; } $sessionsBlock = array( 'id' => $session->getId(), 'name' => $session->getName(), 'image' => isset($imageField['value']) ? $imageField['value'] : null, 'nbr_courses' => $session->getNbrCourses(), 'nbr_users' => $session->getNbrUsers(), 'coach_name' => $session->getGeneralCoach()->getCompleteName(), 'is_subscribed' => SessionManager::isUserSubscribedAsStudent($session->getId(), $userId), 'icon' => $this->getSessionIcon($session->getName()), 'date' => $sessionDates['display'], 'subscribe_button' => $this->getRegisteredInSessionButton( $session->getId(), $session->getName(), $hasRequirements ), 'show_description' => $session->getShowDescription(), 'tags' => $sessionCourseTags, ); $sessionsBlock = array_merge($sessionsBlock, $sequences); $sessionsBlocks[] = $sessionsBlock; } return $sessionsBlocks; } }