-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathindex.ts
65 lines (58 loc) · 2.6 KB
/
index.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
import { transform as linariaTransform } from '@linaria/babel'
import type { Options as LinariaOptions } from '@linaria/babel/types'
import type { OnLoadArgs, OnLoadResult, Plugin, PluginBuild } from 'esbuild'
import { readFile } from 'node:fs/promises'
import * as path from 'node:path'
interface EsbuildPipeableTransformArgs {
readonly args: OnLoadArgs
readonly contents: string
}
interface EsbuildPipeablePlugin extends Plugin {
setup(build: PluginBuild, pipe: { transform: EsbuildPipeableTransformArgs }): OnLoadResult
setup(build: PluginBuild): void
}
interface EsbuildPluginLinariaOptions {
readonly filter?: RegExp
readonly preprocess?: (code: string, args: OnLoadArgs) => string
readonly linariaOptions?: LinariaOptions
}
interface EsbuildPluginLinaria {
(options?: EsbuildPluginLinariaOptions): EsbuildPipeablePlugin
default: EsbuildPluginLinaria
}
const pluginName = 'esbuild-plugin-linaria'
const plugin: EsbuildPluginLinaria = ({ filter, preprocess, linariaOptions } = {}) => {
const cssFileContentsMap = new Map<string, string>()
const transform = ({ args, contents }: EsbuildPipeableTransformArgs) => {
contents = preprocess ? preprocess(contents, args) : contents
let { cssText, code } = linariaTransform(contents, {
filename: args.path,
inputSourceMap: linariaOptions?.inputSourceMap,
preprocessor: linariaOptions?.preprocessor,
pluginOptions: linariaOptions?.pluginOptions ?? {
babelOptions: { presets: ['@babel/preset-react', '@babel/preset-typescript'] },
},
})
if (cssText) {
const cssFilename = `${args.path}.${pluginName}.css`
cssFileContentsMap.set(cssFilename, cssText)
code = `import '${cssFilename.replace(/\\/g, '\\\\')}';\n${code}`
}
return { contents: code, loader: path.extname(args.path).slice(1) as 'js' | 'jsx' | 'ts' | 'tsx' }
}
return {
name: pluginName,
setup: ((build: PluginBuild, pipe?: { transform: EsbuildPipeableTransformArgs }) => {
if (pipe?.transform) {
return transform(pipe.transform)
}
build.onLoad({ filter: filter ?? /\.[cm]?[jt]sx?$/ }, async args => transform({ args, contents: await readFile(args.path, 'utf8') }))
build.onResolve({ filter: RegExp(String.raw`\.${pluginName}\.css`) }, ({ path }) => ({ path, namespace: pluginName }))
build.onLoad({ filter: RegExp(String.raw`\.${pluginName}\.css`), namespace: pluginName }, ({ path }) => {
const contents = cssFileContentsMap.get(path)
return contents ? { contents, loader: 'css' } : undefined
})
}) as EsbuildPipeablePlugin['setup'],
}
}
export = plugin.default = plugin