0
0
mirror of https://github.com/twbs/bootstrap.git synced 2024-12-12 00:08:59 +01:00
Bootstrap/js/dist/dom/selector-engine.js

104 lines
4.1 KiB
JavaScript
Raw Normal View History

2019-03-01 17:31:34 +01:00
/*!
* Bootstrap selector-engine.js v5.3.0-alpha1 (https://getbootstrap.com/)
2022-05-13 08:07:23 +02:00
* Copyright 2011-2022 The Bootstrap Authors (https://github.com/twbs/bootstrap/graphs/contributors)
2020-06-16 20:50:01 +02:00
* Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)
2019-03-01 17:31:34 +01:00
*/
(function (global, factory) {
typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory(require('../util/index.js')) :
2022-05-13 08:07:23 +02:00
typeof define === 'function' && define.amd ? define(['../util/index'], factory) :
(global = typeof globalThis !== 'undefined' ? globalThis : global || self, global.SelectorEngine = factory(global.Index));
})(this, (function (index_js) { 'use strict';
2019-03-01 17:31:34 +01:00
/**
* --------------------------------------------------------------------------
* Bootstrap (v5.3.0-alpha1): dom/selector-engine.js
2020-06-16 20:50:01 +02:00
* Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)
2019-03-01 17:31:34 +01:00
* --------------------------------------------------------------------------
*/
const getSelector = element => {
let selector = element.getAttribute('data-bs-target');
if (!selector || selector === '#') {
let hrefAttribute = element.getAttribute('href');
// The only valid content that could double as a selector are IDs or classes,
// so everything starting with `#` or `.`. If a "real" URL is used as the selector,
// `document.querySelector` will rightfully complain it is invalid.
// See https://github.com/twbs/bootstrap/issues/32273
if (!hrefAttribute || !hrefAttribute.includes('#') && !hrefAttribute.startsWith('.')) {
return null;
}
2022-05-13 08:07:23 +02:00
// Just in case some CMS puts out a full URL with the anchor appended
if (hrefAttribute.includes('#') && !hrefAttribute.startsWith('#')) {
hrefAttribute = `#${hrefAttribute.split('#')[1]}`;
}
selector = hrefAttribute && hrefAttribute !== '#' ? hrefAttribute.trim() : null;
}
return index_js.parseSelector(selector);
};
2021-03-23 17:26:54 +01:00
const SelectorEngine = {
find(selector, element = document.documentElement) {
return [].concat(...Element.prototype.querySelectorAll.call(element, selector));
2019-03-01 17:31:34 +01:00
},
2021-03-23 17:26:54 +01:00
findOne(selector, element = document.documentElement) {
2020-11-11 18:07:37 +01:00
return Element.prototype.querySelector.call(element, selector);
2019-03-01 17:31:34 +01:00
},
2021-03-23 17:26:54 +01:00
children(element, selector) {
return [].concat(...element.children).filter(child => child.matches(selector));
2019-03-01 17:31:34 +01:00
},
2021-03-23 17:26:54 +01:00
parents(element, selector) {
const parents = [];
2022-05-13 08:07:23 +02:00
let ancestor = element.parentNode.closest(selector);
while (ancestor) {
parents.push(ancestor);
ancestor = ancestor.parentNode.closest(selector);
2019-03-01 17:31:34 +01:00
}
return parents;
},
2021-03-23 17:26:54 +01:00
prev(element, selector) {
let previous = element.previousElementSibling;
2020-03-28 11:29:08 +01:00
while (previous) {
if (previous.matches(selector)) {
return [previous];
}
previous = previous.previousElementSibling;
}
return [];
},
2022-05-13 08:07:23 +02:00
// TODO: this is now unused; remove later along with prev()
2021-03-23 17:26:54 +01:00
next(element, selector) {
let next = element.nextElementSibling;
2020-03-28 11:29:08 +01:00
while (next) {
if (next.matches(selector)) {
2020-03-28 11:29:08 +01:00
return [next];
2019-03-01 17:31:34 +01:00
}
2020-03-28 11:29:08 +01:00
next = next.nextElementSibling;
2019-03-01 17:31:34 +01:00
}
2020-03-28 11:29:08 +01:00
return [];
2021-08-04 17:41:51 +02:00
},
focusableChildren(element) {
2022-05-13 08:07:23 +02:00
const focusables = ['a', 'button', 'input', 'textarea', 'select', 'details', '[tabindex]', '[contenteditable="true"]'].map(selector => `${selector}:not([tabindex^="-"])`).join(',');
return this.find(focusables, element).filter(el => !index_js.isDisabled(el) && index_js.isVisible(el));
},
getSelectorFromElement(element) {
const selector = getSelector(element);
if (selector) {
return SelectorEngine.findOne(selector) ? selector : null;
}
return null;
},
getElementFromSelector(element) {
const selector = getSelector(element);
return selector ? SelectorEngine.findOne(selector) : null;
},
getMultipleElementsFromSelector(element) {
const selector = getSelector(element);
return selector ? SelectorEngine.find(selector) : [];
2019-03-01 17:31:34 +01:00
}
};
return SelectorEngine;
2021-10-05 17:50:18 +02:00
}));
//# sourceMappingURL=selector-engine.js.map