diff options
Diffstat (limited to 'school/node_modules/nwsapi')
-rw-r--r-- | school/node_modules/nwsapi/LICENSE | 22 | ||||
-rw-r--r-- | school/node_modules/nwsapi/README.md | 132 | ||||
-rw-r--r-- | school/node_modules/nwsapi/dist/lint.log | 0 | ||||
-rw-r--r-- | school/node_modules/nwsapi/package.json | 43 | ||||
-rw-r--r-- | school/node_modules/nwsapi/src/modules/nwsapi-jquery.js | 135 | ||||
-rw-r--r-- | school/node_modules/nwsapi/src/modules/nwsapi-traversal.js | 90 | ||||
-rw-r--r-- | school/node_modules/nwsapi/src/nwsapi.js | 1807 |
7 files changed, 2229 insertions, 0 deletions
diff --git a/school/node_modules/nwsapi/LICENSE b/school/node_modules/nwsapi/LICENSE new file mode 100644 index 0000000..cc3621a --- /dev/null +++ b/school/node_modules/nwsapi/LICENSE @@ -0,0 +1,22 @@ +Copyright (c) 2007-2019 Diego Perini (http://www.iport.it/) + +Permission is hereby granted, free of charge, to any person +obtaining a copy of this software and associated documentation +files (the "Software"), to deal in the Software without +restriction, including without limitation the rights to use, +copy, modify, merge, publish, distribute, sublicense, and/or +sell copies of the Software, and to permit persons to whom the +Software is furnished to do so, subject to the following +conditions: + +The above copyright notice and this permission notice shall be +included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES +OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND +NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT +HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, +WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR +OTHER DEALINGS IN THE SOFTWARE. diff --git a/school/node_modules/nwsapi/README.md b/school/node_modules/nwsapi/README.md new file mode 100644 index 0000000..f74b87d --- /dev/null +++ b/school/node_modules/nwsapi/README.md @@ -0,0 +1,132 @@ +# [NWSAPI](http://dperini.github.io/nwsapi/) + +Fast CSS Selectors API Engine + +![](https://img.shields.io/npm/v/nwsapi.svg?colorB=orange&style=flat) ![](https://img.shields.io/github/tag/dperini/nwsapi.svg?style=flat) ![](https://img.shields.io/npm/dw/nwsapi.svg?style=flat) ![](https://img.shields.io/github/issues/dperini/nwsapi.svg?style=flat) + +NWSAPI is the development progress of [NWMATCHER](https://github.com/dperini/nwmatcher) aiming at [Selectors Level 4](https://www.w3.org/TR/selectors-4/) conformance. It has been completely reworked to be easily extended and maintained. It is a right-to-left selector parser and compiler written in pure Javascript with no external dependencies. It was initially thought as a cross browser library to improve event delegation and web page scraping in various frameworks but it has become a popular replacement of the native CSS selection and matching functionality in newer browsers and headless environments. + +It uses [regular expressions](https://en.wikipedia.org/wiki/Regular_expression) to parse CSS selector strings and [metaprogramming](https://en.wikipedia.org/wiki/Metaprogramming) to transforms these selector strings into Javascript function resolvers. This process is executed only once for each selector string allowing memoization of the function resolvers and achieving unmatched performances. + +## Installation + +To include NWSAPI in a standard web page: + +```html +<script type="text/javascript" src="nwsapi.js"></script> +``` + +To include NWSAPI in a standard web page and automatically replace the native QSA: + +```html +<script type="text/javascript" src="nwsapi.js" onload="NW.Dom.install()"></script> +``` + +To use NWSAPI with Node.js: + +``` +$ npm install nwsapi +``` + +NWSAPI currently supports browsers (as a global, `NW.Dom`) and headless environments (as a CommonJS module). + + +## Supported Selectors + +Here is a list of all the CSS2/CSS3/CSS4 [Supported selectors](https://github.com/dperini/nwsapi/wiki/CSS-supported-selectors). + + +## Features and Compliance + +You can read more about NWSAPI [features and compliance](https://github.com/dperini/nwsapi/wiki/Features-and-compliance) on the wiki. + + +## API + +### DOM Selection + +#### `ancestor( selector, context, callback )` + +Returns a reference to the nearest ancestor element matching `selector`, starting at `context`. Returns `null` if no element is found. If `callback` is provided, it is invoked for the matched element. + +#### `first( selector, context, callback )` + +Returns a reference to the first element matching `selector`, starting at `context`. Returns `null` if no element matches. If `callback` is provided, it is invoked for the matched element. + +#### `match( selector, element, callback )` + +Returns `true` if `element` matches `selector`, starting at `context`; returns `false` otherwise. If `callback` is provided, it is invoked for the matched element. + +#### `select( selector, context, callback )` + +Returns an array of all the elements matching `selector`, starting at `context`; returns empty `Array` otherwise. If `callback` is provided, it is invoked for each matching element. + + +### DOM Helpers + +#### `byId( id, from )` + +Returns a reference to the first element with ID `id`, optionally filtered to descendants of the element `from`. + +#### `byTag( tag, from )` + +Returns an array of elements having the specified tag name `tag`, optionally filtered to descendants of the element `from`. + +#### `byClass( class, from )` + +Returns an array of elements having the specified class name `class`, optionally filtered to descendants of the element `from`. + + +### Engine Configuration + +#### `configure( options )` + +The following is the list of currently available configuration options, their default values and descriptions, they are boolean flags that can be set to `true` or `false`: + +* `IDS_DUPES`: true - true to allow using multiple elements having the same id, false to disallow +* `LIVECACHE`: true - true for caching both results and resolvers, false for caching only resolvers +* `MIXEDCASE`: true - true to match tag names case insensitive, false to match using case sensitive +* `LOGERRORS`: true - true to print errors and warnings to the console, false to mute both of them + + +### Examples on extending the basic functionalities + +#### `configure( { <configuration-flag>: [ true | false ] } )` + +Disable logging errors/warnings to console, disallow duplicate ids. Example: + +```js +NW.Dom.configure( { LOGERRORS: false, IDS_DUPES: false } ); +``` +NOTE: NW.Dom.configure() without parameters return the current configuration. + +#### `registerCombinator( symbol, resolver )` + +Registers a new symbol and its matching resolver in the combinators table. Example: + +```js +NW.Dom.registerCombinator( '^', 'e.parentElement' ); +``` + +#### `registerOperator( symbol, resolver )` + +Registers a new symbol and its matching resolver in the attribute operators table. Example: + +```js +NW.Dom.registerOperator( '!=', { p1: '^', p2: '$', p3: 'false' } ); +``` + +#### `registerSelector( name, rexp, func )` + +Registers a new selector, the matching RE and the resolver function, in the selectors table. Example: + +```js +NW.Dom.registerSelector('Controls', /^\:(control)(.*)/i, + (function(global) { + return function(match, source, mode, callback) { + var status = true; + source = 'if(/^(button|input|select|textarea)/i.test(e.nodeName)){' + source + '}'; + return { 'source': source, 'status': status }; + }; + })(this)); +``` diff --git a/school/node_modules/nwsapi/dist/lint.log b/school/node_modules/nwsapi/dist/lint.log new file mode 100644 index 0000000..e69de29 --- /dev/null +++ b/school/node_modules/nwsapi/dist/lint.log diff --git a/school/node_modules/nwsapi/package.json b/school/node_modules/nwsapi/package.json new file mode 100644 index 0000000..21f71b0 --- /dev/null +++ b/school/node_modules/nwsapi/package.json @@ -0,0 +1,43 @@ +{ + "name": "nwsapi", + "version": "2.2.2", + "description": "Fast CSS Selectors API Engine", + "homepage": "http://javascript.nwbox.com/nwsapi/", + "main": "./src/nwsapi", + "keywords": [ + "css", + "css3", + "css4", + "matcher", + "selector" + ], + "licenses": [ + { + "type": "MIT", + "url": "http://javascript.nwbox.com/nwsapi/MIT-LICENSE" + } + ], + "license": "MIT", + "author": { + "name": "Diego Perini", + "email": "diego.perini@gmail.com", + "web": "http://www.iport.it/" + }, + "maintainers": [ + { + "name": "Diego Perini", + "email": "diego.perini@gmail.com", + "web": "http://www.iport.it/" + } + ], + "bugs": { + "url": "http://github.com/dperini/nwsapi/issues" + }, + "repository": { + "type": "git", + "url": "git://github.com/dperini/nwsapi.git" + }, + "scripts": { + "lint": "eslint ./src/nwsapi.js" + } +} diff --git a/school/node_modules/nwsapi/src/modules/nwsapi-jquery.js b/school/node_modules/nwsapi/src/modules/nwsapi-jquery.js new file mode 100644 index 0000000..961cb30 --- /dev/null +++ b/school/node_modules/nwsapi/src/modules/nwsapi-jquery.js @@ -0,0 +1,135 @@ +/* + * Copyright (C) 2007-2017 Diego Perini + * All rights reserved. + * + * this is just a small example to show + * how an extension for NWMatcher could be + * adapted to handle special jQuery selectors + * + * Child Selectors + * :even, :odd, :eq, :lt, :gt, :first, :last, :nth + * + * Pseudo Selectors + * :has, :button, :header, :input, :checkbox, :radio, :file, :image + * :password, :reset, :submit, :text, :hidden, :visible, :parent + * + */ + +// for structural pseudo-classes extensions +NW.Dom.registerSelector( + 'jquery:child', + /^\:((?:(nth|eq|lt|gt)\(([^()]*)\))|(?:even|odd|first|last))(.*)/i, + (function(global) { + + return function(match, source, mode, callback) { + + var status = true, + macro = mode ? NW.Dom.S_BODY : NW.Dom.M_BODY; + + macro = macro.replace('@', typeof callback == 'function' ? (mode ? NW.Dom.S_TEST : NW.Dom.M_TEST) : ''); + + switch (match[1].toLowerCase()) { + case 'odd': + source = source.replace(macro, 'if((n=n^1)==0){' + macro + '}'); + break; + case 'even': + source = source.replace(macro, 'if((n=n^1)==1){' + macro + '}'); + break; + case 'first': + source = 'n=s.root.getElementsByTagName(e.nodeName);if(n.length&&n[0]===e){' + source + '}'; + break; + case 'last': + source = 'n=s.root.getElementsByTagName(e.nodeName);if(n.length&&n[n.length-1]===e){' + source + '}'; + break; + default: + switch (match[2].toLowerCase()) { + case 'nth': + source = 'n=s.root.getElementsByTagName(e.nodeName);if(n.length&&n[' + match[3] + ']===e){' + source + '}'; + break; + case 'eq': + source = source.replace(macro, 'if(x++==' + match[3] + '){' + macro + '}'); + break; + case 'lt': + source = source.replace(macro, 'if(x++<' + match[3] + '){' + macro + '}'); + break; + case 'gt': + source = source.replace(macro, 'if(x++>' + match[3] + '){' + macro + '}'); + break; + default: + status = false; + break; + } + break; + } + + // compiler will add this to "source" + return { + 'source': source, + 'status': status, + 'modvar': 'x=0' + }; + + }; + + })(this)); + +// for element pseudo-classes extensions +NW.Dom.registerSelector( + 'jquery:pseudo', + /^\:(has|checkbox|file|image|password|radio|reset|submit|text|button|input|header|hidden|visible|parent)(?:\(\s*(["']*)?([^'"()]*)\2\s*\))?(.*)/i, + (function(global) { + + return function(match, source, mode, callback) { + + var status = true, + macro = mode ? NW.Dom.S_BODY : NW.Dom.M_BODY; + + macro = macro.replace('@', typeof callback == 'function' ? (mode ? NW.Dom.S_TEST : NW.Dom.M_TEST) : ''); + + switch(match[1].toLowerCase()) { + case 'has': + source = source.replace(macro, 'if(e.getElementsByTagName("' + match[3].replace(/^\s|\s$/g, '') + '")[0]){' + macro + '}'); + break; + case 'checkbox': + case 'file': + case 'image': + case 'password': + case 'radio': + case 'reset': + case 'submit': + case 'text': + // :checkbox, :file, :image, :password, :radio, :reset, :submit, :text + source = 'if(/^' + match[1] + '$/i.test(e.type)){' + source + '}'; + break; + case 'button': + source = 'if(/^button$/i.test(e.nodeName)){' + source + '}'; + break; + case 'input': + source = 'if(/^(?:button|input|select|textarea)$/i.test(e.nodeName)){' + source + '}'; + break; + case 'header': + source = 'if(/^h[1-6]$/i.test(e.nodeName)){' + source + '}'; + break; + case 'hidden': + source = 'if(!e.offsetWidth&&!e.offsetHeight){' + source + '}'; + break; + case 'visible': + source = 'if(e.offsetWidth||e.offsetHeight){' + source + '}'; + break; + case 'parent': + source = 'if(e.firstChild){' + source + '}'; + break; + default: + status = false; + break; + } + + // compiler will add this to "source" + return { + 'source': source, + 'status': status + }; + + }; + + })(this)); diff --git a/school/node_modules/nwsapi/src/modules/nwsapi-traversal.js b/school/node_modules/nwsapi/src/modules/nwsapi-traversal.js new file mode 100644 index 0000000..9d2b62f --- /dev/null +++ b/school/node_modules/nwsapi/src/modules/nwsapi-traversal.js @@ -0,0 +1,90 @@ +/* + * Element Traversal methods from Juriy Zaytsev (kangax) + * used to emulate Prototype up/down/previous/next methods + */ + +(function(D){ + + // TODO: all of this needs tests + var match = D.match, select = D.select, root = document.documentElement, + + // Use the Element Traversal API if available. + nextElement = 'nextElementSibling', + previousElement = 'previousElementSibling', + parentElement = 'parentElement'; + + // Fall back to the DOM Level 1 API. + if (!(nextElement in root)) nextElement = 'nextSibling'; + if (!(previousElement in root)) previousElement = 'previousSibling'; + if (!(parentElement in root)) parentElement = 'parentNode'; + + function walkElements(property, element, expr) { + var i = 0, isIndex = typeof expr == 'number'; + if (typeof expr == 'undefined') { + isIndex = true; + expr = 0; + } + while ((element = element[property])) { + if (element.nodeType != 1) continue; + if (isIndex) { + ++i; + if (i == expr) return element; + } else if (match(element, expr)) { + return element; + } + } + return null; + } + + /** + * @method up + * @param {HTMLElement} element element to walk from + * @param {String | Number} expr CSS expression or an index + * @return {HTMLElement | null} + */ + function up(element, expr) { + return walkElements(parentElement, element, expr); + } + /** + * @method next + * @param {HTMLElement} element element to walk from + * @param {String | Number} expr CSS expression or an index + * @return {HTMLElement | null} + */ + function next(element, expr) { + return walkElements(nextElement, element, expr); + } + /** + * @method previous + * @param {HTMLElement} element element to walk from + * @param {String | Number} expr CSS expression or an index + * @return {HTMLElement | null} + */ + function previous(element, expr) { + return walkElements(previousElement, element, expr); + } + /** + * @method down + * @param {HTMLElement} element element to walk from + * @param {String | Number} expr CSS expression or an index + * @return {HTMLElement | null} + */ + function down(element, expr) { + var isIndex = typeof expr == 'number', descendants, index, descendant; + if (expr === null) { + element = element.firstChild; + while (element && element.nodeType != 1) element = element[nextElement]; + return element; + } + if (!isIndex && match(element, expr) || isIndex && expr === 0) return element; + descendants = select('*', element); + if (isIndex) return descendants[expr] || null; + index = 0; + while ((descendant = descendants[index]) && !match(descendant, expr)) { ++index; } + return descendant || null; + } + D.up = up; + D.down = down; + D.next = next; + D.previous = previous; +})(NW.Dom); diff --git a/school/node_modules/nwsapi/src/nwsapi.js b/school/node_modules/nwsapi/src/nwsapi.js new file mode 100644 index 0000000..d5ec57c --- /dev/null +++ b/school/node_modules/nwsapi/src/nwsapi.js @@ -0,0 +1,1807 @@ +/* + * Copyright (C) 2007-2019 Diego Perini + * All rights reserved. + * + * nwsapi.js - Fast CSS Selectors API Engine + * + * Author: Diego Perini <diego.perini at gmail com> + * Version: 2.2.0 + * Created: 20070722 + * Release: 20220901 + * + * License: + * http://javascript.nwbox.com/nwsapi/MIT-LICENSE + * Download: + * http://javascript.nwbox.com/nwsapi/nwsapi.js + */ + +(function Export(global, factory) { + + 'use strict'; + + if (typeof module == 'object' && typeof exports == 'object') { + module.exports = factory; + } else if (typeof define == 'function' && define['amd']) { + define(factory); + } else { + global.NW || (global.NW = { }); + global.NW.Dom = factory(global, Export); + } + +})(this, function Factory(global, Export) { + + var version = 'nwsapi-2.2.2', + + doc = global.document, + root = doc.documentElement, + slice = Array.prototype.slice, + + WSP = '[\\x20\\t\\r\\n\\f]', + + CFG = { + // extensions + operators: '[~*^$|]=|=', + combinators: '[\\x20\\t>+~](?=[^>+~])' + }, + + NOT = { + // not enclosed in double/single/parens/square + double_enc: '(?=(?:[^"]*["][^"]*["])*[^"]*$)', + single_enc: "(?=(?:[^']*['][^']*['])*[^']*$)", + parens_enc: '(?![^\\x28]*\\x29)', + square_enc: '(?![^\\x5b]*\\x5d)' + }, + + REX = { + // regular expressions + HasEscapes: RegExp('\\\\'), + HexNumbers: RegExp('^[0-9a-fA-F]'), + EscOrQuote: RegExp('^\\\\|[\\x22\\x27]'), + RegExpChar: RegExp('(?:(?!\\\\)[\\\\^$.*+?()[\\]{}|\\/])', 'g'), + TrimSpaces: RegExp('[\\r\\n\\f]|^' + WSP + '+|' + WSP + '+$', 'g'), + CommaGroup: RegExp('(\\s*,\\s*)' + NOT.square_enc + NOT.parens_enc, 'g'), + SplitGroup: RegExp('((?:\\x28[^\\x29]*\\x29|\\[[^\\]]*\\]|\\\\.|[^,])+)', 'g'), + FixEscapes: RegExp('\\\\([0-9a-fA-F]{1,6}' + WSP + '?|.)|([\\x22\\x27])', 'g'), + CombineWSP: RegExp('[\\n\\r\\f\\x20]+' + NOT.single_enc + NOT.double_enc, 'g'), + TabCharWSP: RegExp('(\\x20?\\t+\\x20?)' + NOT.single_enc + NOT.double_enc, 'g'), + PseudosWSP: RegExp('\\s+([-+])\\s+' + NOT.square_enc, 'g') + }, + + STD = { + combinator: RegExp('\\s?([>+~])\\s?', 'g'), + apimethods: RegExp('^(?:[a-z]+|\\*)\\|', 'i'), + namespaces: RegExp('(\\*|[a-z]+)\\|[-a-z]+', 'i') + }, + + GROUPS = { + // pseudo-classes requiring parameters + linguistic: '(dir|lang)\\x28\\s?([-\\w]{2,})\\s?(?:\\x29|$)', + logicalsel: '(is|where|matches|not)\\x28\\s?([^()]*|[^\\x28]*\\x28[^\\x29]*\\x29)\\s?(?:\\x29|$)', + treestruct: '(nth(?:-last)?(?:-child|-of-type))(?:\\x28\\s?(even|odd|(?:[-+]?\\d*)(?:n\\s?[-+]?\\s?\\d*)?)\\s?(?:\\x29|$))', + // pseudo-classes not requiring parameters + locationpc: '(any-link|link|visited|target)\\b', + useraction: '(hover|active|focus|focus-within)\\b', + structural: '(root|empty|(?:(?:first|last|only)(?:-child|-of-type)))\\b', + inputstate: '(enabled|disabled|read-only|read-write|placeholder-shown|default)\\b', + inputvalue: '(checked|indeterminate|required|optional|valid|invalid|in-range|out-of-range)\\b', + // pseudo-classes for parsing only selectors + pseudo_nop: '(autofill|-webkit-autofill)\\b', + // pseudo-elements starting with single colon (:) + pseudo_sng: '(after|before|first-letter|first-line)\\b', + // pseudo-elements starting with double colon (::) + pseudo_dbl: ':(after|before|first-letter|first-line|selection|placeholder|-webkit-[-a-zA-Z0-9]{2,})\\b' + }, + + Patterns = { + // pseudo-classes + treestruct: RegExp('^:(?:' + GROUPS.treestruct + ')(.*)', 'i'), + structural: RegExp('^:(?:' + GROUPS.structural + ')(.*)', 'i'), + linguistic: RegExp('^:(?:' + GROUPS.linguistic + ')(.*)', 'i'), + useraction: RegExp('^:(?:' + GROUPS.useraction + ')(.*)', 'i'), + inputstate: RegExp('^:(?:' + GROUPS.inputstate + ')(.*)', 'i'), + inputvalue: RegExp('^:(?:' + GROUPS.inputvalue + ')(.*)', 'i'), + locationpc: RegExp('^:(?:' + GROUPS.locationpc + ')(.*)', 'i'), + logicalsel: RegExp('^:(?:' + GROUPS.logicalsel + ')(.*)', 'i'), + pseudo_nop: RegExp('^:(?:' + GROUPS.pseudo_nop + ')(.*)', 'i'), + pseudo_sng: RegExp('^:(?:' + GROUPS.pseudo_sng + ')(.*)', 'i'), + pseudo_dbl: RegExp('^:(?:' + GROUPS.pseudo_dbl + ')(.*)', 'i'), + // combinator symbols + children: RegExp('^' + WSP + '?\\>' + WSP + '?(.*)'), + adjacent: RegExp('^' + WSP + '?\\+' + WSP + '?(.*)'), + relative: RegExp('^' + WSP + '?\\~' + WSP + '?(.*)'), + ancestor: RegExp('^' + WSP + '+(.*)'), + // universal & namespace + universal: RegExp('^\\*(.*)'), + namespace: RegExp('^(\\w+|\\*)?\\|(.*)') + }, + + // regexp to aproximate detection of RTL languages (Arabic) + RTL = RegExp('^[\\u0591-\\u08ff\\ufb1d-\\ufdfd\\ufe70-\\ufefc ]+$'), + + // emulate firefox error strings + qsNotArgs = 'Not enough arguments', + qsInvalid = ' is not a valid selector', + + // detect structural pseudo-classes in selectors + reNthElem = RegExp('(:nth(?:-last)?-child)', 'i'), + reNthType = RegExp('(:nth(?:-last)?-of-type)', 'i'), + + // placeholder for global regexp + reOptimizer, + reValidator, + + // special handling configuration flags + Config = { + IDS_DUPES: true, + MIXEDCASE: true, + LOGERRORS: true, + VERBOSITY: true + }, + + NAMESPACE, + QUIRKS_MODE, + HTML_DOCUMENT, + + ATTR_STD_OPS = { + '=': 1, '^=': 1, '$=': 1, '|=': 1, '*=': 1, '~=': 1 + }, + + HTML_TABLE = { + 'accept': 1, 'accept-charset': 1, 'align': 1, 'alink': 1, 'axis': 1, + 'bgcolor': 1, 'charset': 1, 'checked': 1, 'clear': 1, 'codetype': 1, 'color': 1, + 'compact': 1, 'declare': 1, 'defer': 1, 'dir': 1, 'direction': 1, 'disabled': 1, + 'enctype': 1, 'face': 1, 'frame': 1, 'hreflang': 1, 'http-equiv': 1, 'lang': 1, + 'language': 1, 'link': 1, 'media': 1, 'method': 1, 'multiple': 1, 'nohref': 1, + 'noresize': 1, 'noshade': 1, 'nowrap': 1, 'readonly': 1, 'rel': 1, 'rev': 1, + 'rules': 1, 'scope': 1, 'scrolling': 1, 'selected': 1, 'shape': 1, 'target': 1, + 'text': 1, 'type': 1, 'valign': 1, 'valuetype': 1, 'vlink': 1 + }, + + Combinators = { }, + + Selectors = { }, + + Operators = { + '=': { p1: '^', + p2: '$', + p3: 'true' }, + '^=': { p1: '^', + p2: '', + p3: 'true' }, + '$=': { p1: '', + p2: '$', + p3: 'true' }, + '*=': { p1: '', + p2: '', + p3: 'true' }, + '|=': { p1: '^', + p2: '(-|$)', + p3: 'true' }, + '~=': { p1: '(^|\\s)', + p2: '(\\s|$)', + p3: 'true' } + }, + + concatCall = + function(nodes, callback) { + var i = 0, l = nodes.length, list = Array(l); + while (l > i) { + if (false === callback(list[i] = nodes[i])) break; + ++i; + } + return list; + }, + + concatList = + function(list, nodes) { + var i = -1, l = nodes.length; + while (l--) { list[list.length] = nodes[++i]; } + return list; + }, + + documentOrder = + function(a, b) { + if (!hasDupes && a === b) { + hasDupes = true; + return 0; + } + return a.compareDocumentPosition(b) & 4 ? -1 : 1; + }, + + hasDupes = false, + + unique = + function(nodes) { + var i = 0, j = -1, l = nodes.length + 1, list = [ ]; + while (--l) { + if (nodes[i++] === nodes[i]) continue; + list[++j] = nodes[i - 1]; + } + hasDupes = false; + return list; + }, + + // check context for mixed content + hasMixedCaseTagNames = + function(context) { + var ns, api = 'getElementsByTagNameNS'; + + // current host context (ownerDocument) + context = context.ownerDocument || context; + + // documentElement (root) element namespace or default html/xhtml namespace + ns = context.documentElement.namespaceURI || 'http://www.w3.org/1999/xhtml'; + + // checking the number of non HTML nodes in the document + return (context[api]('*', '*').length - context[api](ns, '*').length) > 0; + }, + + switchContext = + function(context, force) { + var oldDoc = doc; + doc = context.ownerDocument || context; + if (force || oldDoc !== doc) { + // force a new check for each document change + // performed before the next select operation + root = doc.documentElement; + HTML_DOCUMENT = isHTML(doc); + QUIRKS_MODE = HTML_DOCUMENT && + doc.compatMode.indexOf('CSS') < 0; + NAMESPACE = root && root.namespaceURI; + Snapshot.doc = doc; + Snapshot.root = root; + } + return (Snapshot.from = context); + }, + + // convert single codepoint to UTF-16 encoding + codePointToUTF16 = + function(codePoint) { + // out of range, use replacement character + if (codePoint < 1 || codePoint > 0x10ffff || + (codePoint > 0xd7ff && codePoint < 0xe000)) { + return '\\ufffd'; + } + // javascript strings are UTF-16 encoded + if (codePoint < 0x10000) { + var lowHex = '000' + codePoint.toString(16); + return '\\u' + lowHex.substr(lowHex.length - 4); + } + // supplementary high + low surrogates + return '\\u' + (((codePoint - 0x10000) >> 0x0a) + 0xd800).toString(16) + + '\\u' + (((codePoint - 0x10000) % 0x400) + 0xdc00).toString(16); + }, + + // convert single codepoint to string + stringFromCodePoint = + function(codePoint) { + // out of range, use replacement character + if (codePoint < 1 || codePoint > 0x10ffff || + (codePoint > 0xd7ff && codePoint < 0xe000)) { + return '\ufffd'; + } + if (codePoint < 0x10000) { + return String.fromCharCode(codePoint); + } + return String.fromCodePoint ? + String.fromCodePoint(codePoint) : + String.fromCharCode( + ((codePoint - 0x10000) >> 0x0a) + 0xd800, + ((codePoint - 0x10000) % 0x400) + 0xdc00); + }, + + // convert escape sequence in a CSS string or identifier + // to javascript string with javascript escape sequences + convertEscapes = + function(str) { + return REX.HasEscapes.test(str) ? + str.replace(REX.FixEscapes, + function(substring, p1, p2) { + // unescaped " or ' + return p2 ? '\\' + p2 : + // javascript strings are UTF-16 encoded + REX.HexNumbers.test(p1) ? codePointToUTF16(parseInt(p1, 16)) : + // \' \" + REX.EscOrQuote.test(p1) ? substring : + // \g \h \. \# etc + p1; + } + ) : str; + }, + + // convert escape sequence in a CSS string or identifier + // to javascript string with characters representations + unescapeIdentifier = + function(str) { + return REX.HasEscapes.test(str) ? + str.replace(REX.FixEscapes, + function(substring, p1, p2) { + // unescaped " or ' + return p2 ? p2 : + // javascript strings are UTF-16 encoded + REX.HexNumbers.test(p1) ? stringFromCodePoint(parseInt(p1, 16)) : + // \' \" + REX.EscOrQuote.test(p1) ? substring : + // \g \h \. \# etc + p1; + } + ) : str; + }, + + method = { + '#': 'getElementById', + '*': 'getElementsByTagName', + '|': 'getElementsByTagNameNS', + '.': 'getElementsByClassName' + }, + + compat = { + '#': function(c, n) { REX.HasEscapes.test(n) && (n = unescapeIdentifier(n)); return function(e, f) { return byId(n, c); }; }, + '*': function(c, n) { REX.HasEscapes.test(n) && (n = unescapeIdentifier(n)); return function(e, f) { return byTag(n, c); }; }, + '|': function(c, n) { REX.HasEscapes.test(n) && (n = unescapeIdentifier(n)); return function(e, f) { return byTag(n, c); }; }, + '.': function(c, n) { REX.HasEscapes.test(n) && (n = unescapeIdentifier(n)); return function(e, f) { return byClass(n, c); }; } + }, + + // find duplicate ids using iterative walk + byIdRaw = + function(id, context) { + var node = context, nodes = [ ], next = node.firstElementChild; + while ((node = next)) { + node.id == id && (nodes[nodes.length] = node); + if ((next = node.firstElementChild || node.nextElementSibling)) continue; + while (!next && (node = node.parentElement) && node !== context) { + next = node.nextElementSibling; + } + } + return nodes; + }, + + // context agnostic getElementById + byId = + function(id, context) { + var e, nodes, api = method['#']; + + // duplicates id allowed + if (Config.IDS_DUPES === false) { + if (api in context) { + return (e = context[api](id)) ? [ e ] : none; + } + } else { + if ('all' in context) { + if ((e = context.all[id])) { + if (e.nodeType == 1) return e.getAttribute('id') != id ? [ ] : [ e ]; + else if (id == 'length') return (e = context[api](id)) ? [ e ] : none; + for (i = 0, l = e.length, nodes = [ ]; l > i; ++i) { + if (e[i].id == id) nodes[nodes.length] = e[i]; + } + return nodes && nodes.length ? nodes : [ nodes ]; + } else return none; + } + } + + return byIdRaw(id, context); + }, + + // context agnostic getElementsByTagName + byTag = + function(tag, context) { + var e, nodes, api = method['*']; + // DOCUMENT_NODE (9) & ELEMENT_NODE (1) + if (api in context) { + return slice.call(context[api](tag)); + } else { + tag = tag.toLowerCase(); + // DOCUMENT_FRAGMENT_NODE (11) + if ((e = context.firstElementChild)) { + if (!(e.nextElementSibling || tag == '*' || e.localName == tag)) { + return slice.call(e[api](tag)); + } else { + nodes = [ ]; + do { + if (tag == '*' || e.localName == tag) nodes[nodes.length] = e; + concatList(nodes, e[api](tag)); + } while ((e = e.nextElementSibling)); + } + } else nodes = none; + } + return nodes; + }, + + // context agnostic getElementsByClassName + byClass = + function(cls, context) { + var e, nodes, api = method['.'], reCls; + // DOCUMENT_NODE (9) & ELEMENT_NODE (1) + if (api in context) { + return slice.call(context[api](cls)); + } else { + // DOCUMENT_FRAGMENT_NODE (11) + if ((e = context.firstElementChild)) { + reCls = RegExp('(^|\\s)' + cls + '(\\s|$)', QUIRKS_MODE ? 'i' : ''); + if (!(e.nextElementSibling || reCls.test(e.className))) { + return slice.call(e[api](cls)); + } else { + nodes = [ ]; + do { + if (reCls.test(e.className)) nodes[nodes.length] = e; + concatList(nodes, e[api](cls)); + } while ((e = e.nextElementSibling)); + } + } else nodes = none; + } + return nodes; + }, + + // namespace aware hasAttribute + // helper for XML/XHTML documents + hasAttributeNS = + function(e, name) { + var i, l, attr = e.getAttributeNames(); + name = RegExp(':?' + name + '$', HTML_DOCUMENT ? 'i' : ''); + for (i = 0, l = attr.length; l > i; ++i) { + if (name.test(attr[i])) return true; + } + return false; + }, + + // fast resolver for the :nth-child() and :nth-last-child() pseudo-classes + nthElement = (function() { + var idx = 0, len = 0, set = 0, parent = undefined, parents = Array(), nodes = Array(); + return function(element, dir) { + // ensure caches are emptied after each run, invoking with dir = 2 + if (dir == 2) { + idx = 0; len = 0; set = 0; nodes.length = 0; + parents.length = 0; parent = undefined; + return -1; + } + var e, i, j, k, l; + if (parent === element.parentElement) { + i = set; j = idx; l = len; + } else { + l = parents.length; + parent = element.parentElement; + for (i = -1, j = 0, k = l - 1; l > j; ++j, --k) { + if (parents[j] === parent) { i = j; break; } + if (parents[k] === parent) { i = k; break; } + } + if (i < 0) { + parents[i = l] = parent; + l = 0; nodes[i] = Array(); + e = parent && parent.firstElementChild || element; + while (e) { nodes[i][l] = e; if (e === element) j = l; e = e.nextElementSibling; ++l; } + set = i; idx = 0; len = l; + if (l < 2) return l; + } else { + l = nodes[i].length; + set = i; + } + } + if (element !== nodes[i][j] && element !== nodes[i][j = 0]) { + for (j = 0, e = nodes[i], k = l - 1; l > j; ++j, --k) { + if (e[j] === element) { break; } + if (e[k] === element) { j = k; break; } + } + } + idx = j + 1; len = l; + return dir ? l - j : idx; + }; + })(), + + // fast resolver for the :nth-of-type() and :nth-last-of-type() pseudo-classes + nthOfType = (function() { + var idx = 0, len = 0, set = 0, parent = undefined, parents = Array(), nodes = Array(); + return function(element, dir) { + // ensure caches are emptied after each run, invoking with dir = 2 + if (dir == 2) { + idx = 0; len = 0; set = 0; nodes.length = 0; + parents.length = 0; parent = undefined; + return -1; + } + var e, i, j, k, l, name = element.localName; + if (nodes[set] && nodes[set][name] && parent === element.parentElement) { + i = set; j = idx; l = len; + } else { + l = parents.length; + parent = element.parentElement; + for (i = -1, j = 0, k = l - 1; l > j; ++j, --k) { + if (parents[j] === parent) { i = j; break; } + if (parents[k] === parent) { i = k; break; } + } + if (i < 0 || !nodes[i][name]) { + parents[i = l] = parent; + nodes[i] || (nodes[i] = Object()); + l = 0; nodes[i][name] = Array(); + e = parent && parent.firstElementChild || element; + while (e) { if (e === element) j = l; if (e.localName == name) { nodes[i][name][l] = e; ++l; } e = e.nextElementSibling; } + set = i; idx = j; len = l; + if (l < 2) return l; + } else { + l = nodes[i][name].length; + set = i; + } + } + if (element !== nodes[i][name][j] && element !== nodes[i][name][j = 0]) { + for (j = 0, e = nodes[i][name], k = l - 1; l > j; ++j, --k) { + if (e[j] === element) { break; } + if (e[k] === element) { j = k; break; } + } + } + idx = j + 1; len = l; + return dir ? l - j : idx; + }; + })(), + + // check if the document type is HTML + isHTML = + function(node) { + var doc = node.ownerDocument || node; + return doc.nodeType == 9 && + // contentType not in IE <= 11 + 'contentType' in doc ? + doc.contentType.indexOf('/html') > 0 : + doc.createElement('DiV').localName == 'div'; + }, + + // configure the engine to use special handling + configure = + function(option, clear) { + if (typeof option == 'string') { return !!Config[option]; } + if (typeof option != 'object') { return Config; } + for (var i in option) { + Config[i] = !!option[i]; + } + // clear lambda cache + if (clear) { + matchResolvers = { }; + selectResolvers = { }; + } + setIdentifierSyntax(); + return true; + }, + + // centralized error and exceptions handling + emit = + function(message, proto) { + var err; + if (Config.VERBOSITY) { + if (proto) { + err = new proto(message); + } else { + err = new global.DOMException(message, 'SyntaxError'); + } + throw err; + } + if (Config.LOGERRORS && console && console.log) { + console.log(message); + } + }, + + // execute the engine initialization code + initialize = + function(doc) { + setIdentifierSyntax(); + lastContext = switchContext(doc, true); + }, + + // build validation regexps used by the engine + setIdentifierSyntax = + function() { + + // + // NOTE: SPECIAL CASES IN CSS SYNTAX PARSING RULES + // + // The <EOF-token> https://drafts.csswg.org/css-syntax/#typedef-eof-token + // allow mangled|unclosed selector syntax at the end of selectors strings + // + // Literal equivalent hex representations of the characters: " ' ` ] ) + // + // \\x22 = " - double quotes \\x5b = [ - open square bracket + // \\x27 = ' - single quote \\x5d = ] - closed square bracket + // \\x60 = ` - back tick \\x28 = ( - open round parens + // \\x5c = \ - back slash \\x29 = ) - closed round parens + // + // using hex format prevents false matches of opened/closed instances + // pairs, coloring breakage and other editors highlightning problems. + // + + var identifier = + // doesn't start with a digit + '(?=[^0-9])' + + // can start with double dash + '(?:-{2}' + + // may include ascii chars + '|[a-zA-Z0-9-_]' + + // non-ascii chars + '|[^\\x00-\\x9f]' + + // escaped chars + '|\\\\[^\\r\\n\\f0-9a-fA-F]' + + // unicode chars + '|\\\\[0-9a-fA-F]{1,6}(?:\\r\\n|\\s)?' + + // any escaped chars + '|\\\\.' + + ')+', + + pseudonames = '[-\\w]+', + pseudoparms = '(?:[-+]?\\d*)(?:n\\s?[-+]?\\s?\\d*)', + doublequote = '"[^"\\\\]*(?:\\\\.[^"\\\\]*)*(?:"|$)', + singlequote = "'[^'\\\\]*(?:\\\\.[^'\\\\]*)*(?:'|$)", + + attrparser = identifier + '|' + doublequote + '|' + singlequote, + + attrvalues = '([\\x22\\x27]?)((?!\\3)*|(?:\\\\?.)*?)(?:\\3|$)', + + attributes = + '\\[' + + // attribute presence + '(?:\\*\\|)?' + + WSP + '?' + + '(' + identifier + '(?::' + identifier + ')?)' + + WSP + '?' + + '(?:' + + '(' + CFG.operators + ')' + WSP + '?' + + '(?:' + attrparser + ')' + + ')?' + + // attribute case sensitivity + '(?:' + WSP + '?\\b(i))?' + WSP + '?' + + '(?:\\]|$)', + + attrmatcher = attributes.replace(attrparser, attrvalues), + + pseudoclass = + '(?:\\x28' + WSP + '*' + + '(?:' + pseudoparms + '?)?|' + + // universal * & + // namespace *|* + '(?:\\*|\\|)|' + + '(?:' + + '(?::' + pseudonames + + '(?:\\x28' + pseudoparms + '?(?:\\x29|$))?|' + + ')|' + + '(?:[.#]?' + identifier + ')|' + + '(?:' + attributes + ')' + + ')+|' + + '(?:' + WSP + '?,' + WSP + '?)|' + + '(?:' + WSP + '?)|' + + '(?:\\x29|$))*', + + standardValidator = + '(?=' + WSP + '?[^>+~(){}<>])' + + '(?:' + + // universal * & + // namespace *|* + '(?:\\*|\\|)|' + + '(?:[.#]?' + identifier + ')+|' + + '(?:' + attributes + ')+|' + + '(?:::?' + pseudonames + pseudoclass + ')|' + + '(?:' + WSP + '?' + CFG.combinators + WSP + '?)|' + + '(?:' + WSP + '?,' + WSP + '?)|' + + '(?:' + WSP + '?)' + + ')+'; + + // the following global RE is used to return the + // deepest localName in selector strings and then + // use it to retrieve all possible matching nodes + // that will be filtered by compiled resolvers + reOptimizer = RegExp( + '(?:([.:#*]?)' + + '(' + identifier + ')' + + '(?:' + + ':[-\\w]+|' + + '\\[[^\\]]+(?:\\]|$)|' + + '\\x28[^\\x29]+(?:\\x29|$)' + + ')*)$'); + + // global + reValidator = RegExp(standardValidator, 'g'); + + Patterns.id = RegExp('^#(' + identifier + ')(.*)'); + Patterns.tagName = RegExp('^(' + identifier + ')(.*)'); + Patterns.className = RegExp('^\\.(' + identifier + ')(.*)'); + Patterns.attribute = RegExp('^(?:' + attrmatcher + ')(.*)'); + }, + + F_INIT = '"use strict";return function Resolver(c,f,x,r)', + + S_HEAD = 'var e,n,o,j=r.length-1,k=-1', + M_HEAD = 'var e,n,o', + + S_LOOP = 'main:while((e=c[++k]))', + N_LOOP = 'main:while((e=c.item(++k)))', + M_LOOP = 'e=c;', + + S_BODY = 'r[++j]=c[k];', + N_BODY = 'r[++j]=c.item(k);', + M_BODY = '', + + S_TAIL = 'continue main;', + M_TAIL = 'r=true;', + + S_TEST = 'if(f(c[k])){break main;}', + N_TEST = 'if(f(c.item(k))){break main;}', + M_TEST = 'f(c);', + + S_VARS = [ ], + M_VARS = [ ], + + // compile groups or single selector strings into + // executable functions for matching or selecting + compile = + function(selector, mode, callback) { + var factory, token, head = '', loop = '', macro = '', source = '', vars = ''; + + // 'mode' can be boolean or null + // true = select / false = match + // null to use collection.item() + switch (mode) { + case true: + if (selectLambdas[selector]) { return selectLambdas[selector]; } + macro = S_BODY + (callback ? S_TEST : '') + S_TAIL; + head = S_HEAD; + loop = S_LOOP; + break; + case false: + if (matchLambdas[selector]) { return matchLambdas[selector]; } + macro = M_BODY + (callback ? M_TEST : '') + M_TAIL; + head = M_HEAD; + loop = M_LOOP; + break; + case null: + if (selectLambdas[selector]) { return selectLambdas[selector]; } + macro = N_BODY + (callback ? N_TEST : '') + S_TAIL; + head = S_HEAD; + loop = N_LOOP; + break; + default: + break; + } + + source = compileSelector(selector, macro, mode, callback); + + loop += mode || mode === null ? '{' + source + '}' : source; + + if (mode || mode === null && selector.includes(':nth')) { + loop += reNthElem.test(selector) ? 's.nthElement(null, 2);' : ''; + loop += reNthType.test(selector) ? 's.nthOfType(null, 2);' : ''; + } + + if (S_VARS[0] || M_VARS[0]) { + vars = ',' + (S_VARS.join(',') || M_VARS.join(',')); + S_VARS.length = 0; + M_VARS.length = 0; + } + + factory = Function('s', F_INIT + '{' + head + vars + ';' + loop + 'return r;}')(Snapshot); + + return mode || mode === null ? (selectLambdas[selector] = factory) : (matchLambdas[selector] = factory); + }, + + // build conditional code to check components of selector strings + compileSelector = + function(expression, source, mode, callback) { + + // N is the negation pseudo-class flag + // D is the default inverted negation flag + var a, b, n, f, i, l, name, NS, N = '', D = '!', + compat, expr, match, result, status, symbol, test, + type, selector = expression, selector_string, vars; + + // original 'select' or 'match' selector string before normalization + selector_string = mode ? lastSelected : lastMatched; + + // isolate selector combinators/components and normalize whitespace + selector = selector.replace(STD.combinator, '$1');//.replace(STD.whitespace, ' '); + + // javascript needs a label to break + // out of the while loops processing + selector_recursion_label: + + while (selector) { + + // get namespace prefix if present or get first char of selector + symbol = STD.apimethods.test(selector) ? '|' : selector[0]; + + switch (symbol) { + + // universal resolver + case '*': + match = selector.match(Patterns.universal); + if (N == '!') { + source = 'if(' + N + 'true' + + '){' + source + '}'; + } + break; + + // id resolver + case '#': + match = selector.match(Patterns.id); + source = 'if(' + N + '(/^' + match[1] + '$/.test(e.getAttribute("id"))' + + ')){' + source + '}'; + break; + + // class name resolver + case '.': + match = selector.match(Patterns.className); + compat = (QUIRKS_MODE ? 'i' : '') + '.test(e.getAttribute("class"))'; + source = 'if(' + N + '(/(^|\\s)' + match[1] + '(\\s|$)/' + compat + + ')){' + source + '}'; + break; + + // tag name resolver + case (/[_a-z]/i.test(symbol) ? symbol : undefined): + match = selector.match(Patterns.tagName); + source = 'if(' + N + '(e.localName' + + (Config.MIXEDCASE || hasMixedCaseTagNames(doc) ? + '=="' + match[1].toLowerCase() + '"' : + '=="' + match[1].toUpperCase() + '"') + + ')){' + source + '}'; + break; + + // namespace resolver + case '|': + match = selector.match(Patterns.namespace); + if (match[1] == '*') { + source = 'if(' + N + 'true){' + source + '}'; + } else if (!match[1]) { + source = 'if(' + N + '(!e.namespaceURI)){' + source + '}'; + } else if (typeof match[1] == 'string' && root.prefix == match[1]) { + source = 'if(' + N + '(e.namespaceURI=="' + NAMESPACE + '")){' + source + '}'; + } else { + emit('\'' + selector_string + '\'' + qsInvalid); + } + break; + + // attributes resolver + case '[': + match = selector.match(Patterns.attribute); + NS = match[0].match(STD.namespaces); + name = match[1]; + expr = name.split(':'); + expr = expr.length == 2 ? expr[1] : expr[0]; + if (match[2] && !(test = Operators[match[2]])) { + emit('\'' + selector_string + '\'' + qsInvalid); + return ''; + } + if (match[4] === '') { + test = match[2] == '~=' ? + { p1: '^\\s', p2: '+$', p3: 'true' } : + match[2] in ATTR_STD_OPS && match[2] != '~=' ? + { p1: '^', p2: '$', p3: 'true' } : test; + } else if (match[2] == '~=' && match[4].includes(' ')) { + // whitespace separated list but value contains space + source = 'if(' + N + 'false){' + source + '}'; + break; + } else if (match[4]) { + match[4] = convertEscapes(match[4]).replace(REX.RegExpChar, '\\$&'); + } + type = match[5] == 'i' || (HTML_DOCUMENT && HTML_TABLE[expr.toLowerCase()]) ? 'i' : ''; + source = 'if(' + N + '(' + + (!match[2] ? (NS ? 's.hasAttributeNS(e,"' + name + '")' : 'e.hasAttribute&&e.hasAttribute("' + name + '")') : + !match[4] && ATTR_STD_OPS[match[2]] && match[2] != '~=' ? 'e.getAttribute&&e.getAttribute("' + name + '")==""' : + '(/' + test.p1 + match[4] + test.p2 + '/' + type + ').test(e.getAttribute&&e.getAttribute("' + name + '"))==' + test.p3) + + ')){' + source + '}'; + break; + + // *** General sibling combinator + // E ~ F (F relative sibling of E) + case '~': + match = selector.match(Patterns.relative); + source = 'n=e;while((e=e.previousElementSibling)){' + source + '}e=n;'; + break; + // *** Adjacent sibling combinator + // E + F (F adiacent sibling of E) + case '+': + match = selector.match(Patterns.adjacent); + source = 'n=e;if((e=e.previousElementSibling)){' + source + '}e=n;'; + break; + // *** Descendant combinator + // E F (E ancestor of F) + case '\x09': + case '\x20': + match = selector.match(Patterns.ancestor); + source = 'n=e;while((e=e.parentElement)){' + source + '}e=n;'; + break; + // *** Child combinator + // E > F (F children of E) + case '>': + match = selector.match(Patterns.children); + source = 'n=e;if((e=e.parentElement)){' + source + '}e=n;'; + break; + + // *** user supplied combinators extensions + case (symbol in Combinators ? symbol : undefined): + // for other registered combinators extensions + match[match.length - 1] = '*'; + source = Combinators[symbol](match) + source; + break; + + // *** tree-structural pseudo-classes + // :root, :empty, :first-child, :last-child, :only-child, :first-of-type, :last-of-type, :only-of-type + case ':': + if ((match = selector.match(Patterns.structural))) { + match[1] = match[1].toLowerCase(); + switch (match[1]) { + case 'root': + // there can only be one :root element, so exit the loop once found + source = 'if(' + N + '(e===s.root)){' + source + (mode ? 'break main;' : '') + '}'; + break; + case 'empty': + // matches elements that don't contain elements or text nodes + source = 'n=e.firstChild;while(n&&!(/1|3/).test(n.nodeType)){n=n.nextSibling}if(' + D + 'n){' + source + '}'; + break; + + // *** child-indexed pseudo-classes + // :first-child, :last-child, :only-child + case 'only-child': + source = 'if(' + N + '(!e.nextElementSibling&&!e.previousElementSibling)){' + source + '}'; + break; + case 'last-child': + source = 'if(' + N + '(!e.nextElementSibling)){' + source + '}'; + break; + case 'first-child': + source = 'if(' + N + '(!e.previousElementSibling)){' + source + '}'; + break; + + // *** typed child-indexed pseudo-classes + // :only-of-type, :last-of-type, :first-of-type + case 'only-of-type': + source = 'o=e.localName;' + + 'n=e;while((n=n.nextElementSibling)&&n.localName!=o);if(!n){' + + 'n=e;while((n=n.previousElementSibling)&&n.localName!=o);}if(' + D + 'n){' + source + '}'; + break; + case 'last-of-type': + source = 'n=e;o=e.localName;while((n=n.nextElementSibling)&&n.localName!=o);if(' + D + 'n){' + source + '}'; + break; + case 'first-of-type': + source = 'n=e;o=e.localName;while((n=n.previousElementSibling)&&n.localName!=o);if(' + D + 'n){' + source + '}'; + break; + default: + emit('\'' + selector_string + '\'' + qsInvalid); + break; + } + } + + // *** child-indexed & typed child-indexed pseudo-classes + // :nth-child, :nth-of-type, :nth-last-child, :nth-last-of-type + else if ((match = selector.match(Patterns.treestruct))) { + match[1] = match[1].toLowerCase(); + switch (match[1]) { + case 'nth-child': + case 'nth-of-type': + case 'nth-last-child': + case 'nth-last-of-type': + expr = /-of-type/i.test(match[1]); + if (match[1] && match[2]) { + type = /last/i.test(match[1]); + if (match[2] == 'n') { + source = 'if(' + N + 'true){' + source + '}'; + break; + } else if (match[2] == '1') { + test = type ? 'next' : 'previous'; + source = expr ? 'n=e;o=e.localName;' + + 'while((n=n.' + test + 'ElementSibling)&&n.localName!=o);if(' + D + 'n){' + source + '}' : + 'if(' + N + '!e.' + test + 'ElementSibling){' + source + '}'; + break; + } else if (match[2] == 'even' || match[2] == '2n0' || match[2] == '2n+0' || match[2] == '2n') { + test = 'n%2==0'; + } else if (match[2] == 'odd' || match[2] == '2n1' || match[2] == '2n+1') { + test = 'n%2==1'; + } else { + f = /n/i.test(match[2]); + n = match[2].split('n'); + a = parseInt(n[0], 10) || 0; + b = parseInt(n[1], 10) || 0; + if (n[0] == '-') { a = -1; } + if (n[0] == '+') { a = +1; } + test = (b ? '(n' + (b > 0 ? '-' : '+') + Math.abs(b) + ')' : 'n') + '%' + a + '==0' ; + test = + a >= +1 ? (f ? 'n>' + (b - 1) + (Math.abs(a) != 1 ? '&&' + test : '') : 'n==' + a) : + a <= -1 ? (f ? 'n<' + (b + 1) + (Math.abs(a) != 1 ? '&&' + test : '') : 'n==' + a) : + a === 0 ? (n[0] ? 'n==' + b : 'n>' + (b - 1)) : 'false'; + } + expr = expr ? 'OfType' : 'Element'; + type = type ? 'true' : 'false'; + source = 'n=s.nth' + expr + '(e,' + type + ');if(' + N + '(' + test + ')){' + source + '}'; + } else { + emit('\'' + selector_string + '\'' + qsInvalid); + } + break; + default: + emit('\'' + selector_string + '\'' + qsInvalid); + break; + } + } + + // *** logical combination pseudo-classes + // :is( s1, [ s2, ... ]), :not( s1, [ s2, ... ]) + else if ((match = selector.match(Patterns.logicalsel))) { + match[1] = match[1].toLowerCase(); + switch (match[1]) { + case 'is': + case 'where': + case 'matches': + expr = match[2].replace(REX.CommaGroup, ',').replace(REX.TrimSpaces, ''); + source = 'if(s.match("' + expr.replace(/\x22/g, '\\"') + '",e)){' + source + '}'; + break; + case 'not': + expr = match[2].replace(REX.CommaGroup, ',').replace(REX.TrimSpaces, ''); + source = 'if(!s.match("' + expr.replace(/\x22/g, '\\"') + '",e)){' + source + '}'; + break; + default: + emit('\'' + selector_string + '\'' + qsInvalid); + break; + } + } + + // *** linguistic pseudo-classes + // :dir( ltr / rtl ), :lang( en ) + else if ((match = selector.match(Patterns.linguistic))) { + match[1] = match[1].toLowerCase(); + switch (match[1]) { + case 'dir': + source = 'var p;if(' + N + '(' + + '(/' + match[2] + '/i.test(e.dir))||(p=s.ancestor("[dir]", e))&&' + + '(/' + match[2] + '/i.test(p.dir))||(e.dir==""||e.dir=="auto")&&' + + '(' + (match[2] == 'ltr' ? '!':'')+ RTL +'.test(e.textContent)))' + + '){' + source + '};'; + break; + case 'lang': + expr = '(?:^|-)' + match[2] + '(?:-|$)'; + source = 'var p;if(' + N + '(' + + '(e.isConnected&&(e.lang==""&&(p=s.ancestor("[lang]",e)))&&' + + '(p.lang=="' + match[2] + '")||/'+ expr +'/i.test(e.lang)))' + + '){' + source + '};'; + break; + default: + emit('\'' + selector_string + '\'' + qsInvalid); + break; + } + } + + // *** location pseudo-classes + // :any-link, :link, :visited, :target + else if ((match = selector.match(Patterns.locationpc))) { + match[1] = match[1].toLowerCase(); + switch (match[1]) { + case 'any-link': + source = 'if(' + N + '(/^a|area$/i.test(e.localName)&&e.hasAttribute("href")||e.visited)){' + source + '}'; + break; + case 'link': + source = 'if(' + N + '(/^a|area$/i.test(e.localName)&&e.hasAttribute("href"))){' + source + '}'; + break; + case 'visited': + source = 'if(' + N + '(/^a|area$/i.test(e.localName)&&e.hasAttribute("href")&&e.visited)){' + source + '}'; + break; + case 'target': + source = 'if(' + N + '((s.doc.compareDocumentPosition(e)&16)&&s.doc.location.hash&&e.id==s.doc.location.hash.slice(1))){' + source + '}'; + break; + default: + emit('\'' + selector_string + '\'' + qsInvalid); + break; + } + } + + // *** user actions pseudo-classes + // :hover, :active, :focus + else if ((match = selector.match(Patterns.useraction))) { + match[1] = match[1].toLowerCase(); + switch (match[1]) { + case 'hover': + source = 'hasFocus' in doc && doc.hasFocus() ? + 'if(' + N + '(e===s.doc.hoverElement)){' + source + '}' : + 'if(' + D + 'true){' + source + '}'; + break; + case 'active': + source = 'hasFocus' in doc && doc.hasFocus() ? + 'if(' + N + '(e===s.doc.activeElement)){' + source + '}' : + 'if(' + D + 'true){' + source + '}'; + break; + case 'focus': + source = 'hasFocus' in doc ? + 'if(' + N + '(e===s.doc.activeElement&&s.doc.hasFocus()&&(e.type||e.href||typeof e.tabIndex=="number"))){' + source + '}' : + 'if(' + N + '(e===s.doc.activeElement&&(e.type||e.href))){' + source + '}'; + break; + case 'focus-within': + source = 'hasFocus' in doc ? + 'n=s.doc.activeElement;while(e){if(e===n||e.parentNode===n)break;}' + + 'if(' + N + '(e===n&&s.doc.hasFocus()&&(e.type||e.href||typeof e.tabIndex=="number"))){' + source + '}' : source; + break; + default: + emit('\'' + selector_string + '\'' + qsInvalid); + break; + } + } + + // *** user interface and form pseudo-classes + // :enabled, :disabled, :read-only, :read-write, :placeholder-shown, :default + else if ((match = selector.match(Patterns.inputstate))) { + match[1] = match[1].toLowerCase(); + switch (match[1]) { + case 'enabled': + source = 'if(' + N + '(("form" in e||/^optgroup$/i.test(e.localName))&&"disabled" in e &&e.disabled===false' + + ')){' + source + '}'; + break; + case 'disabled': + // https://www.w3.org/TR/html5/forms.html#enabling-and-disabling-form-controls:-the-disabled-attribute + source = 'if(' + N + '(("form" in e||/^optgroup$/i.test(e.localName))&&"disabled" in e&&' + + '(e.disabled===true||(n=s.ancestor("fieldset",e))&&(n=s.first("legend",n))&&!n.contains(e))' + + ')){' + source + '}'; + break; + case 'read-only': + source = + 'if(' + N + '(' + + '(/^textarea$/i.test(e.localName)&&(e.readOnly||e.disabled))||' + + '("|password|text|".includes("|"+e.type+"|")&&e.readOnly)' + + ')){' + source + '}'; + break; + case 'read-write': + source = + 'if(' + N + '(' + + '((/^textarea$/i.test(e.localName)&&!e.readOnly&&!e.disabled)||' + + '("|password|text|".includes("|"+e.type+"|")&&!e.readOnly&&!e.disabled))||' + + '(e.hasAttribute("contenteditable")||(s.doc.designMode=="on"))' + + ')){' + source + '}'; + break; + case 'placeholder-shown': + source = + 'if(' + N + '(' + + '(/^input|textarea$/i.test(e.localName))&&e.hasAttribute("placeholder")&&' + + '("|textarea|password|number|search|email|text|tel|url|".includes("|"+e.type+"|"))&&' + + '(!s.match(":focus",e))' + + ')){' + source + '}'; + break; + case 'default': + source = + 'if(' + N + '("form" in e && e.form)){' + + 'var x=0;n=[];' + + 'if(e.type=="image")n=e.form.getElementsByTagName("input");' + + 'if(e.type=="submit")n=e.form.elements;' + + 'while(n[x]&&e!==n[x]){' + + 'if(n[x].type=="image")break;' + + 'if(n[x].type=="submit")break;' + + 'x++;' + + '}' + + '}' + + 'if(' + N + '(e.form&&(e===n[x]&&"|image|submit|".includes("|"+e.type+"|"))||' + + '((/^option$/i.test(e.localName))&&e.defaultSelected)||' + + '(("|radio|checkbox|".includes("|"+e.type+"|"))&&e.defaultChecked)' + + ')){' + source + '}'; + break; + default: + emit('\'' + selector_string + '\'' + qsInvalid); + break; + } + } + + // *** input pseudo-classes (for form validation) + // :checked, :indeterminate, :valid, :invalid, :in-range, :out-of-range, :required, :optional + else if ((match = selector.match(Patterns.inputvalue))) { + match[1] = match[1].toLowerCase(); + switch (match[1]) { + case 'checked': + source = 'if(' + N + '(/^input$/i.test(e.localName)&&' + + '("|radio|checkbox|".includes("|"+e.type+"|")&&e.checked)||' + + '(/^option$/i.test(e.localName)&&(e.selected||e.checked))' + + ')){' + source + '}'; + break; + case 'indeterminate': + source = + 'if(' + N + '(/^progress$/i.test(e.localName)&&!e.hasAttribute("value"))||' + + '(/^input$/i.test(e.localName)&&("checkbox"==e.type&&e.indeterminate)||' + + '("radio"==e.type&&e.name&&!s.first("input[name="+e.name+"]:checked",e.form))' + + ')){' + source + '}'; + break; + case 'required': + source = + 'if(' + N + + '(/^input|select|textarea$/i.test(e.localName)&&e.required)' + + '){' + source + '}'; + break; + case 'optional': + source = + 'if(' + N + + '(/^input|select|textarea$/i.test(e.localName)&&!e.required)' + + '){' + source + '}'; + break; + case 'invalid': + source = + 'if(' + N + '((' + + '(/^form$/i.test(e.localName)&&!e.noValidate)||' + + '(e.willValidate&&!e.formNoValidate))&&!e.checkValidity())||' + + '(/^fieldset$/i.test(e.localName)&&s.first(":invalid",e))' + + '){' + source + '}'; + break; + case 'valid': + source = + 'if(' + N + '((' + + '(/^form$/i.test(e.localName)&&!e.noValidate)||' + + '(e.willValidate&&!e.formNoValidate))&&e.checkValidity())||' + + '(/^fieldset$/i.test(e.localName)&&s.first(":valid",e))' + + '){' + source + '}'; + break; + case 'in-range': + source = + 'if(' + N + + '(/^input$/i.test(e.localName))&&' + + '(e.willValidate&&!e.formNoValidate)&&' + + '(!e.validity.rangeUnderflow&&!e.validity.rangeOverflow)&&' + + '("|date|datetime-local|month|number|range|time|week|".includes("|"+e.type+"|"))&&' + + '("range"==e.type||e.getAttribute("min")||e.getAttribute("max"))' + + '){' + source + '}'; + break; + case 'out-of-range': + source = + 'if(' + N + + '(/^input$/i.test(e.localName))&&' + + '(e.willValidate&&!e.formNoValidate)&&' + + '(e.validity.rangeUnderflow||e.validity.rangeOverflow)&&' + + '("|date|datetime-local|month|number|range|time|week|".includes("|"+e.type+"|"))&&' + + '("range"==e.type||e.getAttribute("min")||e.getAttribute("max"))' + + '){' + source + '}'; + break; + default: + emit('\'' + selector_string + '\'' + qsInvalid); + break; + } + } + + // allow pseudo-elements starting with single colon (:) + // :after, :before, :first-letter, :first-line + // assert: e.type is in double-colon format, like ::after + else if ((match = selector.match(Patterns.pseudo_sng))) { + source = 'if(e.element&&e.type.toLowerCase()=="' + + ':' + match[0].toLowerCase() + '"){e=e.element;' + source + '}'; + } + + // allow pseudo-elements starting with double colon (::) + // ::after, ::before, ::marker, ::placeholder, ::inactive-selection, ::selection, ::-webkit-<foo-bar> + // assert: e.type is in double-colon format, like ::after + else if ((match = selector.match(Patterns.pseudo_dbl))) { + source = 'if(e.element&&e.type.toLowerCase()=="' + + match[0].toLowerCase() + '"){e=e.element;' + source + '}'; + } + + // placeholder for parsed only no-op selectors + else if ((match = selector.match(Patterns.pseudo_nop))) { + source = 'if(' + N + 'false' + '){' + source + '}'; + } + + else { + + // reset + expr = false; + status = false; + + // process registered selector extensions + for (expr in Selectors) { + if ((match = selector.match(Selectors[expr].Expression))) { + result = Selectors[expr].Callback(match, source, mode, callback); + if ('match' in result) { match = result.match; } + vars = result.modvar; + if (mode) { + // add extra select() vars + vars && S_VARS.indexOf(vars) < 0 && (S_VARS[S_VARS.length] = vars); + } else { + // add extra match() vars + vars && M_VARS.indexOf(vars) < 0 && (M_VARS[M_VARS.length] = vars); + } + // extension source code + source = result.source; + // extension status code + status = result.status; + // break on status error + if (status) { break; } + } + } + + if (!status) { + emit('unknown pseudo-class selector \'' + selector + '\''); + return ''; + } + + if (!expr) { + emit('unknown token in selector \'' + selector + '\''); + return ''; + } + + } + break; + + default: + emit('\'' + selector_string + '\'' + qsInvalid); + break selector_recursion_label; + + } + // end of switch symbol + + if (!match) { + emit('\'' + selector_string + '\'' + qsInvalid); + return ''; + } + + // pop last component + selector = match.pop(); + } + // end of while selector + + return source; + }, + + // replace ':scope' pseudo-class with element references + makeref = + function(selectors, element) { + return selectors.replace(/:scope/ig, + element.localName + + (element.id ? '#' + element.id : '') + + (element.className ? '.' + element.classList[0] : '')); + }, + + // equivalent of w3c 'closest' method + ancestor = + function _closest(selectors, element, callback) { + + if ((/:scope/i).test(selectors)) { + selectors = makeref(selectors, element); + } + + while (element) { + if (match(selectors, element, callback)) break; + element = element.parentElement; + } + return element; + }, + + match_assert = + function(f, element, callback) { + for (var i = 0, l = f.length, r = false; l > i; ++i) + f[i](element, callback, null, false) && (r = true); + return r; + }, + + match_collect = + function(selectors, callback) { + for (var i = 0, l = selectors.length, f = [ ]; l > i; ++i) + f[i] = compile(selectors[i], false, callback); + return { factory: f }; + }, + + // equivalent of w3c 'matches' method + match = + function _matches(selectors, element, callback) { + + var expressions, parsed; + + if (element && matchResolvers[selectors]) { + return match_assert(matchResolvers[selectors].factory, element, callback); + } + + lastMatched = selectors; + + // arguments validation + if (arguments.length === 0) { + emit(qsNotArgs, TypeError); + return Config.VERBOSITY ? undefined : false; + } else if (arguments[0] === '') { + emit('\'\'' + qsInvalid); + return Config.VERBOSITY ? undefined : false; + } + + // input NULL or UNDEFINED + if (typeof selectors != 'string') { + selectors = '' + selectors; + } + + if ((/:scope/i).test(selectors)) { + selectors = makeref(selectors, element); + } + + // normalize input string + parsed = selectors. + replace(/\x00|\\$/g, '\ufffd'). + replace(REX.CombineWSP, '\x20'). + replace(REX.PseudosWSP, '$1'). + replace(REX.TabCharWSP, '\t'). + replace(REX.CommaGroup, ','). + replace(REX.TrimSpaces, ''); + + // parse, validate and split possible compound selectors + if ((expressions = parsed.match(reValidator)) && expressions.join('') == parsed) { + expressions = parsed.match(REX.SplitGroup); + if (parsed[parsed.length - 1] == ',') { + emit(qsInvalid); + return Config.VERBOSITY ? undefined : false; + } + } else { + emit('\'' + selectors + '\'' + qsInvalid); + return Config.VERBOSITY ? undefined : false; + } + + matchResolvers[selectors] = match_collect(expressions, callback); + + return match_assert(matchResolvers[selectors].factory, element, callback); + }, + + // equivalent of w3c 'querySelector' method + first = + function _querySelector(selectors, context, callback) { + if (arguments.length === 0) { + emit(qsNotArgs, TypeError); + } + return select(selectors, context, + typeof callback == 'function' ? + function firstMatch(element) { + callback(element); + return false; + } : + function firstMatch() { + return false; + } + )[0] || null; + }, + + // equivalent of w3c 'querySelectorAll' method + select = + function _querySelectorAll(selectors, context, callback) { + + var expressions, nodes = [ ], parsed, resolver; + + context || (context = doc); + + if (selectors) { + if ((resolver = selectResolvers[selectors])) { + if (resolver.context === context && resolver.callback === callback) { + var f = resolver.factory, h = resolver.htmlset, n = resolver.nodeset; + if (n.length > 1) { + for (var i = 0, l = n.length, list; l > i; ++i) { + list = compat[n[i][0]](context, n[i].slice(1))(); + if (f[i] !== null) { + f[i](list, callback, context, nodes); + } else { + nodes = nodes.concat(list); + } + } + if (l > 1 && nodes.length > 1) { + nodes.sort(documentOrder); + hasDupes && (nodes = unique(nodes)); + } + } else { + if (f[0]) { + nodes = f[0](h[0](), callback, context, nodes); + } else { + nodes = h[0](); + } + } + return typeof callback == 'function' ? + concatCall(nodes, callback) : nodes; + } + } + } + + lastSelected = selectors; + + // arguments validation + if (arguments.length === 0) { + emit(qsNotArgs, TypeError); + return Config.VERBOSITY ? undefined : none; + } else if (arguments[0] === '') { + emit('\'\'' + qsInvalid); + return Config.VERBOSITY ? undefined : none; + } else if (lastContext !== context) { + lastContext = switchContext(context); + } + + // input NULL or UNDEFINED + if (typeof selectors != 'string') { + selectors = '' + selectors; + } + + if ((/:scope/i).test(selectors)) { + selectors = makeref(selectors, context); + } + + // normalize input string + parsed = selectors. + replace(/\x00|\\$/g, '\ufffd'). + replace(REX.CombineWSP, '\x20'). + replace(REX.PseudosWSP, '$1'). + replace(REX.TabCharWSP, '\t'). + replace(REX.CommaGroup, ','). + replace(REX.TrimSpaces, ''); + + // parse, validate and split possible compound selectors + if ((expressions = parsed.match(reValidator)) && expressions.join('') == parsed) { + expressions = parsed.match(REX.SplitGroup); + if (parsed[parsed.length - 1] == ',') { + emit(qsInvalid); + return Config.VERBOSITY ? undefined : false; + } + } else { + emit('\'' + selectors + '\'' + qsInvalid); + return Config.VERBOSITY ? undefined : false; + } + + // save/reuse factory and closure collection + selectResolvers[selectors] = collect(expressions, context, callback); + + nodes = selectResolvers[selectors].results; + + return typeof callback == 'function' ? + concatCall(nodes, callback) : nodes; + }, + + // optimize selectors avoiding duplicated checks + optimize = + function(selector, token) { + var index = token.index, + length = token[1].length + token[2].length; + return selector.slice(0, index) + + (' >+~'.indexOf(selector.charAt(index - 1)) > -1 ? + (':['.indexOf(selector.charAt(index + length + 1)) > -1 ? + '*' : '') : '') + selector.slice(index + length - (token[1] == '*' ? 1 : 0)); + }, + + // prepare factory resolvers and closure collections + collect = + function(selectors, context, callback) { + + var i, l, seen = { }, token = ['', '*', '*'], optimized = selectors, + factory = [ ], htmlset = [ ], nodeset = [ ], results = [ ], type; + + for (i = 0, l = selectors.length; l > i; ++i) { + + if (!seen[selectors[i]] && (seen[selectors[i]] = true)) { + type = selectors[i].match(reOptimizer); + if (type && type[1] != ':' && (token = type)) { + token[1] || (token[1] = '*'); + optimized[i] = optimize(optimized[i], token); + } else { + token = ['', '*', '*']; + } + } + + nodeset[i] = token[1] + token[2]; + htmlset[i] = compat[token[1]](context, token[2]); + factory[i] = compile(optimized[i], true, null); + + factory[i] ? + factory[i](htmlset[i](), callback, context, results) : + result.concat(htmlset[i]()); + } + + if (l > 1) { + results.sort(documentOrder); + hasDupes && (results = unique(results)); + } + + return { + callback: callback, + context: context, + factory: factory, + htmlset: htmlset, + nodeset: nodeset, + results: results + }; + + }, + + // QSA placeholders to native references + _closest, _matches, _querySelector, _querySelectorAll, + + // overrides QSA methods (only for browsers) + install = + function(all) { + + // save native QSA references + _closest = Element.prototype.closest; + _matches = Element.prototype.matches; + _querySelector = Document.prototype.querySelector; + _querySelectorAll = Document.prototype.querySelectorAll; + + Element.prototype.closest = + function closest() { + var ctor = Object.getPrototypeOf(this).__proto__.__proto__.constructor.name; + if (!('nodeType' in this)) { emit('\'closest\' called on an object that does not implement interface ' + ctor + '.', TypeError); } + return arguments.length < 1 ? ancestor.apply(this, [ ]) : + arguments.length < 2 ? ancestor.apply(this, [ arguments[0], this ]) : + ancestor.apply(this, [ arguments[0], this, typeof arguments[1] == 'function' ? arguments[1] : undefined ]); + }; + + Element.prototype.matches = + function matches() { + var ctor = Object.getPrototypeOf(this).__proto__.__proto__.constructor.name; + if (!('nodeType' in this)) { emit('\'matches\' called on an object that does not implement interface ' + ctor + '.', TypeError); } + return arguments.length < 1 ? match.apply(this, [ ]) : + arguments.length < 2 ? match.apply(this, [ arguments[0], this ]) : + match.apply(this, [ arguments[0], this, typeof arguments[1] == 'function' ? arguments[1] : undefined ]); + }; + + Element.prototype.querySelector = + Document.prototype.querySelector = + DocumentFragment.prototype.querySelector = + function querySelector() { + var ctor = Object.getPrototypeOf(this).__proto__.__proto__.constructor.name; + if (!('nodeType' in this)) { emit('\'querySelector\' called on an object that does not implement interface ' + ctor + '.', TypeError); } + return arguments.length < 1 ? first.apply(this, [ ]) : + arguments.length < 2 ? first.apply(this, [ arguments[0], this ]) : + first.apply(this, [ arguments[0], this, typeof arguments[1] == 'function' ? arguments[1] : undefined ]); + }; + + Element.prototype.querySelectorAll = + Document.prototype.querySelectorAll = + DocumentFragment.prototype.querySelectorAll = + function querySelectorAll() { + var ctor = Object.getPrototypeOf(this).__proto__.__proto__.constructor.name; + if (!('nodeType' in this)) { emit('\'querySelectorAll\' called on an object that does not implement interface ' + ctor + '.', TypeError); } + return arguments.length < 1 ? select.apply(this, [ ]) : + arguments.length < 2 ? select.apply(this, [ arguments[0], this ]) : + select.apply(this, [ arguments[0], this, typeof arguments[1] == 'function' ? arguments[1] : undefined ]); + }; + + if (all) { + document.addEventListener('load', function(e) { + var c, d, r, s, t = e.target; + if (/iframe/i.test(t.localName)) { + c = '(' + Export + ')(this, ' + Factory + ');'; d = t.contentDocument; + s = d.createElement('script'); s.textContent = c + 'NW.Dom.install()'; + r = d.documentElement; r.removeChild(r.insertBefore(s, r.firstChild)); + } + }, true); + } + + }, + + // restore QSA methods (only for browsers) + uninstall = + function() { + // reinstates QSA native references + Element.prototype.closest = _closest; + Element.prototype.matches = _matches; + Element.prototype.querySelector = + Document.prototype.querySelector = + DocumentFragment.prototype.querySelector = _querySelector; + Element.prototype.querySelectorAll = + Document.prototype.querySelectorAll = + DocumentFragment.prototype.querySelectorAll = _querySelectorAll; + }, + + // empty set + none = Array(), + + // context + lastContext, + + // selector + lastMatched, + lastSelected, + + // cached lambdas + matchLambdas = { }, + selectLambdas = { }, + + // cached resolvers + matchResolvers = { }, + selectResolvers = { }, + + // passed to resolvers + Snapshot = { + + doc: doc, + from: doc, + root: root, + + byTag: byTag, + + first: first, + match: match, + + ancestor: ancestor, + + nthOfType: nthOfType, + nthElement: nthElement, + + hasAttributeNS: hasAttributeNS + }, + + // public exported methods/objects + Dom = { + + // exported cache objects + + lastMatched: lastMatched, + lastSelected: lastSelected, + + matchLambdas: matchLambdas, + selectLambdas: selectLambdas, + + matchResolvers: matchResolvers, + selectResolvers: selectResolvers, + + // exported compiler macros + + CFG: CFG, + + M_BODY: M_BODY, + S_BODY: S_BODY, + M_TEST: M_TEST, + S_TEST: S_TEST, + + // exported engine methods + + byId: byId, + byTag: byTag, + byClass: byClass, + + match: match, + first: first, + select: select, + closest: ancestor, + + compile: compile, + configure: configure, + + emit: emit, + Config: Config, + Snapshot: Snapshot, + + Version: version, + + install: install, + uninstall: uninstall, + + Operators: Operators, + Selectors: Selectors, + + // register a new selector combinator symbol and its related function resolver + registerCombinator: + function(combinator, resolver) { + var i = 0, l = combinator.length, symbol; + for (; l > i; ++i) { + if (combinator[i] != '=') { + symbol = combinator[i]; + break; + } + } + if (CFG.combinators.indexOf(symbol) < 0) { + CFG.combinators = CFG.combinators.replace('](', symbol + ']('); + CFG.combinators = CFG.combinators.replace('])', symbol + '])'); + Combinators[combinator] = resolver; + setIdentifierSyntax(); + } else { + console.warn('Warning: the \'' + combinator + '\' combinator is already registered.'); + } + }, + + // register a new attribute operator symbol and its related function resolver + registerOperator: + function(operator, resolver) { + var i = 0, l = operator.length, symbol; + for (; l > i; ++i) { + if (operator[i] != '=') { + symbol = operator[i]; + break; + } + } + if (CFG.operators.indexOf(symbol) < 0 && !Operators[operator]) { + CFG.operators = CFG.operators.replace(']=', symbol + ']='); + Operators[operator] = resolver; + setIdentifierSyntax(); + } else { + console.warn('Warning: the \'' + operator + '\' operator is already registered.'); + } + }, + + // register a new selector symbol and its related function resolver + registerSelector: + function(name, rexp, func) { + Selectors[name] || (Selectors[name] = { + Expression: rexp, + Callback: func + }); + } + + }; + + initialize(doc); + + return Dom; + +}); |