|
|
- var Marker = require('../../tokenizer/marker');
-
- var Selector = {
- ADJACENT_SIBLING: '+',
- DESCENDANT: '>',
- DOT: '.',
- HASH: '#',
- NON_ADJACENT_SIBLING: '~',
- PSEUDO: ':'
- };
-
- var LETTER_PATTERN = /[a-zA-Z]/;
- var NOT_PREFIX = ':not(';
- var SEPARATOR_PATTERN = /[\s,\(>~\+]/;
-
- function specificity(selector) {
- var result = [0, 0, 0];
- var character;
- var isEscaped;
- var isSingleQuoted;
- var isDoubleQuoted;
- var roundBracketLevel = 0;
- var couldIntroduceNewTypeSelector;
- var withinNotPseudoClass = false;
- var wasPseudoClass = false;
- var i, l;
-
- for (i = 0, l = selector.length; i < l; i++) {
- character = selector[i];
-
- if (isEscaped) {
- // noop
- } else if (character == Marker.SINGLE_QUOTE && !isDoubleQuoted && !isSingleQuoted) {
- isSingleQuoted = true;
- } else if (character == Marker.SINGLE_QUOTE && !isDoubleQuoted && isSingleQuoted) {
- isSingleQuoted = false;
- } else if (character == Marker.DOUBLE_QUOTE && !isDoubleQuoted && !isSingleQuoted) {
- isDoubleQuoted = true;
- } else if (character == Marker.DOUBLE_QUOTE && isDoubleQuoted && !isSingleQuoted) {
- isDoubleQuoted = false;
- } else if (isSingleQuoted || isDoubleQuoted) {
- continue;
- } else if (roundBracketLevel > 0 && !withinNotPseudoClass) {
- // noop
- } else if (character == Marker.OPEN_ROUND_BRACKET) {
- roundBracketLevel++;
- } else if (character == Marker.CLOSE_ROUND_BRACKET && roundBracketLevel == 1) {
- roundBracketLevel--;
- withinNotPseudoClass = false;
- } else if (character == Marker.CLOSE_ROUND_BRACKET) {
- roundBracketLevel--;
- } else if (character == Selector.HASH) {
- result[0]++;
- } else if (character == Selector.DOT || character == Marker.OPEN_SQUARE_BRACKET) {
- result[1]++;
- } else if (character == Selector.PSEUDO && !wasPseudoClass && !isNotPseudoClass(selector, i)) {
- result[1]++;
- withinNotPseudoClass = false;
- } else if (character == Selector.PSEUDO) {
- withinNotPseudoClass = true;
- } else if ((i === 0 || couldIntroduceNewTypeSelector) && LETTER_PATTERN.test(character)) {
- result[2]++;
- }
-
- isEscaped = character == Marker.BACK_SLASH;
- wasPseudoClass = character == Selector.PSEUDO;
- couldIntroduceNewTypeSelector = !isEscaped && SEPARATOR_PATTERN.test(character);
- }
-
- return result;
- }
-
- function isNotPseudoClass(selector, index) {
- return selector.indexOf(NOT_PREFIX, index) === index;
- }
-
- module.exports = specificity;
|