|
| 1 | +import { createEslintRule } from '../utils' |
| 2 | + |
| 3 | +export const RULE_NAME = 'top-level-function' |
| 4 | +export type MessageIds = 'topLevelFunctionDeclaration' |
| 5 | +export type Options = [] |
| 6 | + |
| 7 | +export default createEslintRule<Options, MessageIds>({ |
| 8 | + name: RULE_NAME, |
| 9 | + meta: { |
| 10 | + type: 'problem', |
| 11 | + docs: { |
| 12 | + description: 'Enforce top-level functions to be declared with function keyword', |
| 13 | + recommended: 'error', |
| 14 | + }, |
| 15 | + fixable: 'code', |
| 16 | + schema: [], |
| 17 | + messages: { |
| 18 | + topLevelFunctionDeclaration: 'Top-level functions should be declared with function keyword', |
| 19 | + }, |
| 20 | + }, |
| 21 | + defaultOptions: [], |
| 22 | + create: (context) => { |
| 23 | + return { |
| 24 | + VariableDeclaration(node) { |
| 25 | + if (node.parent.type !== 'Program' && node.parent.type !== 'ExportNamedDeclaration') |
| 26 | + return |
| 27 | + |
| 28 | + if (node.declarations.length !== 1) |
| 29 | + return |
| 30 | + if (node.kind !== 'const') |
| 31 | + return |
| 32 | + if (node.declare) |
| 33 | + return |
| 34 | + |
| 35 | + const declaration = node.declarations[0] |
| 36 | + |
| 37 | + if (declaration.init?.type !== 'ArrowFunctionExpression') |
| 38 | + return |
| 39 | + if (declaration.id?.type !== 'Identifier') |
| 40 | + return |
| 41 | + if (declaration.id.typeAnnotation) |
| 42 | + return |
| 43 | + |
| 44 | + const arrowFn = declaration.init |
| 45 | + const body = declaration.init.body |
| 46 | + const id = declaration.id |
| 47 | + |
| 48 | + context.report({ |
| 49 | + node, |
| 50 | + loc: { |
| 51 | + start: node.loc.start, |
| 52 | + end: node.loc.end, |
| 53 | + }, |
| 54 | + messageId: 'topLevelFunctionDeclaration', |
| 55 | + fix(fixer) { |
| 56 | + const code = context.getSourceCode().text |
| 57 | + const textName = code.slice(id.range[0], id.range[1]) |
| 58 | + const textArgs = arrowFn.params.length |
| 59 | + ? code.slice(arrowFn.params[0].range[0], arrowFn.params[arrowFn.params.length - 1].range[1]) |
| 60 | + : '' |
| 61 | + const textBody = body.type === 'BlockStatement' |
| 62 | + ? code.slice(body.range[0], body.range[1]) |
| 63 | + : `{ return ${code.slice(body.range[0], body.range[1])} }` |
| 64 | + const textGeneric = arrowFn.typeParameters |
| 65 | + ? code.slice(arrowFn.typeParameters.range[0], arrowFn.typeParameters.range[1]) |
| 66 | + : '' |
| 67 | + const textTypeReturn = arrowFn.returnType |
| 68 | + ? code.slice(arrowFn.returnType.range[0], arrowFn.returnType.range[1]) |
| 69 | + : '' |
| 70 | + const text = `function ${textName} ${textGeneric}(${textArgs})${textTypeReturn} ${textBody}` |
| 71 | + // console.log({ |
| 72 | + // input: code.slice(node.range[0], node.range[1]), |
| 73 | + // output: text, |
| 74 | + // }) |
| 75 | + return fixer.replaceTextRange([node.range[0], node.range[1]], text) |
| 76 | + }, |
| 77 | + }) |
| 78 | + }, |
| 79 | + } |
| 80 | + }, |
| 81 | +}) |
9 commit comments
felixranesberger commentedon Mar 29, 2023
Hi Anthony, thanks for providing this ESLint configuration!
What is your reason to enforce top-level functions, instead of arrow functions?
Is it because of the
this
binding?mcfarljw commentedon Mar 29, 2023
This change breaks a lot of things in my current codebase so I had to disable it. For example:
https://vueuse.org/shared/createGlobalState/#createglobalstate
antfu commentedon Mar 29, 2023
@mcfarljw I am happy to improve it by including more cases if you could provide more info. The one you sent seems to work fine: 3fa8617
antfu commentedon Mar 29, 2023
@felixranesberger It's just out of personal preference. I might raise the awareness again this is my personal opinionated config. Feel free to disable/config/fork to fit your needs.
mcfarljw commentedon Mar 30, 2023
@antfu thanks for responding! It looks like this was just a fluke, after updating from
0.37.0
to0.38.0
VSCode needed to be fully reloaded on my machine otherwise I just get this error at the top of most of my files.artur-oliva commentedon Apr 14, 2023
What is the syntax to disable the
top-level-function
rulelincenying commentedon Apr 15, 2023
'antfu/top-level-function': 'off',
zanminkian commentedon Apr 17, 2023
I agree most of the configs except this rule. For example, one-line arrow function is convenient in some case. But enforcing this rule will make the code verbose.
@antfu Is it possible that don't format the one-line arrow function?
antfu commentedon Apr 17, 2023
Sure, we could filter out one-liner, PR welcome