Файловый менеджер - Редактировать - /home/harasnat/www/learning/group/amd/src/comboboxsearch/group.js
Назад
// This file is part of Moodle - http://moodle.org/ // // Moodle is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // Moodle is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with Moodle. If not, see <http://www.gnu.org/licenses/>. /** * Allow the user to search for groups. * * @module core_group/comboboxsearch/group * @copyright 2023 Mathew May <mathew.solutions> * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ import search_combobox from 'core/comboboxsearch/search_combobox'; import {groupFetch} from 'core_group/comboboxsearch/repository'; import {renderForPromise, replaceNodeContents} from 'core/templates'; import {debounce} from 'core/utils'; import Notification from 'core/notification'; export default class GroupSearch extends search_combobox { courseID; bannedFilterFields = ['id', 'link', 'groupimageurl']; constructor() { super(); this.selectors = {...this.selectors, courseid: '[data-region="courseid"]', placeholder: '.groupsearchdropdown [data-region="searchplaceholder"]', }; const component = document.querySelector(this.componentSelector()); this.courseID = component.querySelector(this.selectors.courseid).dataset.courseid; this.renderDefault().catch(Notification.exception); } static init() { return new GroupSearch(); } /** * The overall div that contains the searching widget. * * @returns {string} */ componentSelector() { return '.group-search'; } /** * The dropdown div that contains the searching widget result space. * * @returns {string} */ dropdownSelector() { return '.groupsearchdropdown'; } /** * The triggering div that contains the searching widget. * * @returns {string} */ triggerSelector() { return '.groupsearchwidget'; } /** * Build the content then replace the node. */ async renderDropdown() { const {html, js} = await renderForPromise('core_group/comboboxsearch/resultset', { groups: this.getMatchedResults(), hasresults: this.getMatchedResults().length > 0, searchterm: this.getSearchTerm(), }); replaceNodeContents(this.selectors.placeholder, html, js); } /** * Build the content then replace the node by default we want our form to exist. */ async renderDefault() { this.setMatchedResults(await this.filterDataset(await this.getDataset())); this.filterMatchDataset(); await this.renderDropdown(); this.updateNodes(); this.registerInputEvents(); // Add a small BS listener so that we can set the focus correctly on open. this.$component.on('shown.bs.dropdown', () => { this.searchInput.focus({preventScroll: true}); }); } /** * Get the data we will be searching against in this component. * * @returns {Promise<*>} */ async fetchDataset() { return await groupFetch(this.courseID).then((r) => r.groups); } /** * Dictate to the search component how and what we want to match upon. * * @param {Array} filterableData * @returns {Array} The users that match the given criteria. */ async filterDataset(filterableData) { // Sometimes we just want to show everything. if (this.getPreppedSearchTerm() === '') { return filterableData; } return filterableData.filter((group) => Object.keys(group).some((key) => { if (group[key] === "" || this.bannedFilterFields.includes(key)) { return false; } return group[key].toString().toLowerCase().includes(this.getPreppedSearchTerm()); })); } /** * Given we have a subset of the dataset, set the field that we matched upon to inform the end user. */ filterMatchDataset() { this.setMatchedResults( this.getMatchedResults().map((group) => { return { id: group.id, name: group.name, link: this.selectOneLink(group.id), groupimageurl: group.groupimageurl, }; }) ); } /** * Handle any keyboard inputs. */ registerInputEvents() { // Register & handle the text input. this.searchInput.addEventListener('input', debounce(async() => { this.setSearchTerms(this.searchInput.value); // We can also require a set amount of input before search. if (this.searchInput.value === '') { // Hide the "clear" search button in the search bar. this.clearSearchButton.classList.add('d-none'); } else { // Display the "clear" search button in the search bar. this.clearSearchButton.classList.remove('d-none'); } // User has given something for us to filter against. await this.filterrenderpipe(); }, 300)); } /** * The handler for when a user interacts with the component. * * @param {MouseEvent} e The triggering event that we are working with. */ async clickHandler(e) { if (e.target.closest(this.selectors.dropdown)) { // Forcibly prevent BS events so that we can control the open and close. // Really needed because by default input elements cant trigger a dropdown. e.stopImmediatePropagation(); } this.clearSearchButton.addEventListener('click', async() => { this.searchInput.value = ''; this.setSearchTerms(this.searchInput.value); await this.filterrenderpipe(); }); // Prevent normal key presses activating this. if (e.target.closest('.dropdown-item') && e.button === 0) { window.location = e.target.closest('.dropdown-item').href; } } /** * The handler for when a user presses a key within the component. * * @param {KeyboardEvent} e The triggering event that we are working with. */ keyHandler(e) { super.keyHandler(e); // Switch the key presses to handle keyboard nav. switch (e.key) { case 'Tab': if (e.target.closest(this.selectors.input)) { e.preventDefault(); this.clearSearchButton.focus({preventScroll: true}); } break; case 'Escape': if (document.activeElement.getAttribute('role') === 'option') { e.stopPropagation(); this.searchInput.focus({preventScroll: true}); } else if (e.target.closest(this.selectors.input)) { const trigger = this.component.querySelector(this.selectors.trigger); trigger.focus({preventScroll: true}); } break; } } /** * Override the input event listener for the text input area. */ registerInputHandlers() { // Register & handle the text input. this.searchInput.addEventListener('input', debounce(() => { this.setSearchTerms(this.searchInput.value); // We can also require a set amount of input before search. if (this.getSearchTerm() === '') { // Hide the "clear" search button in the search bar. this.clearSearchButton.classList.add('d-none'); } else { // Display the "clear" search button in the search bar. this.clearSearchButton.classList.remove('d-none'); } }, 300)); } /** * Build up the view all link that is dedicated to a particular result. * * @param {Number} groupID The ID of the group selected. */ selectOneLink(groupID) { throw new Error(`selectOneLink(${groupID}) must be implemented in ${this.constructor.name}`); } }
| ver. 1.4 |
Github
|
.
| PHP 8.1.33 | Генерация страницы: 0.02 |
proxy
|
phpinfo
|
Настройка