parserlib/csslint: tweaks and speedups
* "simple-not" rule * enable and fix "selector-newline" rule * fix error attribution * removed the now redundant suppressor of USO var errors * pre-parse escapes
This commit is contained in:
parent
6c7e77bc5a
commit
ac66095ee0
|
@ -100,6 +100,7 @@ define(require => {
|
||||||
'empty-rules': 1,
|
'empty-rules': 1,
|
||||||
'errors': 1,
|
'errors': 1,
|
||||||
'known-properties': 1,
|
'known-properties': 1,
|
||||||
|
'selector-newline': 1,
|
||||||
'simple-not': 1,
|
'simple-not': 1,
|
||||||
'warnings': 1,
|
'warnings': 1,
|
||||||
// disabled
|
// disabled
|
||||||
|
@ -126,7 +127,6 @@ define(require => {
|
||||||
'rules-count': 0,
|
'rules-count': 0,
|
||||||
'selector-max': 0,
|
'selector-max': 0,
|
||||||
'selector-max-approaching': 0,
|
'selector-max-approaching': 0,
|
||||||
'selector-newline': 0,
|
|
||||||
'shorthand': 0,
|
'shorthand': 0,
|
||||||
'star-property-hack': 0,
|
'star-property-hack': 0,
|
||||||
'text-indent': 0,
|
'text-indent': 0,
|
||||||
|
|
|
@ -1409,13 +1409,10 @@ define(require => {
|
||||||
init(parser, reporter) {
|
init(parser, reporter) {
|
||||||
parser.addListener('startrule', event => {
|
parser.addListener('startrule', event => {
|
||||||
for (const {parts} of event.selectors) {
|
for (const {parts} of event.selectors) {
|
||||||
for (let p = 0, pLen = parts.length; p < pLen; p++) {
|
for (let i = 0, p, pn; i < parts.length - 1 && (p = parts[i]); i++) {
|
||||||
for (let n = p + 1; n < pLen; n++) {
|
if (p.type === 'descendant' && (pn = parts[i + 1]).line > p.line) {
|
||||||
if (parts[p].type === 'descendant' &&
|
|
||||||
parts[n].line > parts[p].line) {
|
|
||||||
reporter.report('newline character found in selector (forgot a comma?)',
|
reporter.report('newline character found in selector (forgot a comma?)',
|
||||||
parts[p].line, parts[0].col, this);
|
pn.line, pn.col, this);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1478,6 +1475,37 @@ define(require => {
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
CSSLint.addRule({
|
||||||
|
id: 'simple-not',
|
||||||
|
name: 'Require use of simple selectors inside :not()',
|
||||||
|
desc: 'A complex selector inside :not() is only supported by CSS4-compliant browsers.',
|
||||||
|
browsers: 'All',
|
||||||
|
|
||||||
|
init(parser, reporter) {
|
||||||
|
parser.addListener('startrule', e => {
|
||||||
|
for (const sel of e.selectors) {
|
||||||
|
if (!/:not\(/i.test(sel.text)) continue;
|
||||||
|
for (const part of sel.parts) {
|
||||||
|
if (!part.modifiers) continue;
|
||||||
|
for (const mod of part.modifiers) {
|
||||||
|
if (mod.type !== 'not') continue;
|
||||||
|
const {args} = mod;
|
||||||
|
const {parts} = args[0];
|
||||||
|
if (args.length > 1 ||
|
||||||
|
parts.length !== 1 ||
|
||||||
|
parts[0].modifiers.length + (parts[0].elementName ? 1 : 0) > 1 ||
|
||||||
|
/^:not\(/i.test(parts[0])) {
|
||||||
|
reporter.report(
|
||||||
|
`Simple selector expected, but found '${args.join(', ')}'`,
|
||||||
|
args[0].line, args[0].col, this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
CSSLint.addRule({
|
CSSLint.addRule({
|
||||||
id: 'star-property-hack',
|
id: 'star-property-hack',
|
||||||
name: 'Disallow properties with a star prefix',
|
name: 'Disallow properties with a star prefix',
|
||||||
|
|
|
@ -676,8 +676,9 @@ define(require => {
|
||||||
x: 'resolution',
|
x: 'resolution',
|
||||||
ar: 'dimension',
|
ar: 'dimension',
|
||||||
};
|
};
|
||||||
const rxIdentStart = /[-\\_a-zA-Z\u00A0-\uFFFF]+/yu;
|
// Sticky `y` flag must be used in expressions used with peekTest and readMatch
|
||||||
const rxNameChar = /[-\\_\da-zA-Z\u00A0-\uFFFF]+/yu;
|
const rxIdentStart = /[-\\_a-zA-Z\u00A0-\uFFFF]/u;
|
||||||
|
const rxNameChar = /[-\\_\da-zA-Z\u00A0-\uFFFF]/u;
|
||||||
const rxNameCharNoEsc = /[-_\da-zA-Z\u00A0-\uFFFF]+/yu; // must not match \\
|
const rxNameCharNoEsc = /[-_\da-zA-Z\u00A0-\uFFFF]+/yu; // must not match \\
|
||||||
const rxUnquotedUrlCharNoEsc = /[-!#$%&*-[\]-~\u00A0-\uFFFF]+/yu; // must not match \\
|
const rxUnquotedUrlCharNoEsc = /[-!#$%&*-[\]-~\u00A0-\uFFFF]+/yu; // must not match \\
|
||||||
const rxVendorPrefix = /^-(webkit|moz|ms|o)-(.+)/i;
|
const rxVendorPrefix = /^-(webkit|moz|ms|o)-(.+)/i;
|
||||||
|
@ -1174,6 +1175,7 @@ define(require => {
|
||||||
//#region Tokens
|
//#region Tokens
|
||||||
|
|
||||||
/* https://www.w3.org/TR/css3-syntax/#lexical */
|
/* https://www.w3.org/TR/css3-syntax/#lexical */
|
||||||
|
/** @type {Object<string,number|Object>} */
|
||||||
const Tokens = Object.assign([], {
|
const Tokens = Object.assign([], {
|
||||||
EOF: {}, // must be the first token
|
EOF: {}, // must be the first token
|
||||||
}, {
|
}, {
|
||||||
|
@ -1530,8 +1532,10 @@ define(require => {
|
||||||
|
|
||||||
constructor(matchFunc, toString, options) {
|
constructor(matchFunc, toString, options) {
|
||||||
this.matchFunc = matchFunc;
|
this.matchFunc = matchFunc;
|
||||||
|
/** @type {function(?number):string} */
|
||||||
this.toString = typeof toString === 'function' ? toString : () => toString;
|
this.toString = typeof toString === 'function' ? toString : () => toString;
|
||||||
if (options) this.options = options;
|
/** @type {?Matcher[]} */
|
||||||
|
this.options = options;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -2013,11 +2017,6 @@ define(require => {
|
||||||
|
|
||||||
// individual media query
|
// individual media query
|
||||||
class MediaQuery extends SyntaxUnit {
|
class MediaQuery extends SyntaxUnit {
|
||||||
/**
|
|
||||||
* @param {String} modifier The modifier "not" or "only" (or null).
|
|
||||||
* @param {String} mediaType The type of media (i.e., "print").
|
|
||||||
* @param {Array} features Array of selectors parts making up this selector.
|
|
||||||
*/
|
|
||||||
constructor(modifier, mediaType, features, pos) {
|
constructor(modifier, mediaType, features, pos) {
|
||||||
const text = (modifier ? modifier + ' ' : '') +
|
const text = (modifier ? modifier + ' ' : '') +
|
||||||
(mediaType ? mediaType : '') +
|
(mediaType ? mediaType : '') +
|
||||||
|
@ -2045,9 +2044,6 @@ define(require => {
|
||||||
* including multiple selectors (those separated by commas).
|
* including multiple selectors (those separated by commas).
|
||||||
*/
|
*/
|
||||||
class Selector extends SyntaxUnit {
|
class Selector extends SyntaxUnit {
|
||||||
/**
|
|
||||||
* @param {SelectorPart[]} parts
|
|
||||||
*/
|
|
||||||
constructor(parts, pos) {
|
constructor(parts, pos) {
|
||||||
super(parts.join(' '), pos, TYPES.SELECTOR_TYPE);
|
super(parts.join(' '), pos, TYPES.SELECTOR_TYPE);
|
||||||
this.parts = parts;
|
this.parts = parts;
|
||||||
|
@ -2061,10 +2057,6 @@ define(require => {
|
||||||
* Does not include combinators such as spaces, +, >, etc.
|
* Does not include combinators such as spaces, +, >, etc.
|
||||||
*/
|
*/
|
||||||
class SelectorPart extends SyntaxUnit {
|
class SelectorPart extends SyntaxUnit {
|
||||||
/**
|
|
||||||
* @param {String} elementName or null if there's none
|
|
||||||
* @param {SelectorSubPart[]} modifiers - may be empty
|
|
||||||
*/
|
|
||||||
constructor(elementName, modifiers, text, pos) {
|
constructor(elementName, modifiers, text, pos) {
|
||||||
super(text, pos, TYPES.SELECTOR_PART_TYPE);
|
super(text, pos, TYPES.SELECTOR_PART_TYPE);
|
||||||
this.elementName = elementName;
|
this.elementName = elementName;
|
||||||
|
@ -2076,9 +2068,6 @@ define(require => {
|
||||||
* Selector modifier string
|
* Selector modifier string
|
||||||
*/
|
*/
|
||||||
class SelectorSubPart extends SyntaxUnit {
|
class SelectorSubPart extends SyntaxUnit {
|
||||||
/**
|
|
||||||
* @param {string} type - elementName id class attribute pseudo any not
|
|
||||||
*/
|
|
||||||
constructor(text, type, pos) {
|
constructor(text, type, pos) {
|
||||||
super(text, pos, TYPES.SELECTOR_SUB_PART_TYPE);
|
super(text, pos, TYPES.SELECTOR_SUB_PART_TYPE);
|
||||||
this.type = type;
|
this.type = type;
|
||||||
|
@ -2136,21 +2125,15 @@ define(require => {
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
/**
|
|
||||||
* @return {int} The numeric value for the specificity.
|
|
||||||
*/
|
|
||||||
valueOf() {
|
valueOf() {
|
||||||
return (this.a * 1000) + (this.b * 100) + (this.c * 10) + this.d;
|
return this.a * 1000 + this.b * 100 + this.c * 10 + this.d;
|
||||||
}
|
}
|
||||||
/**
|
|
||||||
* @return {String} The string representation of specificity.
|
|
||||||
*/
|
|
||||||
toString() {
|
toString() {
|
||||||
return this.a + ',' + this.b + ',' + this.c + ',' + this.d;
|
return `${this.a},${this.b},${this.c},${this.d}`;
|
||||||
}
|
}
|
||||||
/**
|
/**
|
||||||
* Calculates the specificity of the given selector.
|
* Calculates the specificity of the given selector.
|
||||||
* @param {Selector} The selector to calculate specificity for.
|
* @param {Selector} selector The selector to calculate specificity for.
|
||||||
* @return {Specificity} The specificity of the selector.
|
* @return {Specificity} The specificity of the selector.
|
||||||
*/
|
*/
|
||||||
static calculate(selector) {
|
static calculate(selector) {
|
||||||
|
@ -2192,7 +2175,6 @@ define(require => {
|
||||||
class PropertyName extends SyntaxUnit {
|
class PropertyName extends SyntaxUnit {
|
||||||
constructor(text, hack, pos) {
|
constructor(text, hack, pos) {
|
||||||
super(text, pos, TYPES.PROPERTY_NAME_TYPE);
|
super(text, pos, TYPES.PROPERTY_NAME_TYPE);
|
||||||
// type of IE hack applied ("*", "_", or null).
|
|
||||||
this.hack = hack;
|
this.hack = hack;
|
||||||
}
|
}
|
||||||
toString() {
|
toString() {
|
||||||
|
@ -2205,9 +2187,6 @@ define(require => {
|
||||||
* separated by commas, this type represents just one of the values.
|
* separated by commas, this type represents just one of the values.
|
||||||
*/
|
*/
|
||||||
class PropertyValue extends SyntaxUnit {
|
class PropertyValue extends SyntaxUnit {
|
||||||
/**
|
|
||||||
* @param {PropertyValuePart[]} parts An array of value parts making up this value.
|
|
||||||
*/
|
|
||||||
constructor(parts, pos) {
|
constructor(parts, pos) {
|
||||||
super(parts.join(' '), pos, TYPES.PROPERTY_VALUE_TYPE);
|
super(parts.join(' '), pos, TYPES.PROPERTY_VALUE_TYPE);
|
||||||
this.parts = parts;
|
this.parts = parts;
|
||||||
|
@ -2225,12 +2204,7 @@ define(require => {
|
||||||
const {value, type} = token;
|
const {value, type} = token;
|
||||||
super(value, token, TYPES.PROPERTY_VALUE_PART_TYPE);
|
super(value, token, TYPES.PROPERTY_VALUE_PART_TYPE);
|
||||||
this.tokenType = type;
|
this.tokenType = type;
|
||||||
if (token.expr) this.expr = token.expr;
|
this.expr = token.expr || null;
|
||||||
// There can be ambiguity with escape sequences in identifiers, as
|
|
||||||
// well as with "color" parts which are also "identifiers", so record
|
|
||||||
// an explicit hint when the token generating this PropertyValuePart
|
|
||||||
// was an identifier.
|
|
||||||
this.wasIdent = type === Tokens.IDENT;
|
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case Tokens.ANGLE:
|
case Tokens.ANGLE:
|
||||||
case Tokens.DIMENSION:
|
case Tokens.DIMENSION:
|
||||||
|
@ -2286,7 +2260,6 @@ define(require => {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// A utility class that allows for easy iteration over the various parts of a property value.
|
|
||||||
class PropertyValueIterator {
|
class PropertyValueIterator {
|
||||||
/**
|
/**
|
||||||
* @param {PropertyValue} value
|
* @param {PropertyValue} value
|
||||||
|
@ -2368,7 +2341,7 @@ define(require => {
|
||||||
|
|
||||||
/** @param {PropertyValuePart} p */
|
/** @param {PropertyValuePart} p */
|
||||||
function vtIsIdent(p) {
|
function vtIsIdent(p) {
|
||||||
return p.type === 'identifier' || p.wasIdent;
|
return p.tokenType === Tokens.IDENT;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** @param {PropertyValuePart} p */
|
/** @param {PropertyValuePart} p */
|
||||||
|
@ -2690,8 +2663,13 @@ define(require => {
|
||||||
line: reader._line,
|
line: reader._line,
|
||||||
offset: reader._cursor,
|
offset: reader._cursor,
|
||||||
};
|
};
|
||||||
const a = tok.value = reader.read();
|
let a = tok.value = reader.read();
|
||||||
const b = reader.peek();
|
let b = reader.peek();
|
||||||
|
if (a === '\\') {
|
||||||
|
if (b === '\n' || b === '\f') return tok;
|
||||||
|
a = this.readEscape();
|
||||||
|
b = reader.peek();
|
||||||
|
}
|
||||||
switch (a) {
|
switch (a) {
|
||||||
case ' ':
|
case ' ':
|
||||||
case '\n':
|
case '\n':
|
||||||
|
@ -2745,7 +2723,7 @@ define(require => {
|
||||||
case "'":
|
case "'":
|
||||||
return this.stringToken(a, tok);
|
return this.stringToken(a, tok);
|
||||||
case '#':
|
case '#':
|
||||||
if ((rxNameChar.lastIndex = 0, rxNameChar.test(b))) {
|
if (rxNameChar.test(b)) {
|
||||||
tok.type = Tokens.HASH;
|
tok.type = Tokens.HASH;
|
||||||
tok.value = this.readName(a);
|
tok.value = this.readName(a);
|
||||||
}
|
}
|
||||||
|
@ -2768,7 +2746,7 @@ define(require => {
|
||||||
}
|
}
|
||||||
} else if (b >= '0' && b <= '9' || b === '.' && reader.peekTest(/\.\d/y)) {
|
} else if (b >= '0' && b <= '9' || b === '.' && reader.peekTest(/\.\d/y)) {
|
||||||
this.numberToken(a, tok);
|
this.numberToken(a, tok);
|
||||||
} else if ((rxIdentStart.lastIndex = 0, rxIdentStart.test(b))) {
|
} else if (rxIdentStart.test(b)) {
|
||||||
this.identOrFunctionToken(a, tok);
|
this.identOrFunctionToken(a, tok);
|
||||||
} else {
|
} else {
|
||||||
tok.type = Tokens.MINUS;
|
tok.type = Tokens.MINUS;
|
||||||
|
@ -2806,10 +2784,6 @@ define(require => {
|
||||||
tok.value = '<!--';
|
tok.value = '<!--';
|
||||||
}
|
}
|
||||||
return tok;
|
return tok;
|
||||||
case '\\':
|
|
||||||
return b !== '\r' && b !== '\n' && b !== '\f' ?
|
|
||||||
this.identOrFunctionToken(this.readEscape(), tok) :
|
|
||||||
tok;
|
|
||||||
// EOF
|
// EOF
|
||||||
case null:
|
case null:
|
||||||
tok.type = Tokens.EOF;
|
tok.type = Tokens.EOF;
|
||||||
|
@ -2822,7 +2796,7 @@ define(require => {
|
||||||
}
|
}
|
||||||
if (a >= '0' && a <= '9') {
|
if (a >= '0' && a <= '9') {
|
||||||
this.numberToken(a, tok);
|
this.numberToken(a, tok);
|
||||||
} else if ((rxIdentStart.lastIndex = 0, rxIdentStart.test(a))) {
|
} else if (rxIdentStart.test(a)) {
|
||||||
this.identOrFunctionToken(a, tok);
|
this.identOrFunctionToken(a, tok);
|
||||||
} else {
|
} else {
|
||||||
tok.type = typeMap.get(a) || Tokens.CHAR;
|
tok.type = typeMap.get(a) || Tokens.CHAR;
|
||||||
|
@ -2912,7 +2886,7 @@ define(require => {
|
||||||
let tt = Tokens.NUMBER;
|
let tt = Tokens.NUMBER;
|
||||||
let units, type;
|
let units, type;
|
||||||
const c = reader.peek();
|
const c = reader.peek();
|
||||||
if ((rxIdentStart.lastIndex = 0, rxIdentStart.test(c))) {
|
if (rxIdentStart.test(c)) {
|
||||||
units = this.readName(reader.read());
|
units = this.readName(reader.read());
|
||||||
type = UNITS[units] || UNITS[lower(units)];
|
type = UNITS[units] || UNITS[lower(units)];
|
||||||
tt = type && Tokens[type.toUpperCase()] ||
|
tt = type && Tokens[type.toUpperCase()] ||
|
||||||
|
@ -3076,8 +3050,11 @@ define(require => {
|
||||||
const value = [];
|
const value = [];
|
||||||
const endings = [];
|
const endings = [];
|
||||||
let end = stopOn;
|
let end = stopOn;
|
||||||
|
const rx = stopOn.includes(';')
|
||||||
|
? /([^;!'"{}()[\]/\\]|\/(?!\*))+/y
|
||||||
|
: /([^'"{}()[\]/\\]|\/(?!\*))+/y;
|
||||||
while (!reader.eof()) {
|
while (!reader.eof()) {
|
||||||
const chunk = reader.readMatch(/([^;!'"{}()[\]/\\]|\/(?!\*))+/y);
|
const chunk = reader.readMatch(rx);
|
||||||
if (chunk) {
|
if (chunk) {
|
||||||
value.push(chunk);
|
value.push(chunk);
|
||||||
}
|
}
|
||||||
|
@ -3431,14 +3408,12 @@ define(require => {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param {String|{type: string, ...}} event
|
* @param {string|Object} event
|
||||||
* @param {Token|SyntaxUnit} [token=this._tokenStream._token] - sets the position
|
* @param {parserlib.Token|SyntaxUnit} [token=this._tokenStream._token] - sets the position
|
||||||
*/
|
*/
|
||||||
fire(event, token = this._tokenStream._token) {
|
fire(event, token = this._tokenStream._token) {
|
||||||
if (typeof event === 'string') {
|
if (typeof event === 'string') {
|
||||||
event = {type: event};
|
event = {type: event};
|
||||||
} else if (event.message && event.message.includes('/*[[')) {
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
if (event.offset === undefined && token) {
|
if (event.offset === undefined && token) {
|
||||||
event.offset = token.offset;
|
event.offset = token.offset;
|
||||||
|
@ -4044,22 +4019,11 @@ define(require => {
|
||||||
|
|
||||||
_negation(start) {
|
_negation(start) {
|
||||||
const stream = this._tokenStream;
|
const stream = this._tokenStream;
|
||||||
let value = start.value + this._ws();
|
const value = [start.value, this._ws()];
|
||||||
const args = this._selectorsGroup();
|
const args = this._selectorsGroup();
|
||||||
if (!args) stream.throwUnexpected(stream.LT(1));
|
if (!args) stream.throwUnexpected(stream.LT(1));
|
||||||
const arg = args[0];
|
value.push(...args, this._ws(), stream.mustMatch(Tokens.RPAREN).value);
|
||||||
const parts = arg.parts;
|
return Object.assign(new SelectorSubPart(fastJoin(value), 'not', start), {args});
|
||||||
if (args.length > 1 ||
|
|
||||||
parts.length !== 1 ||
|
|
||||||
parts[0].modifiers.length + (parts[0].elementName ? 1 : 0) > 1 ||
|
|
||||||
/^:not\b/i.test(parts[0])) {
|
|
||||||
this.fire({
|
|
||||||
type: 'warning',
|
|
||||||
message: `Simple selector expected, but found '${args.join(', ')}'`,
|
|
||||||
}, arg);
|
|
||||||
}
|
|
||||||
value += arg + this._ws() + stream.mustMatch(Tokens.RPAREN).value;
|
|
||||||
return Object.assign(new SelectorSubPart(value, 'not', start), {args: [arg]});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
_declaration(consumeSemicolon) {
|
_declaration(consumeSemicolon) {
|
||||||
|
|
Loading…
Reference in New Issue
Block a user