eslint.config.mjs 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184
  1. import {
  2. GLOB_TESTS, combine, javascript, node,
  3. stylistic, typescript, unicorn,
  4. } from '@antfu/eslint-config'
  5. import path from 'node:path'
  6. import { fileURLToPath } from 'node:url'
  7. import js from '@eslint/js'
  8. import { FlatCompat } from '@eslint/eslintrc'
  9. import globals from 'globals'
  10. import storybook from 'eslint-plugin-storybook'
  11. import { fixupConfigRules } from '@eslint/compat'
  12. import tailwind from 'eslint-plugin-tailwindcss'
  13. const __filename = fileURLToPath(import.meta.url)
  14. const __dirname = path.dirname(__filename)
  15. const compat = new FlatCompat({
  16. baseDirectory: __dirname,
  17. recommendedConfig: js.configs.recommended,
  18. allConfig: js.configs.all,
  19. })
  20. export default combine(
  21. stylistic({
  22. lessOpinionated: true,
  23. // original @antfu/eslint-config does not support jsx
  24. jsx: false,
  25. semi: false,
  26. quotes: 'single',
  27. overrides: {
  28. // original config
  29. 'style/indent': ['error', 2],
  30. 'style/quotes': ['error', 'single'],
  31. 'curly': ['error', 'multi-or-nest', 'consistent'],
  32. 'style/comma-spacing': ['error', { before: false, after: true }],
  33. 'style/quote-props': ['warn', 'consistent-as-needed'],
  34. // these options does not exist in old version
  35. // maybe useless
  36. 'style/indent-binary-ops': 'off',
  37. 'style/multiline-ternary': 'off',
  38. 'antfu/top-level-function': 'off',
  39. 'antfu/curly': 'off',
  40. 'antfu/consistent-chaining': 'off',
  41. // copy from eslint-config-antfu 0.36.0
  42. 'style/brace-style': ['error', 'stroustrup', { allowSingleLine: true }],
  43. 'style/dot-location': ['error', 'property'],
  44. 'style/object-curly-newline': ['error', { consistent: true, multiline: true }],
  45. 'style/object-property-newline': ['error', { allowMultiplePropertiesPerLine: true }],
  46. 'style/template-curly-spacing': ['error', 'never'],
  47. 'style/keyword-spacing': 'off',
  48. // not exist in old version, and big change
  49. 'style/member-delimiter-style': 'off',
  50. },
  51. }),
  52. javascript({
  53. overrides: {
  54. // handled by unused-imports/no-unused-vars
  55. 'no-unused-vars': 'off',
  56. },
  57. }),
  58. typescript({
  59. overrides: {
  60. // original config
  61. 'ts/consistent-type-definitions': ['warn', 'type'],
  62. // useful, but big change
  63. 'ts/no-empty-object-type': 'off',
  64. },
  65. }),
  66. unicorn(),
  67. node(),
  68. // use nextjs config will break @eslint/config-inspector
  69. // use `ESLINT_CONFIG_INSPECTOR=true pnpx @eslint/config-inspector` to check the config
  70. ...process.env.ESLINT_CONFIG_INSPECTOR
  71. ? []
  72. // TODO: remove this when upgrade to nextjs 15
  73. : fixupConfigRules(compat.extends('next')),
  74. {
  75. rules: {
  76. // performance issue, and not used.
  77. '@next/next/no-html-link-for-pages': 'off',
  78. },
  79. },
  80. {
  81. ignores: [
  82. '**/node_modules/*',
  83. '**/node_modules/',
  84. '**/dist/',
  85. '**/build/',
  86. '**/out/',
  87. '**/.next/',
  88. '**/public/*',
  89. '**/*.json',
  90. ],
  91. },
  92. {
  93. // orignal config
  94. rules: {
  95. // orignal ts/no-var-requires
  96. 'ts/no-require-imports': 'off',
  97. 'no-console': 'off',
  98. 'react-hooks/exhaustive-deps': 'warn',
  99. 'react/display-name': 'off',
  100. 'array-callback-return': ['error', {
  101. allowImplicit: false,
  102. checkForEach: false,
  103. }],
  104. // copy from eslint-config-antfu 0.36.0
  105. 'camelcase': 'off',
  106. 'default-case-last': 'error',
  107. // antfu use eslint-plugin-perfectionist to replace this
  108. // will cause big change, so keep the original sort-imports
  109. 'sort-imports': [
  110. 'error',
  111. {
  112. ignoreCase: false,
  113. ignoreDeclarationSort: true,
  114. ignoreMemberSort: false,
  115. memberSyntaxSortOrder: ['none', 'all', 'multiple', 'single'],
  116. allowSeparatedGroups: false,
  117. },
  118. ],
  119. // antfu migrate to eslint-plugin-unused-imports
  120. 'unused-imports/no-unused-vars': 'warn',
  121. 'unused-imports/no-unused-imports': 'warn',
  122. },
  123. languageOptions: {
  124. globals: {
  125. ...globals.browser,
  126. ...globals.es2025,
  127. ...globals.node,
  128. React: 'readable',
  129. JSX: 'readable',
  130. },
  131. },
  132. },
  133. storybook.configs['flat/recommended'],
  134. // need futher research
  135. {
  136. rules: {
  137. // not exist in old version
  138. 'antfu/consistent-list-newline': 'off',
  139. 'node/prefer-global/process': 'off',
  140. 'node/prefer-global/buffer': 'off',
  141. 'node/no-callback-literal': 'off',
  142. // useful, but big change
  143. 'unicorn/prefer-number-properties': 'warn',
  144. 'unicorn/no-new-array': 'warn',
  145. },
  146. },
  147. // suppress error for `no-undef` rule
  148. {
  149. files: GLOB_TESTS,
  150. languageOptions: {
  151. globals: {
  152. ...globals.browser,
  153. ...globals.es2021,
  154. ...globals.node,
  155. ...globals.jest,
  156. },
  157. },
  158. },
  159. tailwind.configs['flat/recommended'],
  160. {
  161. rules: {
  162. // due to 1k lines of tailwind config, these rule have performance issue
  163. 'tailwindcss/no-contradicting-classname': 'off',
  164. 'tailwindcss/no-unnecessary-arbitrary-value': 'off',
  165. 'tailwindcss/enforces-shorthand': 'off',
  166. 'tailwindcss/no-custom-classname': 'off',
  167. // in the future
  168. 'tailwindcss/classnames-order': 'off',
  169. },
  170. },
  171. )