diff --git a/classes/output/core/course_renderer.php b/classes/output/core/course_renderer.php
new file mode 100644
index 00000000000..f3f132b851f
--- /dev/null
+++ b/classes/output/core/course_renderer.php
@@ -0,0 +1,260 @@
+.
+
+namespace theme_boost_union\output\core;
+
+use html_writer;
+use coursecat_helper;
+use stdClass;
+use core_course_list_element;
+use theme_boost_union\util\course;
+use moodle_url;
+
+/**
+ * Renderers to align Boost Union's course elements to what is expect
+ *
+ * @package theme_boost_union
+ * @copyright 2024 Daniel Neis Araujo {@link https://www.adapta.online}
+ * @copyright 2022 Willian Mano {@link https://conecti.me}
+ * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
+ */
+class course_renderer extends \core_course_renderer {
+
+ /**
+ * Renders the list of courses
+ *
+ * This is internal function, please use core_course_renderer::courses_list() or another public
+ * method from outside of the class
+ *
+ * If list of courses is specified in $courses; the argument $chelper is only used
+ * to retrieve display options and attributes, only methods get_show_courses(),
+ * get_courses_display_option() and get_and_erase_attributes() are called.
+ *
+ * @param coursecat_helper $chelper various display options
+ * @param array $courses the list of courses to display
+ * @param int|null $totalcount total number of courses (affects display mode if it is AUTO or pagination if applicable),
+ * defaulted to count($courses)
+ * @return string
+ */
+ protected function coursecat_courses(coursecat_helper $chelper, $courses, $totalcount = null) {
+ if (!get_config('theme_boost_union', 'enablecards')) {
+ return parent::coursecat_courses($chelper, $courses, $totalcount);
+ }
+ global $CFG;
+ if ($totalcount === null) {
+ $totalcount = count($courses);
+ }
+ if (!$totalcount) {
+ // Courses count is cached during courses retrieval.
+ return '';
+ }
+
+ if ($chelper->get_show_courses() == self::COURSECAT_SHOW_COURSES_AUTO) {
+ // In 'auto' course display mode we analyse if number of courses is more or less than $CFG->courseswithsummarieslimit.
+ if ($totalcount <= $CFG->courseswithsummarieslimit) {
+ $chelper->set_show_courses(self::COURSECAT_SHOW_COURSES_EXPANDED);
+ } else {
+ $chelper->set_show_courses(self::COURSECAT_SHOW_COURSES_COLLAPSED);
+ }
+ }
+
+ // Prepare content of paging bar if it is needed.
+ $paginationurl = $chelper->get_courses_display_option('paginationurl');
+ $paginationallowall = $chelper->get_courses_display_option('paginationallowall');
+ if ($totalcount > count($courses)) {
+ // There are more results that can fit on one page.
+ if ($paginationurl) {
+ // The option paginationurl was specified, display pagingbar.
+ $perpage = $chelper->get_courses_display_option('limit', $CFG->coursesperpage);
+ $page = $chelper->get_courses_display_option('offset') / $perpage;
+ $pagingbar = $this->paging_bar($totalcount, $page, $perpage,
+ $paginationurl->out(false, ['perpage' => $perpage]));
+ if ($paginationallowall) {
+ $pagingbar .= html_writer::tag('div', html_writer::link($paginationurl->out(false, ['perpage' => 'all']),
+ get_string('showall', '', $totalcount)), ['class' => 'paging paging-showall']);
+ }
+ } else if ($viewmoreurl = $chelper->get_courses_display_option('viewmoreurl')) {
+ // The option for 'View more' link was specified, display more link.
+ $viewmoretext = $chelper->get_courses_display_option('viewmoretext', new \lang_string('viewmore'));
+ $morelink = html_writer::tag(
+ 'div',
+ html_writer::link($viewmoreurl, $viewmoretext, ['class' => 'btn btn-secondary']),
+ ['class' => 'paging paging-morelink']
+ );
+ }
+ } else if (($totalcount > $CFG->coursesperpage) && $paginationurl && $paginationallowall) {
+ // There are more than one page of results and we are in 'view all' mode, suggest to go back to paginated view mode.
+ $pagingbar = html_writer::tag('div',
+ html_writer::link($paginationurl->out(false, ['perpage' => $CFG->coursesperpage]),
+ get_string('showperpage', '', $CFG->coursesperpage)), ['class' => 'paging paging-showperpage']);
+ }
+
+ // Display list of courses.
+ $attributes = $chelper->get_and_erase_attributes('courses');
+ $content = html_writer::start_tag('div', $attributes);
+
+ if (!empty($pagingbar)) {
+ $content .= $pagingbar;
+ }
+
+ $coursecount = 1;
+ $content .= html_writer::start_tag('div', ['class' => 'card-deck dashboard-card-deck mt-2']);
+ foreach ($courses as $course) {
+ $content .= $this->coursecat_coursebox($chelper, $course);
+
+ if ($coursecount % 3 == 0) {
+ $content .= html_writer::end_tag('div');
+ $content .= html_writer::start_tag('div', ['class' => 'card-deck dashboard-card-deck mt-2']);
+ }
+
+ $coursecount ++;
+ }
+
+ $content .= html_writer::end_tag('div');
+
+ if (!empty($pagingbar)) {
+ $content .= $pagingbar;
+ }
+
+ if (!empty($morelink)) {
+ $content .= $morelink;
+ }
+
+ $content .= html_writer::end_tag('div'); // End courses.
+
+ return $content;
+ }
+
+ /**
+ * Displays one course in the list of courses.
+ *
+ * This is an internal function, to display an information about just one course
+ * please use core_course_renderer::course_info_box()
+ *
+ * @param coursecat_helper $chelper various display options
+ * @param core_course_list_element|stdClass $course
+ * @param string $additionalclasses additional classes to add to the main
tag (usually
+ * depend on the course position in list - first/last/even/odd)
+ * @return string
+ *
+ * @throws \coding_exception
+ * @throws \dml_exception
+ * @throws \moodle_exception
+ */
+ protected function coursecat_coursebox(coursecat_helper $chelper, $course, $additionalclasses = '') {
+ if (!get_config('theme_boost_union', 'enablecards')) {
+ return parent::coursecat_coursebox($chelper, $course, $additionalclasses);
+ }
+ if (!isset($this->strings->summary)) {
+ $this->strings->summary = get_string('summary');
+ }
+
+ if ($chelper->get_show_courses() <= self::COURSECAT_SHOW_COURSES_COUNT) {
+ return '';
+ }
+
+ if ($course instanceof stdClass) {
+ $course = new core_course_list_element($course);
+ }
+
+ return $this->coursecat_coursebox_content($chelper, $course);
+ }
+
+ /**
+ * Returns HTML to display course content (summary, course contacts and optionally category name)
+ *
+ * This method is called from coursecat_coursebox() and may be re-used in AJAX
+ *
+ * @param coursecat_helper $chelper various display options
+ * @param stdClass|core_course_list_element $course
+ *
+ * @return string
+ *
+ * @throws \coding_exception
+ * @throws \dml_exception
+ * @throws \moodle_exception
+ */
+ protected function coursecat_coursebox_content(coursecat_helper $chelper, $course) {
+ if (!get_config('theme_boost_union', 'enablecards')) {
+ return parent::coursecat_coursebox_content($chelper, $course);
+ }
+ if ($course instanceof stdClass) {
+ $course = new core_course_list_element($course);
+ }
+
+ $courseutil = new course($course);
+
+ $coursecontacts = $courseutil->get_course_contacts();
+
+ $courseenrolmenticons = $courseutil->get_enrolment_icons();
+ $courseenrolmenticons = !empty($courseenrolmenticons) ? $this->render_enrolment_icons($courseenrolmenticons) : false;
+
+ $courseprogress = $courseutil->get_progress();
+ $hasprogress = $courseprogress != null;
+
+ $data = [
+ 'id' => $course->id,
+ 'fullname' => $chelper->get_course_formatted_name($course),
+ 'visible' => $course->visible,
+ 'image' => $courseutil->get_summary_image(),
+ 'summary' => $courseutil->get_summary($chelper),
+ 'category' => $courseutil->get_category(),
+ 'customfields' => $courseutil->get_custom_fields(),
+ 'hasprogress' => $hasprogress,
+ 'progress' => (int) $courseprogress,
+ 'hasenrolmenticons' => $courseenrolmenticons != false,
+ 'enrolmenticons' => $courseenrolmenticons,
+ 'hascontacts' => !empty($coursecontacts),
+ 'contacts' => $coursecontacts,
+ 'courseurl' => $this->get_course_url($course->id),
+ ];
+
+ return $this->render_from_template('theme_boost_union/coursecard', $data);
+ }
+
+ /**
+ * Returns enrolment icons
+ *
+ * @param array $icons
+ *
+ * @return array
+ */
+ protected function render_enrolment_icons(array $icons): array {
+ $data = [];
+
+ foreach ($icons as $icon) {
+ $data[] = $this->render($icon);
+ }
+
+ return $data;
+ }
+
+ /**
+ * Returns the course URL based on some criterias.
+ *
+ * @param int $courseid
+ *
+ * @return moodle_url
+ * @throws \moodle_exception
+ */
+ private function get_course_url($courseid) {
+ if (class_exists('\local_course\output\index')) {
+ return new moodle_url('/local/course/index.php', ['id' => $courseid]);
+ }
+
+ return new moodle_url('/course/view.php', ['id' => $courseid]);
+ }
+}
diff --git a/classes/util/course.php b/classes/util/course.php
new file mode 100644
index 00000000000..2d022a8e01c
--- /dev/null
+++ b/classes/util/course.php
@@ -0,0 +1,166 @@
+.
+
+namespace theme_boost_union\util;
+
+use moodle_url;
+use core_course_list_element;
+use coursecat_helper;
+use core_course_category;
+
+/**
+ * Course class utility class
+ *
+ * @package theme_boost_union
+ * @copyright 2022 Willian Mano {@link https://conecti.me}
+ * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
+ */
+class course {
+ /**
+ * @var \stdClass $course The course object.
+ */
+ protected $course;
+
+ /**
+ * Class constructor
+ *
+ * @param core_course_list_element $course
+ *
+ */
+ public function __construct($course) {
+ $this->course = $course;
+ }
+
+ /**
+ * Returns the first course's summary image url
+ *
+ * @return string
+ */
+ public function get_summary_image() {
+ global $CFG, $OUTPUT;
+
+ foreach ($this->course->get_course_overviewfiles() as $file) {
+ if ($file->is_valid_image()) {
+ $url = moodle_url::make_file_url("$CFG->wwwroot/pluginfile.php",
+ '/' . $file->get_contextid() . '/' . $file->get_component() . '/' .
+ $file->get_filearea() . $file->get_filepath() . $file->get_filename(), !$file->is_valid_image());
+
+ return $url->out();
+ }
+ }
+
+ return $OUTPUT->get_generated_image_for_id($this->course->id);
+ }
+
+ /**
+ * Returns HTML to display course contacts.
+ *
+ * @return array
+ */
+ public function get_course_contacts() {
+ $theme = \theme_config::load('boost_union');
+
+ $contacts = [];
+ if ($theme->settings->enableteacherspic && $this->course->has_course_contacts()) {
+ $instructors = $this->course->get_course_contacts();
+
+ foreach ($instructors as $instructor) {
+ $user = $instructor['user'];
+ $userutil = new user($user->id);
+
+ $contacts[] = [
+ 'id' => $user->id,
+ 'fullname' => fullname($user),
+ 'userpicture' => $userutil->get_user_picture(),
+ 'role' => $instructor['role']->displayname,
+ ];
+ }
+ }
+
+ return $contacts;
+ }
+
+ /**
+ * Returns HTML to display course category name.
+ *
+ * @return string
+ *
+ * @throws \moodle_exception
+ */
+ public function get_category(): string {
+ $cat = core_course_category::get($this->course->category, IGNORE_MISSING);
+
+ if (!$cat) {
+ return '';
+ }
+
+ return $cat->get_formatted_name();
+ }
+
+ /**
+ * Returns course summary.
+ *
+ * @param coursecat_helper $chelper
+ */
+ public function get_summary(coursecat_helper $chelper): string {
+ if ($this->course->has_summary()) {
+ return $chelper->get_course_formatted_summary($this->course,
+ ['overflowdiv' => true, 'noclean' => true, 'para' => false]
+ );
+ }
+
+ return false;
+ }
+
+ /**
+ * Returns course custom fields.
+ *
+ * @return string
+ */
+ public function get_custom_fields(): string {
+ if ($this->course->has_custom_fields()) {
+ $handler = \core_course\customfield\course_handler::create();
+
+ return $handler->display_custom_fields_data($this->course->get_custom_fields());
+ }
+
+ return '';
+ }
+
+ /**
+ * Returns HTML to display course enrolment icons.
+ *
+ * @return array
+ */
+ public function get_enrolment_icons(): array {
+ if ($icons = enrol_get_course_info_icons($this->course)) {
+ return $icons;
+ }
+
+ return [];
+ }
+
+ /**
+ * Get the user progress in the course.
+ *
+ * @param null $userid
+ *
+ * @return mixed
+ */
+ public function get_progress($userid = null) {
+ return \core_completion\progress::get_course_progress_percentage(get_course($this->course->id), $userid);
+ }
+}
diff --git a/classes/util/user.php b/classes/util/user.php
new file mode 100644
index 00000000000..60b7f91d21c
--- /dev/null
+++ b/classes/util/user.php
@@ -0,0 +1,72 @@
+.
+
+namespace theme_boost_union\util;
+
+use stdClass;
+use user_picture;
+
+/**
+ * User class utility class
+ *
+ * @package theme_boost_union
+ * @copyright 2022 Willian Mano {@link https://conecti.me}
+ * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
+ */
+class user {
+ /**
+ * @var \stdClass $user The user object.
+ */
+ protected $user;
+
+ /**
+ * Class constructor
+ *
+ * @param stdClass $user
+ *
+ */
+ public function __construct($user = null) {
+ global $USER, $DB;
+
+ if (!is_object($user)) {
+ $user = $DB->get_record('user', ['id' => $user], '*', MUST_EXIST);
+ }
+
+ if (!$user) {
+ $user = $USER;
+ }
+
+ $this->user = $user;
+ }
+
+ /**
+ * Returns the user picture
+ *
+ * @param int $imgsize
+ *
+ * @return \moodle_url
+ * @throws \coding_exception
+ */
+ public function get_user_picture($imgsize = 100) {
+ global $PAGE;
+
+ $userimg = new user_picture($this->user);
+
+ $userimg->size = $imgsize;
+
+ return $userimg->get_url($PAGE)->out();
+ }
+}
diff --git a/lang/en/theme_boost_union.php b/lang/en/theme_boost_union.php
index 200640f7bd7..9b89c24eeb8 100644
--- a/lang/en/theme_boost_union.php
+++ b/lang/en/theme_boost_union.php
@@ -238,6 +238,12 @@
// ... ... Setting: Show course completion progress.
$string['courseoverviewshowprogresssetting'] = 'Show course completion progress';
$string['courseoverviewshowprogresssetting_desc'] = 'With this setting, you can control whether the course completion progress is visible inside the course overview block or not.';
+// ... ... Setting: Enable cards on course index.
+$string['enablecards'] = 'Enable cards on course index';
+$string['enablecardsdesc'] = 'Display courses as cards on site and course index.';
+// ... ... Setting: Disable teachers from cards.
+$string['enableteacherspic'] = 'Enable teachers pictures';
+$string['enableteacherspicdesc'] = 'This setting controls were to hide or display the teachers\' pictures in the course cards.';
// Settings: Course tab.
$string['coursetab'] = 'Course';
diff --git a/scss/boost_union/post.scss b/scss/boost_union/post.scss
index f1f37eb5bdc..e3c7d3da8db 100644
--- a/scss/boost_union/post.scss
+++ b/scss/boost_union/post.scss
@@ -1,3 +1,101 @@
+@media (max-width: 991.98px) {
+ .pagelayout-mydashboard #block-region-content .dashboard-card-deck:not(.fixed-width-cards) .dashboard-card {
+ width: 100%;
+ margin: 1rem 0;
+ }
+}
+.pagelayout-frontpage .dashboard-card,
+.pagelayout-coursecategory .dashboard-card {
+ position: relative;
+ width: calc(33.33% - 2rem);
+ margin: 1rem;
+}
+.pagelayout-frontpage .dashboard-card.dimmed .card-img::after,
+.pagelayout-coursecategory .dashboard-card.dimmed .card-img::after {
+ content: "";
+ position: absolute;
+ background-color: rgba(0, 0, 0, 0.5);
+ top: 0;
+ right: 0;
+ left: 0;
+ bottom: 0;
+ border-top-left-radius: 0.5rem;
+ border-top-right-radius: 0.5rem;
+}
+.pagelayout-frontpage .dashboard-card .dashboard-card-footer,
+.pagelayout-coursecategory .dashboard-card .dashboard-card-footer {
+ border-radius: 0.5rem;
+}
+.pagelayout-frontpage .dashboard-card .card-body,
+.pagelayout-coursecategory .dashboard-card .card-body {
+ padding: 0.8rem;
+}
+.pagelayout-frontpage .dashboard-card .customfields .customfield,
+.pagelayout-coursecategory .dashboard-card .customfields .customfield {
+ font-size: 80%;
+ margin-bottom: 0.25rem;
+}
+.pagelayout-frontpage .dashboard-card .customfields .customfield:last-of-type,
+.pagelayout-coursecategory .dashboard-card .customfields .customfield:last-of-type {
+ margin-bottom: 0;
+}
+.pagelayout-frontpage .dashboard-card .customfields .customfield .customfieldname,
+.pagelayout-frontpage .dashboard-card .customfields .customfield .customfieldseparator,
+.pagelayout-coursecategory .dashboard-card .customfields .customfield .customfieldname,
+.pagelayout-coursecategory .dashboard-card .customfields .customfield .customfieldseparator {
+ font-weight: 500;
+}
+.pagelayout-frontpage .dashboard-card .enrolmenticons,
+.pagelayout-coursecategory .dashboard-card .enrolmenticons {
+ position: absolute;
+ bottom: 8px;
+ right: 0;
+}
+.pagelayout-frontpage .dashboard-card .enrolmenticons .enrolmenticon,
+.pagelayout-coursecategory .dashboard-card .enrolmenticons .enrolmenticon {
+ padding: 4px 8px;
+ background-color: #fff;
+ color: #0f47ad;
+ border-radius: 0.5rem;
+ margin: 0 4px;
+}
+.pagelayout-frontpage .dashboard-card .enrolmenticons .enrolmenticon .icon,
+.pagelayout-coursecategory .dashboard-card .enrolmenticons .enrolmenticon .icon {
+ margin-right: 0;
+}
+.pagelayout-frontpage .dashboard-card .course-contacts .contact .info,
+.pagelayout-coursecategory .dashboard-card .course-contacts .contact .info {
+ margin-left: 1em;
+}
+.pagelayout-frontpage .dashboard-card .course-contacts .contact,
+.pagelayout-coursecategory .dashboard-card .course-contacts .contact {
+ display: flex;
+}
+.pagelayout-frontpage .dashboard-card .course-contacts .contact:not(:first-of-type),
+.pagelayout-coursecategory .dashboard-card .course-contacts .contact:not(:first-of-type) {
+ margin-top: 0.4rem;
+}
+.pagelayout-frontpage .dashboard-card .course-contacts .contact img,
+.pagelayout-coursecategory .dashboard-card .course-contacts .contact img {
+ width: 36px;
+ height: 36px;
+ border: 1px solid #dee2e6;
+ margin-left: 1em;
+}
+.pagelayout-frontpage .dashboard-card .course-contacts .contact p.role,
+.pagelayout-coursecategory .dashboard-card .course-contacts .contact p.role {
+ color: #1d2125;
+ font-size: 80%;
+}
+.pagelayout-frontpage .dashboard-card .dashboard-card-footer,
+.pagelayout-coursecategory .dashboard-card .dashboard-card-footer {
+ padding: 0 0.8rem 0.8rem 0.8rem;
+}
+.pagelayout-frontpage .course_category_tree .category .categoryname.aabtn,
+.pagelayout-coursecategory .course_category_tree .category .categoryname.aabtn {
+ font-size: 1.5em;
+}
+
/*=======================================
* Settings: Look -> Site branding
======================================*/
@@ -2544,7 +2642,6 @@ body.dir-rtl {
}
}
-
/*=======================================
* Supporting third-party plugins
======================================*/
@@ -2635,3 +2732,4 @@ body.dir-rtl {
}
}
}
+
diff --git a/settings.php b/settings.php
index 787192d813d..055f6b028c4 100644
--- a/settings.php
+++ b/settings.php
@@ -725,6 +725,24 @@
$setting->set_updatedcallback('theme_reset_all_caches');
$tab->add($setting);
+ // Enable cards on course index.
+ $name = 'theme_boost_union/enablecards';
+ $title = get_string('enablecards', 'theme_boost_union');
+ $description = get_string('enablecardsdesc', 'theme_boost_union');
+ $default = 0;
+ $choices = [0 => get_string('no'), 1 => get_string('yes')];
+ $setting = new admin_setting_configselect($name, $title, $description, $default, $choices);
+ $tab->add($setting);
+
+ // Disable teachers from cards.
+ $name = 'theme_boost_union/enableteacherspic';
+ $title = get_string('enableteacherspic', 'theme_boost_union');
+ $description = get_string('enableteacherspicdesc', 'theme_boost_union');
+ $default = 0;
+ $choices = [0 => get_string('no'), 1 => get_string('yes')];
+ $setting = new admin_setting_configselect($name, $title, $description, $default, $choices);
+ $tab->add($setting);
+
// Add tab to settings page.
$page->add($tab);
diff --git a/templates/coursecard.mustache b/templates/coursecard.mustache
new file mode 100644
index 00000000000..406be6ef59d
--- /dev/null
+++ b/templates/coursecard.mustache
@@ -0,0 +1,57 @@
+