{"version":3,"file":"coursecategory.min.js","sources":["https:\/\/www.learnifs.com\/theme\/remui\/amd\/src\/coursecategory.js"],"sourcesContent":["\/* eslint-disable no-console *\/\n\/* eslint-disable no-unused-vars *\/\n\/* eslint-disable jsdoc\/require-jsdoc *\/\n\/* eslint-disable no-unused-vars *\/\n\/* eslint-disable babel\/semi *\/\n\/* eslint-disable no-trailing-spaces*\/\n\n\/\/ Moodle is free software: you can redistribute it and\/or modify\n\/\/ it under the terms of the GNU General Public License as published by\n\/\/ the Free Software Foundation, either version 3 of the License, or\n\/\/ (at your option) any later version.\n\/\/\n\/\/ Moodle is distributed in the hope that it will be useful,\n\/\/ but WITHOUT ANY WARRANTY; without even the implied warranty of\n\/\/ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\n\/\/ GNU General Public License for more details.\n\/\/\n\/\/ You should have received a copy of the GNU General Public License\n\/\/ along with Moodle. If not, see .\n\n\/**\n * @module theme_remui\/coursecategory\n * @copyright (c) 2023 WisdmLabs (https:\/\/wisdmlabs.com\/)\n * @license http:\/\/www.gnu.org\/copyleft\/gpl.html GNU GPL v3 or later\n *\/\n\n\"use strict\";\n\ndefine([\n 'jquery',\n 'core\/ajax',\n 'core\/str',\n 'core\/templates',\n 'theme_remui\/jquery-toolbar',\n 'core\/notification',\n 'core_user\/repository',\n 'theme_remui\/bootstrap-select'\n], function($, Ajax, str, templates, toolbar, Notification,UserRepository) {\n\n \/\/ Globals.\n var filterobj;\n var langstrings;\n\n var categoryfilter = $('#categoryfilter');\n\n var sortfilter = $('select#sortfilter');\n\n var categorylink = '[category-filter-link]';\n\n var cardswrapperarea = $('.course-cards');\n var cardspagination = $('.cards-pagination');\n\n var pageheaderactions = '.page-header-actionss';\n \/\/ View templates.\n var gridtemplate = 'theme_remui\/course_card_grid';\n var listtemplate = 'theme_remui\/course_card_list';\n var summarytemplate = 'theme_remui\/course_card_summary';\n\n var searchfilter = $('.layout-1 .filters-wrapper .simplesearchform');\n\n var mycoursescheckbox = $('.custom-switch');\n\n var coursecounter = $('.course-counter span.course-number');\n\n var tagswrapper = $('.tag-wrapper');\n\n var togglebtn = $(\".togglebtn\");\n \/**\n * Main category filters class.\n * @param {Integer} defaultCategory Default category to select.\n * @return {Object} Filter object\n *\/\n var categoryFilters = function(defaultCategory) {\n\n var _pageobj = {courses: 0, mycourses: 0};\n var _obj = {\n \/\/ Category id.\n category: defaultCategory,\n \/\/ Sorting.\n sort: 'ASC',\n \/\/ Searching string.\n search: \"\",\n \/\/ If true, means mycourses tab is active.\n tab: false,\n \/\/ This object consist of page number that is currently active, has mycourses and all courses tab page number.\n page: _pageobj,\n \/\/ If True, regenerate the pagination on any action performed.\n pagination: true,\n \/\/ Initially it is null to detect initial change in view, String grid - view in grid format, String list - list format.\n \/\/ view: null,\n \/\/ This filterModified true will tell that we need to fetch the courses otherwise show old fetched data.\n isFilterModified: true\n };\n\n _obj.initAttributes = function() {\n _obj.category = defaultCategory;\n _obj.sort = 'ASC';\n _obj.search = '';\n _obj.tab = false;\n _obj.page = _pageobj;\n _obj.pagination = true;\n \/\/ _obj.view = null;\n _obj.isFilterModified = true;\n };\n\n _obj.initPagination = function() {\n _obj.page = {courses: 0, mycourses: 0};\n };\n return _obj;\n };\n\n \/**\n * Course content object to handle ajax.\n *\/\n var courseContent = new (function() {\n this.mycourses = false;\n this.courses = false;\n\n \/**\n * Check if course contents are loaded.\n * @returns {Boolean}\n *\/\n this.isLoaded = function() {\n let type = courseContent.getActive();\n return this[type];\n };\n\n \/**\n * Mark courses as loaded\n * @param {Boolean} state Loading state\n *\/\n this.loaded = function(state) {\n let type = courseContent.getActive();\n if (state === undefined) {\n state = true;\n }\n this[type] = state;\n };\n\n \/**\n * Reset courses.\n *\/\n this.reset = function() {\n this.mycourses = false;\n this.courses = false;\n };\n\n \/**\n * Get currently active tab\n * @returns {String}\n *\/\n this.getActive = function() {\n return filterobj.tab === true ? 'mycourses' : 'courses';\n };\n })();\n\n \/**\n * Filters Generation\n * @param {Object} filterdata Filter data\n *\/\n function generateFilters(filterdata) {\n $(\".selectpicker\").each(function() {\n $(this).selectpicker();\n });\n\n if (filterdata.category !== \"\") {\n\n\n if($.isNumeric(filterdata.category)){\n var targetElement = $('.categoryfiltermenu .dropdown-menu a[data-cat-id=\"' + filterdata.category + '\"]');\n $('.categoryfiltermenu .categoryfilter span').text(targetElement.text());\n }\n \/\/ $(\"#categoryfilter.selectpicker\").selectpicker('val', filterdata.category);\n }\n\n if (filterdata.tab == true) {\n $('#switch-label1, #switch-label2').prop('checked', true);\n } else {\n $('#switch-label1, #switch-label2').prop('checked', false);\n }\n if (filterdata.sort !== null && filterdata.sort != undefined && typeof filterdata.sort != \"function\") {\n $(\"#sortfilter.selectpicker\").selectpicker('val', filterdata.sort);\n }\n\n if (filterdata.search !== \"\") {\n $(searchfilter).find('input[type=\"text\"]').val(filterdata.search);\n }\n\n \/\/ Put animation over here.\n \/\/ $(\".filters-wrapper\").removeClass('d-none');\n }\n\n \/**\n * Update page content\n *\/\n function updatePage() {\n \/\/ Destroy the cards from page.\n destroyCourseCards();\n \/\/ Create courses cards again.\n generateCourseCards();\n }\n\n \/*\n * Populate the tags section.\n *\/\n function populate_tags() {\n var serviceName = 'theme_remui_get_tags';\n var getcourses = Ajax.call([{\n methodname: serviceName,\n args: {\n data: JSON.stringify(filterobj)\n }\n }]);\n getcourses[0].done(function(response) {\n tagswrapper.empty().append(response);\n generateHorizontalScroller($(\".tag_list\"));\n\n }).fail(Notification.exception);\n }\n\n const generateHorizontalScroller = (element) => {\n const _leftscroll = \".left-scroll\";\n const _rightscroll = \".right-scroll\";\n\n if (element.length == 0) {\n return;\n }\n\n const _main = element[0];\n\n if (_main.scrollWidth > _main.clientWidth) {\n\n $(_leftscroll + \",\" + _rightscroll).removeClass(\"d-none\");\n\n const updateButtons = () => {\n\n $(_leftscroll).disabled = false;\n $(_rightscroll).disabled = false;\n\n if (_main.scrollLeft == 0) {\n $(_leftscroll).disabled = true;\n }\n\n var _scrollWidth = _main.scrollWidth;\n var _clientWidth = _main.clientWidth;\n var _scrollLeft = _main.scrollLeft;\n var _rightPos = _scrollWidth - _clientWidth;\n \/\/ Here we detect if element if fully scrolled to right.\n if (_scrollLeft == _rightPos) {\n $(_leftscroll).disabled = true;\n }\n };\n\n const moveRight = (_ele, scrollStep = 30) => {\n _ele.scrollLeft += scrollStep;\n updateButtons();\n };\n\n const moveLeft = (_ele, scrollStep = 30) => {\n _ele.scrollLeft -= scrollStep;\n updateButtons();\n };\n\n var _lInterval;\n var _rInterval;\n\n $(_leftscroll).on(\"mouseover\", function() {\n _lInterval = setInterval(function() {\n moveLeft(_main, 1);\n }, 1);\n\n });\n\n $(_rightscroll).on(\"mouseover\", function() {\n _rInterval = setInterval(function() {\n moveRight(_main, 1);\n }, 1);\n });\n\n $(_leftscroll).on(\"mouseout\", function() {\n clearInterval(_lInterval);\n});\n $(_rightscroll).on(\"mouseout\", function() {\n clearInterval(_rInterval);\n});\n }\n };\n\n \/**\n * Course cards initialization function.\n *\/\n function generateCourseCards() {\n \/\/ Check if Filters are modified and need to fetch the courses.\n if (!filterobj.isFilterModified) {\n return;\n }\n toggleContent(true);\n\n \/\/ Fetch the courses.\n getCourses();\n\n\n }\n\n \/**\n * Hide the content.\n * @param {Boolean} hidecontent\n *\/\n function toggleContent(hidecontent) {\n if (hidecontent) {\n $(\"#course-archive-main-container .tag-wrapper\").addClass(\"d-none\");\n $(\"#course-archive-main-container .course-cards\").addClass(\"d-none\");\n $(\"#course-archive-main-container .cards-pagination\").addClass(\"d-none\");\n $(\"#course-archive-main-container .loader-overlay\").addClass(\"d-flex\").removeClass(\"d-none\");\n } else {\n $(\"#course-archive-main-container .tag-wrapper\").removeClass(\"d-none\");\n $(\"#course-archive-main-container .course-cards\").removeClass(\"d-none\");\n $(\"#course-archive-main-container .cards-pagination\").removeClass(\"d-none\");\n $(\"#course-archive-main-container .loader-overlay\").addClass(\"d-none\").removeClass(\"d-flex\");\n }\n }\n \/**\n * Destroy courses cards\n *\/\n function destroyCourseCards() {\n \/\/ Find active tab to append the course cards.\n \/\/ var destroytab = (filterobj.tab) ? mycoursesregion : coursesregion;\n \/\/ Empty the courses region.\n $(cardswrapperarea).empty();\n\n \/\/ Destroy the pagination also.\n if (filterobj.pagination) {\n \/\/ Var destroypagination = (filterobj.tab) ? mycoursespagination : coursespagination;\n \/\/ $(destroypagination).empty();\n cardspagination.empty();\n }\n }\n\n \/**\n * Ajax to fetch the course and also append those courses to the page.\n * If pagination is enabled it will also generate new pagination.\n *\/\n function getCourses() {\n $('.courses-tabs .courses-loader-wrap').show();\n \/\/ Find active tab to append the course cards.\n \/\/ var appendtab = (filterobj.tab) ? mycoursesregion : coursesregion;\n \/\/ var appendpagination = (filterobj.tab) ? mycoursespagination : coursespagination;\n var serviceName = 'theme_remui_get_courses';\n var getcourses = Ajax.call([{\n methodname: serviceName,\n args: {\n data: JSON.stringify(filterobj)\n }\n }]);\n getcourses[0].done(function(response) {\n response = JSON.parse(response);\n \/\/ $(\"#page-header\").load(location.href + \" #page-header\");\n \/\/ $(\".filters-wrapper .navitem\").load(location.href + \" .filters-wrapper .navitem\");\n \/\/ Empty the action button on top header, and add new ones.\n $(coursecounter).text(response.totalcoursescount);\n $(pageheaderactions).empty();\n if (response.hasmanagebutton == true) {\n $(pageheaderactions).append(response.managebuttons);\n }\n\n \/\/ Show category management dropdown button when user has 'moodle\/category:manage' capability.\n if (response.dropdown != undefined) {\n $('#page-header .page-header-actionss').append(response.dropdown);\n } else {\n $('#page-header .page-header-actionss [data-enhance=\"moodle-core-actionmenu\"]').remove();\n }\n\n \/\/ Get the view.\n var viewobj = (typeof filterobj.view === \"undefined\") ? response.view : filterobj.view;\n \/\/ Var viewobj = 'grid'; \/\/ View is always in grid form.\n \/\/ Select the template to render according to view.\n \/\/ Var rendertemplate = (viewobj == 'grid' || response.latest_card) ? gridtemplate : listtemplate;\n\n \/\/ It will handle the view buttons synchronization with myoverview settings\n var filterbuttonsarray = $('.filters-wrapper .view-buttons .btn');\n var prefveiwbuttonavailable = filterbuttonsarray.filter(function() {\n return $(this).attr('data-view') == viewobj;\n }).length > 0;\n if (!prefveiwbuttonavailable) {\n if (filterbuttonsarray.length >= 1) {\n viewobj = $(filterbuttonsarray[0]).attr('data-view');\n }\n }\n var rendertemplate = gridtemplate;\n\n if (viewobj == 'grid' || response.latest_card) {\n rendertemplate = gridtemplate;\n }\n if (viewobj == 'list' || response.latest_card) {\n rendertemplate = listtemplate;\n }\n if (viewobj == 'summary' || response.latest_card) {\n rendertemplate = summarytemplate;\n }\n\n \/\/ Always render grid teplate on mobile screen and when latest cards setting is on.\n if (window.screen.width <= 480 || response.latest_card) {\n rendertemplate = gridtemplate;\n viewobj = 'grid';\n }\n\n if (prefveiwbuttonavailable) {\n \/\/ Update the view.\n updateView(viewobj);\n }else{\n \/\/ Update the card container.\n updateCardContainer(viewobj);\n }\n updateCards(response.latest_card);\n\n var courses = response.courses;\n if (courses.length > 0) {\n for (var i = 0; i < courses.length; i++) {\n \/\/ This will call the function to load and render our template.\n templates.render(rendertemplate, courses[i])\n \/\/ It returns a promise that needs to be resoved.\n \/* eslint no-loop-func: 0 *\/\n .then(function(html, js) {\n \/\/ Here eventually I have my compiled template, and any javascript that it generated.\n \/\/ The templates object has append, prepend and replace functions.\n \/\/ templates.appendNodeContents(appendtab, html, js);\n templates.appendNodeContents(cardswrapperarea, html, js);\n\n \/\/ Show options button on course card.\n \/\/ check if not mycourse tab.\n \/\/ This is very bad code, couldn't do it another way.\n \/\/ it get called each time a single card is added to dom, try to improve it.\n if (!filterobj.tab && !response.latest_card) {\n \/* eslint promise\/always-return: 0 *\/\n $('.showoptions').each(function() {\n $(this).toolbar({\n content: $(this).data('toolbar'),\n style: 'primary'\n });\n });\n }\n }).fail(Notification.exception);\n }\n\n } else {\n var htmldata = '
';\n htmldata += '