2023-03-08 21:13:06 +00:00
|
|
|
import { createUnplugin } from 'unplugin'
|
|
|
|
import type { ComponentsOptions } from '@nuxt/schema'
|
|
|
|
import MagicString from 'magic-string'
|
|
|
|
import { isAbsolute, relative } from 'pathe'
|
|
|
|
import { hash } from 'ohash'
|
2023-04-19 18:17:36 +00:00
|
|
|
import { isVue } from '../core/utils'
|
2024-04-05 18:08:32 +00:00
|
|
|
|
2023-03-08 21:13:06 +00:00
|
|
|
interface LoaderOptions {
|
|
|
|
sourcemap?: boolean
|
2024-04-05 18:08:32 +00:00
|
|
|
transform?: ComponentsOptions['transform']
|
2023-03-08 21:13:06 +00:00
|
|
|
rootDir: string
|
|
|
|
}
|
2024-05-14 17:54:37 +00:00
|
|
|
const CLIENT_FALLBACK_RE = /<(?:NuxtClientFallback|nuxt-client-fallback)(?: [^>]*)?>/
|
2023-03-08 21:13:06 +00:00
|
|
|
const CLIENT_FALLBACK_GLOBAL_RE = /<(NuxtClientFallback|nuxt-client-fallback)( [^>]*)?>/g
|
|
|
|
export const clientFallbackAutoIdPlugin = createUnplugin((options: LoaderOptions) => {
|
|
|
|
const exclude = options.transform?.exclude || []
|
|
|
|
const include = options.transform?.include || []
|
|
|
|
|
|
|
|
return {
|
|
|
|
name: 'nuxt:client-fallback-auto-id',
|
|
|
|
enforce: 'pre',
|
|
|
|
transformInclude (id) {
|
2023-05-22 20:25:42 +00:00
|
|
|
if (exclude.some(pattern => pattern.test(id))) {
|
2023-03-08 21:13:06 +00:00
|
|
|
return false
|
|
|
|
}
|
2023-05-22 20:25:42 +00:00
|
|
|
if (include.some(pattern => pattern.test(id))) {
|
2023-03-08 21:13:06 +00:00
|
|
|
return true
|
|
|
|
}
|
2023-05-13 21:39:50 +00:00
|
|
|
return isVue(id)
|
2023-03-08 21:13:06 +00:00
|
|
|
},
|
|
|
|
transform (code, id) {
|
|
|
|
if (!CLIENT_FALLBACK_RE.test(code)) { return }
|
|
|
|
|
|
|
|
const s = new MagicString(code)
|
|
|
|
const relativeID = isAbsolute(id) ? relative(options.rootDir, id) : id
|
|
|
|
let count = 0
|
|
|
|
|
|
|
|
s.replace(CLIENT_FALLBACK_GLOBAL_RE, (full, name, attrs) => {
|
|
|
|
count++
|
2024-05-14 17:54:37 +00:00
|
|
|
if (/ :?uid=/.test(attrs)) { return full }
|
2023-03-08 21:13:06 +00:00
|
|
|
return `<${name} :uid="'${hash(relativeID)}' + JSON.stringify($props) + '${count}'" ${attrs ?? ''}>`
|
|
|
|
})
|
|
|
|
|
|
|
|
if (s.hasChanged()) {
|
|
|
|
return {
|
|
|
|
code: s.toString(),
|
|
|
|
map: options.sourcemap
|
2023-04-14 17:21:08 +00:00
|
|
|
? s.generateMap({ hires: true })
|
2024-04-05 18:08:32 +00:00
|
|
|
: undefined,
|
2023-03-08 21:13:06 +00:00
|
|
|
}
|
|
|
|
}
|
2024-04-05 18:08:32 +00:00
|
|
|
},
|
2023-03-08 21:13:06 +00:00
|
|
|
}
|
|
|
|
})
|