-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathvue-dts-gen.ts
99 lines (88 loc) · 2.95 KB
/
vue-dts-gen.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
// https://github.com/egoist/vue-dts-gen/blob/main/src/index.ts
import path from 'path'
import fs from 'fs'
import { Project, SourceFile, CompilerOptions } from 'ts-morph'
import glob from 'fast-glob'
import resolveFrom from 'resolve-from'
export type Options = {
input: string | string[]
outDir?: string
}
let vueCompiler: typeof import('@vue/compiler-sfc')
const getVueCompiler = () => {
if (!vueCompiler) {
const id = resolveFrom.silent(process.cwd(), '@vue/compiler-sfc')
if (!id) {
throw new Error(`@vue/compiler-sfc is not founded in ./node_modules`)
}
vueCompiler = require(id)
}
return vueCompiler
}
export async function build({ input, outDir }: Options) {
const vueCompiler = getVueCompiler()
const tsConfigFilePath = fs.existsSync('tsconfig.json') ? 'tsconfig.json' : undefined
const compilerOptions: CompilerOptions = {
allowJs: true,
declaration: true,
emitDeclarationOnly: true,
noEmitOnError: true
}
if (outDir) {
compilerOptions.outDir = outDir
}
const project = new Project({
compilerOptions,
tsConfigFilePath,
skipAddingFilesFromTsConfig: true
})
const files = await glob(input)
const sourceFiles: SourceFile[] = []
await Promise.all(
files.map(async (file) => {
const content = await fs.promises.readFile(file, 'utf8')
if (file.endsWith('.ts')) {
const sourceFile = project.createSourceFile(path.relative(process.cwd(), file), content, {
overwrite: true
})
sourceFiles.push(sourceFile)
return
}
const sfc = vueCompiler.parse(content)
const { script, scriptSetup } = sfc.descriptor
if (script || scriptSetup) {
let content = ''
let isTS = false
if (script && script.content) {
content += script.content
if (script.lang === 'ts') isTS = true
}
if (scriptSetup) {
const compiled = vueCompiler.compileScript(sfc.descriptor, {
id: 'xxx'
})
content += compiled.content
if (scriptSetup.lang === 'ts') isTS = true
}
const sourceFile = project.createSourceFile(
path.relative(process.cwd(), file) + (isTS ? '.ts' : '.js'),
content
)
sourceFiles.push(sourceFile)
}
})
)
const diagnostics = project.getPreEmitDiagnostics()
console.log(project.formatDiagnosticsWithColorAndContext(diagnostics))
project.emitToMemory()
for (const sourceFile of sourceFiles) {
const emitOutput = sourceFile.getEmitOutput()
for (const outputFile of emitOutput.getOutputFiles()) {
const filepath = outputFile.getFilePath().replace('/src', '').replace('.vue.d.ts', '.d.ts')
await fs.promises.mkdir(path.dirname(filepath), { recursive: true })
await fs.promises.writeFile(filepath, outputFile.getText(), 'utf8')
console.log(`Emitted ${filepath}`)
}
}
}
build({ input: ['src/packages/*/index.vue', 'src/index.ts'], outDir: 'dist' })