/* pseudo selectors --- they are available in two forms: * filters called when the selector is compiled and return a function that needs to return next() * pseudos get called on execution they need to return a boolean */ var getNCheck = require("nth-check"); var BaseFuncs = require("boolbase"); var attributes = require("./attributes.js"); var trueFunc = BaseFuncs.trueFunc; var falseFunc = BaseFuncs.falseFunc; var checkAttrib = attributes.rules.equals; function getAttribFunc(name, value) { var data = { name: name, value: value }; return function attribFunc(next, rule, options) { return checkAttrib(next, data, options); }; } function getChildFunc(next, adapter) { return function(elem) { return !!adapter.getParent(elem) && next(elem); }; } var filters = { contains: function(next, text, options) { var adapter = options.adapter; return function contains(elem) { return next(elem) && adapter.getText(elem).indexOf(text) >= 0; }; }, icontains: function(next, text, options) { var itext = text.toLowerCase(); var adapter = options.adapter; return function icontains(elem) { return ( next(elem) && adapter .getText(elem) .toLowerCase() .indexOf(itext) >= 0 ); }; }, //location specific methods "nth-child": function(next, rule, options) { var func = getNCheck(rule); var adapter = options.adapter; if (func === falseFunc) return func; if (func === trueFunc) return getChildFunc(next, adapter); return function nthChild(elem) { var siblings = adapter.getSiblings(elem); for (var i = 0, pos = 0; i < siblings.length; i++) { if (adapter.isTag(siblings[i])) { if (siblings[i] === elem) break; else pos++; } } return func(pos) && next(elem); }; }, "nth-last-child": function(next, rule, options) { var func = getNCheck(rule); var adapter = options.adapter; if (func === falseFunc) return func; if (func === trueFunc) return getChildFunc(next, adapter); return function nthLastChild(elem) { var siblings = adapter.getSiblings(elem); for (var pos = 0, i = siblings.length - 1; i >= 0; i--) { if (adapter.isTag(siblings[i])) { if (siblings[i] === elem) break; else pos++; } } return func(pos) && next(elem); }; }, "nth-of-type": function(next, rule, options) { var func = getNCheck(rule); var adapter = options.adapter; if (func === falseFunc) return func; if (func === trueFunc) return getChildFunc(next, adapter); return function nthOfType(elem) { var siblings = adapter.getSiblings(elem); for (var pos = 0, i = 0; i < siblings.length; i++) { if (adapter.isTag(siblings[i])) { if (siblings[i] === elem) break; if (adapter.getName(siblings[i]) === adapter.getName(elem)) pos++; } } return func(pos) && next(elem); }; }, "nth-last-of-type": function(next, rule, options) { var func = getNCheck(rule); var adapter = options.adapter; if (func === falseFunc) return func; if (func === trueFunc) return getChildFunc(next, adapter); return function nthLastOfType(elem) { var siblings = adapter.getSiblings(elem); for (var pos = 0, i = siblings.length - 1; i >= 0; i--) { if (adapter.isTag(siblings[i])) { if (siblings[i] === elem) break; if (adapter.getName(siblings[i]) === adapter.getName(elem)) pos++; } } return func(pos) && next(elem); }; }, //TODO determine the actual root element root: function(next, rule, options) { var adapter = options.adapter; return function(elem) { return !adapter.getParent(elem) && next(elem); }; }, scope: function(next, rule, options, context) { var adapter = options.adapter; if (!context || context.length === 0) { //equivalent to :root return filters.root(next, rule, options); } function equals(a, b) { if (typeof adapter.equals === "function") return adapter.equals(a, b); return a === b; } if (context.length === 1) { //NOTE: can't be unpacked, as :has uses this for side-effects return function(elem) { return equals(context[0], elem) && next(elem); }; } return function(elem) { return context.indexOf(elem) >= 0 && next(elem); }; }, //jQuery extensions (others follow as pseudos) checkbox: getAttribFunc("type", "checkbox"), file: getAttribFunc("type", "file"), password: getAttribFunc("type", "password"), radio: getAttribFunc("type", "radio"), reset: getAttribFunc("type", "reset"), image: getAttribFunc("type", "image"), submit: getAttribFunc("type", "submit"), //dynamic state pseudos. These depend on optional Adapter methods. hover: function(next, rule, options) { var adapter = options.adapter; if (typeof adapter.isHovered === 'function') { return function hover(elem) { return next(elem) && adapter.isHovered(elem); }; } return falseFunc; }, visited: function(next, rule, options) { var adapter = options.adapter; if (typeof adapter.isVisited === 'function') { return function visited(elem) { return next(elem) && adapter.isVisited(elem); }; } return falseFunc; }, active: function(next, rule, options) { var adapter = options.adapter; if (typeof adapter.isActive === 'function') { return function active(elem) { return next(elem) && adapter.isActive(elem); }; } return falseFunc; } }; //helper methods function getFirstElement(elems, adapter) { for (var i = 0; elems && i < elems.length; i++) { if (adapter.isTag(elems[i])) return elems[i]; } } //while filters are precompiled, pseudos get called when they are needed var pseudos = { empty: function(elem, adapter) { return !adapter.getChildren(elem).some(function(elem) { return adapter.isTag(elem) || elem.type === "text"; }); }, "first-child": function(elem, adapter) { return getFirstElement(adapter.getSiblings(elem), adapter) === elem; }, "last-child": function(elem, adapter) { var siblings = adapter.getSiblings(elem); for (var i = siblings.length - 1; i >= 0; i--) { if (siblings[i] === elem) return true; if (adapter.isTag(siblings[i])) break; } return false; }, "first-of-type": function(elem, adapter) { var siblings = adapter.getSiblings(elem); for (var i = 0; i < siblings.length; i++) { if (adapter.isTag(siblings[i])) { if (siblings[i] === elem) return true; if (adapter.getName(siblings[i]) === adapter.getName(elem)) break; } } return false; }, "last-of-type": function(elem, adapter) { var siblings = adapter.getSiblings(elem); for (var i = siblings.length - 1; i >= 0; i--) { if (adapter.isTag(siblings[i])) { if (siblings[i] === elem) return true; if (adapter.getName(siblings[i]) === adapter.getName(elem)) break; } } return false; }, "only-of-type": function(elem, adapter) { var siblings = adapter.getSiblings(elem); for (var i = 0, j = siblings.length; i < j; i++) { if (adapter.isTag(siblings[i])) { if (siblings[i] === elem) continue; if (adapter.getName(siblings[i]) === adapter.getName(elem)) { return false; } } } return true; }, "only-child": function(elem, adapter) { var siblings = adapter.getSiblings(elem); for (var i = 0; i < siblings.length; i++) { if (adapter.isTag(siblings[i]) && siblings[i] !== elem) return false; } return true; }, //:matches(a, area, link)[href] link: function(elem, adapter) { return adapter.hasAttrib(elem, "href"); }, //TODO: :any-link once the name is finalized (as an alias of :link) //forms //to consider: :target //:matches([selected], select:not([multiple]):not(> option[selected]) > option:first-of-type) selected: function(elem, adapter) { if (adapter.hasAttrib(elem, "selected")) return true; else if (adapter.getName(elem) !== "option") return false; //the first