2023-05-15 22:43:53 +00:00
import { pathToFileURL } from 'node:url'
2023-12-19 12:21:29 +00:00
import fs from 'node:fs'
2024-02-05 11:24:39 +00:00
import { join } from 'pathe'
2023-05-15 22:43:53 +00:00
import type { Component } from '@nuxt/schema'
import { parseURL } from 'ufo'
import { createUnplugin } from 'unplugin'
import MagicString from 'magic-string'
import { ELEMENT_NODE , parse , walk } from 'ultrahtml'
2023-12-19 12:21:29 +00:00
import { hash } from 'ohash'
import { resolvePath } from '@nuxt/kit'
2023-06-26 18:32:02 +00:00
import { isVue } from '../core/utils'
2023-05-15 22:43:53 +00:00
interface ServerOnlyComponentTransformPluginOptions {
2024-01-16 13:22:50 +00:00
getComponents : ( ) = > Component [ ]
/ * *
2024-01-16 16:33:45 +00:00
* passed down to ` NuxtTeleportIslandComponent `
2024-01-16 13:22:50 +00:00
* should be done only in dev mode as we use build :manifest result in production
* /
rootDir? : string
isDev? : boolean
/ * *
* allow using ` nuxt-client ` attribute on components
* /
2024-03-06 15:26:19 +00:00
selectiveClient? : boolean | 'deep'
2023-12-19 12:21:29 +00:00
}
interface ComponentChunkOptions {
getComponents : ( ) = > Component [ ]
buildDir : string
2023-05-15 22:43:53 +00:00
}
const SCRIPT_RE = /<script[^>]*>/g
2023-12-19 12:21:29 +00:00
const HAS_SLOT_OR_CLIENT_RE = /(<slot[^>]*>)|(nuxt-client)/
2023-06-26 18:32:02 +00:00
const TEMPLATE_RE = /<template>([\s\S]*)<\/template>/
2024-01-27 21:45:34 +00:00
const NUXTCLIENT_ATTR_RE = /\s:?nuxt-client(="[^"]*")?/g
2024-01-16 16:33:45 +00:00
const IMPORT_CODE = '\nimport { vforToArray as __vforToArray } from \'#app/components/utils\'' + '\nimport NuxtTeleportIslandComponent from \'#app/components/nuxt-teleport-island-component\'' + '\nimport NuxtTeleportSsrSlot from \'#app/components/nuxt-teleport-island-slot\''
2024-03-25 10:19:02 +00:00
const EXTRACTED_ATTRS_RE = /v-(?:if|else-if|else)(="[^"]*")?/g
2024-01-16 13:22:50 +00:00
function wrapWithVForDiv ( code : string , vfor : string ) : string {
return ` <div v-for=" ${ vfor } " style="display: contents;"> ${ code } </div> `
}
2023-05-15 22:43:53 +00:00
2023-12-19 12:21:29 +00:00
export const islandsTransform = createUnplugin ( ( options : ServerOnlyComponentTransformPluginOptions , meta ) = > {
const isVite = meta . framework === 'vite'
const { isDev , rootDir } = options
2023-05-15 22:43:53 +00:00
return {
name : 'server-only-component-transform' ,
enforce : 'pre' ,
transformInclude ( id ) {
2023-06-26 18:32:02 +00:00
if ( ! isVue ( id ) ) { return false }
2024-03-06 15:26:19 +00:00
if ( options . selectiveClient === 'deep' ) { return true }
2023-05-15 22:43:53 +00:00
const components = options . getComponents ( )
2023-12-19 12:21:29 +00:00
2023-05-15 22:43:53 +00:00
const islands = components . filter ( component = >
component . island || ( component . mode === 'server' && ! components . some ( c = > c . pascalName === component . pascalName && c . mode === 'client' ) )
)
const { pathname } = parseURL ( decodeURIComponent ( pathToFileURL ( id ) . href ) )
return islands . some ( c = > c . filePath === pathname )
} ,
async transform ( code , id ) {
2023-12-19 12:21:29 +00:00
if ( ! HAS_SLOT_OR_CLIENT_RE . test ( code ) ) { return }
2023-06-26 18:32:02 +00:00
const template = code . match ( TEMPLATE_RE )
2023-05-15 22:43:53 +00:00
if ( ! template ) { return }
2023-06-26 18:32:02 +00:00
const startingIndex = template . index || 0
2023-05-15 22:43:53 +00:00
const s = new MagicString ( code )
2024-01-11 14:40:02 +00:00
if ( ! code . match ( SCRIPT_RE ) ) {
s . prepend ( '<script setup>' + IMPORT_CODE + '</script>' )
} else {
s . replace ( SCRIPT_RE , ( full ) = > {
2024-01-16 13:22:50 +00:00
return full + IMPORT_CODE
2024-01-11 14:40:02 +00:00
} )
}
2023-12-19 12:21:29 +00:00
let hasNuxtClient = false
2023-05-15 22:43:53 +00:00
const ast = parse ( template [ 0 ] )
await walk ( ast , ( node ) = > {
2023-12-19 12:21:29 +00:00
if ( node . type === ELEMENT_NODE ) {
if ( node . name === 'slot' ) {
2024-01-16 13:22:50 +00:00
const { attributes , children , loc } = node
2023-12-19 12:21:29 +00:00
const slotName = attributes . name ? ? 'default'
2024-03-25 10:19:02 +00:00
let vfor : string | undefined
2023-12-19 12:21:29 +00:00
if ( attributes [ 'v-for' ] ) {
2024-03-25 10:19:02 +00:00
vfor = attributes [ 'v-for' ]
2023-12-19 12:21:29 +00:00
}
2024-01-16 13:22:50 +00:00
delete attributes [ 'v-for' ]
2023-12-19 12:21:29 +00:00
if ( attributes . name ) { delete attributes . name }
if ( attributes [ 'v-bind' ] ) {
2024-03-25 10:19:02 +00:00
attributes . _bind = extractAttributes ( attributes , [ 'v-bind' ] ) [ 'v-bind' ]
2023-12-19 12:21:29 +00:00
}
2024-03-25 10:19:02 +00:00
const teleportAttributes = extractAttributes ( attributes , [ 'v-if' , 'v-else-if' , 'v-else' ] )
const bindings = getPropsToString ( attributes , vfor ? . split ( ' in ' ) . map ( ( v : string ) = > v . trim ( ) ) as [ string , string ] )
2024-01-16 13:22:50 +00:00
// add the wrapper
2024-03-25 10:19:02 +00:00
s . appendLeft ( startingIndex + loc [ 0 ] . start , ` <NuxtTeleportSsrSlot ${ attributeToString ( teleportAttributes ) } name=" ${ slotName } " :props=" ${ bindings } "> ` )
if ( children . length ) {
// pass slot fallback to NuxtTeleportSsrSlot fallback
const attrString = attributeToString ( attributes )
const slice = code . slice ( startingIndex + loc [ 0 ] . end , startingIndex + loc [ 1 ] . start ) . replaceAll ( /:?key="[^"]"/g , '' )
s . overwrite ( startingIndex + loc [ 0 ] . start , startingIndex + loc [ 1 ] . end , ` <slot ${ attrString . replaceAll ( EXTRACTED_ATTRS_RE , '' ) } /><template #fallback> ${ vfor ? wrapWithVForDiv ( slice , vfor ) : slice } </template> ` )
} else {
s . overwrite ( startingIndex + loc [ 0 ] . start , startingIndex + loc [ 0 ] . end , code . slice ( startingIndex + loc [ 0 ] . start , startingIndex + loc [ 0 ] . end ) . replaceAll ( EXTRACTED_ATTRS_RE , '' ) )
}
2024-01-16 13:22:50 +00:00
s . appendRight ( startingIndex + loc [ 1 ] . end , '</NuxtTeleportSsrSlot>' )
2023-12-19 12:21:29 +00:00
} else if ( options . selectiveClient && ( 'nuxt-client' in node . attributes || ':nuxt-client' in node . attributes ) ) {
hasNuxtClient = true
2024-03-26 13:47:40 +00:00
const { loc , attributes } = node
const attributeValue = attributes [ ':nuxt-client' ] || attributes [ 'nuxt-client' ] || 'true'
2023-12-19 12:21:29 +00:00
if ( isVite ) {
const uid = hash ( id + node . loc [ 0 ] . start + node . loc [ 0 ] . end )
2024-03-26 13:47:40 +00:00
const wrapperAttributes = extractAttributes ( attributes , [ 'v-if' , 'v-else-if' , 'v-else' ] )
2023-05-15 22:43:53 +00:00
2024-03-26 13:47:40 +00:00
let startTag = code . slice ( startingIndex + loc [ 0 ] . start , startingIndex + loc [ 0 ] . end ) . replace ( NUXTCLIENT_ATTR_RE , '' )
if ( wrapperAttributes ) {
startTag = startTag . replaceAll ( EXTRACTED_ATTRS_RE , '' )
}
s . appendLeft ( startingIndex + loc [ 0 ] . start , ` <NuxtTeleportIslandComponent ${ attributeToString ( wrapperAttributes ) } to=" ${ node . name } - ${ uid } " ${ rootDir && isDev ? ` root-dir=" ${ rootDir } " ` : '' } :nuxt-client=" ${ attributeValue } "> ` )
s . overwrite ( startingIndex + loc [ 0 ] . start , startingIndex + loc [ 0 ] . end , startTag )
s . appendRight ( startingIndex + loc [ 1 ] . end , '</NuxtTeleportIslandComponent>' )
2023-05-15 22:43:53 +00:00
}
}
}
} )
2023-12-19 12:21:29 +00:00
if ( ! isVite && hasNuxtClient ) {
// eslint-disable-next-line no-console
console . warn ( ` nuxt-client attribute and client components within islands is only supported with Vite. file: ${ id } ` )
}
2023-05-15 22:43:53 +00:00
if ( s . hasChanged ( ) ) {
return {
code : s.toString ( ) ,
map : s.generateMap ( { source : id , includeContent : true } )
}
}
}
}
} )
2024-03-25 10:19:02 +00:00
/ * *
* extract attributes from a node
* /
function extractAttributes ( attributes : Record < string , string > , names : string [ ] ) {
const extracted :Record < string , string > = { }
for ( const name of names ) {
if ( name in attributes ) {
extracted [ name ] = attributes [ name ]
delete attributes [ name ]
}
}
return extracted
}
function attributeToString ( attributes : Record < string , string > ) {
return Object . entries ( attributes ) . map ( ( [ name , value ] ) = > value ? ` ${ name } =" ${ value } " ` : ` ${ name } ` ) . join ( '' )
}
2023-05-15 22:43:53 +00:00
function isBinding ( attr : string ) : boolean {
return attr . startsWith ( ':' )
}
2024-01-16 13:22:50 +00:00
function getPropsToString ( bindings : Record < string , string > , vfor ? : [ string , string ] ) : string {
if ( Object . keys ( bindings ) . length === 0 ) { return 'undefined' }
2024-03-25 10:19:02 +00:00
const content = Object . entries ( bindings ) . filter ( b = > b [ 0 ] && b [ 0 ] !== '_bind' ) . map ( ( [ name , value ] ) = > isBinding ( name ) ? ` [ \` ${ name . slice ( 1 ) } \` ]: ${ value } ` : ` [ \` ${ name } \` ]: \` ${ value } \` ` ) . join ( ',' )
2023-05-15 22:43:53 +00:00
const data = bindings . _bind ? ` mergeProps( ${ bindings . _bind } , { ${ content } }) ` : ` { ${ content } } `
if ( ! vfor ) {
2024-01-16 13:22:50 +00:00
return ` [ ${ data } ] `
2023-05-15 22:43:53 +00:00
} else {
2024-01-16 13:22:50 +00:00
return ` __vforToArray( ${ vfor [ 1 ] } ).map( ${ vfor [ 0 ] } => ( ${ data } )) `
2023-05-15 22:43:53 +00:00
}
}
2023-12-19 12:21:29 +00:00
export const componentsChunkPlugin = createUnplugin ( ( options : ComponentChunkOptions ) = > {
const { buildDir } = options
return {
name : 'componentsChunkPlugin' ,
vite : {
2024-01-16 13:22:50 +00:00
async config ( config ) {
2023-12-19 12:21:29 +00:00
const components = options . getComponents ( )
config . build = config . build || { }
config . build . rollupOptions = config . build . rollupOptions || { }
config . build . rollupOptions . output = config . build . rollupOptions . output || { }
config . build . rollupOptions . input = config . build . rollupOptions . input || { }
// don't use 'strict', this would create another "facade" chunk for the entry file, causing the ssr styles to not detect everything
config . build . rollupOptions . preserveEntrySignatures = 'allow-extension'
for ( const component of components ) {
if ( component . mode === 'client' || component . mode === 'all' ) {
( config . build . rollupOptions . input as Record < string , string > ) [ component . pascalName ] = await resolvePath ( component . filePath )
}
}
} ,
async generateBundle ( _opts , bundle ) {
const components = options . getComponents ( ) . filter ( c = > c . mode === 'client' || c . mode === 'all' )
const pathAssociation : Record < string , string > = { }
for ( const [ chunkPath , chunkInfo ] of Object . entries ( bundle ) ) {
if ( chunkInfo . type !== 'chunk' ) { continue }
for ( const component of components ) {
if ( chunkInfo . facadeModuleId && chunkInfo . exports . length > 0 ) {
const { pathname } = parseURL ( decodeURIComponent ( pathToFileURL ( chunkInfo . facadeModuleId ) . href ) )
const isPath = await resolvePath ( component . filePath ) === pathname
if ( isPath ) {
// avoid importing the component chunk in all pages
chunkInfo . isEntry = false
pathAssociation [ component . pascalName ] = chunkPath
}
}
}
}
fs . writeFileSync ( join ( buildDir , 'components-chunk.mjs' ) , ` export const paths = ${ JSON . stringify ( pathAssociation , null , 2 ) } ` )
}
}
}
} )