import piniaPluginPersistedstate from 'pinia-plugin-persistedstate' import { PageContext } from 'vike/types' import { createSSRApp, defineComponent, h, markRaw, reactive, Component } from 'vue' import PageShell from '#components/PageShell.vue' import { setPageContext } from '#context/usePageContext' import i18n from '#plugins/i18n' import pinia from '#plugins/pinia' import CreateVuetify from '#plugins/vuetify' const vuetify = CreateVuetify(i18n) function createApp(pageContext: PageContext, isClient = true) { // eslint-disable-next-line no-use-before-define let rootComponent: InstanceType const PageWithWrapper = defineComponent({ data: () => ({ Page: markRaw(pageContext.Page), pageProps: markRaw(pageContext.pageProps || {}), isClient, }), created() { // eslint-disable-next-line @typescript-eslint/no-this-alias rootComponent = this }, render() { return h( PageShell as Component, {}, { default: () => { return h(this.Page, this.pageProps) }, }, ) }, }) if (isClient) { pinia.use(piniaPluginPersistedstate) } const app = createSSRApp(PageWithWrapper) app.use(pinia) app.use(i18n) app.use(vuetify) objectAssign(app, { changePage: (pageContext: PageContext) => { Object.assign(pageContextReactive, pageContext) rootComponent.Page = markRaw(pageContext.Page) rootComponent.pageProps = markRaw(pageContext.pageProps || {}) }, }) const pageContextReactive = reactive(pageContext) setPageContext(app, pageContextReactive) return { app, i18n } } // Same as `Object.assign()` but with type inference function objectAssign( obj: Obj, objAddendum: ObjAddendum, ): asserts obj is Obj & ObjAddendum { Object.assign(obj, objAddendum) } export { createApp }