| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453 | /*	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 methodsfunction 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 neededvar 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 <option> in a <select> is also selected        var parent = adapter.getParent(elem);        if (!parent || adapter.getName(parent) !== "select" || adapter.hasAttrib(parent, "multiple")) {            return false;        }        var siblings = adapter.getChildren(parent);        var sawElem = false;        for (var i = 0; i < siblings.length; i++) {            if (adapter.isTag(siblings[i])) {                if (siblings[i] === elem) {                    sawElem = true;                } else if (!sawElem) {                    return false;                } else if (adapter.hasAttrib(siblings[i], "selected")) {                    return false;                }            }        }        return sawElem;    },    //https://html.spec.whatwg.org/multipage/scripting.html#disabled-elements    //:matches(    //  :matches(button, input, select, textarea, menuitem, optgroup, option)[disabled],    //  optgroup[disabled] > option),    // fieldset[disabled] * //TODO not child of first <legend>    //)    disabled: function(elem, adapter) {        return adapter.hasAttrib(elem, "disabled");    },    enabled: function(elem, adapter) {        return !adapter.hasAttrib(elem, "disabled");    },    //:matches(:matches(:radio, :checkbox)[checked], :selected) (TODO menuitem)    checked: function(elem, adapter) {        return adapter.hasAttrib(elem, "checked") || pseudos.selected(elem, adapter);    },    //:matches(input, select, textarea)[required]    required: function(elem, adapter) {        return adapter.hasAttrib(elem, "required");    },    //:matches(input, select, textarea):not([required])    optional: function(elem, adapter) {        return !adapter.hasAttrib(elem, "required");    },    //jQuery extensions    //:not(:empty)    parent: function(elem, adapter) {        return !pseudos.empty(elem, adapter);    },    //:matches(h1, h2, h3, h4, h5, h6)    header: namePseudo(["h1", "h2", "h3", "h4", "h5", "h6"]),    //:matches(button, input[type=button])    button: function(elem, adapter) {        var name = adapter.getName(elem);        return (            name === "button" || (name === "input" && adapter.getAttributeValue(elem, "type") === "button")        );    },    //:matches(input, textarea, select, button)    input: namePseudo(["input", "textarea", "select", "button"]),    //input:matches(:not([type!='']), [type='text' i])    text: function(elem, adapter) {        var attr;        return (            adapter.getName(elem) === "input" &&            (!(attr = adapter.getAttributeValue(elem, "type")) || attr.toLowerCase() === "text")        );    }};function namePseudo(names) {    if (typeof Set !== "undefined") {        // eslint-disable-next-line no-undef        var nameSet = new Set(names);        return function(elem, adapter) {            return nameSet.has(adapter.getName(elem));        };    }    return function(elem, adapter) {        return names.indexOf(adapter.getName(elem)) >= 0;    };}function verifyArgs(func, name, subselect) {    if (subselect === null) {        if (func.length > 2 && name !== "scope") {            throw new Error("pseudo-selector :" + name + " requires an argument");        }    } else {        if (func.length === 2) {            throw new Error("pseudo-selector :" + name + " doesn't have any arguments");        }    }}//FIXME this feels hackyvar re_CSS3 = /^(?:(?:nth|last|first|only)-(?:child|of-type)|root|empty|(?:en|dis)abled|checked|not)$/;module.exports = {    compile: function(next, data, options, context) {        var name = data.name;        var subselect = data.data;        var adapter = options.adapter;        if (options && options.strict && !re_CSS3.test(name)) {            throw new Error(":" + name + " isn't part of CSS3");        }        if (typeof filters[name] === "function") {            return filters[name](next, subselect, options, context);        } else if (typeof pseudos[name] === "function") {            var func = pseudos[name];            verifyArgs(func, name, subselect);            if (func === falseFunc) {                return func;            }            if (next === trueFunc) {                return function pseudoRoot(elem) {                    return func(elem, adapter, subselect);                };            }            return function pseudoArgs(elem) {                return func(elem, adapter, subselect) && next(elem);            };        } else {            throw new Error("unmatched pseudo-class :" + name);        }    },    filters: filters,    pseudos: pseudos};
 |