Compare commits
	
		
			53 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | fa24ca77a4 | ||
|  | 8bceb3ef50 | ||
|  | 108f55bbe8 | ||
|  | 87419670fd | ||
|  | cdb1849476 | ||
|  | 55ffba3288 | ||
|  | 8ddaee3bf8 | ||
|  | af2c3273cf | ||
|  | 12e57fce54 | ||
|  | b08b769a8a | ||
|  | 65cef9437a | ||
|  | f1b9dc396b | ||
|  | b8fe7fe4e7 | ||
|  | 001de3b060 | ||
|  | 761a0ce4ee | ||
|  | a9d330cac3 | ||
|  | 540786d482 | ||
|  | b876b7b1d9 | ||
|  | d463449db4 | ||
|  | 37d30e9e7d | ||
|  | 58b8383d4c | ||
|  | 35a5d3ef9d | ||
|  | f618005402 | ||
|  | a9b6d5f9ee | ||
|  | e6a4d6db8b | ||
|  | ed0f5b1d76 | ||
|  | 65222e2e3a | ||
|  | a1e890844f | ||
|  | a2de527401 | ||
|  | 6080d3f74d | ||
|  | b996c58bac | ||
|  | b8fe1fabea | ||
|  | 0ab9f88603 | ||
|  | 17670e8e38 | ||
|  | 686f285a37 | ||
|  | 2d9efd4df6 | ||
|  | b8e6cdf047 | ||
|  | 50c962197b | ||
|  | bf19dc0adb | ||
|  | 38f64ae9af | ||
|  | 22527f61de | ||
|  | 1ff53f2f04 | ||
|  | 316bbd903b | ||
|  | bce5016b53 | ||
|  | ddd34a9465 | ||
|  | 2f1c785eef | ||
|  | 4eb7aee28e | ||
|  | 769bee0139 | ||
|  | 07a15be095 | ||
|  | 0af15d4bf4 | ||
|  | d23abea648 | ||
|  | eaff018580 | ||
|  | 327ab9dd35 | 
							
								
								
									
										22
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | ||||
| name: Release | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     tags: | ||||
|       - 'v*' | ||||
|  | ||||
| jobs: | ||||
|   release: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/checkout@v3 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|  | ||||
|       - uses: actions/setup-node@v3 | ||||
|         with: | ||||
|           node-version: 16.x | ||||
|  | ||||
|       - run: npx changelogithub | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{secrets.GITHUB_TOKEN}} | ||||
| @@ -34,7 +34,8 @@ For example: | ||||
| ```json | ||||
| { | ||||
|   "scripts": { | ||||
|     "lint": "eslint ." | ||||
|     "lint": "eslint .", | ||||
|     "lint:fix": "eslint . --fix" | ||||
|   } | ||||
| } | ||||
| ``` | ||||
| @@ -54,10 +55,12 @@ Create `.vscode/settings.json` | ||||
|  | ||||
| ## Check Also | ||||
|  | ||||
| - [antfu/dotfiles](https://github.com/antfu/dotfiles) - My dotfiles | ||||
| - [antfu/vscode-settings](https://github.com/antfu/vscode-settings) - My VS Code settings | ||||
| - [antfu/eslint-config](https://github.com/antfu/eslint-config) - My ESLint config | ||||
| - [antfu/ts-starter](https://github.com/antfu/ts-starter) - My starter template for TypeScript library | ||||
| - [antfu/vitesse](https://github.com/antfu/vitesse) - My starter template for Vue & Vite app | ||||
|  | ||||
| ## License | ||||
|  | ||||
| MIT | ||||
| [MIT](./LICENSE) License © 2019-PRESENT [Anthony Fu](https://github.com/antfu) | ||||
|   | ||||
							
								
								
									
										12
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										12
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,19 +1,21 @@ | ||||
| { | ||||
|   "name": "@antfu/eslint-config-monorepo", | ||||
|   "version": "0.19.4", | ||||
|   "version": "0.26.0", | ||||
|   "private": true, | ||||
|   "license": "MIT", | ||||
|   "packageManager": "pnpm@7.1.0", | ||||
|   "author": "Anthony Fu <anthonyfu117@hotmail.com> (https://github.com/antfu/)", | ||||
|   "license": "MIT", | ||||
|   "scripts": { | ||||
|     "lint": "eslint .", | ||||
|     "test": "pnpm -r run test", | ||||
|     "prepare": "pnpm -r run stub", | ||||
|     "release": "bumpp package.json packages/*/package.json --commit --push --tag && pnpm -r publish --access public" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@antfu/eslint-config": "*", | ||||
|     "bumpp": "^7.1.1", | ||||
|     "eslint": "^8.12.0", | ||||
|     "bumpp": "^8.2.1", | ||||
|     "eslint": "^8.21.0", | ||||
|     "eslint-plugin-antfu": "workspace:*", | ||||
|     "typescript": "^4.6.3" | ||||
|     "typescript": "^4.7.4" | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1,10 +1,13 @@ | ||||
| { | ||||
|   "name": "@antfu/eslint-config", | ||||
|   "version": "0.19.4", | ||||
|   "description": "", | ||||
|   "keywords": [], | ||||
|   "license": "MIT", | ||||
|   "version": "0.26.0", | ||||
|   "description": "Anthony's ESLint config", | ||||
|   "author": "Anthony Fu <anthonyfu117@hotmail.com> (https://github.com/antfu/)", | ||||
|   "license": "MIT", | ||||
|   "homepage": "https://github.com/antfu/eslint-config", | ||||
|   "keywords": [ | ||||
|     "eslint-config" | ||||
|   ], | ||||
|   "main": "index.js", | ||||
|   "files": [ | ||||
|     "index.js" | ||||
| @@ -15,24 +18,21 @@ | ||||
|   "dependencies": { | ||||
|     "@antfu/eslint-config-react": "workspace:*", | ||||
|     "@antfu/eslint-config-vue": "workspace:*", | ||||
|     "@typescript-eslint/eslint-plugin": "^5.17.0", | ||||
|     "@typescript-eslint/parser": "^5.17.0", | ||||
|     "@typescript-eslint/eslint-plugin": "^5.32.0", | ||||
|     "@typescript-eslint/parser": "^5.32.0", | ||||
|     "eslint-plugin-eslint-comments": "^3.2.0", | ||||
|     "eslint-plugin-html": "^6.2.0", | ||||
|     "eslint-plugin-import": "^2.25.4", | ||||
|     "eslint-plugin-jsonc": "^2.2.1", | ||||
|     "eslint-plugin-n": "^15.1.0", | ||||
|     "eslint-plugin-html": "^7.1.0", | ||||
|     "eslint-plugin-import": "^2.26.0", | ||||
|     "eslint-plugin-jsonc": "^2.3.1", | ||||
|     "eslint-plugin-n": "^15.2.4", | ||||
|     "eslint-plugin-promise": "^6.0.0", | ||||
|     "eslint-plugin-unicorn": "^41.0.1", | ||||
|     "eslint-plugin-vue": "^8.5.0", | ||||
|     "eslint-plugin-yml": "^0.14.0", | ||||
|     "eslint-plugin-unicorn": "^43.0.2", | ||||
|     "eslint-plugin-vue": "^9.3.0", | ||||
|     "eslint-plugin-yml": "^1.1.0", | ||||
|     "jsonc-eslint-parser": "^2.1.0", | ||||
|     "yaml-eslint-parser": "^0.5.0" | ||||
|     "yaml-eslint-parser": "^1.1.0" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "eslint": "^8.12.0" | ||||
|   }, | ||||
|   "publishConfig": { | ||||
|     "access": "public" | ||||
|     "eslint": "^8.21.0" | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -4,6 +4,7 @@ module.exports = { | ||||
|     browser: true, | ||||
|     node: true, | ||||
|   }, | ||||
|   reportUnusedDisableDirectives: true, | ||||
|   extends: [ | ||||
|     './standard', | ||||
|     'plugin:import/recommended', | ||||
| @@ -14,6 +15,7 @@ module.exports = { | ||||
|   ], | ||||
|   ignorePatterns: [ | ||||
|     '*.min.*', | ||||
|     '*.d.ts', | ||||
|     'CHANGELOG.md', | ||||
|     'dist', | ||||
|     'LICENSE*', | ||||
| @@ -44,9 +46,15 @@ module.exports = { | ||||
|       files: ['*.json', '*.json5'], | ||||
|       parser: 'jsonc-eslint-parser', | ||||
|       rules: { | ||||
|         'quotes': ['error', 'double'], | ||||
|         'quote-props': ['error', 'always'], | ||||
|         'comma-dangle': ['error', 'never'], | ||||
|         'jsonc/array-bracket-spacing': ['error', 'never'], | ||||
|         'jsonc/comma-dangle': ['error', 'never'], | ||||
|         'jsonc/comma-style': ['error', 'last'], | ||||
|         'jsonc/indent': ['error', 2], | ||||
|         'jsonc/key-spacing': ['error', { beforeColon: false, afterColon: true }], | ||||
|         'jsonc/no-octal-escape': 'error', | ||||
|         'jsonc/object-curly-newline': ['error', { multiline: true, consistent: true }], | ||||
|         'jsonc/object-curly-spacing': ['error', 'always'], | ||||
|         'jsonc/object-property-newline': ['error', { allowMultiplePropertiesPerLine: true }], | ||||
|       }, | ||||
|     }, | ||||
|     { | ||||
| @@ -65,33 +73,47 @@ module.exports = { | ||||
|           { | ||||
|             pathPattern: '^$', | ||||
|             order: [ | ||||
|               'publisher', | ||||
|               'name', | ||||
|               'displayName', | ||||
|               'type', | ||||
|               'version', | ||||
|               'private', | ||||
|               'packageManager', | ||||
|               'description', | ||||
|               'keywords', | ||||
|               'license', | ||||
|               'author', | ||||
|               'repository', | ||||
|               'license', | ||||
|               'funding', | ||||
|               'homepage', | ||||
|               'repository', | ||||
|               'bugs', | ||||
|               'keywords', | ||||
|               'categories', | ||||
|               'sideEffects', | ||||
|               'exports', | ||||
|               'main', | ||||
|               'module', | ||||
|               'types', | ||||
|               'unpkg', | ||||
|               'jsdelivr', | ||||
|               'exports', | ||||
|               'files', | ||||
|               'types', | ||||
|               'typesVersions', | ||||
|               'bin', | ||||
|               'sideEffects', | ||||
|               'icon', | ||||
|               'files', | ||||
|               'engines', | ||||
|               'activationEvents', | ||||
|               'contributes', | ||||
|               'scripts', | ||||
|               'peerDependencies', | ||||
|               'peerDependenciesMeta', | ||||
|               'dependencies', | ||||
|               'optionalDependencies', | ||||
|               'devDependencies', | ||||
|               'pnpm', | ||||
|               'overrides', | ||||
|               'resolutions', | ||||
|               'husky', | ||||
|               'simple-git-hooks', | ||||
|               'lint-staged', | ||||
|               'eslintConfig', | ||||
|             ], | ||||
| @@ -100,6 +122,14 @@ module.exports = { | ||||
|             pathPattern: '^(?:dev|peer|optional|bundled)?[Dd]ependencies$', | ||||
|             order: { type: 'asc' }, | ||||
|           }, | ||||
|           { | ||||
|             pathPattern: '^exports.*$', | ||||
|             order: [ | ||||
|               'types', | ||||
|               'require', | ||||
|               'import', | ||||
|             ], | ||||
|           }, | ||||
|         ], | ||||
|       }, | ||||
|     }, | ||||
| @@ -135,6 +165,7 @@ module.exports = { | ||||
|         '@typescript-eslint/no-unused-vars': 'off', | ||||
|         '@typescript-eslint/no-use-before-define': 'off', | ||||
|         '@typescript-eslint/no-var-requires': 'off', | ||||
|         '@typescript-eslint/comma-dangle': 'off', | ||||
|         'import/no-unresolved': 'off', | ||||
|         'no-alert': 'off', | ||||
|         'no-console': 'off', | ||||
| @@ -182,7 +213,15 @@ module.exports = { | ||||
|     ], | ||||
|     'object-curly-spacing': ['error', 'always'], | ||||
|     'no-return-await': 'off', | ||||
|     'space-before-function-paren': ['error', 'never'], | ||||
|     'space-before-function-paren': [ | ||||
|       'error', | ||||
|       { | ||||
|         anonymous: 'always', | ||||
|         named: 'never', | ||||
|         asyncArrow: 'always', | ||||
|       }, | ||||
|     ], | ||||
|     'no-multiple-empty-lines': ['error', { max: 1, maxBOF: 0, maxEOF: 1 }], | ||||
|  | ||||
|     // es6 | ||||
|     'no-var': 'error', | ||||
| @@ -208,6 +247,7 @@ module.exports = { | ||||
|         avoidQuotes: true, | ||||
|       }, | ||||
|     ], | ||||
|     'prefer-exponentiation-operator': 'error', | ||||
|     'prefer-rest-params': 'error', | ||||
|     'prefer-spread': 'error', | ||||
|     'prefer-template': 'error', | ||||
| @@ -250,15 +290,13 @@ module.exports = { | ||||
|     // Uppercase regex escapes | ||||
|     'unicorn/escape-case': 'error', | ||||
|     // Array.isArray instead of instanceof | ||||
|     'unicorn/no-array-instanceof': 'error', | ||||
|     'unicorn/no-instanceof-array': 'error', | ||||
|     // Prevent deprecated `new Buffer()` | ||||
|     'unicorn/no-new-buffer': 'error', | ||||
|     // Keep regex literals safe! | ||||
|     'unicorn/no-unsafe-regex': 'off', | ||||
|     // Lowercase number formatting for octal, hex, binary (0x1'error' instead of 0X1'error') | ||||
|     'unicorn/number-literal-case': 'error', | ||||
|     // ** instead of Math.pow() | ||||
|     'unicorn/prefer-exponentiation-operator': 'error', | ||||
|     // includes over indexOf when checking for existence | ||||
|     'unicorn/prefer-includes': 'error', | ||||
|     // String methods startsWith/endsWith instead of more complicated stuff | ||||
| @@ -273,6 +311,8 @@ module.exports = { | ||||
|     'no-use-before-define': ['error', { functions: false, classes: false, variables: true }], | ||||
|     'eslint-comments/disable-enable-pair': 'off', | ||||
|     'import/no-named-as-default-member': 'off', | ||||
|     'import/no-named-as-default': 'off', | ||||
|     'import/namespace': 'off', | ||||
|     'n/no-callback-literal': 'off', | ||||
|  | ||||
|     'sort-imports': [ | ||||
| @@ -291,7 +331,8 @@ module.exports = { | ||||
|     'yml/no-empty-document': 'off', | ||||
|  | ||||
|     // antfu | ||||
|     'antfu/no-leading-newline': 'error', | ||||
|     'antfu/if-newline': 'error', | ||||
|     'antfu/import-dedupe': 'error', | ||||
|     // 'antfu/prefer-inline-type-import': 'error', | ||||
|   }, | ||||
| } | ||||
|   | ||||
| @@ -1,10 +1,13 @@ | ||||
| { | ||||
|   "name": "@antfu/eslint-config-basic", | ||||
|   "version": "0.19.4", | ||||
|   "version": "0.26.0", | ||||
|   "description": "", | ||||
|   "keywords": [], | ||||
|   "license": "MIT", | ||||
|   "author": "Anthony Fu <anthonyfu117@hotmail.com> (https://github.com/antfu/)", | ||||
|   "license": "MIT", | ||||
|   "homepage": "https://github.com/antfu/eslint-config", | ||||
|   "keywords": [ | ||||
|     "eslint-config" | ||||
|   ], | ||||
|   "main": "index.js", | ||||
|   "files": [ | ||||
|     "*.js" | ||||
| @@ -18,21 +21,18 @@ | ||||
|   "dependencies": { | ||||
|     "eslint-plugin-antfu": "workspace:*", | ||||
|     "eslint-plugin-eslint-comments": "^3.2.0", | ||||
|     "eslint-plugin-html": "^6.2.0", | ||||
|     "eslint-plugin-import": "^2.25.4", | ||||
|     "eslint-plugin-jsonc": "^2.2.1", | ||||
|     "eslint-plugin-markdown": "^2.2.1", | ||||
|     "eslint-plugin-n": "^15.1.0", | ||||
|     "eslint-plugin-html": "^7.1.0", | ||||
|     "eslint-plugin-import": "^2.26.0", | ||||
|     "eslint-plugin-jsonc": "^2.3.1", | ||||
|     "eslint-plugin-markdown": "^3.0.0", | ||||
|     "eslint-plugin-n": "^15.2.4", | ||||
|     "eslint-plugin-promise": "^6.0.0", | ||||
|     "eslint-plugin-unicorn": "^41.0.1", | ||||
|     "eslint-plugin-yml": "^0.14.0", | ||||
|     "eslint-plugin-unicorn": "^43.0.2", | ||||
|     "eslint-plugin-yml": "^1.1.0", | ||||
|     "jsonc-eslint-parser": "^2.1.0", | ||||
|     "yaml-eslint-parser": "^0.5.0" | ||||
|     "yaml-eslint-parser": "^1.1.0" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "eslint": "^8.12.0" | ||||
|   }, | ||||
|   "publishConfig": { | ||||
|     "access": "public" | ||||
|     "eslint": "^8.21.0" | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -202,7 +202,14 @@ module.exports = { | ||||
|     'semi': ['error', 'never'], | ||||
|     'semi-spacing': ['error', { before: false, after: true }], | ||||
|     'space-before-blocks': ['error', 'always'], | ||||
|     'space-before-function-paren': ['error', 'always'], | ||||
|     'space-before-function-paren': [ | ||||
|       'error', | ||||
|       { | ||||
|         anonymous: 'always', | ||||
|         named: 'never', | ||||
|         asyncArrow: 'always', | ||||
|       }, | ||||
|     ], | ||||
|     'space-in-parens': ['error', 'never'], | ||||
|     'space-infix-ops': 'error', | ||||
|     'space-unary-ops': ['error', { words: true, nonwords: false }], | ||||
|   | ||||
| @@ -2,8 +2,5 @@ | ||||
|   "extends": "@antfu", | ||||
|   "plugins": [ | ||||
|     "antfu" | ||||
|   ], | ||||
|   "rules": { | ||||
|     "antfu/no-leading-newline": "error" | ||||
|   } | ||||
|   ] | ||||
| } | ||||
|   | ||||
| @@ -1,17 +1,11 @@ | ||||
| { | ||||
|   "name": "eslint-plugin-antfu", | ||||
|   "version": "0.19.4", | ||||
|   "version": "0.26.0", | ||||
|   "license": "MIT", | ||||
|   "homepage": "https://github.com/antfu/eslint-config", | ||||
|   "main": "./dist/index.cjs", | ||||
|   "module": "./dist/index.mjs", | ||||
|   "types": "./dist/index.d.ts", | ||||
|   "exports": { | ||||
|     ".": { | ||||
|       "require": "./dist/index.cjs", | ||||
|       "import": "./dist/index.mjs", | ||||
|       "types": "./dist/index.d.ts" | ||||
|     } | ||||
|   }, | ||||
|   "files": [ | ||||
|     "dist" | ||||
|   ], | ||||
| @@ -22,10 +16,10 @@ | ||||
|     "prepublishOnly": "nr build" | ||||
|   }, | ||||
|   "dependencies": { | ||||
|     "@typescript-eslint/utils": "^5.17.0" | ||||
|     "@typescript-eslint/utils": "^5.32.0" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "unbuild": "^0.7.0", | ||||
|     "vitest": "^0.8.2" | ||||
|     "unbuild": "^0.7.6", | ||||
|     "vitest": "^0.21.0" | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1,9 +1,11 @@ | ||||
| import ifNewline from './rules/if-newline' | ||||
| import noLeadingNewline from './rules/no-leading-newline' | ||||
| import importDedupe from './rules/import-dedupe' | ||||
| import preferInlineTypeImport from './rules/prefer-inline-type-import' | ||||
|  | ||||
| export default { | ||||
|   rules: { | ||||
|     'no-leading-newline': noLeadingNewline, | ||||
|     'if-newline': ifNewline, | ||||
|     'import-dedupe': importDedupe, | ||||
|     'prefer-inline-type-import': preferInlineTypeImport, | ||||
|   }, | ||||
| } | ||||
|   | ||||
| @@ -1,14 +1,15 @@ | ||||
| import { RuleTester } from '@typescript-eslint/utils/dist/ts-eslint' | ||||
| import { it } from 'vitest' | ||||
| import rule, { RULE_NAME } from './no-leading-newline' | ||||
| import rule, { RULE_NAME } from './import-dedupe' | ||||
| 
 | ||||
| const valids = [ | ||||
|   'import {} from \'foo\'', | ||||
|   `// comment
 | ||||
| import {} from ''`,
 | ||||
|   'import { a } from \'foo\'', | ||||
| ] | ||||
| const invalids = [ | ||||
|   '\n\nimport {} from \'fo\'', | ||||
|   [ | ||||
|     'import { a, b, a, a, c, a } from \'foo\'', | ||||
|     'import { a, b,   c,  } from \'foo\'', | ||||
|   ], | ||||
| ] | ||||
| 
 | ||||
| it('runs', () => { | ||||
| @@ -19,9 +20,9 @@ it('runs', () => { | ||||
|   ruleTester.run(RULE_NAME, rule, { | ||||
|     valid: valids, | ||||
|     invalid: invalids.map(i => ({ | ||||
|       code: i, | ||||
|       output: i.trim(), | ||||
|       errors: [{ messageId: 'noLeadingNewline' }], | ||||
|       code: i[0], | ||||
|       output: i[1], | ||||
|       errors: [{ messageId: 'importDedupe' }, { messageId: 'importDedupe' }, { messageId: 'importDedupe' }], | ||||
|     })), | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										55
									
								
								packages/eslint-plugin-antfu/src/rules/import-dedupe.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								packages/eslint-plugin-antfu/src/rules/import-dedupe.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,55 @@ | ||||
| import { createEslintRule } from '../utils' | ||||
|  | ||||
| export const RULE_NAME = 'import-dedupe' | ||||
| export type MessageIds = 'importDedupe' | ||||
| export type Options = [] | ||||
|  | ||||
| export default createEslintRule<Options, MessageIds>({ | ||||
|   name: RULE_NAME, | ||||
|   meta: { | ||||
|     type: 'problem', | ||||
|     docs: { | ||||
|       description: 'Fix duplication in imports', | ||||
|       recommended: 'error', | ||||
|     }, | ||||
|     fixable: 'code', | ||||
|     schema: [], | ||||
|     messages: { | ||||
|       importDedupe: 'Expect no duplication in imports', | ||||
|     }, | ||||
|   }, | ||||
|   defaultOptions: [], | ||||
|   create: (context) => { | ||||
|     return { | ||||
|       ImportDeclaration(node) { | ||||
|         if (node.specifiers.length <= 1) | ||||
|           return | ||||
|  | ||||
|         const names = new Set<string>() | ||||
|         node.specifiers.forEach((n) => { | ||||
|           const id = n.local.name | ||||
|           if (names.has(id)) { | ||||
|             context.report({ | ||||
|               node, | ||||
|               loc: { | ||||
|                 start: n.loc.end, | ||||
|                 end: n.loc.start, | ||||
|               }, | ||||
|               messageId: 'importDedupe', | ||||
|               fix(fixer) { | ||||
|                 const s = n.range[0] | ||||
|                 let e = n.range[1] | ||||
|                 if (context.getSourceCode().text[e] === ',') | ||||
|                   e += 1 | ||||
|                 return fixer.removeRange([s, e]) | ||||
|               }, | ||||
|             }) | ||||
|           } | ||||
|           names.add(id) | ||||
|         }) | ||||
|  | ||||
|         // console.log(node) | ||||
|       }, | ||||
|     } | ||||
|   }, | ||||
| }) | ||||
| @@ -1,44 +0,0 @@ | ||||
| import { createEslintRule } from '../utils' | ||||
|  | ||||
| export const RULE_NAME = 'no-leading-newline' | ||||
| export type MessageIds = 'noLeadingNewline' | ||||
| export type Options = [] | ||||
|  | ||||
| export default createEslintRule<Options, MessageIds>({ | ||||
|   name: RULE_NAME, | ||||
|   meta: { | ||||
|     type: 'problem', | ||||
|     docs: { | ||||
|       description: 'Do not allow leading newline', | ||||
|       recommended: 'error', | ||||
|     }, | ||||
|     fixable: 'whitespace', | ||||
|     schema: [], | ||||
|     messages: { | ||||
|       noLeadingNewline: 'No leading newline allowed', | ||||
|     }, | ||||
|   }, | ||||
|   defaultOptions: [], | ||||
|   create: (context) => { | ||||
|     return { | ||||
|       Program(node) { | ||||
|         const code = context.getSourceCode() | ||||
|         const match = code.text.match(/^[\s]+/)?.[0] || '' | ||||
|         if (match.includes('\n')) { | ||||
|           const line = match.split('\n') | ||||
|           context.report({ | ||||
|             node, | ||||
|             loc: { | ||||
|               start: { line: 0, column: 0 }, | ||||
|               end: { line: line.length - 1, column: line[line.length - 1].length }, | ||||
|             }, | ||||
|             messageId: 'noLeadingNewline', | ||||
|             fix(fixer) { | ||||
|               return fixer.replaceTextRange([0, match.length], '') | ||||
|             }, | ||||
|           }) | ||||
|         } | ||||
|       }, | ||||
|     } | ||||
|   }, | ||||
| }) | ||||
| @@ -0,0 +1,27 @@ | ||||
| import { RuleTester } from '@typescript-eslint/utils/dist/ts-eslint' | ||||
| import { it } from 'vitest' | ||||
| import rule, { RULE_NAME } from './prefer-inline-type-import' | ||||
|  | ||||
| const valids = [ | ||||
|   'import { type Foo } from \'foo\'', | ||||
|   'import type Foo from \'foo\'', | ||||
|   'import type * as Foo from \'foo\'', | ||||
| ] | ||||
| const invalids = [ | ||||
|   ['import type { Foo } from \'foo\'', 'import { type Foo } from \'foo\''], | ||||
| ] | ||||
|  | ||||
| it('runs', () => { | ||||
|   const ruleTester: RuleTester = new RuleTester({ | ||||
|     parser: require.resolve('@typescript-eslint/parser'), | ||||
|   }) | ||||
|  | ||||
|   ruleTester.run(RULE_NAME, rule, { | ||||
|     valid: valids, | ||||
|     invalid: invalids.map(i => ({ | ||||
|       code: i[0], | ||||
|       output: i[1].trim(), | ||||
|       errors: [{ messageId: 'preferInlineTypeImport' }], | ||||
|     })), | ||||
|   }) | ||||
| }) | ||||
| @@ -0,0 +1,63 @@ | ||||
| // Ported from https://github.com/gajus/eslint-plugin-canonical/blob/master/src/rules/preferInlineTypeImport.js | ||||
| // by Gajus Kuizinas https://github.com/gajus | ||||
|  | ||||
| import { createEslintRule } from '../utils' | ||||
|  | ||||
| export const RULE_NAME = 'prefer-inline-type-import' | ||||
| export type MessageIds = 'preferInlineTypeImport' | ||||
| export type Options = [] | ||||
|  | ||||
| export default createEslintRule<Options, MessageIds>({ | ||||
|   name: RULE_NAME, | ||||
|   meta: { | ||||
|     type: 'suggestion', | ||||
|     docs: { | ||||
|       description: 'Newline after if', | ||||
|       recommended: 'error', | ||||
|     }, | ||||
|     fixable: 'code', | ||||
|     schema: [], | ||||
|     messages: { | ||||
|       preferInlineTypeImport: 'Prefer inline type import', | ||||
|     }, | ||||
|   }, | ||||
|   defaultOptions: [], | ||||
|   create: (context) => { | ||||
|     const sourceCode = context.getSourceCode() | ||||
|     return { | ||||
|       ImportDeclaration: (node) => { | ||||
|         // ignore bare type imports | ||||
|         if (node.specifiers.length === 1 && ['ImportNamespaceSpecifier', 'ImportDefaultSpecifier'].includes(node.specifiers[0].type)) | ||||
|           return | ||||
|         if (node.importKind === 'type') { | ||||
|           context.report({ | ||||
|             *fix(fixer) { | ||||
|               yield * removeTypeSpecifier(fixer, sourceCode, node) | ||||
|  | ||||
|               for (const specifier of node.specifiers) | ||||
|                 yield fixer.insertTextBefore(specifier, 'type ') | ||||
|             }, | ||||
|             loc: node.loc, | ||||
|             messageId: 'preferInlineTypeImport', | ||||
|             node, | ||||
|           }) | ||||
|         } | ||||
|       }, | ||||
|     } | ||||
|   }, | ||||
| }) | ||||
|  | ||||
| function *removeTypeSpecifier(fixer, sourceCode, node) { | ||||
|   const importKeyword = sourceCode.getFirstToken(node) | ||||
|  | ||||
|   const typeIdentifier = sourceCode.getTokenAfter(importKeyword) | ||||
|  | ||||
|   yield fixer.remove(typeIdentifier) | ||||
|  | ||||
|   if (importKeyword.loc.end.column + 1 === typeIdentifier.loc.start.column) { | ||||
|     yield fixer.removeRange([ | ||||
|       importKeyword.range[1], | ||||
|       importKeyword.range[1] + 1, | ||||
|     ]) | ||||
|   } | ||||
| } | ||||
| @@ -1,10 +1,13 @@ | ||||
| { | ||||
|   "name": "@antfu/eslint-config-react", | ||||
|   "version": "0.19.4", | ||||
|   "version": "0.26.0", | ||||
|   "description": "", | ||||
|   "keywords": [], | ||||
|   "license": "MIT", | ||||
|   "author": "Anthony Fu <anthonyfu117@hotmail.com> (https://github.com/antfu/)", | ||||
|   "license": "MIT", | ||||
|   "homepage": "https://github.com/antfu/eslint-config", | ||||
|   "keywords": [ | ||||
|     "eslint-config" | ||||
|   ], | ||||
|   "main": "index.js", | ||||
|   "files": [ | ||||
|     "index.js" | ||||
| @@ -15,16 +18,13 @@ | ||||
|   "peerDependencies": { | ||||
|     "eslint": ">=7.4.0" | ||||
|   }, | ||||
|   "publishConfig": { | ||||
|     "access": "public" | ||||
|   }, | ||||
|   "dependencies": { | ||||
|     "@antfu/eslint-config-ts": "workspace:*", | ||||
|     "eslint-plugin-react": "^7.29.4" | ||||
|     "eslint-plugin-react": "^7.30.1" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "eslint": "^8.12.0", | ||||
|     "react": "^18.0.0", | ||||
|     "typescript": "^4.6.3" | ||||
|     "eslint": "^8.21.0", | ||||
|     "react": "^18.2.0", | ||||
|     "typescript": "^4.7.4" | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1,4 +1,3 @@ | ||||
| // eslint-disable-next-line @typescript-eslint/no-var-requires | ||||
| const basic = require('@antfu/eslint-config-basic') | ||||
|  | ||||
| module.exports = { | ||||
| @@ -17,19 +16,51 @@ module.exports = { | ||||
|     'import/named': 'off', | ||||
|  | ||||
|     // TS | ||||
|     '@typescript-eslint/semi': ['error', 'never'], | ||||
|     '@typescript-eslint/ban-ts-comment': ['error', { 'ts-ignore': 'allow-with-description' }], | ||||
|     '@typescript-eslint/member-delimiter-style': ['error', { multiline: { delimiter: 'none' } }], | ||||
|     '@typescript-eslint/type-annotation-spacing': ['error', {}], | ||||
|     '@typescript-eslint/consistent-type-imports': ['error', { prefer: 'type-imports', disallowTypeAnnotations: false }], | ||||
|     '@typescript-eslint/consistent-type-definitions': ['error', 'interface'], | ||||
|     '@typescript-eslint/consistent-indexed-object-style': ['error', 'record'], | ||||
|     '@typescript-eslint/prefer-ts-expect-error': 'error', | ||||
|  | ||||
|     // Override JS | ||||
|     'no-useless-constructor': 'off', | ||||
|     'indent': 'off', | ||||
|     '@typescript-eslint/indent': ['error', 2], | ||||
|     '@typescript-eslint/indent': ['error', 2, { | ||||
|       SwitchCase: 1, | ||||
|       VariableDeclarator: 1, | ||||
|       outerIIFEBody: 1, | ||||
|       MemberExpression: 1, | ||||
|       FunctionDeclaration: { parameters: 1, body: 1 }, | ||||
|       FunctionExpression: { parameters: 1, body: 1 }, | ||||
|       CallExpression: { arguments: 1 }, | ||||
|       ArrayExpression: 1, | ||||
|       ObjectExpression: 1, | ||||
|       ImportDeclaration: 1, | ||||
|       flatTernaryExpressions: false, | ||||
|       ignoreComments: false, | ||||
|       ignoredNodes: [ | ||||
|         'TemplateLiteral *', | ||||
|         'JSXElement', | ||||
|         'JSXElement > *', | ||||
|         'JSXAttribute', | ||||
|         'JSXIdentifier', | ||||
|         'JSXNamespacedName', | ||||
|         'JSXMemberExpression', | ||||
|         'JSXSpreadAttribute', | ||||
|         'JSXExpressionContainer', | ||||
|         'JSXOpeningElement', | ||||
|         'JSXClosingElement', | ||||
|         'JSXFragment', | ||||
|         'JSXOpeningFragment', | ||||
|         'JSXClosingFragment', | ||||
|         'JSXText', | ||||
|         'JSXEmptyExpression', | ||||
|         'JSXSpreadChild', | ||||
|         'TSTypeParameterInstantiation', | ||||
|       ], | ||||
|       offsetTernaryExpressions: true, | ||||
|     }], | ||||
|     'no-unused-vars': 'off', | ||||
|     '@typescript-eslint/no-unused-vars': ['error', { argsIgnorePattern: '^_' }], | ||||
|     'no-redeclare': 'off', | ||||
| @@ -42,13 +73,51 @@ module.exports = { | ||||
|     '@typescript-eslint/comma-dangle': ['error', 'always-multiline'], | ||||
|     'object-curly-spacing': 'off', | ||||
|     '@typescript-eslint/object-curly-spacing': ['error', 'always'], | ||||
|     'semi': 'off', | ||||
|     '@typescript-eslint/semi': ['error', 'never'], | ||||
|     'quotes': 'off', | ||||
|     '@typescript-eslint/quotes': ['error', 'single'], | ||||
|     'space-before-blocks': 'off', | ||||
|     '@typescript-eslint/space-before-blocks': ['error', 'always'], | ||||
|     'space-before-function-paren': 'off', | ||||
|     '@typescript-eslint/space-before-function-paren': [ | ||||
|       'error', | ||||
|       { | ||||
|         anonymous: 'always', | ||||
|         named: 'never', | ||||
|         asyncArrow: 'always', | ||||
|       }, | ||||
|     ], | ||||
|     'space-infix-ops': 'off', | ||||
|     '@typescript-eslint/space-infix-ops': 'error', | ||||
|     'keyword-spacing': 'off', | ||||
|     '@typescript-eslint/keyword-spacing': ['error', { before: true, after: true }], | ||||
|     'comma-spacing': 'off', | ||||
|     '@typescript-eslint/comma-spacing': ['error', { before: false, after: true }], | ||||
|     'no-extra-parens': 'off', | ||||
|     '@typescript-eslint/no-extra-parens': ['error', 'functions'], | ||||
|     'no-dupe-class-members': 'off', | ||||
|     '@typescript-eslint/no-dupe-class-members': 'error', | ||||
|     'no-loss-of-precision': 'off', | ||||
|     '@typescript-eslint/no-loss-of-precision': 'error', | ||||
|     'lines-between-class-members': 'off', | ||||
|     '@typescript-eslint/lines-between-class-members': ['error', 'always', { exceptAfterSingleLine: true }], | ||||
|     // The following rule overrides require a parser service, aka. require a `typescript.json` path. | ||||
|     // This needs to be done individually for each project, and it slows down linting significantly. | ||||
|     // 'no-throw-literal': 'off', | ||||
|     // '@typescript-eslint/no-throw-literal': 'error', | ||||
|     // 'no-implied-eval': 'off', | ||||
|     // '@typescript-eslint/no-implied-eval': 'error', | ||||
|     // 'dot-notation': 'off', | ||||
|     // '@typescript-eslint/dot-notation': ['error', { allowKeywords: true }], | ||||
|  | ||||
|     // off | ||||
|     '@typescript-eslint/camelcase': 'off', | ||||
|     '@typescript-eslint/consistent-indexed-object-style': 'off', | ||||
|     '@typescript-eslint/naming-convention': 'off', | ||||
|     '@typescript-eslint/explicit-function-return-type': 'off', | ||||
|     '@typescript-eslint/explicit-member-accessibility': 'off', | ||||
|     '@typescript-eslint/no-explicit-any': 'off', | ||||
|     '@typescript-eslint/no-parameter-properties': 'off', | ||||
|     '@typescript-eslint/parameter-properties': 'off', | ||||
|     '@typescript-eslint/no-empty-interface': 'off', | ||||
|     '@typescript-eslint/ban-ts-ignore': 'off', | ||||
|     '@typescript-eslint/no-empty-function': 'off', | ||||
| @@ -56,5 +125,6 @@ module.exports = { | ||||
|     '@typescript-eslint/explicit-module-boundary-types': 'off', | ||||
|     '@typescript-eslint/ban-types': 'off', | ||||
|     '@typescript-eslint/no-namespace': 'off', | ||||
|     '@typescript-eslint/triple-slash-reference': 'off', | ||||
|   }, | ||||
| } | ||||
|   | ||||
| @@ -1,27 +1,27 @@ | ||||
| { | ||||
|   "name": "@antfu/eslint-config-ts", | ||||
|   "version": "0.19.4", | ||||
|   "version": "0.26.0", | ||||
|   "description": "", | ||||
|   "keywords": [], | ||||
|   "license": "MIT", | ||||
|   "author": "Anthony Fu <anthonyfu117@hotmail.com> (https://github.com/antfu/)", | ||||
|   "license": "MIT", | ||||
|   "homepage": "https://github.com/antfu/eslint-config", | ||||
|   "keywords": [ | ||||
|     "eslint-config" | ||||
|   ], | ||||
|   "main": "index.js", | ||||
|   "files": [ | ||||
|     "index.js" | ||||
|   ], | ||||
|   "publishConfig": { | ||||
|     "access": "public" | ||||
|   }, | ||||
|   "peerDependencies": { | ||||
|     "eslint": ">=7.4.0", | ||||
|     "typescript": ">=3.9" | ||||
|   }, | ||||
|   "dependencies": { | ||||
|     "@antfu/eslint-config-basic": "workspace:*", | ||||
|     "@typescript-eslint/eslint-plugin": "^5.17.0", | ||||
|     "@typescript-eslint/parser": "^5.17.0" | ||||
|     "@typescript-eslint/eslint-plugin": "^5.32.0", | ||||
|     "@typescript-eslint/parser": "^5.32.0" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "eslint": "^8.12.0" | ||||
|     "eslint": "^8.21.0" | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -23,6 +23,7 @@ module.exports = { | ||||
|     'vue/require-prop-types': 'off', | ||||
|     'vue/require-default-prop': 'off', | ||||
|     'vue/multi-word-component-names': 'off', | ||||
|     'vue/prefer-import-from-vue': 'off', | ||||
|  | ||||
|     // reactivity transform | ||||
|     'vue/no-setup-props-destructure': 'off', | ||||
| @@ -30,5 +31,68 @@ module.exports = { | ||||
|     'vue/component-tags-order': ['error', { | ||||
|       order: ['script', 'template', 'style'], | ||||
|     }], | ||||
|     'vue/block-tag-newline': ['error', { | ||||
|       singleline: 'always', | ||||
|       multiline: 'always', | ||||
|     }], | ||||
|     'vue/component-name-in-template-casing': ['error', 'PascalCase'], | ||||
|     'vue/component-options-name-casing': ['error', 'PascalCase'], | ||||
|     'vue/custom-event-name-casing': ['error', 'camelCase'], | ||||
|     'vue/define-macros-order': ['error', { | ||||
|       order: ['defineProps', 'defineEmits'], | ||||
|     }], | ||||
|     'vue/html-comment-content-spacing': ['error', 'always', { | ||||
|       exceptions: ['-'], | ||||
|     }], | ||||
|     'vue/no-restricted-v-bind': ['error', '/^v-/'], | ||||
|     'vue/no-useless-v-bind': 'error', | ||||
|     'vue/no-v-text-v-html-on-component': 'error', | ||||
|     'vue/padding-line-between-blocks': ['error', 'always'], | ||||
|     'vue/prefer-separate-static-class': 'error', | ||||
|  | ||||
|     // extensions | ||||
|     'vue/array-bracket-spacing': ['error', 'never'], | ||||
|     'vue/arrow-spacing': ['error', { before: true, after: true }], | ||||
|     'vue/block-spacing': ['error', 'always'], | ||||
|     'vue/brace-style': ['error', 'stroustrup', { allowSingleLine: true }], | ||||
|     'vue/comma-dangle': ['error', 'always-multiline'], | ||||
|     'vue/comma-spacing': ['error', { before: false, after: true }], | ||||
|     'vue/comma-style': ['error', 'last'], | ||||
|     'vue/dot-location': ['error', 'property'], | ||||
|     'vue/dot-notation': ['error', { allowKeywords: true }], | ||||
|     'vue/eqeqeq': ['error', 'smart'], | ||||
|     // 'vue/func-call-spacing': ['off', 'never'], | ||||
|     'vue/key-spacing': ['error', { beforeColon: false, afterColon: true }], | ||||
|     'vue/keyword-spacing': ['error', { before: true, after: true }], | ||||
|     'vue/no-constant-condition': 'warn', | ||||
|     'vue/no-empty-pattern': 'error', | ||||
|     'vue/no-extra-parens': ['error', 'functions'], | ||||
|     'vue/no-irregular-whitespace': 'error', | ||||
|     'vue/no-loss-of-precision': 'error', | ||||
|     'vue/no-restricted-syntax': [ | ||||
|       'error', | ||||
|       'DebuggerStatement', | ||||
|       'LabeledStatement', | ||||
|       'WithStatement', | ||||
|     ], | ||||
|     'vue/no-sparse-arrays': 'error', | ||||
|     'vue/object-curly-newline': ['error', { multiline: true, consistent: true }], | ||||
|     'vue/object-curly-spacing': ['error', 'always'], | ||||
|     'vue/object-property-newline': ['error', { allowMultiplePropertiesPerLine: true }], | ||||
|     'vue/object-shorthand': [ | ||||
|       'error', | ||||
|       'always', | ||||
|       { | ||||
|         ignoreConstructors: false, | ||||
|         avoidQuotes: true, | ||||
|       }, | ||||
|     ], | ||||
|     'vue/operator-linebreak': ['error', 'before'], | ||||
|     'vue/prefer-template': 'error', | ||||
|     'vue/quote-props': ['error', 'consistent-as-needed'], | ||||
|     'vue/space-in-parens': ['error', 'never'], | ||||
|     'vue/space-infix-ops': 'error', | ||||
|     'vue/space-unary-ops': ['error', { words: true, nonwords: false }], | ||||
|     'vue/template-curly-spacing': 'error', | ||||
|   }, | ||||
| } | ||||
|   | ||||
| @@ -1,10 +1,12 @@ | ||||
| { | ||||
|   "name": "@antfu/eslint-config-vue", | ||||
|   "version": "0.19.4", | ||||
|   "version": "0.26.0", | ||||
|   "description": "", | ||||
|   "keywords": [], | ||||
|   "license": "MIT", | ||||
|   "author": "Anthony Fu <anthonyfu117@hotmail.com> (https://github.com/antfu/)", | ||||
|   "license": "MIT", | ||||
|   "keywords": [ | ||||
|     "eslint-config" | ||||
|   ], | ||||
|   "main": "index.js", | ||||
|   "files": [ | ||||
|     "index.js" | ||||
| @@ -12,14 +14,11 @@ | ||||
|   "peerDependencies": { | ||||
|     "eslint": ">=7.4.0" | ||||
|   }, | ||||
|   "publishConfig": { | ||||
|     "access": "public" | ||||
|   }, | ||||
|   "dependencies": { | ||||
|     "@antfu/eslint-config-ts": "workspace:*", | ||||
|     "eslint-plugin-vue": "^8.5.0" | ||||
|     "eslint-plugin-vue": "^9.3.0" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "eslint": "^8.12.0" | ||||
|     "eslint": "^8.21.0" | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										1700
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										1700
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
		Reference in New Issue
	
	Block a user