2021-08-24 13:36:48 +08:00
|
|
|
import path from 'path'
|
2021-09-26 01:29:07 +08:00
|
|
|
import fs from 'fs/promises'
|
|
|
|
import * as vueCompiler from '@vue/compiler-sfc'
|
2021-09-17 15:27:31 +08:00
|
|
|
import { Project } from 'ts-morph'
|
2021-09-26 01:29:07 +08:00
|
|
|
import glob from 'fast-glob'
|
|
|
|
import { bold } from 'chalk'
|
|
|
|
|
2021-12-10 18:18:16 +08:00
|
|
|
import { errorAndExit, green, yellow } from './utils/log'
|
2021-11-06 10:47:03 +08:00
|
|
|
import { buildOutput, epRoot, pkgRoot, projRoot } from './utils/paths'
|
2021-09-26 01:29:07 +08:00
|
|
|
|
2021-10-25 17:07:48 +08:00
|
|
|
import { excludeFiles, pathRewriter } from './utils/pkg'
|
2021-09-17 15:27:31 +08:00
|
|
|
import type { SourceFile } from 'ts-morph'
|
2021-08-24 13:36:48 +08:00
|
|
|
|
2021-09-26 01:29:07 +08:00
|
|
|
const TSCONFIG_PATH = path.resolve(projRoot, 'tsconfig.json')
|
|
|
|
const outDir = path.resolve(buildOutput, 'types')
|
2021-08-24 13:36:48 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* fork = require( https://github.com/egoist/vue-dts-gen/blob/main/src/index.ts
|
|
|
|
*/
|
2021-10-25 17:07:48 +08:00
|
|
|
export const generateTypesDefinitions = async () => {
|
2021-08-24 13:36:48 +08:00
|
|
|
const project = new Project({
|
|
|
|
compilerOptions: {
|
|
|
|
emitDeclarationOnly: true,
|
|
|
|
outDir,
|
2021-09-26 01:29:07 +08:00
|
|
|
baseUrl: projRoot,
|
2021-08-24 13:36:48 +08:00
|
|
|
paths: {
|
|
|
|
'@element-plus/*': ['packages/*'],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
tsConfigFilePath: TSCONFIG_PATH,
|
|
|
|
skipAddingFilesFromTsConfig: true,
|
|
|
|
})
|
|
|
|
|
2021-10-25 17:07:48 +08:00
|
|
|
const filePaths = excludeFiles(
|
2021-11-06 10:47:03 +08:00
|
|
|
await glob(['**/*.{js,ts,vue}', '!element-plus/**/*'], {
|
2021-10-25 17:07:48 +08:00
|
|
|
cwd: pkgRoot,
|
2021-10-20 09:42:32 +08:00
|
|
|
absolute: true,
|
2021-10-25 17:07:48 +08:00
|
|
|
onlyFiles: true,
|
2021-09-26 01:29:07 +08:00
|
|
|
})
|
2021-09-15 01:13:36 +08:00
|
|
|
)
|
2021-11-06 10:47:03 +08:00
|
|
|
const epPaths = excludeFiles(
|
|
|
|
await glob('**/*.{js,ts,vue}', {
|
|
|
|
cwd: epRoot,
|
|
|
|
onlyFiles: true,
|
|
|
|
})
|
|
|
|
)
|
2021-08-24 13:36:48 +08:00
|
|
|
|
2021-09-26 01:29:07 +08:00
|
|
|
const sourceFiles: SourceFile[] = []
|
2021-11-06 10:47:03 +08:00
|
|
|
await Promise.all([
|
|
|
|
...filePaths.map(async (file) => {
|
2021-08-24 13:36:48 +08:00
|
|
|
if (file.endsWith('.vue')) {
|
2021-09-26 01:29:07 +08:00
|
|
|
const content = await fs.readFile(file, 'utf-8')
|
2021-08-24 13:36:48 +08:00
|
|
|
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'),
|
2021-09-04 19:29:28 +08:00
|
|
|
content
|
2021-08-24 13:36:48 +08:00
|
|
|
)
|
|
|
|
sourceFiles.push(sourceFile)
|
|
|
|
}
|
2021-11-06 10:47:03 +08:00
|
|
|
} else {
|
2021-08-24 13:36:48 +08:00
|
|
|
const sourceFile = project.addSourceFileAtPath(file)
|
|
|
|
sourceFiles.push(sourceFile)
|
|
|
|
}
|
2021-11-06 10:47:03 +08:00
|
|
|
}),
|
|
|
|
...epPaths.map(async (file) => {
|
|
|
|
const content = await fs.readFile(path.resolve(epRoot, file), 'utf-8')
|
|
|
|
sourceFiles.push(
|
|
|
|
project.createSourceFile(path.resolve(pkgRoot, file), content)
|
|
|
|
)
|
|
|
|
}),
|
|
|
|
])
|
2021-08-24 13:36:48 +08:00
|
|
|
|
|
|
|
const diagnostics = project.getPreEmitDiagnostics()
|
|
|
|
console.log(project.formatDiagnosticsWithColorAndContext(diagnostics))
|
|
|
|
|
|
|
|
await project.emit({
|
|
|
|
emitOnlyDtsFiles: true,
|
|
|
|
})
|
|
|
|
|
2021-09-26 01:29:07 +08:00
|
|
|
const tasks = sourceFiles.map(async (sourceFile) => {
|
2021-10-25 17:07:48 +08:00
|
|
|
const relativePath = path.relative(pkgRoot, sourceFile.getFilePath())
|
2021-09-26 01:29:07 +08:00
|
|
|
yellow(`Generating definition for file: ${bold(relativePath)}`)
|
2021-08-24 13:36:48 +08:00
|
|
|
|
|
|
|
const emitOutput = sourceFile.getEmitOutput()
|
2021-10-06 19:56:24 +08:00
|
|
|
const emitFiles = emitOutput.getOutputFiles()
|
|
|
|
if (emitFiles.length === 0) {
|
2021-12-10 18:18:16 +08:00
|
|
|
errorAndExit(new Error(`Emit no file: ${bold(relativePath)}`))
|
2021-10-06 19:56:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
const tasks = emitFiles.map(async (outputFile) => {
|
2021-08-24 13:36:48 +08:00
|
|
|
const filepath = outputFile.getFilePath()
|
2021-09-26 01:29:07 +08:00
|
|
|
await fs.mkdir(path.dirname(filepath), {
|
2021-08-24 13:36:48 +08:00
|
|
|
recursive: true,
|
|
|
|
})
|
|
|
|
|
2021-09-26 01:29:07 +08:00
|
|
|
await fs.writeFile(
|
2021-09-04 19:29:28 +08:00
|
|
|
filepath,
|
2021-10-25 17:07:48 +08:00
|
|
|
pathRewriter('esm')(outputFile.getText()),
|
2021-09-04 19:29:28 +08:00
|
|
|
'utf8'
|
|
|
|
)
|
2021-08-24 13:36:48 +08:00
|
|
|
|
2021-09-26 01:29:07 +08:00
|
|
|
green(`Definition for file: ${bold(relativePath)} generated`)
|
|
|
|
})
|
2021-10-25 17:07:48 +08:00
|
|
|
|
2021-09-26 01:29:07 +08:00
|
|
|
await Promise.all(tasks)
|
|
|
|
})
|
|
|
|
|
|
|
|
await Promise.all(tasks)
|
|
|
|
}
|