UNPKG

eslint-plugin-solid

Version:
103 lines (96 loc) 3.75 kB
/** * FIXME: remove this comments and import when below issue is fixed. * This import is necessary for type generation due to a bug in the TypeScript compiler. * See: https://github.com/microsoft/TypeScript/issues/42873 */ // eslint-disable-next-line @typescript-eslint/no-unused-vars import type { TSESLint } from "@typescript-eslint/utils"; import { TSESTree as T, ESLintUtils } from "@typescript-eslint/utils"; import { isJSXElementOrFragment } from "../utils"; import { getSourceCode } from "../compat"; const createRule = ESLintUtils.RuleCreator.withoutDocs; const EXPENSIVE_TYPES = ["JSXElement", "JSXFragment", "Identifier"]; export default createRule({ meta: { type: "problem", docs: { description: "Enforce using Solid's `<Show />` component for conditionally showing content. Solid's compiler covers this case, so it's a stylistic rule only.", url: "https://github.com/solidjs-community/eslint-plugin-solid/blob/main/packages/eslint-plugin-solid/docs/prefer-show.md", }, fixable: "code", schema: [], messages: { preferShowAnd: "Use Solid's `<Show />` component for conditionally showing content.", preferShowTernary: "Use Solid's `<Show />` component for conditionally showing content with a fallback.", }, }, defaultOptions: [], create(context) { const sourceCode = getSourceCode(context); const putIntoJSX = (node: T.Node): string => { const text = sourceCode.getText(node); return isJSXElementOrFragment(node) ? text : `{${text}}`; }; const logicalExpressionHandler = (node: T.LogicalExpression) => { if (node.operator === "&&" && EXPENSIVE_TYPES.includes(node.right.type)) { context.report({ node, messageId: "preferShowAnd", fix: (fixer) => fixer.replaceText( node.parent?.type === "JSXExpressionContainer" && isJSXElementOrFragment(node.parent.parent) ? node.parent : node, `<Show when={${sourceCode.getText(node.left)}}>${putIntoJSX(node.right)}</Show>` ), }); } }; const conditionalExpressionHandler = (node: T.ConditionalExpression) => { if ( EXPENSIVE_TYPES.includes(node.consequent.type) || EXPENSIVE_TYPES.includes(node.alternate.type) ) { context.report({ node, messageId: "preferShowTernary", fix: (fixer) => fixer.replaceText( node.parent?.type === "JSXExpressionContainer" && isJSXElementOrFragment(node.parent.parent) ? node.parent : node, `<Show when={${sourceCode.getText(node.test)}} fallback={${sourceCode.getText( node.alternate )}}>${putIntoJSX(node.consequent)}</Show>` ), }); } }; return { JSXExpressionContainer(node) { if (!isJSXElementOrFragment(node.parent)) { return; } if (node.expression.type === "LogicalExpression") { logicalExpressionHandler(node.expression); } else if ( node.expression.type === "ArrowFunctionExpression" && node.expression.body.type === "LogicalExpression" ) { logicalExpressionHandler(node.expression.body); } else if (node.expression.type === "ConditionalExpression") { conditionalExpressionHandler(node.expression); } else if ( node.expression.type === "ArrowFunctionExpression" && node.expression.body.type === "ConditionalExpression" ) { conditionalExpressionHandler(node.expression.body); } }, }; }, });