whatsappicloudtweetdeckhipchattelegramhangoutsslackgmailskypefacebook-workplaceoutlookemailmicrosoft-teamsdiscordmessengercustom-servicesmacoslinuxwindowsinbox
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1236 lines
45 KiB
1236 lines
45 KiB
9 years ago
|
/**
|
||
|
* @class Ext.dom.Query
|
||
|
* @alternateClassName Ext.DomQuery
|
||
|
* @alternateClassName Ext.core.DomQuery
|
||
|
* @singleton
|
||
|
*
|
||
|
* Provides high performance selector/xpath processing by compiling queries into reusable functions. New pseudo classes
|
||
|
* and matchers can be plugged. It works on HTML and XML documents (if a content node is passed in).
|
||
|
*
|
||
|
* DomQuery supports most of the [CSS3 selectors spec][1], along with some custom selectors and basic XPath.
|
||
|
*
|
||
|
* All selectors, attribute filters and pseudos below can be combined infinitely in any order. For example
|
||
|
* `div.foo:nth-child(odd)[@foo=bar].bar:first` would be a perfectly valid selector. Node filters are processed
|
||
|
* in the order in which they appear, which allows you to optimize your queries for your document structure.
|
||
|
*
|
||
|
* ## Simple Selectors
|
||
|
*
|
||
|
* For performance reasons, some query methods accept selectors that are termed as **simple selectors**. A simple
|
||
|
* selector is a selector that does not include contextual information about any parent/sibling elements.
|
||
|
*
|
||
|
* Some examples of valid simple selectors:
|
||
|
*
|
||
|
* var simple = '.foo'; // Only asking for the class name on the element
|
||
|
* var simple = 'div.bar'; // Only asking for the tag/class name on the element
|
||
|
* var simple = '[href];' // Asking for an attribute on the element.
|
||
|
* var simple = ':not(.foo)'; // Only asking for the non-matches against the class name
|
||
|
* var simple = 'span:first-child'; // Doesn't require any contextual information about the parent node
|
||
|
*
|
||
|
* Simple examples of invalid simple selectors:
|
||
|
*
|
||
|
* var notSimple = 'div.foo div.bar'; // Requires matching a parent node by class name
|
||
|
* var notSimple = 'span + div'; // Requires matching a sibling by tag name
|
||
|
*
|
||
|
* ## Element Selectors:
|
||
|
*
|
||
|
* - **`*`** any element
|
||
|
* - **`E`** an element with the tag E
|
||
|
* - **`E F`** All descendent elements of E that have the tag F
|
||
|
* - **`E > F`** or **E/F** all direct children elements of E that have the tag F
|
||
|
* - **`E + F`** all elements with the tag F that are immediately preceded by an element with the tag E
|
||
|
* - **`E ~ F`** all elements with the tag F that are preceded by a sibling element with the tag E
|
||
|
*
|
||
|
* ## Attribute Selectors:
|
||
|
*
|
||
|
* The use of `@` and quotes are optional. For example, `div[@foo='bar']` is also a valid attribute selector.
|
||
|
*
|
||
|
* - **`E[foo]`** has an attribute "foo"
|
||
|
* - **`E[foo=bar]`** has an attribute "foo" that equals "bar"
|
||
|
* - **`E[foo^=bar]`** has an attribute "foo" that starts with "bar"
|
||
|
* - **`E[foo$=bar]`** has an attribute "foo" that ends with "bar"
|
||
|
* - **`E[foo*=bar]`** has an attribute "foo" that contains the substring "bar"
|
||
|
* - **`E[foo%=2]`** has an attribute "foo" that is evenly divisible by 2
|
||
|
* - **`E[foo!=bar]`** attribute "foo" does not equal "bar"
|
||
|
*
|
||
|
* ## Pseudo Classes:
|
||
|
*
|
||
|
* - **`E:first-child`** E is the first child of its parent
|
||
|
* - **`E:last-child`** E is the last child of its parent
|
||
|
* - **`E:nth-child(_n_)`** E is the _n_th child of its parent (1 based as per the spec)
|
||
|
* - **`E:nth-child(odd)`** E is an odd child of its parent
|
||
|
* - **`E:nth-child(even)`** E is an even child of its parent
|
||
|
* - **`E:only-child`** E is the only child of its parent
|
||
|
* - **`E:checked`** E is an element that is has a checked attribute that is true (e.g. a radio or checkbox)
|
||
|
* - **`E:first`** the first E in the resultset
|
||
|
* - **`E:last`** the last E in the resultset
|
||
|
* - **`E:nth(_n_)`** the _n_th E in the resultset (1 based)
|
||
|
* - **`E:odd`** shortcut for :nth-child(odd)
|
||
|
* - **`E:even`** shortcut for :nth-child(even)
|
||
|
* - **`E:contains(foo)`** E's innerHTML contains the substring "foo"
|
||
|
* - **`E:nodeValue(foo)`** E contains a textNode with a nodeValue that equals "foo"
|
||
|
* - **`E:not(S)`** an E element that does not match simple selector S
|
||
|
* - **`E:has(S)`** an E element that has a descendent that matches simple selector S
|
||
|
* - **`E:next(S)`** an E element whose next sibling matches simple selector S
|
||
|
* - **`E:prev(S)`** an E element whose previous sibling matches simple selector S
|
||
|
* - **`E:any(S1|S2|S2)`** an E element which matches any of the simple selectors S1, S2 or S3
|
||
|
* - **`E:visible(true)`** an E element which is deeply visible according to {@link Ext.dom.Element#isVisible}
|
||
|
*
|
||
|
* ## CSS Value Selectors:
|
||
|
*
|
||
|
* - **`E{display=none}`** css value "display" that equals "none"
|
||
|
* - **`E{display^=none}`** css value "display" that starts with "none"
|
||
|
* - **`E{display$=none}`** css value "display" that ends with "none"
|
||
|
* - **`E{display*=none}`** css value "display" that contains the substring "none"
|
||
|
* - **`E{display%=2}`** css value "display" that is evenly divisible by 2
|
||
|
* - **`E{display!=none}`** css value "display" that does not equal "none"
|
||
|
*
|
||
|
* ## XML Namespaces:
|
||
|
* - **`ns|E`** an element with tag E and namespace prefix ns
|
||
|
*
|
||
|
* [1]: http://www.w3.org/TR/2005/WD-css3-selectors-20051215/#selectors
|
||
|
*/
|
||
|
Ext.define('Ext.dom.Query', function() {
|
||
|
var DQ,
|
||
|
doc = document,
|
||
|
cache, simpleCache, valueCache,
|
||
|
useClassList = !!doc.documentElement.classList,
|
||
|
useElementPointer = !!doc.documentElement.firstElementChild,
|
||
|
useChildrenCollection = (function() {
|
||
|
var d = doc.createElement('div');
|
||
|
d.innerHTML = '<!-- -->text<!-- -->';
|
||
|
return d.children && (d.children.length === 0);
|
||
|
})(),
|
||
|
nonSpace = /\S/,
|
||
|
trimRe = /^\s+|\s+$/g,
|
||
|
tplRe = /\{(\d+)\}/g,
|
||
|
modeRe = /^(\s?[\/>+~]\s?|\s|$)/,
|
||
|
tagTokenRe = /^(#)?([\w\-\*\|\\]+)/,
|
||
|
nthRe = /(\d*)n\+?(\d*)/,
|
||
|
nthRe2 = /\D/,
|
||
|
startIdRe = /^\s*#/,
|
||
|
// This is for IE MSXML which does not support expandos.
|
||
|
// IE runs the same speed using setAttribute, however FF slows way down
|
||
|
// and Safari completely fails so they need to continue to use expandos.
|
||
|
isIE = window.ActiveXObject ? true : false,
|
||
|
key = 30803,
|
||
|
longHex = /\\([0-9a-fA-F]{6})/g,
|
||
|
shortHex = /\\([0-9a-fA-F]{1,6})\s{0,1}/g,
|
||
|
nonHex = /\\([^0-9a-fA-F]{1})/g,
|
||
|
escapes = /\\/g,
|
||
|
num, hasEscapes,
|
||
|
// True if the browser supports the following syntax:
|
||
|
// document.getElementsByTagName('namespacePrefix:tagName')
|
||
|
supportsColonNsSeparator = (function () {
|
||
|
var xmlDoc,
|
||
|
xmlString = '<r><a:b xmlns:a="n"></a:b></r>';
|
||
|
|
||
|
if (window.DOMParser) {
|
||
|
xmlDoc = (new DOMParser()).parseFromString(xmlString, "application/xml");
|
||
|
} else {
|
||
|
xmlDoc = new ActiveXObject("Microsoft.XMLDOM");
|
||
|
xmlDoc.loadXML(xmlString);
|
||
|
}
|
||
|
|
||
|
return !!xmlDoc.getElementsByTagName('a:b').length;
|
||
|
})(),
|
||
|
|
||
|
// replaces a long hex regex match group with the appropriate ascii value
|
||
|
// $args indicate regex match pos
|
||
|
longHexToChar = function($0, $1) {
|
||
|
return String.fromCharCode(parseInt($1, 16));
|
||
|
},
|
||
|
|
||
|
// converts a shortHex regex match to the long form
|
||
|
shortToLongHex = function($0, $1) {
|
||
|
while ($1.length < 6) {
|
||
|
$1 = '0' + $1;
|
||
|
}
|
||
|
return '\\' + $1;
|
||
|
},
|
||
|
|
||
|
// converts a single char escape to long escape form
|
||
|
charToLongHex = function($0, $1) {
|
||
|
num = $1.charCodeAt(0).toString(16);
|
||
|
if (num.length === 1) {
|
||
|
num = '0' + num;
|
||
|
}
|
||
|
return '\\0000' + num;
|
||
|
},
|
||
|
|
||
|
// Un-escapes an input selector string. Assumes all escape sequences have been
|
||
|
// normalized to the css '\\0000##' 6-hex-digit style escape sequence :
|
||
|
// will not handle any other escape formats
|
||
|
unescapeCssSelector = function(selector) {
|
||
|
return (hasEscapes) ? selector.replace(longHex, longHexToChar) : selector;
|
||
|
},
|
||
|
|
||
|
// checks if the path has escaping & does any appropriate replacements
|
||
|
setupEscapes = function(path) {
|
||
|
hasEscapes = (path.indexOf('\\') > -1);
|
||
|
if (hasEscapes) {
|
||
|
path = path
|
||
|
.replace(shortHex, shortToLongHex)
|
||
|
.replace(nonHex, charToLongHex)
|
||
|
.replace(escapes, '\\\\'); // double the '\' for js compilation
|
||
|
}
|
||
|
return path;
|
||
|
};
|
||
|
|
||
|
// this eval is stop the compressor from
|
||
|
// renaming the variable to something shorter
|
||
|
eval("var batch = 30803, child, next, prev, byClassName;");
|
||
|
|
||
|
// Retrieve the child node from a particular
|
||
|
// parent at the specified index.
|
||
|
child = useChildrenCollection ?
|
||
|
function child(parent, index) {
|
||
|
return parent.children[index];
|
||
|
} :
|
||
|
function child(parent, index) {
|
||
|
var i = 0,
|
||
|
n = parent.firstChild;
|
||
|
while (n) {
|
||
|
if (n.nodeType == 1) {
|
||
|
if (++i == index) {
|
||
|
return n;
|
||
|
}
|
||
|
}
|
||
|
n = n.nextSibling;
|
||
|
}
|
||
|
return null;
|
||
|
};
|
||
|
|
||
|
// retrieve the next element node
|
||
|
next = useElementPointer ?
|
||
|
function(n) {
|
||
|
return n.nextElementSibling;
|
||
|
} :
|
||
|
function(n) {
|
||
|
while ((n = n.nextSibling) && n.nodeType != 1);
|
||
|
return n;
|
||
|
};
|
||
|
|
||
|
// retrieve the previous element node
|
||
|
prev = useElementPointer ?
|
||
|
function(n) {
|
||
|
return n.previousElementSibling;
|
||
|
} :
|
||
|
function(n) {
|
||
|
while ((n = n.previousSibling) && n.nodeType != 1);
|
||
|
return n;
|
||
|
};
|
||
|
|
||
|
// Mark each child node with a nodeIndex skipping and
|
||
|
// removing empty text nodes.
|
||
|
function children(parent) {
|
||
|
var n = parent.firstChild,
|
||
|
nodeIndex = -1,
|
||
|
nextNode;
|
||
|
|
||
|
while (n) {
|
||
|
nextNode = n.nextSibling;
|
||
|
// clean worthless empty nodes.
|
||
|
if (n.nodeType == 3 && !nonSpace.test(n.nodeValue)) {
|
||
|
parent.removeChild(n);
|
||
|
} else {
|
||
|
// add an expando nodeIndex
|
||
|
n.nodeIndex = ++nodeIndex;
|
||
|
}
|
||
|
n = nextNode;
|
||
|
}
|
||
|
return this;
|
||
|
}
|
||
|
|
||
|
// nodeSet - array of nodes
|
||
|
// cls - CSS Class
|
||
|
byClassName = useClassList ? // Use classList API where available: http://jsperf.com/classlist-vs-old-school-check/
|
||
|
function (nodeSet, cls) {
|
||
|
cls = unescapeCssSelector(cls);
|
||
|
if (!cls) {
|
||
|
return nodeSet;
|
||
|
}
|
||
|
var result = [], ri = -1,
|
||
|
i, ci, classList;
|
||
|
|
||
|
for (i = 0; ci = nodeSet[i]; i++) {
|
||
|
classList = ci.classList;
|
||
|
if (classList) {
|
||
|
if (classList.contains(cls)) {
|
||
|
result[++ri] = ci;
|
||
|
}
|
||
|
} else if ((' ' + ci.className + ' ').indexOf(cls) !== -1) {
|
||
|
// Some elements types (SVG) may not always have a classList
|
||
|
// in some browsers, so fallback to the old style here
|
||
|
result[++ri] = ci;
|
||
|
}
|
||
|
}
|
||
|
return result;
|
||
|
} :
|
||
|
function (nodeSet, cls) {
|
||
|
cls = unescapeCssSelector(cls);
|
||
|
if (!cls) {
|
||
|
return nodeSet;
|
||
|
}
|
||
|
var result = [], ri = -1,
|
||
|
i, ci;
|
||
|
|
||
|
for (i = 0; ci = nodeSet[i]; i++) {
|
||
|
if ((' ' + ci.className + ' ').indexOf(cls) !== -1) {
|
||
|
result[++ri] = ci;
|
||
|
}
|
||
|
}
|
||
|
return result;
|
||
|
};
|
||
|
|
||
|
function attrValue(n, attr) {
|
||
|
// if its an array, use the first node.
|
||
|
if (!n.tagName && typeof n.length != "undefined") {
|
||
|
n = n[0];
|
||
|
}
|
||
|
if (!n) {
|
||
|
return null;
|
||
|
}
|
||
|
|
||
|
if (attr == "for") {
|
||
|
return n.htmlFor;
|
||
|
}
|
||
|
if (attr == "class" || attr == "className") {
|
||
|
return n.className;
|
||
|
}
|
||
|
return n.getAttribute(attr) || n[attr];
|
||
|
|
||
|
}
|
||
|
|
||
|
// ns - nodes
|
||
|
// mode - false, /, >, +, ~
|
||
|
// tagName - defaults to "*"
|
||
|
function getNodes(ns, mode, tagName) {
|
||
|
var result = [], ri = -1, cs,
|
||
|
i, ni, j, ci, cn, utag, n, cj;
|
||
|
if (!ns) {
|
||
|
return result;
|
||
|
}
|
||
|
tagName = tagName.replace('|', ':') || "*";
|
||
|
// convert to array
|
||
|
if (typeof ns.getElementsByTagName != "undefined") {
|
||
|
ns = [ns];
|
||
|
}
|
||
|
|
||
|
// no mode specified, grab all elements by tagName
|
||
|
// at any depth
|
||
|
if (!mode) {
|
||
|
tagName = unescapeCssSelector(tagName);
|
||
|
if (!supportsColonNsSeparator && DQ.isXml(ns[0]) &&
|
||
|
tagName.indexOf(':') !== -1) {
|
||
|
// Some browsers (e.g. WebKit and Opera do not support the following syntax
|
||
|
// in xml documents: getElementsByTagName('ns:tagName'). To work around
|
||
|
// this, we remove the namespace prefix from the tagName, get the elements
|
||
|
// by tag name only, and then compare each element's tagName property to
|
||
|
// the tagName with namespace prefix attached to ensure that the tag is in
|
||
|
// the proper namespace.
|
||
|
for (i = 0; ni = ns[i]; i++) {
|
||
|
cs = ni.getElementsByTagName(tagName.split(':').pop());
|
||
|
for (j = 0; ci = cs[j]; j++) {
|
||
|
if (ci.tagName === tagName) {
|
||
|
result[++ri] = ci;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
} else {
|
||
|
for (i = 0; ni = ns[i]; i++) {
|
||
|
cs = ni.getElementsByTagName(tagName);
|
||
|
for (j = 0; ci = cs[j]; j++) {
|
||
|
result[++ri] = ci;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
// Direct Child mode (/ or >)
|
||
|
// E > F or E/F all direct children elements of E that have the tag
|
||
|
} else if (mode == "/" || mode == ">") {
|
||
|
utag = tagName.toUpperCase();
|
||
|
for (i = 0; ni = ns[i]; i++) {
|
||
|
cn = ni.childNodes;
|
||
|
for (j = 0; cj = cn[j]; j++) {
|
||
|
if (cj.nodeName == utag || cj.nodeName == tagName || tagName == '*') {
|
||
|
result[++ri] = cj;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
// Immediately Preceding mode (+)
|
||
|
// E + F all elements with the tag F that are immediately preceded by an element with the tag E
|
||
|
} else if (mode == "+") {
|
||
|
utag = tagName.toUpperCase();
|
||
|
for (i = 0; n = ns[i]; i++) {
|
||
|
while ((n = n.nextSibling) && n.nodeType != 1);
|
||
|
if (n && (n.nodeName == utag || n.nodeName == tagName || tagName == '*')) {
|
||
|
result[++ri] = n;
|
||
|
}
|
||
|
}
|
||
|
// Sibling mode (~)
|
||
|
// E ~ F all elements with the tag F that are preceded by a sibling element with the tag E
|
||
|
} else if (mode == "~") {
|
||
|
utag = tagName.toUpperCase();
|
||
|
for (i = 0; n = ns[i]; i++) {
|
||
|
while ((n = n.nextSibling)) {
|
||
|
if (n.nodeName == utag || n.nodeName == tagName || tagName == '*') {
|
||
|
result[++ri] = n;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
function concat(a, b) {
|
||
|
a.push.apply(a, b);
|
||
|
return a;
|
||
|
}
|
||
|
|
||
|
function byTag(cs, tagName) {
|
||
|
if (cs.tagName || cs === doc) {
|
||
|
cs = [cs];
|
||
|
}
|
||
|
if (!tagName) {
|
||
|
return cs;
|
||
|
}
|
||
|
var result = [], ri = -1,
|
||
|
i, ci;
|
||
|
tagName = tagName.toLowerCase();
|
||
|
for (i = 0; ci = cs[i]; i++) {
|
||
|
if (ci.nodeType == 1 && ci.tagName.toLowerCase() == tagName) {
|
||
|
result[++ri] = ci;
|
||
|
}
|
||
|
}
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
function byId(cs, id) {
|
||
|
id = unescapeCssSelector(id);
|
||
|
if (cs.tagName || cs === doc) {
|
||
|
cs = [cs];
|
||
|
}
|
||
|
if (!id) {
|
||
|
return cs;
|
||
|
}
|
||
|
var result = [], ri = -1,
|
||
|
i, ci;
|
||
|
for (i = 0; ci = cs[i]; i++) {
|
||
|
if (ci && ci.id == id) {
|
||
|
result[++ri] = ci;
|
||
|
return result;
|
||
|
}
|
||
|
}
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
// operators are =, !=, ^=, $=, *=, %=, |= and ~=
|
||
|
// custom can be "{"
|
||
|
function byAttribute(cs, attr, value, op, custom) {
|
||
|
var result = [],
|
||
|
ri = -1,
|
||
|
useGetStyle = custom == "{",
|
||
|
fn = DQ.operators[op],
|
||
|
a,
|
||
|
xml,
|
||
|
hasXml,
|
||
|
i, ci;
|
||
|
|
||
|
value = unescapeCssSelector(value);
|
||
|
|
||
|
for (i = 0; ci = cs[i]; i++) {
|
||
|
// skip non-element nodes.
|
||
|
if (ci.nodeType === 1) {
|
||
|
// only need to do this for the first node
|
||
|
if (!hasXml) {
|
||
|
xml = DQ.isXml(ci);
|
||
|
hasXml = true;
|
||
|
}
|
||
|
|
||
|
// we only need to change the property names if we're dealing with html nodes, not XML
|
||
|
if (!xml) {
|
||
|
if (useGetStyle) {
|
||
|
a = DQ.getStyle(ci, attr);
|
||
|
} else if (attr == "class" || attr == "className") {
|
||
|
a = ci.className;
|
||
|
} else if (attr == "for") {
|
||
|
a = ci.htmlFor;
|
||
|
} else if (attr == "href") {
|
||
|
// getAttribute href bug
|
||
|
// http://www.glennjones.net/Post/809/getAttributehrefbug.htm
|
||
|
a = ci.getAttribute("href", 2);
|
||
|
} else {
|
||
|
a = ci.getAttribute(attr);
|
||
|
}
|
||
|
} else {
|
||
|
a = ci.getAttribute(attr);
|
||
|
}
|
||
|
if ((fn && fn(a, value)) || (!fn && a)) {
|
||
|
result[++ri] = ci;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
function byPseudo(cs, name, value) {
|
||
|
value = unescapeCssSelector(value);
|
||
|
return DQ.pseudos[name](cs, value);
|
||
|
}
|
||
|
|
||
|
function nodupIEXml(cs) {
|
||
|
var d = ++key,
|
||
|
r,
|
||
|
i, len, c;
|
||
|
cs[0].setAttribute("_nodup", d);
|
||
|
r = [cs[0]];
|
||
|
for (i = 1, len = cs.length; i < len; i++) {
|
||
|
c = cs[i];
|
||
|
if (!c.getAttribute("_nodup") != d) {
|
||
|
c.setAttribute("_nodup", d);
|
||
|
r[r.length] = c;
|
||
|
}
|
||
|
}
|
||
|
for (i = 0, len = cs.length; i < len; i++) {
|
||
|
cs[i].removeAttribute("_nodup");
|
||
|
}
|
||
|
return r;
|
||
|
}
|
||
|
|
||
|
function nodup(cs) {
|
||
|
if (!cs) {
|
||
|
return [];
|
||
|
}
|
||
|
var len = cs.length, c, i, r = cs, cj, ri = -1, d, j;
|
||
|
if (!len || typeof cs.nodeType != "undefined" || len == 1) {
|
||
|
return cs;
|
||
|
}
|
||
|
if (isIE && typeof cs[0].selectSingleNode != "undefined") {
|
||
|
return nodupIEXml(cs);
|
||
|
}
|
||
|
d = ++key;
|
||
|
cs[0]._nodup = d;
|
||
|
for (i = 1; c = cs[i]; i++) {
|
||
|
if (c._nodup != d) {
|
||
|
c._nodup = d;
|
||
|
} else {
|
||
|
r = [];
|
||
|
for (j = 0; j < i; j++) {
|
||
|
r[++ri] = cs[j];
|
||
|
}
|
||
|
for (j = i + 1; cj = cs[j]; j++) {
|
||
|
if (cj._nodup != d) {
|
||
|
cj._nodup = d;
|
||
|
r[++ri] = cj;
|
||
|
}
|
||
|
}
|
||
|
return r;
|
||
|
}
|
||
|
}
|
||
|
return r;
|
||
|
}
|
||
|
|
||
|
function quickDiffIEXml(c1, c2) {
|
||
|
var d = ++key,
|
||
|
r = [],
|
||
|
i, len;
|
||
|
for (i = 0, len = c1.length; i < len; i++) {
|
||
|
c1[i].setAttribute("_qdiff", d);
|
||
|
}
|
||
|
for (i = 0, len = c2.length; i < len; i++) {
|
||
|
if (c2[i].getAttribute("_qdiff") != d) {
|
||
|
r[r.length] = c2[i];
|
||
|
}
|
||
|
}
|
||
|
for (i = 0, len = c1.length; i < len; i++) {
|
||
|
c1[i].removeAttribute("_qdiff");
|
||
|
}
|
||
|
return r;
|
||
|
}
|
||
|
|
||
|
function quickDiff(c1, c2) {
|
||
|
var len1 = c1.length,
|
||
|
d = ++key,
|
||
|
r = [],
|
||
|
i, len;
|
||
|
if (!len1) {
|
||
|
return c2;
|
||
|
}
|
||
|
if (isIE && typeof c1[0].selectSingleNode != "undefined") {
|
||
|
return quickDiffIEXml(c1, c2);
|
||
|
}
|
||
|
for (i = 0; i < len1; i++) {
|
||
|
c1[i]._qdiff = d;
|
||
|
}
|
||
|
for (i = 0, len = c2.length; i < len; i++) {
|
||
|
if (c2[i]._qdiff != d) {
|
||
|
r[r.length] = c2[i];
|
||
|
}
|
||
|
}
|
||
|
return r;
|
||
|
}
|
||
|
|
||
|
function quickId(ns, mode, root, id) {
|
||
|
if (ns == root) {
|
||
|
id = unescapeCssSelector(id);
|
||
|
var d = root.ownerDocument || root;
|
||
|
return d.getElementById(id);
|
||
|
}
|
||
|
ns = getNodes(ns, mode, "*");
|
||
|
return byId(ns, id);
|
||
|
}
|
||
|
|
||
|
return {
|
||
|
singleton: true,
|
||
|
|
||
|
alternateClassName: [
|
||
|
'Ext.core.DomQuery',
|
||
|
'Ext.DomQuery'
|
||
|
],
|
||
|
|
||
|
requires: [
|
||
|
'Ext.dom.Helper',
|
||
|
'Ext.util.Operators'
|
||
|
],
|
||
|
|
||
|
_init: function() {
|
||
|
DQ = this;
|
||
|
DQ.operators = Ext.Object.chain(Ext.util.Operators); // can capture now
|
||
|
DQ._cache = cache = new Ext.util.LruCache({
|
||
|
maxSize: 200
|
||
|
});
|
||
|
DQ._valueCache = valueCache = new Ext.util.LruCache({
|
||
|
maxSize: 200
|
||
|
});
|
||
|
DQ._simpleCache = simpleCache = new Ext.util.LruCache({
|
||
|
maxSize: 200
|
||
|
});
|
||
|
},
|
||
|
|
||
|
clearCache: function () {
|
||
|
cache.clear();
|
||
|
valueCache.clear();
|
||
|
simpleCache.clear();
|
||
|
},
|
||
|
|
||
|
getStyle: function(el, name) {
|
||
|
return Ext.fly(el, '_DomQuery').getStyle(name);
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Compiles a selector/xpath query into a reusable function. The returned function
|
||
|
* takes one parameter "root" (optional), which is the context node from where the query should start.
|
||
|
* @param {String} selector The selector/xpath query
|
||
|
* @param {String} [type="select"] Either "select" or "simple" for a simple selector match
|
||
|
* @return {Function}
|
||
|
*/
|
||
|
compile: function(path, type) {
|
||
|
type = type || "select";
|
||
|
|
||
|
// setup fn preamble
|
||
|
var fn = ["var f = function(root) {\n var mode; ++batch; var n = root || document;\n"],
|
||
|
lastPath,
|
||
|
matchers = DQ.matchers,
|
||
|
matchersLn = matchers.length,
|
||
|
modeMatch,
|
||
|
// accept leading mode switch
|
||
|
lmode = path.match(modeRe),
|
||
|
tokenMatch, matched, j, t, m;
|
||
|
|
||
|
path = setupEscapes(path);
|
||
|
|
||
|
if (lmode && lmode[1]) {
|
||
|
fn[fn.length] = 'mode="' + lmode[1].replace(trimRe, "") + '";';
|
||
|
path = path.replace(lmode[1], "");
|
||
|
}
|
||
|
|
||
|
// strip leading slashes
|
||
|
while (path.substr(0, 1) == "/") {
|
||
|
path = path.substr(1);
|
||
|
}
|
||
|
|
||
|
while (path && lastPath != path) {
|
||
|
lastPath = path;
|
||
|
tokenMatch = path.match(tagTokenRe);
|
||
|
if (type == "select") {
|
||
|
if (tokenMatch) {
|
||
|
// ID Selector
|
||
|
if (tokenMatch[1] == "#") {
|
||
|
fn[fn.length] = 'n = quickId(n, mode, root, "' + tokenMatch[2] + '");';
|
||
|
} else {
|
||
|
fn[fn.length] = 'n = getNodes(n, mode, "' + tokenMatch[2] + '");';
|
||
|
}
|
||
|
path = path.replace(tokenMatch[0], "");
|
||
|
} else if (path.substr(0, 1) != '@') {
|
||
|
fn[fn.length] = 'n = getNodes(n, mode, "*");';
|
||
|
}
|
||
|
// type of "simple"
|
||
|
} else {
|
||
|
if (tokenMatch) {
|
||
|
if (tokenMatch[1] == "#") {
|
||
|
fn[fn.length] = 'n = byId(n, "' + tokenMatch[2] + '");';
|
||
|
} else {
|
||
|
fn[fn.length] = 'n = byTag(n, "' + tokenMatch[2] + '");';
|
||
|
}
|
||
|
path = path.replace(tokenMatch[0], "");
|
||
|
}
|
||
|
}
|
||
|
while (!(modeMatch = path.match(modeRe))) {
|
||
|
matched = false;
|
||
|
for (j = 0; j < matchersLn; j++) {
|
||
|
t = matchers[j];
|
||
|
m = path.match(t.re);
|
||
|
if (m) {
|
||
|
fn[fn.length] = t.select.replace(tplRe, function(x, i) {
|
||
|
return m[i];
|
||
|
});
|
||
|
path = path.replace(m[0], "");
|
||
|
matched = true;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
// prevent infinite loop on bad selector
|
||
|
if (!matched) {
|
||
|
Ext.Error.raise({
|
||
|
sourceClass:'Ext.DomQuery',
|
||
|
sourceMethod:'compile',
|
||
|
msg:'Error parsing selector. Parsing failed at "' + path + '"'
|
||
|
});
|
||
|
}
|
||
|
}
|
||
|
if (modeMatch[1]) {
|
||
|
fn[fn.length] = 'mode="' + modeMatch[1].replace(trimRe, "") + '";';
|
||
|
path = path.replace(modeMatch[1], "");
|
||
|
}
|
||
|
}
|
||
|
// close fn out
|
||
|
fn[fn.length] = "return nodup(n);\n}";
|
||
|
|
||
|
// eval fn and return it
|
||
|
eval(fn.join(""));
|
||
|
return f;
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Selects an array of DOM nodes using JavaScript-only implementation.
|
||
|
*
|
||
|
* Use {@link #select} to take advantage of browsers built-in support for CSS selectors.
|
||
|
* @param {String} selector The selector/xpath query (can be a comma separated list of selectors)
|
||
|
* @param {HTMLElement/String} [root=document] The start of the query.
|
||
|
* @return {HTMLElement[]} An Array of DOM elements which match the selector. If there are
|
||
|
* no matches, and empty Array is returned.
|
||
|
*/
|
||
|
jsSelect: function(path, root, type) {
|
||
|
// set root to doc if not specified.
|
||
|
root = root || doc;
|
||
|
|
||
|
if (typeof root == "string") {
|
||
|
root = doc.getElementById(root);
|
||
|
}
|
||
|
var paths = Ext.splitAndUnescape(path, ","),
|
||
|
results = [],
|
||
|
query, i, len, subPath, result;
|
||
|
|
||
|
// loop over each selector
|
||
|
for (i = 0, len = paths.length; i < len; i++) {
|
||
|
subPath = paths[i].replace(trimRe, "");
|
||
|
// compile and place in cache
|
||
|
query = cache.get(subPath);
|
||
|
if (!query) {
|
||
|
// When we compile, escaping is handled inside the compile method
|
||
|
query = DQ.compile(subPath, type);
|
||
|
if (!query) {
|
||
|
Ext.Error.raise({
|
||
|
sourceClass:'Ext.DomQuery',
|
||
|
sourceMethod:'jsSelect',
|
||
|
msg:subPath + ' is not a valid selector'
|
||
|
});
|
||
|
}
|
||
|
cache.add(subPath, query);
|
||
|
} else {
|
||
|
// If we've already compiled, we still need to check if the
|
||
|
// selector has escaping and setup the appropriate flags
|
||
|
setupEscapes(subPath);
|
||
|
}
|
||
|
result = query(root);
|
||
|
if (result && result !== doc) {
|
||
|
results = results.concat(result);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// if there were multiple selectors, make sure dups
|
||
|
// are eliminated
|
||
|
if (paths.length > 1) {
|
||
|
return nodup(results);
|
||
|
}
|
||
|
return results;
|
||
|
},
|
||
|
|
||
|
isXml: function(el) {
|
||
|
var docEl = (el ? el.ownerDocument || el : 0).documentElement;
|
||
|
return docEl ? docEl.nodeName !== "HTML" : false;
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Selects an array of DOM nodes by CSS/XPath selector.
|
||
|
*
|
||
|
* Uses [document.querySelectorAll][0] if browser supports that, otherwise falls back to
|
||
|
* {@link Ext.dom.Query#jsSelect} to do the work.
|
||
|
*
|
||
|
* [0]: https://developer.mozilla.org/en/DOM/document.querySelectorAll
|
||
|
*
|
||
|
* @param {String} path The selector/xpath query
|
||
|
* @param {HTMLElement} [root=document] The start of the query.
|
||
|
* @return {HTMLElement[]} An array of DOM elements (not a NodeList as returned by `querySelectorAll`).
|
||
|
* @param {String} [type="select"] Either "select" or "simple" for a simple selector match (only valid when
|
||
|
* used when the call is deferred to the jsSelect method)
|
||
|
* @param {Boolean} [single] Pass `true` to select only the first matching node using `document.querySelector` (where available)
|
||
|
* @method
|
||
|
*/
|
||
|
select: doc.querySelectorAll ? function(path, root, type, single) {
|
||
|
root = root || doc;
|
||
|
if (!DQ.isXml(root)) {
|
||
|
try {
|
||
|
/*
|
||
|
* This checking here is to "fix" the behaviour of querySelectorAll
|
||
|
* for non root document queries. The way qsa works is intentional,
|
||
|
* however it's definitely not the expected way it should work.
|
||
|
* When descendant selectors are used, only the lowest selector must be inside the root!
|
||
|
* More info: http://ejohn.org/blog/thoughts-on-queryselectorall/
|
||
|
* So we create a descendant selector by prepending the root's ID, and query the parent node.
|
||
|
* UNLESS the root has no parent in which qsa will work perfectly.
|
||
|
*
|
||
|
* We only modify the path for single selectors (ie, no multiples),
|
||
|
* without a full parser it makes it difficult to do this correctly.
|
||
|
*/
|
||
|
if (root.parentNode && (root.nodeType !== 9) && path.indexOf(',') === -1 && !startIdRe.test(path)) {
|
||
|
path = Ext.makeIdSelector(Ext.id(root)) + ' ' + path;
|
||
|
root = root.parentNode;
|
||
|
}
|
||
|
return single ? [ root.querySelector(path) ]
|
||
|
: Ext.Array.toArray(root.querySelectorAll(path));
|
||
|
}
|
||
|
catch (e) {
|
||
|
}
|
||
|
}
|
||
|
return DQ.jsSelect.call(this, path, root, type);
|
||
|
} : function(path, root, type) {
|
||
|
return DQ.jsSelect.call(this, path, root, type);
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Selects a single element.
|
||
|
* @param {String} selector The selector/xpath query
|
||
|
* @param {HTMLElement} [root=document] The start of the query.
|
||
|
* @return {HTMLElement} The DOM element which matched the selector.
|
||
|
*/
|
||
|
selectNode: function(path, root){
|
||
|
return Ext.DomQuery.select(path, root, null, true)[0];
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Selects the value of a node, optionally replacing null with the defaultValue.
|
||
|
* @param {String} selector The selector/xpath query
|
||
|
* @param {HTMLElement} [root=document] The start of the query.
|
||
|
* @param {String} [defaultValue] When specified, this is return as empty value.
|
||
|
* @return {String}
|
||
|
*/
|
||
|
selectValue: function(path, root, defaultValue) {
|
||
|
path = path.replace(trimRe, "");
|
||
|
var query = valueCache.get(path),
|
||
|
n, v;
|
||
|
|
||
|
if (!query) {
|
||
|
query = DQ.compile(path, "select");
|
||
|
valueCache.add(path, query);
|
||
|
} else {
|
||
|
setupEscapes(path);
|
||
|
}
|
||
|
|
||
|
n = query(root);
|
||
|
return DQ.getNodeValue(n[0] ? n[0] : n);
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Get the text value for a node, optionally replacing null with the defaultValue.
|
||
|
* @param {Object} The node
|
||
|
* @param {String} [defaultValue] When specified, this is return as empty value.
|
||
|
* @return {String} The value
|
||
|
*/
|
||
|
getNodeValue: function(node, defaultValue) {
|
||
|
// overcome a limitation of maximum textnode size
|
||
|
// Rumored to potentially crash IE6 but has not been confirmed.
|
||
|
// http://reference.sitepoint.com/javascript/Node/normalize
|
||
|
// https://developer.mozilla.org/En/DOM/Node.normalize
|
||
|
if (typeof node.normalize == 'function') {
|
||
|
node.normalize();
|
||
|
}
|
||
|
|
||
|
var v = (node && node.firstChild ? node.firstChild.nodeValue : null);
|
||
|
return ((v === null || v === undefined || v === '') ? defaultValue : v);
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Selects the value of a node, parsing integers and floats.
|
||
|
* Returns the defaultValue, or 0 if none is specified.
|
||
|
* @param {String} selector The selector/xpath query
|
||
|
* @param {HTMLElement} [root=document] The start of the query.
|
||
|
* @param {Number} [defaultValue] When specified, this is return as empty value.
|
||
|
* @return {Number}
|
||
|
*/
|
||
|
selectNumber: function(path, root, defaultValue) {
|
||
|
var v = DQ.selectValue(path, root, defaultValue || 0);
|
||
|
return parseFloat(v);
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Returns true if the passed element(s) match the passed simple selector
|
||
|
* @param {String/HTMLElement/HTMLElement[]} el An element id, element or array of elements
|
||
|
* @param {String} selector The simple selector to test
|
||
|
* @return {Boolean}
|
||
|
*/
|
||
|
is: function(el, ss) {
|
||
|
if (typeof el == "string") {
|
||
|
el = doc.getElementById(el);
|
||
|
}
|
||
|
var isArray = Ext.isArray(el),
|
||
|
result = DQ.filter(isArray ? el : [el], ss);
|
||
|
return isArray ? (result.length == el.length) : (result.length > 0);
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Filters an array of elements to only include matches of a simple selector
|
||
|
* @param {HTMLElement[]} el An array of elements to filter
|
||
|
* @param {String} selector The simple selector to test
|
||
|
* @param {Boolean} nonMatches If true, it returns the elements that DON'T match the selector instead of the
|
||
|
* ones that match
|
||
|
* @return {HTMLElement[]} An Array of DOM elements which match the selector. If there are no matches, and empty
|
||
|
* Array is returned.
|
||
|
*/
|
||
|
filter: function(els, ss, nonMatches) {
|
||
|
ss = ss.replace(trimRe, "");
|
||
|
var query = simpleCache.get(ss),
|
||
|
result;
|
||
|
|
||
|
if (!query) {
|
||
|
query = DQ.compile(ss, "simple");
|
||
|
simpleCache.add(ss, query);
|
||
|
} else {
|
||
|
setupEscapes(ss);
|
||
|
}
|
||
|
|
||
|
result = query(els);
|
||
|
return nonMatches ? quickDiff(result, els) : result;
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Collection of matching regular expressions and code snippets.
|
||
|
* Each capture group within `()` will be replace the `{}` in the select
|
||
|
* statement as specified by their index.
|
||
|
*/
|
||
|
matchers: [{
|
||
|
re: /^\.([\w\-\\]+)/,
|
||
|
select: useClassList ? 'n = byClassName(n, "{1}");' : 'n = byClassName(n, " {1} ");'
|
||
|
}, {
|
||
|
re: /^\:([\w\-]+)(?:\(((?:[^\s>\/]*|.*?))\))?/,
|
||
|
select: 'n = byPseudo(n, "{1}", "{2}");'
|
||
|
}, {
|
||
|
re: /^(?:([\[\{])(?:@)?([\w\-]+)\s?(?:(=|.=)\s?['"]?(.*?)["']?)?[\]\}])/,
|
||
|
select: 'n = byAttribute(n, "{2}", "{4}", "{3}", "{1}");'
|
||
|
}, {
|
||
|
re: /^#([\w\-\\]+)/,
|
||
|
select: 'n = byId(n, "{1}");'
|
||
|
}, {
|
||
|
re: /^@([\w\-\.]+)/,
|
||
|
select: 'return {firstChild:{nodeValue:attrValue(n, "{1}")}};'
|
||
|
}],
|
||
|
|
||
|
/**
|
||
|
* Collection of operator comparison functions.
|
||
|
* The default operators are `=`, `!=`, `^=`, `$=`, `*=`, `%=`, `|=` and `~=`.
|
||
|
*
|
||
|
* New operators can be added as long as the match the format *c*`=` where *c*
|
||
|
* is any character other than space, `>`, or `<`.
|
||
|
*
|
||
|
* Operator functions are passed the following parameters:
|
||
|
*
|
||
|
* * `propValue` : The property value to test.
|
||
|
* * `compareTo` : The value to compare to.
|
||
|
*
|
||
|
* @property {Object} operators
|
||
|
*/
|
||
|
|
||
|
/**
|
||
|
* Object hash of "pseudo class" filter functions which are used when filtering selections.
|
||
|
* Each function is passed two parameters:
|
||
|
*
|
||
|
* - **c** : Array
|
||
|
* An Array of DOM elements to filter.
|
||
|
*
|
||
|
* - **v** : String
|
||
|
* The argument (if any) supplied in the selector.
|
||
|
*
|
||
|
* A filter function returns an Array of DOM elements which conform to the pseudo class.
|
||
|
* In addition to the provided pseudo classes listed above such as `first-child` and `nth-child`,
|
||
|
* developers may add additional, custom psuedo class filters to select elements according to application-specific requirements.
|
||
|
*
|
||
|
* For example, to filter `a` elements to only return links to __external__ resources:
|
||
|
*
|
||
|
* Ext.DomQuery.pseudos.external = function(c, v) {
|
||
|
* var r = [], ri = -1;
|
||
|
* for(var i = 0, ci; ci = c[i]; i++) {
|
||
|
* // Include in result set only if it's a link to an external resource
|
||
|
* if (ci.hostname != location.hostname) {
|
||
|
* r[++ri] = ci;
|
||
|
* }
|
||
|
* }
|
||
|
* return r;
|
||
|
* };
|
||
|
*
|
||
|
* Then external links could be gathered with the following statement:
|
||
|
*
|
||
|
* var externalLinks = Ext.select("a:external");
|
||
|
*/
|
||
|
pseudos: {
|
||
|
"first-child": function(c) {
|
||
|
var r = [], ri = -1, n,
|
||
|
i, ci;
|
||
|
for (i = 0; (ci = n = c[i]); i++) {
|
||
|
while ((n = n.previousSibling) && n.nodeType != 1);
|
||
|
if (!n) {
|
||
|
r[++ri] = ci;
|
||
|
}
|
||
|
}
|
||
|
return r;
|
||
|
},
|
||
|
|
||
|
"last-child": function(c) {
|
||
|
var r = [], ri = -1, n,
|
||
|
i, ci;
|
||
|
for (i = 0; (ci = n = c[i]); i++) {
|
||
|
while ((n = n.nextSibling) && n.nodeType != 1);
|
||
|
if (!n) {
|
||
|
r[++ri] = ci;
|
||
|
}
|
||
|
}
|
||
|
return r;
|
||
|
},
|
||
|
|
||
|
"nth-child": function(c, a) {
|
||
|
var r = [], ri = -1,
|
||
|
m = nthRe.exec(a == "even" && "2n" || a == "odd" && "2n+1" || !nthRe2.test(a) && "n+" + a || a),
|
||
|
f = (m[1] || 1) - 0, l = m[2] - 0,
|
||
|
i, n, j, cn, pn;
|
||
|
for (i = 0; n = c[i]; i++) {
|
||
|
pn = n.parentNode;
|
||
|
if (batch != pn._batch) {
|
||
|
j = 0;
|
||
|
for (cn = pn.firstChild; cn; cn = cn.nextSibling) {
|
||
|
if (cn.nodeType == 1) {
|
||
|
cn.nodeIndex = ++j;
|
||
|
}
|
||
|
}
|
||
|
pn._batch = batch;
|
||
|
}
|
||
|
if (f == 1) {
|
||
|
if (l === 0 || n.nodeIndex == l) {
|
||
|
r[++ri] = n;
|
||
|
}
|
||
|
} else if ((n.nodeIndex + l) % f === 0) {
|
||
|
r[++ri] = n;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return r;
|
||
|
},
|
||
|
|
||
|
"only-child": function(c) {
|
||
|
var r = [], ri = -1,
|
||
|
i, ci;
|
||
|
for (i = 0; ci = c[i]; i++) {
|
||
|
if (!prev(ci) && !next(ci)) {
|
||
|
r[++ri] = ci;
|
||
|
}
|
||
|
}
|
||
|
return r;
|
||
|
},
|
||
|
|
||
|
"empty": function(c) {
|
||
|
var r = [], ri = -1,
|
||
|
i, ci, cns, j, cn, empty;
|
||
|
for (i = 0; ci = c[i]; i++) {
|
||
|
cns = ci.childNodes;
|
||
|
j = 0;
|
||
|
empty = true;
|
||
|
while (cn = cns[j]) {
|
||
|
++j;
|
||
|
if (cn.nodeType == 1 || cn.nodeType == 3) {
|
||
|
empty = false;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
if (empty) {
|
||
|
r[++ri] = ci;
|
||
|
}
|
||
|
}
|
||
|
return r;
|
||
|
},
|
||
|
|
||
|
"contains": function(c, v) {
|
||
|
var r = [], ri = -1,
|
||
|
i, ci;
|
||
|
for (i = 0; ci = c[i]; i++) {
|
||
|
if ((ci.textContent || ci.innerText || ci.text || '').indexOf(v) != -1) {
|
||
|
r[++ri] = ci;
|
||
|
}
|
||
|
}
|
||
|
return r;
|
||
|
},
|
||
|
|
||
|
"nodeValue": function(c, v) {
|
||
|
var r = [], ri = -1,
|
||
|
i, ci;
|
||
|
for (i = 0; ci = c[i]; i++) {
|
||
|
if (ci.firstChild && ci.firstChild.nodeValue == v) {
|
||
|
r[++ri] = ci;
|
||
|
}
|
||
|
}
|
||
|
return r;
|
||
|
},
|
||
|
|
||
|
"checked": function(c) {
|
||
|
var r = [], ri = -1,
|
||
|
i, ci;
|
||
|
for (i = 0; ci = c[i]; i++) {
|
||
|
if (ci.checked === true) {
|
||
|
r[++ri] = ci;
|
||
|
}
|
||
|
}
|
||
|
return r;
|
||
|
},
|
||
|
|
||
|
"not": function(c, ss) {
|
||
|
return DQ.filter(c, ss, true);
|
||
|
},
|
||
|
|
||
|
"any": function(c, selectors) {
|
||
|
var ss = selectors.split('|'),
|
||
|
r = [], ri = -1, s,
|
||
|
i, ci, j;
|
||
|
for (i = 0; ci = c[i]; i++) {
|
||
|
for (j = 0; s = ss[j]; j++) {
|
||
|
if (DQ.is(ci, s)) {
|
||
|
r[++ri] = ci;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return r;
|
||
|
},
|
||
|
|
||
|
"odd": function(c) {
|
||
|
return this["nth-child"](c, "odd");
|
||
|
},
|
||
|
|
||
|
"even": function(c) {
|
||
|
return this["nth-child"](c, "even");
|
||
|
},
|
||
|
|
||
|
"nth": function(c, a) {
|
||
|
return c[a - 1] || [];
|
||
|
},
|
||
|
|
||
|
"first": function(c) {
|
||
|
return c[0] || [];
|
||
|
},
|
||
|
|
||
|
"last": function(c) {
|
||
|
return c[c.length - 1] || [];
|
||
|
},
|
||
|
|
||
|
"has": function(c, ss) {
|
||
|
var s = DQ.select,
|
||
|
r = [], ri = -1,
|
||
|
i, ci;
|
||
|
for (i = 0; ci = c[i]; i++) {
|
||
|
if (s(ss, ci).length > 0) {
|
||
|
r[++ri] = ci;
|
||
|
}
|
||
|
}
|
||
|
return r;
|
||
|
},
|
||
|
|
||
|
"next": function(c, ss) {
|
||
|
var is = DQ.is,
|
||
|
r = [], ri = -1,
|
||
|
i, ci, n;
|
||
|
for (i = 0; ci = c[i]; i++) {
|
||
|
n = next(ci);
|
||
|
if (n && is(n, ss)) {
|
||
|
r[++ri] = ci;
|
||
|
}
|
||
|
}
|
||
|
return r;
|
||
|
},
|
||
|
|
||
|
"prev": function(c, ss) {
|
||
|
var is = DQ.is,
|
||
|
r = [], ri = -1,
|
||
|
i, ci, n;
|
||
|
for (i = 0; ci = c[i]; i++) {
|
||
|
n = prev(ci);
|
||
|
if (n && is(n, ss)) {
|
||
|
r[++ri] = ci;
|
||
|
}
|
||
|
}
|
||
|
return r;
|
||
|
},
|
||
|
|
||
|
focusable: function(candidates) {
|
||
|
var len = candidates.length,
|
||
|
results = [],
|
||
|
i = 0,
|
||
|
c;
|
||
|
|
||
|
for (; i < len; i++) {
|
||
|
c = candidates[i];
|
||
|
if (Ext.fly(c, '_DomQuery').isFocusable()) {
|
||
|
results.push(c);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return results;
|
||
|
},
|
||
|
|
||
|
visible: function(candidates, deep) {
|
||
|
var len = candidates.length,
|
||
|
results = [],
|
||
|
i = 0,
|
||
|
c;
|
||
|
|
||
|
for (; i < len; i++) {
|
||
|
c = candidates[i];
|
||
|
if (Ext.fly(c, '_DomQuery').isVisible(deep)) {
|
||
|
results.push(c);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return results;
|
||
|
},
|
||
|
|
||
|
isScrolled: function(c) {
|
||
|
var r = [], ri = -1,
|
||
|
i, ci, s;
|
||
|
for (i = 0; ci = c[i]; i++) {
|
||
|
s = Ext.fly(ci, '_DomQuery').getScroll();
|
||
|
if (s.top > 0 || s.left > 0) {
|
||
|
r[++ri] = ci;
|
||
|
}
|
||
|
}
|
||
|
return r;
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
}, function() {
|
||
|
this._init();
|
||
|
});
|