-
Notifications
You must be signed in to change notification settings - Fork 713
/
Copy pathcomponent-example.ts
128 lines (115 loc) · 3.69 KB
/
component-example.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
import { existsSync, readFileSync } from 'node:fs'
import fsp from 'node:fs/promises'
import { dirname, join } from 'pathe'
import { defineNuxtModule, addTemplate, addServerHandler, createResolver } from '@nuxt/kit'
export default defineNuxtModule({
meta: {
name: 'component-example'
},
async setup(_options, nuxt) {
const resolver = createResolver(import.meta.url)
let _configResolved: any
let components: Record<string, any>
const outputPath = join(nuxt.options.buildDir, 'component-example')
async function stubOutput() {
if (existsSync(outputPath + '.mjs')) {
return
}
await updateOutput('export default {}')
}
async function fetchComponent(component: string | any) {
if (typeof component === 'string') {
if (components[component]) {
component = components[component]
} else {
component = Object.entries(components).find(
([, comp]: any) => comp.filePath === component
)
if (!component) {
return
}
component = component[1]
}
}
if (!component?.filePath || !component?.pascalName) {
return
}
const code = await fsp.readFile(component.filePath, 'utf-8')
components[component.pascalName] = {
code,
filePath: component.filePath,
pascalName: component.pascalName
}
}
const getStringifiedComponents = () => JSON.stringify(components, null, 2)
const getVirtualModuleContent = () =>
`export default ${getStringifiedComponents()}`
async function updateOutput(content?: string) {
const path = outputPath + '.mjs'
if (!existsSync(dirname(path))) {
await fsp.mkdir(dirname(path), { recursive: true })
}
if (existsSync(path)) {
await fsp.unlink(path)
}
await fsp.writeFile(path, content || getVirtualModuleContent(), 'utf-8')
}
async function fetchComponents() {
await Promise.all(Object.keys(components).map(fetchComponent))
}
nuxt.hook('components:extend', async (_components) => {
components = _components
.filter(v => v.shortPath.includes('components/content/examples/'))
.reduce((acc, component) => {
acc[component.pascalName] = component
return acc
}, {} as Record<string, any>)
await stubOutput()
})
addTemplate({
filename: 'component-example.mjs',
getContents: () => 'export default {}',
write: true
})
nuxt.hook('vite:extend', (vite: any) => {
vite.config.plugins = vite.config.plugins || []
vite.config.plugins.push({
name: 'component-example',
enforce: 'post',
async buildStart() {
if (_configResolved?.build.ssr) {
return
}
await fetchComponents()
await updateOutput()
},
configResolved(config: any) {
_configResolved = config
},
async handleHotUpdate({ file }: { file: any }) {
if (
Object.entries(components).some(
([, comp]: any) => comp.filePath === file
)
) {
await fetchComponent(file)
await updateOutput()
}
}
})
})
nuxt.hook('nitro:config', (nitroConfig) => {
nitroConfig.virtual = nitroConfig.virtual || {}
nitroConfig.virtual['#component-example/nitro'] = () =>
readFileSync(
join(nuxt.options.buildDir, '/component-example.mjs'),
'utf-8'
)
})
addServerHandler({
method: 'get',
route: '/api/component-example/:component?',
handler: resolver.resolve('../server/api/component-example.get')
})
}
})