Files
med-notes/.pnpm-store/v10/files/c4/c60a0759c566fe3743b8d22b6fe5bdaa3db77d3e02718b7c7dc70962ea859c5f432f9acbb228183bff78b5cf1058990b960440a1ede4fc4ac248d8f697b80d
2025-05-09 05:30:08 +02:00

126 lines
3.9 KiB
Plaintext

import * as React from 'react'
import { Outlet } from './Match'
import type { AsyncRouteComponent } from './route'
// If the load fails due to module not found, it may mean a new version of
// the build was deployed and the user's browser is still using an old version.
// If this happens, the old version in the user's browser would have an outdated
// URL to the lazy module.
// In that case, we want to attempt one window refresh to get the latest.
function isModuleNotFoundError(error: any): boolean {
// chrome: "Failed to fetch dynamically imported module: http://localhost:5173/src/routes/posts.index.tsx?tsr-split"
// firefox: "error loading dynamically imported module: http://localhost:5173/src/routes/posts.index.tsx?tsr-split"
// safari: "Importing a module script failed."
if (typeof error?.message !== 'string') return false
return (
error.message.startsWith('Failed to fetch dynamically imported module') ||
error.message.startsWith('error loading dynamically imported module') ||
error.message.startsWith('Importing a module script failed')
)
}
export function ClientOnly({
children,
fallback = null,
}: React.PropsWithChildren<{ fallback?: React.ReactNode }>) {
return useHydrated() ? <>{children}</> : <>{fallback}</>
}
function subscribe() {
return () => {}
}
export function useHydrated() {
return React.useSyncExternalStore(
subscribe,
() => true,
() => false,
)
}
export function lazyRouteComponent<
T extends Record<string, any>,
TKey extends keyof T = 'default',
>(
importer: () => Promise<T>,
exportName?: TKey,
ssr?: () => boolean,
): T[TKey] extends (props: infer TProps) => any
? AsyncRouteComponent<TProps>
: never {
let loadPromise: Promise<any> | undefined
let comp: T[TKey] | T['default']
let error: any
let reload: boolean
const load = () => {
if (typeof document === 'undefined' && ssr?.() === false) {
comp = (() => null) as any
return Promise.resolve()
}
if (!loadPromise) {
loadPromise = importer()
.then((res) => {
loadPromise = undefined
comp = res[exportName ?? 'default']
})
.catch((err) => {
// We don't want an error thrown from preload in this case, because
// there's nothing we want to do about module not found during preload.
// Record the error, the rest is handled during the render path.
error = err
if (isModuleNotFoundError(error)) {
if (
error instanceof Error &&
typeof window !== 'undefined' &&
typeof sessionStorage !== 'undefined'
) {
// Again, we want to reload one time on module not found error and not enter
// a reload loop if there is some other issue besides an old deploy.
// That's why we store our reload attempt in sessionStorage.
// Use error.message as key because it contains the module path that failed.
const storageKey = `tanstack_router_reload:${error.message}`
if (!sessionStorage.getItem(storageKey)) {
sessionStorage.setItem(storageKey, '1')
reload = true
}
}
}
})
}
return loadPromise
}
const lazyComp = function Lazy(props: any) {
// Now that we're out of preload and into actual render path,
if (reload) {
// If it was a module loading error,
// throw eternal suspense while we wait for window to reload
window.location.reload()
throw new Promise(() => {})
}
if (error) {
// Otherwise, just throw the error
throw error
}
if (!comp) {
throw load()
}
if (ssr?.() === false) {
return (
<ClientOnly fallback={<Outlet />}>
{React.createElement(comp, props)}
</ClientOnly>
)
}
return React.createElement(comp, props)
}
;(lazyComp as any).preload = load
return lazyComp as any
}