forked from webfansplz/vite-plugin-vue-inspector
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtemplate.ts
107 lines (93 loc) · 3.17 KB
/
template.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
import path from 'node:path'
import MagicString from 'magic-string'
import { parse as vueParse, transform as vueTransform } from '@vue/compiler-dom'
import { parse as babelParse, traverse as babelTraverse } from '@babel/core'
import vueJsxPlugin from '@vue/babel-plugin-jsx'
import typescriptPlugin from '@babel/plugin-transform-typescript'
import importMeta from '@babel/plugin-syntax-import-meta'
import decoratorsPlugin from '@babel/plugin-proposal-decorators'
import importAttributesPlugin from '@babel/plugin-syntax-import-attributes'
import { normalizePath } from 'vite'
const EXCLUDE_TAG = ['template', 'script', 'style']
const KEY_DATA = 'data-v-inspector'
interface CompileSFCTemplateOptions {
code: string
id: string
type: 'template' | 'jsx'
}
export async function compileSFCTemplate(
{ code, id, type }: CompileSFCTemplateOptions,
) {
const s = new MagicString(code)
const relativePath = normalizePath(path.relative(process.cwd(), id))
const result = await new Promise((resolve) => {
switch (type) {
case 'template': {
const ast = vueParse(code, { comments: true })
vueTransform(ast, {
nodeTransforms: [
(node) => {
if (node.type === 1) {
if ((node.tagType === 0 || node.tagType === 1) && !EXCLUDE_TAG.includes(node.tag)) {
if (node.loc.source.includes(KEY_DATA))
return
const insertPosition = node.props.length ? Math.max(...node.props.map(i => i.loc.end.offset)) : node.loc.start.offset + node.tag.length + 1
const { line, column } = node.loc.start
const content = ` ${KEY_DATA}="${relativePath}:${line}:${column}"`
s.prependLeft(
insertPosition,
content,
)
}
}
},
],
})
break
}
case 'jsx': {
const ast = babelParse(code, {
babelrc: false,
configFile: false,
comments: true,
plugins: [
importMeta,
[vueJsxPlugin, {}],
[
typescriptPlugin,
{ isTSX: true, allowExtensions: true },
],
[
decoratorsPlugin,
{ legacy: true },
],
[
importAttributesPlugin,
{ deprecatedAssertSyntax: true },
],
],
})
babelTraverse(ast, {
enter({ node }) {
if (node.type === 'JSXElement') {
if (node.openingElement.attributes.some(attr => attr.type !== 'JSXSpreadAttribute' && attr.name.name === KEY_DATA,
))
return
const insertPosition = node.openingElement.end - (node.openingElement.selfClosing ? 2 : 1)
const { line, column } = node.loc.start
const content = ` ${KEY_DATA}="${relativePath}:${line}:${column}"`
s.prependLeft(
insertPosition,
content)
}
},
})
break
}
default:
break
}
resolve(s.toString())
})
return result
}