UNPKG

uikit

Version:

UIkit is a lightweight and modular front-end framework for developing fast and powerful web interfaces.

157 lines (105 loc) 4.01 kB
import {removeAttr} from './attr'; import {isDocument, isNode, isString, startsWith, toNode, toNodes} from './lang'; export function query(selector, context) { return toNode(selector) || find(selector, getContext(selector, context)); } export function queryAll(selector, context) { const nodes = toNodes(selector); return nodes.length && nodes || findAll(selector, getContext(selector, context)); } function getContext(selector, context = document) { return isContextSelector(selector) || isDocument(context) ? context : context.ownerDocument; } export function find(selector, context) { return toNode(_query(selector, context, 'querySelector')); } export function findAll(selector, context) { return toNodes(_query(selector, context, 'querySelectorAll')); } function _query(selector, context = document, queryFn) { if (!selector || !isString(selector)) { return null; } selector = selector.replace(contextSanitizeRe, '$1 *'); let removes; if (isContextSelector(selector)) { removes = []; selector = splitSelector(selector).map((selector, i) => { let ctx = context; if (selector[0] === '!') { const selectors = selector.substr(1).trim().split(' '); ctx = closest(context.parentNode, selectors[0]); selector = selectors.slice(1).join(' ').trim(); } if (selector[0] === '-') { const selectors = selector.substr(1).trim().split(' '); const prev = (ctx || context).previousElementSibling; ctx = matches(prev, selector.substr(1)) ? prev : null; selector = selectors.slice(1).join(' '); } if (!ctx) { return null; } if (!ctx.id) { ctx.id = `uk-${Date.now()}${i}`; removes.push(() => removeAttr(ctx, 'id')); } return `#${escape(ctx.id)} ${selector}`; }).filter(Boolean).join(','); context = document; } try { return context[queryFn](selector); } catch (e) { return null; } finally { removes && removes.forEach(remove => remove()); } } const contextSelectorRe = /(^|[^\\],)\s*[!>+~-]/; const contextSanitizeRe = /([!>+~-])(?=\s+[!>+~-]|\s*$)/g; function isContextSelector(selector) { return isString(selector) && selector.match(contextSelectorRe); } const selectorRe = /.*?[^\\](?:,|$)/g; function splitSelector(selector) { return selector.match(selectorRe).map(selector => selector.replace(/,$/, '').trim()); } const elProto = Element.prototype; const matchesFn = elProto.matches || elProto.webkitMatchesSelector || elProto.msMatchesSelector; export function matches(element, selector) { return toNodes(element).some(element => matchesFn.call(element, selector)); } const closestFn = elProto.closest || function (selector) { let ancestor = this; do { if (matches(ancestor, selector)) { return ancestor; } ancestor = ancestor.parentNode; } while (ancestor && ancestor.nodeType === 1); }; export function closest(element, selector) { if (startsWith(selector, '>')) { selector = selector.slice(1); } return isNode(element) ? closestFn.call(element, selector) : toNodes(element).map(element => closest(element, selector)).filter(Boolean); } export function parents(element, selector) { const elements = []; element = toNode(element); while ((element = element.parentNode) && element.nodeType === 1) { if (matches(element, selector)) { elements.push(element); } } return elements; } const escapeFn = window.CSS && CSS.escape || function (css) { return css.replace(/([^\x7f-\uFFFF\w-])/g, match => `\\${match}`); }; export function escape(css) { return isString(css) ? escapeFn.call(null, css) : ''; }