Skip to content

Commit 9c9f8f3

Browse files
authoredMar 20, 2024··
feat(module)!: generate withNuxt function for easer composition (#344)
1 parent 065fbe2 commit 9c9f8f3

File tree

10 files changed

+129
-43
lines changed

10 files changed

+129
-43
lines changed
 

‎docs/content/1.packages/0.module.md

+27-20
Original file line numberDiff line numberDiff line change
@@ -56,22 +56,31 @@ export default defineNuxtConfig({
5656
And create an `eslint.config.mjs` file under **your project root**, with the following content:
5757

5858
```js [eslint.config.mjs]
59-
import nuxt from './.nuxt/eslint.config.mjs'
60-
61-
export default [
62-
...nuxt,
63-
// your custom flat config here.
64-
]
59+
import withNuxt from './.nuxt/eslint.config.mjs'
60+
61+
export default withNuxt(
62+
// your custom flat configs go here, for example:
63+
// {
64+
// files: ['**/*.ts', '**/*.tsx'],
65+
// rules: {
66+
// 'no-console': 'off' // allow console.log in TypeScript files
67+
// }
68+
// },
69+
// {
70+
// ...
71+
// }
72+
)
6573
```
6674

75+
`withNuxt` will takes rest arguments of flat configs and append them after Nuxt flat config items. You can use the [Nuxt DevTools](https://github.com/nuxt/devtools) panel to inspect the resolved ESLint flat config. Or run [`npx eslint-flat-config-viewer`](https://github.com/antfu/eslint-flat-config-viewer) manually to inspect.
76+
6777
## Receipts
6878

6979
### Work with VS Code
7080

7181
Note that ESLint Flat config is not yet enabled by default in the [ESLint VS Code extension](https://marketplace.visualstudio.com/items?itemName=dbaeumer.vscode-eslint), you will need to enable it via the `eslint.experimental.useFlatConfig` to get ESLint working in VS Code. (This is likely not needed after ESLint v9).
7282

73-
```json
74-
// .vscode/settings.json
83+
```json [.vscode/settings.json]
7584
{
7685
// Enable the ESlint flat config support
7786
"eslint.experimental.useFlatConfig": true
@@ -80,11 +89,11 @@ Note that ESLint Flat config is not yet enabled by default in the [ESLint VS Cod
8089

8190
### Use with Prettier
8291

83-
This module does not enable any stylistic/formatting rules by default. You can use Prettier alongside directly.
92+
This module does not enable stylistic/formatting rules by default. You can use Prettier alongside directly.
8493

8594
### Use with ESLint Stylistic
8695

87-
If instead, you prefer to use ESLint for formatting as well, we also integrated [ESLint Stylistic](https://eslint.style/) to make it easy. You can opt-in by setting `config.stylistic` to `true` in the `eslint` module options.
96+
If you prefer to use ESLint for formatting as well, we also integrated [ESLint Stylistic](https://eslint.style/) to make it easy. You can opt-in by setting `config.stylistic` to `true` in the `eslint` module options.
8897

8998
```ts [nuxt.config.ts]
9099
export default defineNuxtConfig({
@@ -168,17 +177,15 @@ This will make this module only generate the Nuxt-specific rules and disables, s
168177

169178
For example, with [`@antfu/eslint-config`](https://github.com/antfu/eslint-config):
170179

171-
```js
172-
// eslint.config.js
180+
```js [eslint.config.mjs]
181+
// @ts-check
173182
import antfu from '@antfu/eslint-config'
174-
import nuxt from './.nuxt/eslint.config.mjs'
175-
176-
export default antfu(
177-
{
178-
// ...@antfu/eslint-config options,
179-
},
180-
// Add the Nuxt rules
181-
nuxt,
183+
import withNuxt from './.nuxt/eslint.config.mjs'
184+
185+
export default withNuxt(
186+
antfu({
187+
// ...@antfu/eslint-config options
188+
}),
182189
// ...your other rules
183190
)
184191
```

‎docs/content/1.packages/1.config.md

+31-2
Original file line numberDiff line numberDiff line change
@@ -49,14 +49,13 @@ bun add -D @nuxt/eslint-config eslint
4949
2. Import the config factory function from `@nuxt/eslint-config/flat` entry in your `eslint.config.mjs`:
5050

5151
```js [eslint.config.mjs]
52-
import createConfigForNuxt from '@nuxt/eslint-config/flat'
52+
import { createConfigForNuxt } from '@nuxt/eslint-config/flat'
5353

5454
export default createConfigForNuxt({
5555
// options here
5656
})
5757
```
5858

59-
6059
You might also want to add a script entry to your `package.json:
6160

6261
```json [package.json]
@@ -67,6 +66,36 @@ You might also want to add a script entry to your `package.json:
6766
}
6867
```
6968

69+
### Customizing the Config
70+
71+
Note that `createConfigForNuxt()` returns `Promise<FlatConfig[]>`. ESLint allows you to run the promise directly to the default export. If you want to combine with other configs, you will need await and spread the result. For example:
72+
73+
```js [eslint.config.mjs]
74+
import { createConfigForNuxt } from '@nuxt/eslint-config/flat'
75+
76+
export default [
77+
...await createConfigForNuxt({
78+
// options here
79+
}),
80+
// other configs
81+
]
82+
// (which uses Top-level await)
83+
```
84+
85+
To make it easier, you can also use our helper `defineFlatConfigs`, which will also resolve and flatten the configs for you:
86+
87+
```js [eslint.config.mjs]
88+
import { defineFlatConfigs, createConfigForNuxt } from '@nuxt/eslint-config/flat'
89+
90+
export default defineFlatConfigs(
91+
createConfigForNuxt({
92+
// options here
93+
}),
94+
// other configs
95+
)
96+
```
97+
98+
7099
## Legacy Config Format
71100

72101
The legacy config configures for TypeScript and Vue integrations for ESLint. It is unopinionated and static, that does not contains stylistic rules nor project-aware settings.

‎docs/nuxt.config.ts

+8-3
Original file line numberDiff line numberDiff line change
@@ -33,9 +33,6 @@ export default defineNuxtConfig({
3333
routes: ['/api/search.json'],
3434
autoSubfolderIndex: false,
3535
},
36-
experimental: {
37-
wasm: true,
38-
},
3936
},
4037

4138
hooks: {
@@ -49,4 +46,12 @@ export default defineNuxtConfig({
4946
}
5047
},
5148
},
49+
50+
$production: {
51+
nitro: {
52+
experimental: {
53+
wasm: true,
54+
},
55+
},
56+
},
5257
})

‎eslint.config.js

+4-4
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,13 @@
11
// @ts-check
2-
import createConfig from '@nuxt/eslint-config/flat'
2+
import { createConfigForNuxt, defineFlatConfigs } from '@nuxt/eslint-config/flat'
33

4-
export default [
4+
export default defineFlatConfigs(
55
{
66
ignores: [
77
'packages-legacy/**',
88
],
99
},
10-
...createConfig({
10+
createConfigForNuxt({
1111
features: {
1212
stylistic: true,
1313
},
@@ -33,4 +33,4 @@ export default [
3333
'vue/no-v-html': 'off',
3434
},
3535
},
36-
]
36+
)

‎packages/eslint-config/src/flat.ts

-1
Original file line numberDiff line numberDiff line change
@@ -1,2 +1 @@
11
export * from './flat/index'
2-
export { default } from './flat/index'

‎packages/eslint-config/src/flat/index.ts

+20-2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import type { FlatConfig, NuxtESLintConfigOptions } from './types'
1+
import type { Awaitable, FlatConfig, NuxtESLintConfigOptions } from './types'
22
import disables from './configs/disables'
33
import nuxt from './configs/nuxt'
44
import base from './configs/base'
@@ -9,7 +9,25 @@ import stylistic from './configs/stylistic'
99

1010
export * from './types'
1111

12-
export default function createConfigForNuxt(options: NuxtESLintConfigOptions = {}): FlatConfig[] {
12+
/**
13+
* Provide type definitions for constructing ESLint flat config items.
14+
*
15+
* This function takes flat config item, or an array of them as rest arguments.
16+
* It also automatically resolves the promise if the config item is a promise.
17+
*/
18+
export async function defineFlatConfigs(...configs: Awaitable<FlatConfig | FlatConfig[]>[]): Promise<FlatConfig[]> {
19+
const resolved = await Promise.all(configs)
20+
return resolved.flat()
21+
}
22+
23+
/**
24+
* Create an array of ESLint flat configs for Nuxt 3, based on the given options.
25+
*
26+
* Usually it would be use `@nuxt/eslint` module which will generate the necessary configuration based on your project.
27+
*
28+
* @see https://eslint.nuxt.com/packages/module
29+
*/
30+
export async function createConfigForNuxt(options: NuxtESLintConfigOptions = {}): Promise<FlatConfig[]> {
1331
const items: FlatConfig[] = []
1432

1533
if (options.features?.standalone !== false) {

‎packages/eslint-config/src/flat/types.ts

+2
Original file line numberDiff line numberDiff line change
@@ -79,3 +79,5 @@ export interface NuxtESLintConfigOptions {
7979
export interface FlatConfig extends Linter.FlatConfig {
8080
name?: string
8181
}
82+
83+
export type Awaitable<T> = T | Promise<T>

‎packages/module/src/modules/config.ts

+33-8
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,21 @@ export async function setupConfigGen(options: ModuleOptions, nuxt: Nuxt) {
2727
},
2828
})
2929

30+
addTemplate({
31+
filename: 'eslint.config.d.mts',
32+
write: true,
33+
async getContents() {
34+
return [
35+
'import type { FlatConfig } from "@nuxt/eslint-config/flat"',
36+
'export { defineFlatConfigs } from "@nuxt/eslint-config/flat"',
37+
'declare const configs: Promise<FlatConfig[]>',
38+
'declare const withNuxt: typeof defineFlatConfigs',
39+
'export default withNuxt',
40+
'export { withNuxt }',
41+
].join('\n')
42+
},
43+
})
44+
3045
setupDevToolsIntegration(nuxt)
3146
}
3247

@@ -39,11 +54,15 @@ async function generateESLintConfig(options: ModuleOptions, nuxt: Nuxt, addons:
3954
...typeof options.config !== 'boolean' ? options.config || {} : {},
4055
}
4156

42-
importLines.push({
43-
from: '@nuxt/eslint-config/flat',
44-
name: 'default',
45-
as: 'createConfigForNuxt',
46-
})
57+
importLines.push(
58+
{
59+
from: '@nuxt/eslint-config/flat',
60+
name: 'createConfigForNuxt',
61+
}, {
62+
from: '@nuxt/eslint-config/flat',
63+
name: 'defineFlatConfigs',
64+
},
65+
)
4766

4867
const basicOptions: NuxtESLintConfigOptions = {
4968
features: {
@@ -53,7 +72,7 @@ async function generateESLintConfig(options: ModuleOptions, nuxt: Nuxt, addons:
5372
dirs: getDirs(nuxt),
5473
}
5574

56-
configItems.push(`// Nuxt Configs\n...createConfigForNuxt(${JSON.stringify(basicOptions, null, 2)})`)
75+
configItems.push(`// Nuxt Configs\ncreateConfigForNuxt(${JSON.stringify(basicOptions, null, 2)})`)
5776

5877
for (const addon of addons) {
5978
const resolved = await addon()
@@ -66,11 +85,17 @@ async function generateESLintConfig(options: ModuleOptions, nuxt: Nuxt, addons:
6685
return [
6786
'// ESLint config generated by Nuxt',
6887
stringifyImports(importLines, false),
88+
'export { defineFlatConfigs }',
6989
'',
70-
`export default [`,
90+
`export const configs = defineFlatConfigs(`,
7191
configItems.join(',\n\n'),
72-
`]`,
92+
`)`,
93+
'',
94+
'export function withNuxt(...customs) {',
95+
' return defineFlatConfigs(configs, ...customs)',
96+
'}',
7397
'',
98+
'export default withNuxt',
7499
].join('\n')
75100
}
76101

‎playground/eslint.config.js

+3-2
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
1-
import nuxt from './.nuxt/eslint.config.mjs'
1+
// @ts-check
2+
import withNuxt from './.nuxt/eslint.config.mjs'
23

3-
export default nuxt
4+
export default withNuxt()

‎playground/package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@
55
"scripts": {
66
"play:dev": "nuxi dev",
77
"play:build": "nuxi build",
8-
"lint": "nuxi preprare && eslint ."
8+
"lint": "nuxi prepare && eslint ."
99
},
1010
"devDependencies": {
1111
"nuxt": "^3.11.1"

0 commit comments

Comments
 (0)
Please sign in to comment.