blob: 2c52ac616ce37767937c3113ae5525313fdfd604 [file] [log] [blame]
'use strict';
const styleSearch = require('style-search');
const isOnlyWhitespace = require('../../utils/isOnlyWhitespace');
const isStandardSyntaxComment = require('../../utils/isStandardSyntaxComment');
const optionsMatches = require('../../utils/optionsMatches');
const report = require('../../utils/report');
const ruleMessages = require('../../utils/ruleMessages');
const { isAtRule, isComment, isDeclaration, isRule } = require('../../utils/typeGuards');
const validateOptions = require('../../utils/validateOptions');
const ruleName = 'no-eol-whitespace';
const messages = ruleMessages(ruleName, {
rejected: 'Unexpected whitespace at end of line',
});
const whitespacesToReject = new Set([' ', '\t']);
/**
* @param {string} str
* @returns {string}
*/
function fixString(str) {
return str.replace(/[ \t]+$/, '');
}
/**
* @param {number} lastEOLIndex
* @param {string} string
* @param {{ ignoreEmptyLines?: boolean, isRootFirst?: boolean }} [options]
* @returns {number}
*/
function findErrorStartIndex(
lastEOLIndex,
string,
{ ignoreEmptyLines, isRootFirst } = {
ignoreEmptyLines: false,
isRootFirst: false,
},
) {
const eolWhitespaceIndex = lastEOLIndex - 1;
// If the character before newline is not whitespace, ignore
if (!whitespacesToReject.has(string[eolWhitespaceIndex])) {
return -1;
}
if (ignoreEmptyLines) {
// If there is only whitespace between the previous newline and
// this newline, ignore
const beforeNewlineIndex = string.lastIndexOf('\n', eolWhitespaceIndex);
if (beforeNewlineIndex >= 0 || isRootFirst) {
const line = string.substring(beforeNewlineIndex, eolWhitespaceIndex);
if (isOnlyWhitespace(line)) {
return -1;
}
}
}
return eolWhitespaceIndex;
}
/** @type {import('stylelint').Rule} */
const rule = (primary, secondaryOptions, context) => {
return (root, result) => {
const validOptions = validateOptions(
result,
ruleName,
{
actual: primary,
},
{
optional: true,
actual: secondaryOptions,
possible: {
ignore: ['empty-lines'],
},
},
);
if (!validOptions) {
return;
}
const ignoreEmptyLines = optionsMatches(secondaryOptions, 'ignore', 'empty-lines');
if (context.fix) {
fix(root);
}
const rootString = context.fix ? root.toString() : (root.source && root.source.input.css) || '';
/**
* @param {number} index
*/
const reportFromIndex = (index) => {
report({
message: messages.rejected,
node: root,
index,
result,
ruleName,
});
};
eachEolWhitespace(rootString, reportFromIndex, true);
const errorIndex = findErrorStartIndex(rootString.length, rootString, {
ignoreEmptyLines,
isRootFirst: true,
});
if (errorIndex > -1) {
reportFromIndex(errorIndex);
}
/**
* Iterate each whitespace at the end of each line of the given string.
* @param {string} string - the source code string
* @param {(index: number) => void} callback - callback the whitespace index at the end of each line.
* @param {boolean} isRootFirst - set `true` if the given string is the first token of the root.
* @returns {void}
*/
function eachEolWhitespace(string, callback, isRootFirst) {
styleSearch(
{
source: string,
target: ['\n', '\r'],
comments: 'check',
},
(match) => {
const index = findErrorStartIndex(match.startIndex, string, {
ignoreEmptyLines,
isRootFirst,
});
if (index > -1) {
callback(index);
}
},
);
}
/**
* @param {import('postcss').Root} rootNode
*/
function fix(rootNode) {
let isRootFirst = true;
rootNode.walk((node) => {
fixText(
node.raws.before,
(fixed) => {
node.raws.before = fixed;
},
isRootFirst,
);
isRootFirst = false;
if (isAtRule(node)) {
fixText(node.raws.afterName, (fixed) => {
node.raws.afterName = fixed;
});
const rawsParams = node.raws.params;
if (rawsParams) {
fixText(rawsParams.raw, (fixed) => {
rawsParams.raw = fixed;
});
} else {
fixText(node.params, (fixed) => {
node.params = fixed;
});
}
}
if (isRule(node)) {
const rawsSelector = node.raws.selector;
if (rawsSelector) {
fixText(rawsSelector.raw, (fixed) => {
rawsSelector.raw = fixed;
});
} else {
fixText(node.selector, (fixed) => {
node.selector = fixed;
});
}
}
if (isAtRule(node) || isRule(node) || isDeclaration(node)) {
fixText(node.raws.between, (fixed) => {
node.raws.between = fixed;
});
}
if (isDeclaration(node)) {
if (node.raws.value) {
fixText(node.raws.value.raw, (fixed) => {
node.raws.value.raw = fixed;
});
} else {
fixText(node.value, (fixed) => {
node.value = fixed;
});
}
}
if (isComment(node)) {
fixText(node.raws.left, (fixed) => {
node.raws.left = fixed;
});
if (!isStandardSyntaxComment(node)) {
node.raws.right = node.raws.right && fixString(node.raws.right);
} else {
fixText(node.raws.right, (fixed) => {
node.raws.right = fixed;
});
}
fixText(node.text, (fixed) => {
node.text = fixed;
});
}
if (isAtRule(node) || isRule(node)) {
fixText(node.raws.after, (fixed) => {
node.raws.after = fixed;
});
}
});
fixText(
rootNode.raws.after,
(fixed) => {
rootNode.raws.after = fixed;
},
isRootFirst,
);
if (typeof rootNode.raws.after === 'string') {
const lastEOL = Math.max(
rootNode.raws.after.lastIndexOf('\n'),
rootNode.raws.after.lastIndexOf('\r'),
);
if (lastEOL !== rootNode.raws.after.length - 1) {
rootNode.raws.after =
rootNode.raws.after.slice(0, lastEOL + 1) +
fixString(rootNode.raws.after.slice(lastEOL + 1));
}
}
}
/**
* @param {string | undefined} value
* @param {(text: string) => void} fixFn
* @param {boolean} isRootFirst
*/
function fixText(value, fixFn, isRootFirst = false) {
if (!value) {
return;
}
let fixed = '';
let lastIndex = 0;
eachEolWhitespace(
value,
(index) => {
const newlineIndex = index + 1;
fixed += fixString(value.slice(lastIndex, newlineIndex));
lastIndex = newlineIndex;
},
isRootFirst,
);
if (lastIndex) {
fixed += value.slice(lastIndex);
fixFn(fixed);
}
}
};
};
rule.ruleName = ruleName;
rule.messages = messages;
module.exports = rule;