| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151 | /** * @fileoverview Disallows multiple blank lines. * implementation adapted from the no-trailing-spaces rule. * @author Greg Cochard */"use strict";//------------------------------------------------------------------------------// Rule Definition//------------------------------------------------------------------------------/** @type {import('../shared/types').Rule} */module.exports = {    meta: {        type: "layout",        docs: {            description: "Disallow multiple empty lines",            recommended: false,            url: "https://eslint.org/docs/rules/no-multiple-empty-lines"        },        fixable: "whitespace",        schema: [            {                type: "object",                properties: {                    max: {                        type: "integer",                        minimum: 0                    },                    maxEOF: {                        type: "integer",                        minimum: 0                    },                    maxBOF: {                        type: "integer",                        minimum: 0                    }                },                required: ["max"],                additionalProperties: false            }        ],        messages: {            blankBeginningOfFile: "Too many blank lines at the beginning of file. Max of {{max}} allowed.",            blankEndOfFile: "Too many blank lines at the end of file. Max of {{max}} allowed.",            consecutiveBlank: "More than {{max}} blank {{pluralizedLines}} not allowed."        }    },    create(context) {        // Use options.max or 2 as default        let max = 2,            maxEOF = max,            maxBOF = max;        if (context.options.length) {            max = context.options[0].max;            maxEOF = typeof context.options[0].maxEOF !== "undefined" ? context.options[0].maxEOF : max;            maxBOF = typeof context.options[0].maxBOF !== "undefined" ? context.options[0].maxBOF : max;        }        const sourceCode = context.getSourceCode();        // Swallow the final newline, as some editors add it automatically and we don't want it to cause an issue        const allLines = sourceCode.lines[sourceCode.lines.length - 1] === "" ? sourceCode.lines.slice(0, -1) : sourceCode.lines;        const templateLiteralLines = new Set();        //--------------------------------------------------------------------------        // Public        //--------------------------------------------------------------------------        return {            TemplateLiteral(node) {                node.quasis.forEach(literalPart => {                    // Empty lines have a semantic meaning if they're inside template literals. Don't count these as empty lines.                    for (let ignoredLine = literalPart.loc.start.line; ignoredLine < literalPart.loc.end.line; ignoredLine++) {                        templateLiteralLines.add(ignoredLine);                    }                });            },            "Program:exit"(node) {                return allLines                    // Given a list of lines, first get a list of line numbers that are non-empty.                    .reduce((nonEmptyLineNumbers, line, index) => {                        if (line.trim() || templateLiteralLines.has(index + 1)) {                            nonEmptyLineNumbers.push(index + 1);                        }                        return nonEmptyLineNumbers;                    }, [])                    // Add a value at the end to allow trailing empty lines to be checked.                    .concat(allLines.length + 1)                    // Given two line numbers of non-empty lines, report the lines between if the difference is too large.                    .reduce((lastLineNumber, lineNumber) => {                        let messageId, maxAllowed;                        if (lastLineNumber === 0) {                            messageId = "blankBeginningOfFile";                            maxAllowed = maxBOF;                        } else if (lineNumber === allLines.length + 1) {                            messageId = "blankEndOfFile";                            maxAllowed = maxEOF;                        } else {                            messageId = "consecutiveBlank";                            maxAllowed = max;                        }                        if (lineNumber - lastLineNumber - 1 > maxAllowed) {                            context.report({                                node,                                loc: {                                    start: { line: lastLineNumber + maxAllowed + 1, column: 0 },                                    end: { line: lineNumber, column: 0 }                                },                                messageId,                                data: {                                    max: maxAllowed,                                    pluralizedLines: maxAllowed === 1 ? "line" : "lines"                                },                                fix(fixer) {                                    const rangeStart = sourceCode.getIndexFromLoc({ line: lastLineNumber + 1, column: 0 });                                    /*                                     * The end of the removal range is usually the start index of the next line.                                     * However, at the end of the file there is no next line, so the end of the                                     * range is just the length of the text.                                     */                                    const lineNumberAfterRemovedLines = lineNumber - maxAllowed;                                    const rangeEnd = lineNumberAfterRemovedLines <= allLines.length                                        ? sourceCode.getIndexFromLoc({ line: lineNumberAfterRemovedLines, column: 0 })                                        : sourceCode.text.length;                                    return fixer.removeRange([rangeStart, rangeEnd]);                                }                            });                        }                        return lineNumber;                    }, 0);            }        };    }};
 |