123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899 |
- /**
- * @file
- * Module page behaviors.
- */
- (function ($, Drupal, debounce) {
- /**
- * Filters the module list table by a text input search string.
- *
- * Additionally accounts for multiple tables being wrapped in "package" details
- * elements.
- *
- * Text search input: input.table-filter-text
- * Target table: input.table-filter-text[data-table]
- * Source text: .table-filter-text-source, .module-name, .module-description
- *
- * @type {Drupal~behavior}
- */
- Drupal.behaviors.tableFilterByText = {
- attach(context, settings) {
- const $input = $('input.table-filter-text').once('table-filter-text');
- const $table = $($input.attr('data-table'));
- let $rowsAndDetails;
- let $rows;
- let $details;
- let searching = false;
- function hidePackageDetails(index, element) {
- const $packDetails = $(element);
- const $visibleRows = $packDetails.find('tbody tr:visible');
- $packDetails.toggle($visibleRows.length > 0);
- }
- function filterModuleList(e) {
- const query = $(e.target).val();
- // Case insensitive expression to find query at the beginning of a word.
- const re = new RegExp(`\\b${query}`, 'i');
- function showModuleRow(index, row) {
- const $row = $(row);
- const $sources = $row.find('.table-filter-text-source, .module-name, .module-description');
- const textMatch = $sources.text().search(re) !== -1;
- $row.closest('tr').toggle(textMatch);
- }
- // Search over all rows and packages.
- $rowsAndDetails.show();
- // Filter if the length of the query is at least 2 characters.
- if (query.length >= 2) {
- searching = true;
- $rows.each(showModuleRow);
- // Note that we first open all <details> to be able to use ':visible'.
- // Mark the <details> elements that were closed before filtering, so
- // they can be reclosed when filtering is removed.
- $details.not('[open]').attr('data-drupal-system-state', 'forced-open');
- // Hide the package <details> if they don't have any visible rows.
- // Note that we first show() all <details> to be able to use ':visible'.
- $details.attr('open', true).each(hidePackageDetails);
- Drupal.announce(
- Drupal.t(
- '!modules modules are available in the modified list.',
- { '!modules': $rowsAndDetails.find('tbody tr:visible').length },
- ),
- );
- }
- else if (searching) {
- searching = false;
- $rowsAndDetails.show();
- // Return <details> elements that had been closed before filtering
- // to a closed state.
- $details.filter('[data-drupal-system-state="forced-open"]')
- .removeAttr('data-drupal-system-state')
- .attr('open', false);
- }
- }
- function preventEnterKey(event) {
- if (event.which === 13) {
- event.preventDefault();
- event.stopPropagation();
- }
- }
- if ($table.length) {
- $rowsAndDetails = $table.find('tr, details');
- $rows = $table.find('tbody tr');
- $details = $rowsAndDetails.filter('.package-listing');
- $input.on({
- keyup: debounce(filterModuleList, 200),
- keydown: preventEnterKey,
- });
- }
- },
- };
- }(jQuery, Drupal, Drupal.debounce));
|