|
|
/** * @fileoverview Enforce newlines between operands of ternary expressions * @author Kai Cataldo */
"use strict";
const astUtils = require("./utils/ast-utils");
//------------------------------------------------------------------------------
// Rule Definition
//------------------------------------------------------------------------------
module.exports = { meta: { type: "layout",
docs: { description: "enforce newlines between operands of ternary expressions", category: "Stylistic Issues", recommended: false, url: "https://eslint.org/docs/rules/multiline-ternary" },
schema: [ { enum: ["always", "always-multiline", "never"] } ], messages: { expectedTestCons: "Expected newline between test and consequent of ternary expression.", expectedConsAlt: "Expected newline between consequent and alternate of ternary expression.", unexpectedTestCons: "Unexpected newline between test and consequent of ternary expression.", unexpectedConsAlt: "Unexpected newline between consequent and alternate of ternary expression." } },
create(context) { const option = context.options[0]; const multiline = option !== "never"; const allowSingleLine = option === "always-multiline"; const sourceCode = context.getSourceCode();
//--------------------------------------------------------------------------
// Public
//--------------------------------------------------------------------------
return { ConditionalExpression(node) { const questionToken = sourceCode.getTokenAfter(node.test, astUtils.isNotClosingParenToken); const colonToken = sourceCode.getTokenAfter(node.consequent, astUtils.isNotClosingParenToken);
const firstTokenOfTest = sourceCode.getFirstToken(node); const lastTokenOfTest = sourceCode.getTokenBefore(questionToken); const firstTokenOfConsequent = sourceCode.getTokenAfter(questionToken); const lastTokenOfConsequent = sourceCode.getTokenBefore(colonToken); const firstTokenOfAlternate = sourceCode.getTokenAfter(colonToken);
const areTestAndConsequentOnSameLine = astUtils.isTokenOnSameLine(lastTokenOfTest, firstTokenOfConsequent); const areConsequentAndAlternateOnSameLine = astUtils.isTokenOnSameLine(lastTokenOfConsequent, firstTokenOfAlternate);
if (!multiline) { if (!areTestAndConsequentOnSameLine) { context.report({ node: node.test, loc: { start: firstTokenOfTest.loc.start, end: lastTokenOfTest.loc.end }, messageId: "unexpectedTestCons" }); }
if (!areConsequentAndAlternateOnSameLine) { context.report({ node: node.consequent, loc: { start: firstTokenOfConsequent.loc.start, end: lastTokenOfConsequent.loc.end }, messageId: "unexpectedConsAlt" }); } } else { if (allowSingleLine && node.loc.start.line === node.loc.end.line) { return; }
if (areTestAndConsequentOnSameLine) { context.report({ node: node.test, loc: { start: firstTokenOfTest.loc.start, end: lastTokenOfTest.loc.end }, messageId: "expectedTestCons" }); }
if (areConsequentAndAlternateOnSameLine) { context.report({ node: node.consequent, loc: { start: firstTokenOfConsequent.loc.start, end: lastTokenOfConsequent.loc.end }, messageId: "expectedConsAlt" }); } } } }; } };
|