module.exports = { extends: [ 'airbnb', 'prettier', 'plugin:compat/recommended', 'plugin:vitest/recommended', 'plugin:react/recommended', 'plugin:import/typescript', 'plugin:markdown/recommended', ], env: { browser: true, node: true, jasmine: true, es6: true, }, settings: { react: { version: 'detect', }, polyfills: ['Promise', 'URL'], 'import/resolver': { typescript: {}, }, }, parser: '@typescript-eslint/parser', plugins: [ 'react', '@babel', 'vitest', '@typescript-eslint', 'react-hooks', 'unicorn', 'markdown', ], // https://github.com/typescript-eslint/typescript-eslint/issues/46#issuecomment-470486034 overrides: [ { files: ['*.ts', '*.tsx'], rules: { '@typescript-eslint/no-unused-vars': [2, { args: 'none' }], 'no-unused-expressions': 'off', '@typescript-eslint/no-unused-expressions': 2, '@typescript-eslint/consistent-type-imports': [2, { disallowTypeAnnotations: false }], }, }, { // In v2, explicitly apply eslint-plugin-markdown's `markdown` // processor on any Markdown files you want to lint. files: ['components/*/demo/*.md'], processor: 'markdown/markdown', }, { // In v2, configuration for fenced code blocks is separate from the // containing Markdown file. Each code block has a virtual filename // appended to the Markdown file's path. files: [ 'components/*/demo/*.md/*.ts', 'components/*/demo/*.md/*.tsx', 'components/*/demo/*.md/*.js', 'components/*/demo/*.md/*.jsx', ], // Configuration for fenced code blocks goes with the override for // the code block's virtual filename, for example: parserOptions: { ecmaFeatures: { impliedStrict: true, }, }, globals: { React: true, ReactDOM: true, mountNode: true, }, rules: { indent: 0, '@babel/new-cap': 0, '@babel/no-invalid-this': 0, '@babel/no-unused-expressions': 2, '@babel/object-curly-spacing': 0, '@babel/semi': 2, 'default-case': 0, 'eol-last': 0, 'no-console': 0, 'no-plusplus': 0, 'no-script-url': 0, 'prefer-rest-params': 0, 'compat/compat': 0, 'class-methods-use-this': 0, 'react/no-access-state-in-setstate': 0, 'react/destructuring-assignment': 0, 'react/no-multi-comp': 0, 'react/no-array-index-key': 0, 'jsx-a11y/href-no-hash': 0, 'jsx-a11y/control-has-associated-label': 0, 'import/no-extraneous-dependencies': 0, 'react/jsx-no-constructed-context-values': 0, 'react/no-unstable-nested-components': 0, }, }, { files: ['components/**/demo/*.tsx'], rules: { 'import/no-extraneous-dependencies': 0, 'no-console': 0, 'compat/compat': 0, 'react/no-unstable-nested-components': 0, 'jsx-a11y/control-has-associated-label': 0, 'class-methods-use-this': 0, 'react/no-access-state-in-setstate': 0, }, }, { files: ['.dumi/**/*.ts', '.dumi/**/*.tsx', '.dumi/**/*.js', '.dumi/**/*.jsx'], rules: { 'import/no-extraneous-dependencies': 0, 'no-console': 0, 'compat/compat': 0, 'react/no-unstable-nested-components': 0, 'jsx-a11y/control-has-associated-label': 0, 'class-methods-use-this': 0, 'react/no-access-state-in-setstate': 0, 'react/no-unknown-property': ['error', { ignore: ['css'] }], 'react/no-array-index-key': 0, 'react/button-has-type': 0, 'react/no-danger': 0, }, }, ], rules: { 'react/jsx-one-expression-per-line': 0, 'react/prop-types': 0, 'react/forbid-prop-types': 0, 'react/jsx-indent': 0, 'react/jsx-wrap-multilines': ['error', { declaration: false, assignment: false }], 'react/jsx-filename-extension': 0, 'react/state-in-constructor': 0, 'react/jsx-props-no-spreading': 0, 'react/destructuring-assignment': 0, // TODO: remove later 'react/require-default-props': 0, 'react/sort-comp': 0, 'react/display-name': 0, 'react/static-property-placement': 0, 'react/jsx-no-bind': 0, // Should not check test file 'react/no-find-dom-node': 0, 'react/no-unused-prop-types': 0, 'react/default-props-match-prop-types': 0, 'react-hooks/rules-of-hooks': 2, // Checks rules of Hooks 'react/function-component-definition': 0, 'react/no-unused-class-component-methods': 0, 'import/extensions': 0, 'import/no-cycle': 0, 'import/no-extraneous-dependencies': [ 'error', { devDependencies: [ 'site/**', 'tests/**', 'scripts/**', '**/*.test.js', '**/__tests__/*', '__mocks__/**', '*.config.js', '**/*.md', 'vitest*config.ts', ], }, ], 'jsx-a11y/no-static-element-interactions': 0, 'jsx-a11y/anchor-has-content': 0, 'jsx-a11y/click-events-have-key-events': 0, 'jsx-a11y/anchor-is-valid': 0, 'jsx-a11y/no-noninteractive-element-interactions': 0, // label-has-for has been deprecated // https://github.com/evcohen/eslint-plugin-jsx-a11y/blob/master/docs/rules/label-has-for.md 'jsx-a11y/label-has-for': 0, 'comma-dangle': ['error', 'always-multiline'], 'consistent-return': 0, // TODO: remove later 'no-param-reassign': 0, // TODO: remove later 'no-underscore-dangle': 0, // for (let i = 0; i < len; i++) 'no-plusplus': 0, // https://eslint.org/docs/rules/no-continue // labeledLoop is conflicted with `eslint . --fix` 'no-continue': 0, // ban this for Number.isNaN needs polyfill 'no-restricted-globals': 0, 'max-classes-per-file': 0, 'vitest/prefer-to-be': 0, 'vitest/expect-expect': 0, 'vitest/no-done-callback': 0, 'vitest/valid-title': 0, 'vitest/no-conditional-expect': 0, 'vitest/no-standalone-expect': 0, 'unicorn/better-regex': 2, 'unicorn/prefer-string-trim-start-end': 2, 'unicorn/expiring-todo-comments': 2, 'unicorn/no-abusive-eslint-disable': 0, // https://github.com/typescript-eslint/typescript-eslint/issues/2540#issuecomment-692866111 'no-use-before-define': 0, '@typescript-eslint/no-use-before-define': 2, 'no-shadow': 0, '@typescript-eslint/no-shadow': [2, { ignoreTypeValueShadow: true }], // https://github.com/typescript-eslint/typescript-eslint/issues/2528#issuecomment-689369395 'no-undef': 0, }, globals: { gtag: true, }, };