vinext 0.0.31 → 0.0.33
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +10 -7
- package/dist/build/report.d.ts +1 -1
- package/dist/build/report.js +334 -0
- package/dist/build/report.js.map +1 -1
- package/dist/build/run-prerender.js +2 -2
- package/dist/build/run-prerender.js.map +1 -1
- package/dist/check.js +93 -3
- package/dist/check.js.map +1 -1
- package/dist/cli.js +3 -1
- package/dist/cli.js.map +1 -1
- package/dist/config/next-config.d.ts +2 -0
- package/dist/config/next-config.js +9 -3
- package/dist/config/next-config.js.map +1 -1
- package/dist/entries/app-browser-entry.js +3 -330
- package/dist/entries/app-browser-entry.js.map +1 -1
- package/dist/entries/app-rsc-entry.js +286 -644
- package/dist/entries/app-rsc-entry.js.map +1 -1
- package/dist/entries/app-ssr-entry.js +4 -460
- package/dist/entries/app-ssr-entry.js.map +1 -1
- package/dist/entries/pages-server-entry.js +2 -1
- package/dist/entries/pages-server-entry.js.map +1 -1
- package/dist/entries/runtime-entry-module.d.ts +13 -0
- package/dist/entries/runtime-entry-module.js +27 -0
- package/dist/entries/runtime-entry-module.js.map +1 -0
- package/dist/index.d.ts +12 -0
- package/dist/index.js +16 -35
- package/dist/index.js.map +1 -1
- package/dist/plugins/optimize-imports.d.ts +38 -0
- package/dist/plugins/optimize-imports.js +557 -0
- package/dist/plugins/optimize-imports.js.map +1 -0
- package/dist/routing/pages-router.js +1 -1
- package/dist/routing/pages-router.js.map +1 -1
- package/dist/server/api-handler.d.ts +2 -2
- package/dist/server/api-handler.js +3 -4
- package/dist/server/api-handler.js.map +1 -1
- package/dist/server/app-browser-entry.d.ts +1 -0
- package/dist/server/app-browser-entry.js +161 -0
- package/dist/server/app-browser-entry.js.map +1 -0
- package/dist/server/app-browser-stream.d.ts +33 -0
- package/dist/server/app-browser-stream.js +54 -0
- package/dist/server/app-browser-stream.js.map +1 -0
- package/dist/server/app-page-cache.d.ts +61 -0
- package/dist/server/app-page-cache.js +133 -0
- package/dist/server/app-page-cache.js.map +1 -0
- package/dist/server/app-page-response.d.ts +51 -0
- package/dist/server/app-page-response.js +90 -0
- package/dist/server/app-page-response.js.map +1 -0
- package/dist/server/app-route-handler-cache.d.ts +42 -0
- package/dist/server/app-route-handler-cache.js +69 -0
- package/dist/server/app-route-handler-cache.js.map +1 -0
- package/dist/server/app-route-handler-execution.d.ts +64 -0
- package/dist/server/app-route-handler-execution.js +100 -0
- package/dist/server/app-route-handler-execution.js.map +1 -0
- package/dist/server/app-route-handler-policy.d.ts +51 -0
- package/dist/server/app-route-handler-policy.js +57 -0
- package/dist/server/app-route-handler-policy.js.map +1 -0
- package/dist/server/app-route-handler-response.d.ts +26 -0
- package/dist/server/app-route-handler-response.js +61 -0
- package/dist/server/app-route-handler-response.js.map +1 -0
- package/dist/server/app-route-handler-runtime.d.ts +27 -0
- package/dist/server/app-route-handler-runtime.js +99 -0
- package/dist/server/app-route-handler-runtime.js.map +1 -0
- package/dist/server/app-ssr-entry.d.ts +19 -0
- package/dist/server/app-ssr-entry.js +105 -0
- package/dist/server/app-ssr-entry.js.map +1 -0
- package/dist/server/app-ssr-stream.d.ts +30 -0
- package/dist/server/app-ssr-stream.js +116 -0
- package/dist/server/app-ssr-stream.js.map +1 -0
- package/dist/server/dev-server.d.ts +3 -2
- package/dist/server/dev-server.js +29 -30
- package/dist/server/dev-server.js.map +1 -1
- package/dist/server/instrumentation.d.ts +11 -39
- package/dist/server/instrumentation.js +14 -15
- package/dist/server/instrumentation.js.map +1 -1
- package/dist/server/middleware.d.ts +3 -2
- package/dist/server/middleware.js +12 -29
- package/dist/server/middleware.js.map +1 -1
- package/dist/server/prod-server.js +3 -2
- package/dist/server/prod-server.js.map +1 -1
- package/dist/shims/compat-router.d.ts +3 -1
- package/dist/shims/compat-router.js.map +1 -1
- package/dist/shims/error-boundary.d.ts +1 -1
- package/dist/shims/fetch-cache.js +2 -0
- package/dist/shims/fetch-cache.js.map +1 -1
- package/dist/shims/head.d.ts +2 -1
- package/dist/shims/head.js +27 -5
- package/dist/shims/head.js.map +1 -1
- package/dist/shims/internal/router-context.d.ts +2 -1
- package/dist/shims/internal/router-context.js.map +1 -1
- package/dist/shims/metadata.js +3 -3
- package/dist/shims/metadata.js.map +1 -1
- package/dist/shims/request-state-types.d.ts +2 -2
- package/dist/shims/router.d.ts +1 -1
- package/dist/shims/router.js.map +1 -1
- package/dist/shims/server.d.ts +41 -3
- package/dist/shims/server.js +90 -7
- package/dist/shims/server.js.map +1 -1
- package/dist/shims/unified-request-context.d.ts +1 -1
- package/package.json +1 -1
|
@@ -1,11 +1,11 @@
|
|
|
1
1
|
import { PrivateCacheState } from "./cache-runtime.js";
|
|
2
2
|
import { ExecutionContextLike } from "./request-context.js";
|
|
3
3
|
import { CacheLifeConfig, CacheState } from "./cache.js";
|
|
4
|
+
import { HeadersAccessPhase, HeadersContext, VinextHeadersShimState } from "./headers.js";
|
|
5
|
+
import { NavigationContext } from "./navigation.js";
|
|
4
6
|
import { RouterState, SSRContext } from "./router-state.js";
|
|
5
7
|
import { FetchCacheState } from "./fetch-cache.js";
|
|
6
8
|
import { HeadState } from "./head-state.js";
|
|
7
|
-
import { HeadersAccessPhase, HeadersContext, VinextHeadersShimState } from "./headers.js";
|
|
8
9
|
import { I18nState } from "./i18n-state.js";
|
|
9
|
-
import { NavigationContext } from "./navigation.js";
|
|
10
10
|
import { NavigationState } from "./navigation-state.js";
|
|
11
11
|
export { type CacheLifeConfig, type CacheState, type ExecutionContextLike, type FetchCacheState, type HeadState, type HeadersAccessPhase, type HeadersContext, type I18nState, type NavigationContext, type NavigationState, type PrivateCacheState, type RouterState, type SSRContext, type VinextHeadersShimState };
|
package/dist/shims/router.d.ts
CHANGED
|
@@ -118,5 +118,5 @@ declare const Router: {
|
|
|
118
118
|
events: RouterEvents;
|
|
119
119
|
};
|
|
120
120
|
//#endregion
|
|
121
|
-
export { _registerRouterStateAccessors, applyNavigationLocale, Router as default, isExternalUrl, isHashOnlyChange, setSSRContext, useRouter, wrapWithRouterContext };
|
|
121
|
+
export { NextRouter, _registerRouterStateAccessors, applyNavigationLocale, Router as default, isExternalUrl, isHashOnlyChange, setSSRContext, useRouter, wrapWithRouterContext };
|
|
122
122
|
//# sourceMappingURL=router.d.ts.map
|
package/dist/shims/router.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"router.js","names":[],"sources":["../../src/shims/router.ts"],"sourcesContent":["/**\n * next/router shim\n *\n * Provides useRouter() hook and Router singleton for Pages Router.\n * Backed by the browser History API. Supports client-side navigation\n * by fetching new page data and re-rendering the React root.\n */\nimport { useState, useEffect, useCallback, useMemo, createElement, type ReactElement } from \"react\";\nimport { RouterContext } from \"./internal/router-context.js\";\nimport type { VinextNextData } from \"../client/vinext-next-data.js\";\nimport { isValidModulePath } from \"../client/validate-module-path.js\";\nimport { toBrowserNavigationHref, toSameOriginAppPath } from \"./url-utils.js\";\nimport { stripBasePath } from \"../utils/base-path.js\";\nimport { addLocalePrefix, getDomainLocaleUrl, type DomainLocale } from \"../utils/domain-locale.js\";\nimport {\n addQueryParam,\n appendSearchParamsToUrl,\n type UrlQuery,\n urlQueryToSearchParams,\n} from \"../utils/query.js\";\n\n/** basePath from next.config.js, injected by the plugin at build time */\nconst __basePath: string = process.env.__NEXT_ROUTER_BASEPATH ?? \"\";\n\ntype BeforePopStateCallback = (state: {\n url: string;\n as: string;\n options: { shallow: boolean };\n}) => boolean;\n\ninterface NextRouter {\n /** Current pathname */\n pathname: string;\n /** Current route pattern (e.g., \"/posts/[id]\") */\n route: string;\n /** Query parameters */\n query: Record<string, string | string[]>;\n /** Full URL including query string */\n asPath: string;\n /** Base path */\n basePath: string;\n /** Current locale */\n locale?: string;\n /** Available locales */\n locales?: string[];\n /** Default locale */\n defaultLocale?: string;\n /** Configured domain locales */\n domainLocales?: VinextNextData[\"domainLocales\"];\n /** Whether the router is ready */\n isReady: boolean;\n /** Whether this is a preview */\n isPreview: boolean;\n /** Whether this is a fallback page */\n isFallback: boolean;\n\n /** Navigate to a new URL */\n push(url: string | UrlObject, as?: string, options?: TransitionOptions): Promise<boolean>;\n /** Replace current URL */\n replace(url: string | UrlObject, as?: string, options?: TransitionOptions): Promise<boolean>;\n /** Go back */\n back(): void;\n /** Reload the page */\n reload(): void;\n /** Prefetch a page (injects <link rel=\"prefetch\">) */\n prefetch(url: string): Promise<void>;\n /** Register a callback to run before popstate navigation */\n beforePopState(cb: BeforePopStateCallback): void;\n /** Listen for route changes */\n events: RouterEvents;\n}\n\ninterface UrlObject {\n pathname?: string;\n query?: UrlQuery;\n}\n\ninterface TransitionOptions {\n shallow?: boolean;\n scroll?: boolean;\n locale?: string;\n}\n\ninterface RouterEvents {\n on(event: string, handler: (...args: unknown[]) => void): void;\n off(event: string, handler: (...args: unknown[]) => void): void;\n emit(event: string, ...args: unknown[]): void;\n}\n\nfunction createRouterEvents(): RouterEvents {\n const listeners = new Map<string, Set<(...args: unknown[]) => void>>();\n\n return {\n on(event: string, handler: (...args: unknown[]) => void) {\n if (!listeners.has(event)) listeners.set(event, new Set());\n (listeners.get(event) as Set<(...args: unknown[]) => void>).add(handler);\n },\n off(event: string, handler: (...args: unknown[]) => void) {\n listeners.get(event)?.delete(handler);\n },\n emit(event: string, ...args: unknown[]) {\n listeners.get(event)?.forEach((handler) => handler(...args));\n },\n };\n}\n\n// Singleton events instance\nconst routerEvents = createRouterEvents();\n\nfunction resolveUrl(url: string | UrlObject): string {\n if (typeof url === \"string\") return url;\n let result = url.pathname ?? \"/\";\n if (url.query) {\n const params = urlQueryToSearchParams(url.query);\n result = appendSearchParamsToUrl(result, params);\n }\n return result;\n}\n\n/**\n * When `as` is provided, use it as the navigation target. This is a\n * simplification: Next.js keeps `url` and `as` as separate values (url for\n * data fetching, as for the browser URL). We collapse them because vinext's\n * navigateClient() fetches HTML from the target URL, so `as` must be a\n * server-resolvable path. Purely decorative `as` values are not supported.\n */\nfunction resolveNavigationTarget(\n url: string | UrlObject,\n as: string | undefined,\n locale: string | undefined,\n): string {\n return applyNavigationLocale(as ?? resolveUrl(url), locale);\n}\n\nfunction getDomainLocales(): readonly DomainLocale[] | undefined {\n return (window.__NEXT_DATA__ as VinextNextData | undefined)?.domainLocales;\n}\n\nfunction getCurrentHostname(): string | undefined {\n return window.location?.hostname;\n}\n\nfunction getDomainLocalePath(url: string, locale: string): string | undefined {\n return getDomainLocaleUrl(url, locale, {\n basePath: __basePath,\n currentHostname: getCurrentHostname(),\n domainItems: getDomainLocales(),\n });\n}\n\n/**\n * Apply locale prefix to a URL for client-side navigation.\n * Same logic as Link's applyLocaleToHref but reads from window globals.\n */\nexport function applyNavigationLocale(url: string, locale?: string): string {\n if (!locale || typeof window === \"undefined\") return url;\n // Absolute and protocol-relative URLs must not be prefixed — locale\n // only applies to local paths.\n if (url.startsWith(\"http://\") || url.startsWith(\"https://\") || url.startsWith(\"//\")) {\n return url;\n }\n\n const domainLocalePath = getDomainLocalePath(url, locale);\n if (domainLocalePath) return domainLocalePath;\n\n return addLocalePrefix(url, locale, window.__VINEXT_DEFAULT_LOCALE__ ?? \"\");\n}\n\n/** Check if a URL is external (any URL scheme per RFC 3986, or protocol-relative) */\nexport function isExternalUrl(url: string): boolean {\n return /^[a-z][a-z0-9+.-]*:/i.test(url) || url.startsWith(\"//\");\n}\n\n/** Resolve a hash URL to a basePath-stripped app URL for event payloads */\nfunction resolveHashUrl(url: string): string {\n if (typeof window === \"undefined\") return url;\n if (url.startsWith(\"#\"))\n return stripBasePath(window.location.pathname, __basePath) + window.location.search + url;\n // Full-path hash URL — strip basePath for consistency with other events\n try {\n const parsed = new URL(url, window.location.href);\n return stripBasePath(parsed.pathname, __basePath) + parsed.search + parsed.hash;\n } catch {\n return url;\n }\n}\n\n/** Check if a href is only a hash change relative to the current URL */\nexport function isHashOnlyChange(href: string): boolean {\n if (href.startsWith(\"#\")) return true;\n if (typeof window === \"undefined\") return false;\n try {\n const current = new URL(window.location.href);\n const next = new URL(href, window.location.href);\n return current.pathname === next.pathname && current.search === next.search && next.hash !== \"\";\n } catch {\n return false;\n }\n}\n\n/** Scroll to hash target element, or top if no hash */\nfunction scrollToHash(hash: string): void {\n if (!hash || hash === \"#\") {\n window.scrollTo(0, 0);\n return;\n }\n const el = document.getElementById(hash.slice(1));\n if (el) el.scrollIntoView({ behavior: \"auto\" });\n}\n\n/** Save current scroll position into history state for back/forward restoration */\nfunction saveScrollPosition(): void {\n const state = window.history.state ?? {};\n window.history.replaceState(\n { ...state, __vinext_scrollX: window.scrollX, __vinext_scrollY: window.scrollY },\n \"\",\n );\n}\n\n/** Restore scroll position from history state */\nfunction restoreScrollPosition(state: unknown): void {\n if (state && typeof state === \"object\" && \"__vinext_scrollY\" in state) {\n const { __vinext_scrollX: x, __vinext_scrollY: y } = state as {\n __vinext_scrollX: number;\n __vinext_scrollY: number;\n };\n requestAnimationFrame(() => window.scrollTo(x, y));\n }\n}\n\n/**\n * SSR context - set by the dev server before rendering each page.\n */\ninterface SSRContext {\n pathname: string;\n query: Record<string, string | string[]>;\n asPath: string;\n locale?: string;\n locales?: string[];\n defaultLocale?: string;\n domainLocales?: VinextNextData[\"domainLocales\"];\n}\n\n// ---------------------------------------------------------------------------\n// Server-side SSR state uses a registration pattern so this module can be\n// bundled for the browser. The ALS-backed implementation lives in\n// router-state.ts (server-only) and registers itself on import.\n// ---------------------------------------------------------------------------\n\nlet _ssrContext: SSRContext | null = null;\n\nlet _getSSRContext = (): SSRContext | null => _ssrContext;\nlet _setSSRContextImpl = (ctx: SSRContext | null): void => {\n _ssrContext = ctx;\n};\n\n/**\n * Register ALS-backed state accessors. Called by router-state.ts on import.\n * @internal\n */\nexport function _registerRouterStateAccessors(accessors: {\n getSSRContext: () => SSRContext | null;\n setSSRContext: (ctx: SSRContext | null) => void;\n}): void {\n _getSSRContext = accessors.getSSRContext;\n _setSSRContextImpl = accessors.setSSRContext;\n}\n\nexport function setSSRContext(ctx: SSRContext | null): void {\n _setSSRContextImpl(ctx);\n}\n\n/**\n * Extract param names from a Next.js route pattern.\n * E.g., \"/posts/[id]\" → [\"id\"], \"/docs/[...slug]\" → [\"slug\"],\n * \"/shop/[[...path]]\" → [\"path\"], \"/blog/[year]/[month]\" → [\"year\", \"month\"]\n * Also handles internal format: \"/posts/:id\" → [\"id\"], \"/docs/:slug+\" → [\"slug\"]\n */\nfunction extractRouteParamNames(pattern: string): string[] {\n const names: string[] = [];\n // Match Next.js bracket format: [id], [...slug], [[...slug]]\n const bracketMatches = pattern.matchAll(/\\[{1,2}(?:\\.\\.\\.)?([\\w-]+)\\]{1,2}/g);\n for (const m of bracketMatches) {\n names.push(m[1]);\n }\n if (names.length > 0) return names;\n // Fallback: match internal :param format\n const colonMatches = pattern.matchAll(/:([\\w-]+)[+*]?/g);\n for (const m of colonMatches) {\n names.push(m[1]);\n }\n return names;\n}\n\nfunction getPathnameAndQuery(): {\n pathname: string;\n query: Record<string, string | string[]>;\n asPath: string;\n} {\n if (typeof window === \"undefined\") {\n const _ssrCtx = _getSSRContext();\n if (_ssrCtx) {\n const query: Record<string, string | string[]> = {};\n for (const [key, value] of Object.entries(_ssrCtx.query)) {\n query[key] = Array.isArray(value) ? [...value] : value;\n }\n return { pathname: _ssrCtx.pathname, query, asPath: _ssrCtx.asPath };\n }\n return { pathname: \"/\", query: {}, asPath: \"/\" };\n }\n const resolvedPath = stripBasePath(window.location.pathname, __basePath);\n // In Next.js, router.pathname is the route pattern (e.g., \"/posts/[id]\"),\n // not the resolved path (\"/posts/42\"). __NEXT_DATA__.page holds the route\n // pattern and is updated by navigateClient() on every client-side navigation.\n const pathname = window.__NEXT_DATA__?.page ?? resolvedPath;\n const routeQuery: Record<string, string | string[]> = {};\n // Include dynamic route params from __NEXT_DATA__ (e.g., { id: \"42\" } from /posts/[id]).\n // Only include keys that are part of the route pattern (not stale query params).\n const nextData = window.__NEXT_DATA__;\n if (nextData && nextData.query && nextData.page) {\n const routeParamNames = extractRouteParamNames(nextData.page);\n for (const key of routeParamNames) {\n const value = nextData.query[key];\n if (typeof value === \"string\") {\n routeQuery[key] = value;\n } else if (Array.isArray(value)) {\n routeQuery[key] = [...value];\n }\n }\n }\n // URL search params always reflect the current URL\n const searchQuery: Record<string, string | string[]> = {};\n const params = new URLSearchParams(window.location.search);\n for (const [key, value] of params) {\n addQueryParam(searchQuery, key, value);\n }\n const query = { ...searchQuery, ...routeQuery };\n // asPath uses the resolved browser path, not the route pattern\n const asPath = resolvedPath + window.location.search + window.location.hash;\n return { pathname, query, asPath };\n}\n\n/**\n * Perform client-side navigation: fetch the target page's HTML,\n * extract __NEXT_DATA__, and re-render the React root.\n */\nlet _navInProgress = false;\nasync function navigateClient(url: string): Promise<void> {\n if (typeof window === \"undefined\") return;\n\n const root = window.__VINEXT_ROOT__;\n if (!root) {\n // No React root yet — fall back to hard navigation\n window.location.href = url;\n return;\n }\n\n // Prevent re-entrant navigation (e.g., double popstate events)\n if (_navInProgress) return;\n _navInProgress = true;\n\n try {\n // Fetch the target page's SSR HTML\n const res = await fetch(url, { headers: { Accept: \"text/html\" } });\n if (!res.ok) {\n window.location.href = url;\n return;\n }\n\n const html = await res.text();\n\n // Extract __NEXT_DATA__ from the HTML\n const match = html.match(/<script>window\\.__NEXT_DATA__\\s*=\\s*(.*?)<\\/script>/);\n if (!match) {\n window.location.href = url;\n return;\n }\n\n const nextData = JSON.parse(match[1]);\n const { pageProps } = nextData.props;\n window.__NEXT_DATA__ = nextData;\n\n // Get the page module URL from __NEXT_DATA__.__vinext (preferred),\n // or fall back to parsing the hydration script\n let pageModuleUrl: string | undefined = nextData.__vinext?.pageModuleUrl;\n\n if (!pageModuleUrl) {\n // Legacy fallback: try to find the module URL in the inline script\n const moduleMatch = html.match(/import\\(\"([^\"]+)\"\\);\\s*\\n\\s*const PageComponent/);\n const altMatch = html.match(/await import\\(\"([^\"]+pages\\/[^\"]+)\"\\)/);\n pageModuleUrl = moduleMatch?.[1] ?? altMatch?.[1] ?? undefined;\n }\n\n if (!pageModuleUrl) {\n window.location.href = url;\n return;\n }\n\n // Validate the module URL before importing — defense-in-depth against\n // unexpected __NEXT_DATA__ or malformed HTML responses\n if (!isValidModulePath(pageModuleUrl)) {\n console.error(\"[vinext] Blocked import of invalid page module path:\", pageModuleUrl);\n window.location.href = url;\n return;\n }\n\n // Dynamically import the new page module\n const pageModule = await import(/* @vite-ignore */ pageModuleUrl);\n const PageComponent = pageModule.default;\n\n if (!PageComponent) {\n window.location.href = url;\n return;\n }\n\n // Import React for createElement\n const React = (await import(\"react\")).default;\n\n // Re-render with the new page, loading _app if needed\n let AppComponent = window.__VINEXT_APP__;\n const appModuleUrl: string | undefined = nextData.__vinext?.appModuleUrl;\n\n if (!AppComponent && appModuleUrl) {\n if (!isValidModulePath(appModuleUrl)) {\n console.error(\"[vinext] Blocked import of invalid app module path:\", appModuleUrl);\n } else {\n try {\n const appModule = await import(/* @vite-ignore */ appModuleUrl);\n AppComponent = appModule.default;\n window.__VINEXT_APP__ = AppComponent;\n } catch {\n // _app not available — continue without it\n }\n }\n }\n\n let element;\n if (AppComponent) {\n element = React.createElement(AppComponent, {\n Component: PageComponent,\n pageProps,\n });\n } else {\n element = React.createElement(PageComponent, pageProps);\n }\n\n // Wrap with RouterContext.Provider so next/compat/router works\n element = wrapWithRouterContext(element);\n\n root.render(element);\n } catch (err) {\n console.error(\"[vinext] Client navigation failed:\", err);\n routerEvents.emit(\"routeChangeError\", err, url, { shallow: false });\n window.location.href = url;\n } finally {\n _navInProgress = false;\n }\n}\n\n/**\n * Build the full router value object from the current pathname, query, asPath,\n * and a set of navigation methods. Shared by useRouter() (which passes\n * hook-derived callbacks) and wrapWithRouterContext() (which passes the Router\n * singleton methods) so the shape stays in sync.\n */\nfunction buildRouterValue(\n pathname: string,\n query: Record<string, string | string[]>,\n asPath: string,\n methods: {\n push: NextRouter[\"push\"];\n replace: NextRouter[\"replace\"];\n back: NextRouter[\"back\"];\n reload: NextRouter[\"reload\"];\n prefetch: NextRouter[\"prefetch\"];\n beforePopState: NextRouter[\"beforePopState\"];\n },\n): NextRouter {\n const _ssrState = _getSSRContext();\n const nextData =\n typeof window !== \"undefined\"\n ? (window.__NEXT_DATA__ as VinextNextData | undefined)\n : undefined;\n const locale = typeof window === \"undefined\" ? _ssrState?.locale : window.__VINEXT_LOCALE__;\n const locales = typeof window === \"undefined\" ? _ssrState?.locales : window.__VINEXT_LOCALES__;\n const defaultLocale =\n typeof window === \"undefined\" ? _ssrState?.defaultLocale : window.__VINEXT_DEFAULT_LOCALE__;\n const domainLocales =\n typeof window === \"undefined\" ? _ssrState?.domainLocales : nextData?.domainLocales;\n\n const route = typeof window !== \"undefined\" ? (nextData?.page ?? pathname) : pathname;\n\n return {\n pathname,\n route,\n query,\n asPath,\n basePath: __basePath,\n locale,\n locales,\n defaultLocale,\n domainLocales,\n isReady: true,\n isPreview: false,\n isFallback: typeof window !== \"undefined\" && nextData?.isFallback === true,\n ...methods,\n events: routerEvents,\n };\n}\n\n/**\n * useRouter hook - Pages Router compatible.\n */\nexport function useRouter(): NextRouter {\n const [{ pathname, query, asPath }, setState] = useState(getPathnameAndQuery);\n\n // Popstate is handled by the module-level listener below so beforePopState()\n // is consistently enforced even when multiple components mount useRouter().\n useEffect(() => {\n const onNavigate = ((_e: CustomEvent) => {\n setState(getPathnameAndQuery());\n }) as EventListener;\n window.addEventListener(\"vinext:navigate\", onNavigate);\n return () => window.removeEventListener(\"vinext:navigate\", onNavigate);\n }, []);\n\n const push = useCallback(\n async (url: string | UrlObject, as?: string, options?: TransitionOptions): Promise<boolean> => {\n let resolved = resolveNavigationTarget(url, as, options?.locale);\n\n // External URLs — delegate to browser (unless same-origin)\n if (isExternalUrl(resolved)) {\n const localPath = toSameOriginAppPath(resolved, __basePath);\n if (localPath == null) {\n window.location.assign(resolved);\n return true;\n }\n resolved = localPath;\n }\n\n const full = toBrowserNavigationHref(resolved, window.location.href, __basePath);\n\n // Hash-only change — no page fetch needed\n if (isHashOnlyChange(resolved)) {\n const eventUrl = resolveHashUrl(resolved);\n routerEvents.emit(\"hashChangeStart\", eventUrl, {\n shallow: options?.shallow ?? false,\n });\n const hash = resolved.includes(\"#\") ? resolved.slice(resolved.indexOf(\"#\")) : \"\";\n window.history.pushState({}, \"\", resolved.startsWith(\"#\") ? resolved : full);\n _lastPathnameAndSearch = window.location.pathname + window.location.search;\n scrollToHash(hash);\n setState(getPathnameAndQuery());\n routerEvents.emit(\"hashChangeComplete\", eventUrl, {\n shallow: options?.shallow ?? false,\n });\n window.dispatchEvent(new CustomEvent(\"vinext:navigate\"));\n return true;\n }\n\n saveScrollPosition();\n routerEvents.emit(\"routeChangeStart\", resolved, { shallow: options?.shallow ?? false });\n routerEvents.emit(\"beforeHistoryChange\", resolved, { shallow: options?.shallow ?? false });\n window.history.pushState({}, \"\", full);\n _lastPathnameAndSearch = window.location.pathname + window.location.search;\n if (!options?.shallow) {\n await navigateClient(full);\n }\n setState(getPathnameAndQuery());\n routerEvents.emit(\"routeChangeComplete\", resolved, { shallow: options?.shallow ?? false });\n\n // Scroll: handle hash target, else scroll to top unless scroll:false\n const hash = resolved.includes(\"#\") ? resolved.slice(resolved.indexOf(\"#\")) : \"\";\n if (hash) {\n scrollToHash(hash);\n } else if (options?.scroll !== false) {\n window.scrollTo(0, 0);\n }\n window.dispatchEvent(new CustomEvent(\"vinext:navigate\"));\n return true;\n },\n [],\n );\n\n const replace = useCallback(\n async (url: string | UrlObject, as?: string, options?: TransitionOptions): Promise<boolean> => {\n let resolved = resolveNavigationTarget(url, as, options?.locale);\n\n // External URLs — delegate to browser (unless same-origin)\n if (isExternalUrl(resolved)) {\n const localPath = toSameOriginAppPath(resolved, __basePath);\n if (localPath == null) {\n window.location.replace(resolved);\n return true;\n }\n resolved = localPath;\n }\n\n const full = toBrowserNavigationHref(resolved, window.location.href, __basePath);\n\n // Hash-only change — no page fetch needed\n if (isHashOnlyChange(resolved)) {\n const eventUrl = resolveHashUrl(resolved);\n routerEvents.emit(\"hashChangeStart\", eventUrl, {\n shallow: options?.shallow ?? false,\n });\n const hash = resolved.includes(\"#\") ? resolved.slice(resolved.indexOf(\"#\")) : \"\";\n window.history.replaceState({}, \"\", resolved.startsWith(\"#\") ? resolved : full);\n _lastPathnameAndSearch = window.location.pathname + window.location.search;\n scrollToHash(hash);\n setState(getPathnameAndQuery());\n routerEvents.emit(\"hashChangeComplete\", eventUrl, {\n shallow: options?.shallow ?? false,\n });\n window.dispatchEvent(new CustomEvent(\"vinext:navigate\"));\n return true;\n }\n\n routerEvents.emit(\"routeChangeStart\", resolved, { shallow: options?.shallow ?? false });\n routerEvents.emit(\"beforeHistoryChange\", resolved, { shallow: options?.shallow ?? false });\n window.history.replaceState({}, \"\", full);\n _lastPathnameAndSearch = window.location.pathname + window.location.search;\n if (!options?.shallow) {\n await navigateClient(full);\n }\n setState(getPathnameAndQuery());\n routerEvents.emit(\"routeChangeComplete\", resolved, { shallow: options?.shallow ?? false });\n\n // Scroll: handle hash target, else scroll to top unless scroll:false\n const hash = resolved.includes(\"#\") ? resolved.slice(resolved.indexOf(\"#\")) : \"\";\n if (hash) {\n scrollToHash(hash);\n } else if (options?.scroll !== false) {\n window.scrollTo(0, 0);\n }\n window.dispatchEvent(new CustomEvent(\"vinext:navigate\"));\n return true;\n },\n [],\n );\n\n const back = useCallback(() => {\n window.history.back();\n }, []);\n\n const reload = useCallback(() => {\n window.location.reload();\n }, []);\n\n const prefetch = useCallback(async (url: string): Promise<void> => {\n // Inject a <link rel=\"prefetch\"> for the target page\n if (typeof document !== \"undefined\") {\n const link = document.createElement(\"link\");\n link.rel = \"prefetch\";\n link.href = url;\n link.as = \"document\";\n document.head.appendChild(link);\n }\n }, []);\n\n const router = useMemo(\n (): NextRouter =>\n buildRouterValue(pathname, query, asPath, {\n push,\n replace,\n back,\n reload,\n prefetch,\n beforePopState: (cb: BeforePopStateCallback) => {\n _beforePopStateCb = cb;\n },\n }),\n [pathname, query, asPath, push, replace, back, reload, prefetch],\n );\n\n return router;\n}\n\n// beforePopState callback: called before handling browser back/forward.\n// If it returns false, the navigation is cancelled.\nlet _beforePopStateCb: BeforePopStateCallback | undefined;\n\n// Track pathname+search for detecting hash-only back/forward in the popstate\n// handler. Updated after every pushState/replaceState so that popstate can\n// compare the previous value with the (already-changed) window.location.\nlet _lastPathnameAndSearch =\n typeof window !== \"undefined\" ? window.location.pathname + window.location.search : \"\";\n\n// Module-level popstate listener: handles browser back/forward by re-rendering\n// the React root with the page at the new URL. This runs regardless of whether\n// any component calls useRouter().\nif (typeof window !== \"undefined\") {\n window.addEventListener(\"popstate\", (e: PopStateEvent) => {\n const browserUrl = window.location.pathname + window.location.search;\n const appUrl = stripBasePath(window.location.pathname, __basePath) + window.location.search;\n\n // Detect hash-only back/forward: pathname+search unchanged, only hash differs.\n const isHashOnly = browserUrl === _lastPathnameAndSearch;\n\n // Check beforePopState callback\n if (_beforePopStateCb !== undefined) {\n const shouldContinue = (_beforePopStateCb as BeforePopStateCallback)({\n url: appUrl,\n as: appUrl,\n options: { shallow: false },\n });\n if (!shouldContinue) return;\n }\n\n // Update tracker only after beforePopState confirms navigation proceeds.\n // If beforePopState cancels, the tracker must retain the previous value\n // so the next popstate compares against the correct baseline.\n _lastPathnameAndSearch = browserUrl;\n\n if (isHashOnly) {\n // Hash-only back/forward — no page fetch needed\n const hashUrl = appUrl + window.location.hash;\n routerEvents.emit(\"hashChangeStart\", hashUrl, { shallow: false });\n scrollToHash(window.location.hash);\n routerEvents.emit(\"hashChangeComplete\", hashUrl, { shallow: false });\n window.dispatchEvent(new CustomEvent(\"vinext:navigate\"));\n return;\n }\n\n const fullAppUrl = appUrl + window.location.hash;\n routerEvents.emit(\"routeChangeStart\", fullAppUrl, { shallow: false });\n // Note: The browser has already updated window.location by the time popstate\n // fires, so this is not truly \"before\" the URL change. In Next.js the popstate\n // handler calls replaceState to store history metadata — beforeHistoryChange\n // precedes that call, not the URL change itself. We emit it here for API\n // compatibility.\n routerEvents.emit(\"beforeHistoryChange\", fullAppUrl, { shallow: false });\n void navigateClient(browserUrl).then(() => {\n routerEvents.emit(\"routeChangeComplete\", fullAppUrl, { shallow: false });\n restoreScrollPosition(e.state);\n window.dispatchEvent(new CustomEvent(\"vinext:navigate\"));\n });\n });\n}\n\n/**\n * Wrap a React element in a RouterContext.Provider so that\n * next/compat/router's useRouter() returns the real Pages Router value.\n *\n * This is a plain function, NOT a React component — it builds the router\n * value object directly from the current SSR context (server) or\n * window.location + Router singleton (client), avoiding duplicate state\n * that a hook-based component would create.\n */\nexport function wrapWithRouterContext(element: ReactElement): ReactElement {\n const { pathname, query, asPath } = getPathnameAndQuery();\n\n const routerValue = buildRouterValue(pathname, query, asPath, {\n push: Router.push,\n replace: Router.replace,\n back: Router.back,\n reload: Router.reload,\n prefetch: Router.prefetch,\n beforePopState: Router.beforePopState,\n });\n\n return createElement(RouterContext.Provider, { value: routerValue }, element) as ReactElement;\n}\n\n// Also export a default Router singleton for `import Router from 'next/router'`\nconst Router = {\n push: async (url: string | UrlObject, as?: string, options?: TransitionOptions) => {\n let resolved = resolveNavigationTarget(url, as, options?.locale);\n\n // External URLs (unless same-origin)\n if (isExternalUrl(resolved)) {\n const localPath = toSameOriginAppPath(resolved, __basePath);\n if (localPath == null) {\n window.location.assign(resolved);\n return true;\n }\n resolved = localPath;\n }\n\n const full = toBrowserNavigationHref(resolved, window.location.href, __basePath);\n\n // Hash-only change\n if (isHashOnlyChange(resolved)) {\n const eventUrl = resolveHashUrl(resolved);\n routerEvents.emit(\"hashChangeStart\", eventUrl, {\n shallow: options?.shallow ?? false,\n });\n const hash = resolved.includes(\"#\") ? resolved.slice(resolved.indexOf(\"#\")) : \"\";\n window.history.pushState({}, \"\", resolved.startsWith(\"#\") ? resolved : full);\n _lastPathnameAndSearch = window.location.pathname + window.location.search;\n scrollToHash(hash);\n routerEvents.emit(\"hashChangeComplete\", eventUrl, {\n shallow: options?.shallow ?? false,\n });\n window.dispatchEvent(new CustomEvent(\"vinext:navigate\"));\n return true;\n }\n\n saveScrollPosition();\n routerEvents.emit(\"routeChangeStart\", resolved, { shallow: options?.shallow ?? false });\n routerEvents.emit(\"beforeHistoryChange\", resolved, { shallow: options?.shallow ?? false });\n window.history.pushState({}, \"\", full);\n _lastPathnameAndSearch = window.location.pathname + window.location.search;\n if (!options?.shallow) {\n await navigateClient(full);\n }\n routerEvents.emit(\"routeChangeComplete\", resolved, { shallow: options?.shallow ?? false });\n\n const hash = resolved.includes(\"#\") ? resolved.slice(resolved.indexOf(\"#\")) : \"\";\n if (hash) {\n scrollToHash(hash);\n } else if (options?.scroll !== false) {\n window.scrollTo(0, 0);\n }\n window.dispatchEvent(new CustomEvent(\"vinext:navigate\"));\n return true;\n },\n replace: async (url: string | UrlObject, as?: string, options?: TransitionOptions) => {\n let resolved = resolveNavigationTarget(url, as, options?.locale);\n\n // External URLs (unless same-origin)\n if (isExternalUrl(resolved)) {\n const localPath = toSameOriginAppPath(resolved, __basePath);\n if (localPath == null) {\n window.location.replace(resolved);\n return true;\n }\n resolved = localPath;\n }\n\n const full = toBrowserNavigationHref(resolved, window.location.href, __basePath);\n\n // Hash-only change\n if (isHashOnlyChange(resolved)) {\n const eventUrl = resolveHashUrl(resolved);\n routerEvents.emit(\"hashChangeStart\", eventUrl, {\n shallow: options?.shallow ?? false,\n });\n const hash = resolved.includes(\"#\") ? resolved.slice(resolved.indexOf(\"#\")) : \"\";\n window.history.replaceState({}, \"\", resolved.startsWith(\"#\") ? resolved : full);\n _lastPathnameAndSearch = window.location.pathname + window.location.search;\n scrollToHash(hash);\n routerEvents.emit(\"hashChangeComplete\", eventUrl, {\n shallow: options?.shallow ?? false,\n });\n window.dispatchEvent(new CustomEvent(\"vinext:navigate\"));\n return true;\n }\n\n routerEvents.emit(\"routeChangeStart\", resolved, { shallow: options?.shallow ?? false });\n routerEvents.emit(\"beforeHistoryChange\", resolved, { shallow: options?.shallow ?? false });\n window.history.replaceState({}, \"\", full);\n _lastPathnameAndSearch = window.location.pathname + window.location.search;\n if (!options?.shallow) {\n await navigateClient(full);\n }\n routerEvents.emit(\"routeChangeComplete\", resolved, { shallow: options?.shallow ?? false });\n\n const hash = resolved.includes(\"#\") ? resolved.slice(resolved.indexOf(\"#\")) : \"\";\n if (hash) {\n scrollToHash(hash);\n } else if (options?.scroll !== false) {\n window.scrollTo(0, 0);\n }\n window.dispatchEvent(new CustomEvent(\"vinext:navigate\"));\n return true;\n },\n back: () => window.history.back(),\n reload: () => window.location.reload(),\n prefetch: async (url: string) => {\n if (typeof document !== \"undefined\") {\n const link = document.createElement(\"link\");\n link.rel = \"prefetch\";\n link.href = url;\n link.as = \"document\";\n document.head.appendChild(link);\n }\n },\n beforePopState: (cb: BeforePopStateCallback) => {\n _beforePopStateCb = cb;\n },\n events: routerEvents,\n};\n\nexport default Router;\n"],"mappings":";;;;;;;;;;;;;;;;AAsBA,MAAM,aAAqB,QAAQ,IAAI,0BAA0B;AAmEjE,SAAS,qBAAmC;CAC1C,MAAM,4BAAY,IAAI,KAAgD;AAEtE,QAAO;EACL,GAAG,OAAe,SAAuC;AACvD,OAAI,CAAC,UAAU,IAAI,MAAM,CAAE,WAAU,IAAI,uBAAO,IAAI,KAAK,CAAC;AACzD,aAAU,IAAI,MAAM,CAAuC,IAAI,QAAQ;;EAE1E,IAAI,OAAe,SAAuC;AACxD,aAAU,IAAI,MAAM,EAAE,OAAO,QAAQ;;EAEvC,KAAK,OAAe,GAAG,MAAiB;AACtC,aAAU,IAAI,MAAM,EAAE,SAAS,YAAY,QAAQ,GAAG,KAAK,CAAC;;EAE/D;;AAIH,MAAM,eAAe,oBAAoB;AAEzC,SAAS,WAAW,KAAiC;AACnD,KAAI,OAAO,QAAQ,SAAU,QAAO;CACpC,IAAI,SAAS,IAAI,YAAY;AAC7B,KAAI,IAAI,OAAO;EACb,MAAM,SAAS,uBAAuB,IAAI,MAAM;AAChD,WAAS,wBAAwB,QAAQ,OAAO;;AAElD,QAAO;;;;;;;;;AAUT,SAAS,wBACP,KACA,IACA,QACQ;AACR,QAAO,sBAAsB,MAAM,WAAW,IAAI,EAAE,OAAO;;AAG7D,SAAS,mBAAwD;AAC/D,QAAQ,OAAO,eAA8C;;AAG/D,SAAS,qBAAyC;AAChD,QAAO,OAAO,UAAU;;AAG1B,SAAS,oBAAoB,KAAa,QAAoC;AAC5E,QAAO,mBAAmB,KAAK,QAAQ;EACrC,UAAU;EACV,iBAAiB,oBAAoB;EACrC,aAAa,kBAAkB;EAChC,CAAC;;;;;;AAOJ,SAAgB,sBAAsB,KAAa,QAAyB;AAC1E,KAAI,CAAC,UAAU,OAAO,WAAW,YAAa,QAAO;AAGrD,KAAI,IAAI,WAAW,UAAU,IAAI,IAAI,WAAW,WAAW,IAAI,IAAI,WAAW,KAAK,CACjF,QAAO;CAGT,MAAM,mBAAmB,oBAAoB,KAAK,OAAO;AACzD,KAAI,iBAAkB,QAAO;AAE7B,QAAO,gBAAgB,KAAK,QAAQ,OAAO,6BAA6B,GAAG;;;AAI7E,SAAgB,cAAc,KAAsB;AAClD,QAAO,uBAAuB,KAAK,IAAI,IAAI,IAAI,WAAW,KAAK;;;AAIjE,SAAS,eAAe,KAAqB;AAC3C,KAAI,OAAO,WAAW,YAAa,QAAO;AAC1C,KAAI,IAAI,WAAW,IAAI,CACrB,QAAO,cAAc,OAAO,SAAS,UAAU,WAAW,GAAG,OAAO,SAAS,SAAS;AAExF,KAAI;EACF,MAAM,SAAS,IAAI,IAAI,KAAK,OAAO,SAAS,KAAK;AACjD,SAAO,cAAc,OAAO,UAAU,WAAW,GAAG,OAAO,SAAS,OAAO;SACrE;AACN,SAAO;;;;AAKX,SAAgB,iBAAiB,MAAuB;AACtD,KAAI,KAAK,WAAW,IAAI,CAAE,QAAO;AACjC,KAAI,OAAO,WAAW,YAAa,QAAO;AAC1C,KAAI;EACF,MAAM,UAAU,IAAI,IAAI,OAAO,SAAS,KAAK;EAC7C,MAAM,OAAO,IAAI,IAAI,MAAM,OAAO,SAAS,KAAK;AAChD,SAAO,QAAQ,aAAa,KAAK,YAAY,QAAQ,WAAW,KAAK,UAAU,KAAK,SAAS;SACvF;AACN,SAAO;;;;AAKX,SAAS,aAAa,MAAoB;AACxC,KAAI,CAAC,QAAQ,SAAS,KAAK;AACzB,SAAO,SAAS,GAAG,EAAE;AACrB;;CAEF,MAAM,KAAK,SAAS,eAAe,KAAK,MAAM,EAAE,CAAC;AACjD,KAAI,GAAI,IAAG,eAAe,EAAE,UAAU,QAAQ,CAAC;;;AAIjD,SAAS,qBAA2B;CAClC,MAAM,QAAQ,OAAO,QAAQ,SAAS,EAAE;AACxC,QAAO,QAAQ,aACb;EAAE,GAAG;EAAO,kBAAkB,OAAO;EAAS,kBAAkB,OAAO;EAAS,EAChF,GACD;;;AAIH,SAAS,sBAAsB,OAAsB;AACnD,KAAI,SAAS,OAAO,UAAU,YAAY,sBAAsB,OAAO;EACrE,MAAM,EAAE,kBAAkB,GAAG,kBAAkB,MAAM;AAIrD,8BAA4B,OAAO,SAAS,GAAG,EAAE,CAAC;;;AAuBtD,IAAI,cAAiC;AAErC,IAAI,uBAA0C;AAC9C,IAAI,sBAAsB,QAAiC;AACzD,eAAc;;;;;;AAOhB,SAAgB,8BAA8B,WAGrC;AACP,kBAAiB,UAAU;AAC3B,sBAAqB,UAAU;;AAGjC,SAAgB,cAAc,KAA8B;AAC1D,oBAAmB,IAAI;;;;;;;;AASzB,SAAS,uBAAuB,SAA2B;CACzD,MAAM,QAAkB,EAAE;CAE1B,MAAM,iBAAiB,QAAQ,SAAS,qCAAqC;AAC7E,MAAK,MAAM,KAAK,eACd,OAAM,KAAK,EAAE,GAAG;AAElB,KAAI,MAAM,SAAS,EAAG,QAAO;CAE7B,MAAM,eAAe,QAAQ,SAAS,kBAAkB;AACxD,MAAK,MAAM,KAAK,aACd,OAAM,KAAK,EAAE,GAAG;AAElB,QAAO;;AAGT,SAAS,sBAIP;AACA,KAAI,OAAO,WAAW,aAAa;EACjC,MAAM,UAAU,gBAAgB;AAChC,MAAI,SAAS;GACX,MAAM,QAA2C,EAAE;AACnD,QAAK,MAAM,CAAC,KAAK,UAAU,OAAO,QAAQ,QAAQ,MAAM,CACtD,OAAM,OAAO,MAAM,QAAQ,MAAM,GAAG,CAAC,GAAG,MAAM,GAAG;AAEnD,UAAO;IAAE,UAAU,QAAQ;IAAU;IAAO,QAAQ,QAAQ;IAAQ;;AAEtE,SAAO;GAAE,UAAU;GAAK,OAAO,EAAE;GAAE,QAAQ;GAAK;;CAElD,MAAM,eAAe,cAAc,OAAO,SAAS,UAAU,WAAW;CAIxE,MAAM,WAAW,OAAO,eAAe,QAAQ;CAC/C,MAAM,aAAgD,EAAE;CAGxD,MAAM,WAAW,OAAO;AACxB,KAAI,YAAY,SAAS,SAAS,SAAS,MAAM;EAC/C,MAAM,kBAAkB,uBAAuB,SAAS,KAAK;AAC7D,OAAK,MAAM,OAAO,iBAAiB;GACjC,MAAM,QAAQ,SAAS,MAAM;AAC7B,OAAI,OAAO,UAAU,SACnB,YAAW,OAAO;YACT,MAAM,QAAQ,MAAM,CAC7B,YAAW,OAAO,CAAC,GAAG,MAAM;;;CAKlC,MAAM,cAAiD,EAAE;CACzD,MAAM,SAAS,IAAI,gBAAgB,OAAO,SAAS,OAAO;AAC1D,MAAK,MAAM,CAAC,KAAK,UAAU,OACzB,eAAc,aAAa,KAAK,MAAM;AAKxC,QAAO;EAAE;EAAU,OAHL;GAAE,GAAG;GAAa,GAAG;GAAY;EAGrB,QADX,eAAe,OAAO,SAAS,SAAS,OAAO,SAAS;EACrC;;;;;;AAOpC,IAAI,iBAAiB;AACrB,eAAe,eAAe,KAA4B;AACxD,KAAI,OAAO,WAAW,YAAa;CAEnC,MAAM,OAAO,OAAO;AACpB,KAAI,CAAC,MAAM;AAET,SAAO,SAAS,OAAO;AACvB;;AAIF,KAAI,eAAgB;AACpB,kBAAiB;AAEjB,KAAI;EAEF,MAAM,MAAM,MAAM,MAAM,KAAK,EAAE,SAAS,EAAE,QAAQ,aAAa,EAAE,CAAC;AAClE,MAAI,CAAC,IAAI,IAAI;AACX,UAAO,SAAS,OAAO;AACvB;;EAGF,MAAM,OAAO,MAAM,IAAI,MAAM;EAG7B,MAAM,QAAQ,KAAK,MAAM,sDAAsD;AAC/E,MAAI,CAAC,OAAO;AACV,UAAO,SAAS,OAAO;AACvB;;EAGF,MAAM,WAAW,KAAK,MAAM,MAAM,GAAG;EACrC,MAAM,EAAE,cAAc,SAAS;AAC/B,SAAO,gBAAgB;EAIvB,IAAI,gBAAoC,SAAS,UAAU;AAE3D,MAAI,CAAC,eAAe;GAElB,MAAM,cAAc,KAAK,MAAM,kDAAkD;GACjF,MAAM,WAAW,KAAK,MAAM,wCAAwC;AACpE,mBAAgB,cAAc,MAAM,WAAW,MAAM,KAAA;;AAGvD,MAAI,CAAC,eAAe;AAClB,UAAO,SAAS,OAAO;AACvB;;AAKF,MAAI,CAAC,kBAAkB,cAAc,EAAE;AACrC,WAAQ,MAAM,wDAAwD,cAAc;AACpF,UAAO,SAAS,OAAO;AACvB;;EAKF,MAAM,iBADa,MAAM;;GAA0B;GAClB;AAEjC,MAAI,CAAC,eAAe;AAClB,UAAO,SAAS,OAAO;AACvB;;EAIF,MAAM,SAAS,MAAM,OAAO,UAAU;EAGtC,IAAI,eAAe,OAAO;EAC1B,MAAM,eAAmC,SAAS,UAAU;AAE5D,MAAI,CAAC,gBAAgB,aACnB,KAAI,CAAC,kBAAkB,aAAa,CAClC,SAAQ,MAAM,uDAAuD,aAAa;MAElF,KAAI;AAEF,mBADkB,MAAM;;IAA0B;GACzB;AACzB,UAAO,iBAAiB;UAClB;EAMZ,IAAI;AACJ,MAAI,aACF,WAAU,MAAM,cAAc,cAAc;GAC1C,WAAW;GACX;GACD,CAAC;MAEF,WAAU,MAAM,cAAc,eAAe,UAAU;AAIzD,YAAU,sBAAsB,QAAQ;AAExC,OAAK,OAAO,QAAQ;UACb,KAAK;AACZ,UAAQ,MAAM,sCAAsC,IAAI;AACxD,eAAa,KAAK,oBAAoB,KAAK,KAAK,EAAE,SAAS,OAAO,CAAC;AACnE,SAAO,SAAS,OAAO;WACf;AACR,mBAAiB;;;;;;;;;AAUrB,SAAS,iBACP,UACA,OACA,QACA,SAQY;CACZ,MAAM,YAAY,gBAAgB;CAClC,MAAM,WACJ,OAAO,WAAW,cACb,OAAO,gBACR,KAAA;CACN,MAAM,SAAS,OAAO,WAAW,cAAc,WAAW,SAAS,OAAO;CAC1E,MAAM,UAAU,OAAO,WAAW,cAAc,WAAW,UAAU,OAAO;CAC5E,MAAM,gBACJ,OAAO,WAAW,cAAc,WAAW,gBAAgB,OAAO;CACpE,MAAM,gBACJ,OAAO,WAAW,cAAc,WAAW,gBAAgB,UAAU;AAIvE,QAAO;EACL;EACA,OAJY,OAAO,WAAW,cAAe,UAAU,QAAQ,WAAY;EAK3E;EACA;EACA,UAAU;EACV;EACA;EACA;EACA;EACA,SAAS;EACT,WAAW;EACX,YAAY,OAAO,WAAW,eAAe,UAAU,eAAe;EACtE,GAAG;EACH,QAAQ;EACT;;;;;AAMH,SAAgB,YAAwB;CACtC,MAAM,CAAC,EAAE,UAAU,OAAO,UAAU,YAAY,SAAS,oBAAoB;AAI7E,iBAAgB;EACd,MAAM,eAAe,OAAoB;AACvC,YAAS,qBAAqB,CAAC;;AAEjC,SAAO,iBAAiB,mBAAmB,WAAW;AACtD,eAAa,OAAO,oBAAoB,mBAAmB,WAAW;IACrE,EAAE,CAAC;CAEN,MAAM,OAAO,YACX,OAAO,KAAyB,IAAa,YAAkD;EAC7F,IAAI,WAAW,wBAAwB,KAAK,IAAI,SAAS,OAAO;AAGhE,MAAI,cAAc,SAAS,EAAE;GAC3B,MAAM,YAAY,oBAAoB,UAAU,WAAW;AAC3D,OAAI,aAAa,MAAM;AACrB,WAAO,SAAS,OAAO,SAAS;AAChC,WAAO;;AAET,cAAW;;EAGb,MAAM,OAAO,wBAAwB,UAAU,OAAO,SAAS,MAAM,WAAW;AAGhF,MAAI,iBAAiB,SAAS,EAAE;GAC9B,MAAM,WAAW,eAAe,SAAS;AACzC,gBAAa,KAAK,mBAAmB,UAAU,EAC7C,SAAS,SAAS,WAAW,OAC9B,CAAC;GACF,MAAM,OAAO,SAAS,SAAS,IAAI,GAAG,SAAS,MAAM,SAAS,QAAQ,IAAI,CAAC,GAAG;AAC9E,UAAO,QAAQ,UAAU,EAAE,EAAE,IAAI,SAAS,WAAW,IAAI,GAAG,WAAW,KAAK;AAC5E,4BAAyB,OAAO,SAAS,WAAW,OAAO,SAAS;AACpE,gBAAa,KAAK;AAClB,YAAS,qBAAqB,CAAC;AAC/B,gBAAa,KAAK,sBAAsB,UAAU,EAChD,SAAS,SAAS,WAAW,OAC9B,CAAC;AACF,UAAO,cAAc,IAAI,YAAY,kBAAkB,CAAC;AACxD,UAAO;;AAGT,sBAAoB;AACpB,eAAa,KAAK,oBAAoB,UAAU,EAAE,SAAS,SAAS,WAAW,OAAO,CAAC;AACvF,eAAa,KAAK,uBAAuB,UAAU,EAAE,SAAS,SAAS,WAAW,OAAO,CAAC;AAC1F,SAAO,QAAQ,UAAU,EAAE,EAAE,IAAI,KAAK;AACtC,2BAAyB,OAAO,SAAS,WAAW,OAAO,SAAS;AACpE,MAAI,CAAC,SAAS,QACZ,OAAM,eAAe,KAAK;AAE5B,WAAS,qBAAqB,CAAC;AAC/B,eAAa,KAAK,uBAAuB,UAAU,EAAE,SAAS,SAAS,WAAW,OAAO,CAAC;EAG1F,MAAM,OAAO,SAAS,SAAS,IAAI,GAAG,SAAS,MAAM,SAAS,QAAQ,IAAI,CAAC,GAAG;AAC9E,MAAI,KACF,cAAa,KAAK;WACT,SAAS,WAAW,MAC7B,QAAO,SAAS,GAAG,EAAE;AAEvB,SAAO,cAAc,IAAI,YAAY,kBAAkB,CAAC;AACxD,SAAO;IAET,EAAE,CACH;CAED,MAAM,UAAU,YACd,OAAO,KAAyB,IAAa,YAAkD;EAC7F,IAAI,WAAW,wBAAwB,KAAK,IAAI,SAAS,OAAO;AAGhE,MAAI,cAAc,SAAS,EAAE;GAC3B,MAAM,YAAY,oBAAoB,UAAU,WAAW;AAC3D,OAAI,aAAa,MAAM;AACrB,WAAO,SAAS,QAAQ,SAAS;AACjC,WAAO;;AAET,cAAW;;EAGb,MAAM,OAAO,wBAAwB,UAAU,OAAO,SAAS,MAAM,WAAW;AAGhF,MAAI,iBAAiB,SAAS,EAAE;GAC9B,MAAM,WAAW,eAAe,SAAS;AACzC,gBAAa,KAAK,mBAAmB,UAAU,EAC7C,SAAS,SAAS,WAAW,OAC9B,CAAC;GACF,MAAM,OAAO,SAAS,SAAS,IAAI,GAAG,SAAS,MAAM,SAAS,QAAQ,IAAI,CAAC,GAAG;AAC9E,UAAO,QAAQ,aAAa,EAAE,EAAE,IAAI,SAAS,WAAW,IAAI,GAAG,WAAW,KAAK;AAC/E,4BAAyB,OAAO,SAAS,WAAW,OAAO,SAAS;AACpE,gBAAa,KAAK;AAClB,YAAS,qBAAqB,CAAC;AAC/B,gBAAa,KAAK,sBAAsB,UAAU,EAChD,SAAS,SAAS,WAAW,OAC9B,CAAC;AACF,UAAO,cAAc,IAAI,YAAY,kBAAkB,CAAC;AACxD,UAAO;;AAGT,eAAa,KAAK,oBAAoB,UAAU,EAAE,SAAS,SAAS,WAAW,OAAO,CAAC;AACvF,eAAa,KAAK,uBAAuB,UAAU,EAAE,SAAS,SAAS,WAAW,OAAO,CAAC;AAC1F,SAAO,QAAQ,aAAa,EAAE,EAAE,IAAI,KAAK;AACzC,2BAAyB,OAAO,SAAS,WAAW,OAAO,SAAS;AACpE,MAAI,CAAC,SAAS,QACZ,OAAM,eAAe,KAAK;AAE5B,WAAS,qBAAqB,CAAC;AAC/B,eAAa,KAAK,uBAAuB,UAAU,EAAE,SAAS,SAAS,WAAW,OAAO,CAAC;EAG1F,MAAM,OAAO,SAAS,SAAS,IAAI,GAAG,SAAS,MAAM,SAAS,QAAQ,IAAI,CAAC,GAAG;AAC9E,MAAI,KACF,cAAa,KAAK;WACT,SAAS,WAAW,MAC7B,QAAO,SAAS,GAAG,EAAE;AAEvB,SAAO,cAAc,IAAI,YAAY,kBAAkB,CAAC;AACxD,SAAO;IAET,EAAE,CACH;CAED,MAAM,OAAO,kBAAkB;AAC7B,SAAO,QAAQ,MAAM;IACpB,EAAE,CAAC;CAEN,MAAM,SAAS,kBAAkB;AAC/B,SAAO,SAAS,QAAQ;IACvB,EAAE,CAAC;CAEN,MAAM,WAAW,YAAY,OAAO,QAA+B;AAEjE,MAAI,OAAO,aAAa,aAAa;GACnC,MAAM,OAAO,SAAS,cAAc,OAAO;AAC3C,QAAK,MAAM;AACX,QAAK,OAAO;AACZ,QAAK,KAAK;AACV,YAAS,KAAK,YAAY,KAAK;;IAEhC,EAAE,CAAC;AAiBN,QAfe,cAEX,iBAAiB,UAAU,OAAO,QAAQ;EACxC;EACA;EACA;EACA;EACA;EACA,iBAAiB,OAA+B;AAC9C,uBAAoB;;EAEvB,CAAC,EACJ;EAAC;EAAU;EAAO;EAAQ;EAAM;EAAS;EAAM;EAAQ;EAAS,CACjE;;AAOH,IAAI;AAKJ,IAAI,yBACF,OAAO,WAAW,cAAc,OAAO,SAAS,WAAW,OAAO,SAAS,SAAS;AAKtF,IAAI,OAAO,WAAW,YACpB,QAAO,iBAAiB,aAAa,MAAqB;CACxD,MAAM,aAAa,OAAO,SAAS,WAAW,OAAO,SAAS;CAC9D,MAAM,SAAS,cAAc,OAAO,SAAS,UAAU,WAAW,GAAG,OAAO,SAAS;CAGrF,MAAM,aAAa,eAAe;AAGlC,KAAI,sBAAsB,KAAA;MAMpB,CALoB,kBAA6C;GACnE,KAAK;GACL,IAAI;GACJ,SAAS,EAAE,SAAS,OAAO;GAC5B,CAAC,CACmB;;AAMvB,0BAAyB;AAEzB,KAAI,YAAY;EAEd,MAAM,UAAU,SAAS,OAAO,SAAS;AACzC,eAAa,KAAK,mBAAmB,SAAS,EAAE,SAAS,OAAO,CAAC;AACjE,eAAa,OAAO,SAAS,KAAK;AAClC,eAAa,KAAK,sBAAsB,SAAS,EAAE,SAAS,OAAO,CAAC;AACpE,SAAO,cAAc,IAAI,YAAY,kBAAkB,CAAC;AACxD;;CAGF,MAAM,aAAa,SAAS,OAAO,SAAS;AAC5C,cAAa,KAAK,oBAAoB,YAAY,EAAE,SAAS,OAAO,CAAC;AAMrE,cAAa,KAAK,uBAAuB,YAAY,EAAE,SAAS,OAAO,CAAC;AACnE,gBAAe,WAAW,CAAC,WAAW;AACzC,eAAa,KAAK,uBAAuB,YAAY,EAAE,SAAS,OAAO,CAAC;AACxE,wBAAsB,EAAE,MAAM;AAC9B,SAAO,cAAc,IAAI,YAAY,kBAAkB,CAAC;GACxD;EACF;;;;;;;;;;AAYJ,SAAgB,sBAAsB,SAAqC;CACzE,MAAM,EAAE,UAAU,OAAO,WAAW,qBAAqB;CAEzD,MAAM,cAAc,iBAAiB,UAAU,OAAO,QAAQ;EAC5D,MAAM,OAAO;EACb,SAAS,OAAO;EAChB,MAAM,OAAO;EACb,QAAQ,OAAO;EACf,UAAU,OAAO;EACjB,gBAAgB,OAAO;EACxB,CAAC;AAEF,QAAO,cAAc,cAAc,UAAU,EAAE,OAAO,aAAa,EAAE,QAAQ;;AAI/E,MAAM,SAAS;CACb,MAAM,OAAO,KAAyB,IAAa,YAAgC;EACjF,IAAI,WAAW,wBAAwB,KAAK,IAAI,SAAS,OAAO;AAGhE,MAAI,cAAc,SAAS,EAAE;GAC3B,MAAM,YAAY,oBAAoB,UAAU,WAAW;AAC3D,OAAI,aAAa,MAAM;AACrB,WAAO,SAAS,OAAO,SAAS;AAChC,WAAO;;AAET,cAAW;;EAGb,MAAM,OAAO,wBAAwB,UAAU,OAAO,SAAS,MAAM,WAAW;AAGhF,MAAI,iBAAiB,SAAS,EAAE;GAC9B,MAAM,WAAW,eAAe,SAAS;AACzC,gBAAa,KAAK,mBAAmB,UAAU,EAC7C,SAAS,SAAS,WAAW,OAC9B,CAAC;GACF,MAAM,OAAO,SAAS,SAAS,IAAI,GAAG,SAAS,MAAM,SAAS,QAAQ,IAAI,CAAC,GAAG;AAC9E,UAAO,QAAQ,UAAU,EAAE,EAAE,IAAI,SAAS,WAAW,IAAI,GAAG,WAAW,KAAK;AAC5E,4BAAyB,OAAO,SAAS,WAAW,OAAO,SAAS;AACpE,gBAAa,KAAK;AAClB,gBAAa,KAAK,sBAAsB,UAAU,EAChD,SAAS,SAAS,WAAW,OAC9B,CAAC;AACF,UAAO,cAAc,IAAI,YAAY,kBAAkB,CAAC;AACxD,UAAO;;AAGT,sBAAoB;AACpB,eAAa,KAAK,oBAAoB,UAAU,EAAE,SAAS,SAAS,WAAW,OAAO,CAAC;AACvF,eAAa,KAAK,uBAAuB,UAAU,EAAE,SAAS,SAAS,WAAW,OAAO,CAAC;AAC1F,SAAO,QAAQ,UAAU,EAAE,EAAE,IAAI,KAAK;AACtC,2BAAyB,OAAO,SAAS,WAAW,OAAO,SAAS;AACpE,MAAI,CAAC,SAAS,QACZ,OAAM,eAAe,KAAK;AAE5B,eAAa,KAAK,uBAAuB,UAAU,EAAE,SAAS,SAAS,WAAW,OAAO,CAAC;EAE1F,MAAM,OAAO,SAAS,SAAS,IAAI,GAAG,SAAS,MAAM,SAAS,QAAQ,IAAI,CAAC,GAAG;AAC9E,MAAI,KACF,cAAa,KAAK;WACT,SAAS,WAAW,MAC7B,QAAO,SAAS,GAAG,EAAE;AAEvB,SAAO,cAAc,IAAI,YAAY,kBAAkB,CAAC;AACxD,SAAO;;CAET,SAAS,OAAO,KAAyB,IAAa,YAAgC;EACpF,IAAI,WAAW,wBAAwB,KAAK,IAAI,SAAS,OAAO;AAGhE,MAAI,cAAc,SAAS,EAAE;GAC3B,MAAM,YAAY,oBAAoB,UAAU,WAAW;AAC3D,OAAI,aAAa,MAAM;AACrB,WAAO,SAAS,QAAQ,SAAS;AACjC,WAAO;;AAET,cAAW;;EAGb,MAAM,OAAO,wBAAwB,UAAU,OAAO,SAAS,MAAM,WAAW;AAGhF,MAAI,iBAAiB,SAAS,EAAE;GAC9B,MAAM,WAAW,eAAe,SAAS;AACzC,gBAAa,KAAK,mBAAmB,UAAU,EAC7C,SAAS,SAAS,WAAW,OAC9B,CAAC;GACF,MAAM,OAAO,SAAS,SAAS,IAAI,GAAG,SAAS,MAAM,SAAS,QAAQ,IAAI,CAAC,GAAG;AAC9E,UAAO,QAAQ,aAAa,EAAE,EAAE,IAAI,SAAS,WAAW,IAAI,GAAG,WAAW,KAAK;AAC/E,4BAAyB,OAAO,SAAS,WAAW,OAAO,SAAS;AACpE,gBAAa,KAAK;AAClB,gBAAa,KAAK,sBAAsB,UAAU,EAChD,SAAS,SAAS,WAAW,OAC9B,CAAC;AACF,UAAO,cAAc,IAAI,YAAY,kBAAkB,CAAC;AACxD,UAAO;;AAGT,eAAa,KAAK,oBAAoB,UAAU,EAAE,SAAS,SAAS,WAAW,OAAO,CAAC;AACvF,eAAa,KAAK,uBAAuB,UAAU,EAAE,SAAS,SAAS,WAAW,OAAO,CAAC;AAC1F,SAAO,QAAQ,aAAa,EAAE,EAAE,IAAI,KAAK;AACzC,2BAAyB,OAAO,SAAS,WAAW,OAAO,SAAS;AACpE,MAAI,CAAC,SAAS,QACZ,OAAM,eAAe,KAAK;AAE5B,eAAa,KAAK,uBAAuB,UAAU,EAAE,SAAS,SAAS,WAAW,OAAO,CAAC;EAE1F,MAAM,OAAO,SAAS,SAAS,IAAI,GAAG,SAAS,MAAM,SAAS,QAAQ,IAAI,CAAC,GAAG;AAC9E,MAAI,KACF,cAAa,KAAK;WACT,SAAS,WAAW,MAC7B,QAAO,SAAS,GAAG,EAAE;AAEvB,SAAO,cAAc,IAAI,YAAY,kBAAkB,CAAC;AACxD,SAAO;;CAET,YAAY,OAAO,QAAQ,MAAM;CACjC,cAAc,OAAO,SAAS,QAAQ;CACtC,UAAU,OAAO,QAAgB;AAC/B,MAAI,OAAO,aAAa,aAAa;GACnC,MAAM,OAAO,SAAS,cAAc,OAAO;AAC3C,QAAK,MAAM;AACX,QAAK,OAAO;AACZ,QAAK,KAAK;AACV,YAAS,KAAK,YAAY,KAAK;;;CAGnC,iBAAiB,OAA+B;AAC9C,sBAAoB;;CAEtB,QAAQ;CACT"}
|
|
1
|
+
{"version":3,"file":"router.js","names":[],"sources":["../../src/shims/router.ts"],"sourcesContent":["/**\n * next/router shim\n *\n * Provides useRouter() hook and Router singleton for Pages Router.\n * Backed by the browser History API. Supports client-side navigation\n * by fetching new page data and re-rendering the React root.\n */\nimport { useState, useEffect, useCallback, useMemo, createElement, type ReactElement } from \"react\";\nimport { RouterContext } from \"./internal/router-context.js\";\nimport type { VinextNextData } from \"../client/vinext-next-data.js\";\nimport { isValidModulePath } from \"../client/validate-module-path.js\";\nimport { toBrowserNavigationHref, toSameOriginAppPath } from \"./url-utils.js\";\nimport { stripBasePath } from \"../utils/base-path.js\";\nimport { addLocalePrefix, getDomainLocaleUrl, type DomainLocale } from \"../utils/domain-locale.js\";\nimport {\n addQueryParam,\n appendSearchParamsToUrl,\n type UrlQuery,\n urlQueryToSearchParams,\n} from \"../utils/query.js\";\n\n/** basePath from next.config.js, injected by the plugin at build time */\nconst __basePath: string = process.env.__NEXT_ROUTER_BASEPATH ?? \"\";\n\ntype BeforePopStateCallback = (state: {\n url: string;\n as: string;\n options: { shallow: boolean };\n}) => boolean;\n\nexport interface NextRouter {\n /** Current pathname */\n pathname: string;\n /** Current route pattern (e.g., \"/posts/[id]\") */\n route: string;\n /** Query parameters */\n query: Record<string, string | string[]>;\n /** Full URL including query string */\n asPath: string;\n /** Base path */\n basePath: string;\n /** Current locale */\n locale?: string;\n /** Available locales */\n locales?: string[];\n /** Default locale */\n defaultLocale?: string;\n /** Configured domain locales */\n domainLocales?: VinextNextData[\"domainLocales\"];\n /** Whether the router is ready */\n isReady: boolean;\n /** Whether this is a preview */\n isPreview: boolean;\n /** Whether this is a fallback page */\n isFallback: boolean;\n\n /** Navigate to a new URL */\n push(url: string | UrlObject, as?: string, options?: TransitionOptions): Promise<boolean>;\n /** Replace current URL */\n replace(url: string | UrlObject, as?: string, options?: TransitionOptions): Promise<boolean>;\n /** Go back */\n back(): void;\n /** Reload the page */\n reload(): void;\n /** Prefetch a page (injects <link rel=\"prefetch\">) */\n prefetch(url: string): Promise<void>;\n /** Register a callback to run before popstate navigation */\n beforePopState(cb: BeforePopStateCallback): void;\n /** Listen for route changes */\n events: RouterEvents;\n}\n\ninterface UrlObject {\n pathname?: string;\n query?: UrlQuery;\n}\n\ninterface TransitionOptions {\n shallow?: boolean;\n scroll?: boolean;\n locale?: string;\n}\n\ninterface RouterEvents {\n on(event: string, handler: (...args: unknown[]) => void): void;\n off(event: string, handler: (...args: unknown[]) => void): void;\n emit(event: string, ...args: unknown[]): void;\n}\n\nfunction createRouterEvents(): RouterEvents {\n const listeners = new Map<string, Set<(...args: unknown[]) => void>>();\n\n return {\n on(event: string, handler: (...args: unknown[]) => void) {\n if (!listeners.has(event)) listeners.set(event, new Set());\n (listeners.get(event) as Set<(...args: unknown[]) => void>).add(handler);\n },\n off(event: string, handler: (...args: unknown[]) => void) {\n listeners.get(event)?.delete(handler);\n },\n emit(event: string, ...args: unknown[]) {\n listeners.get(event)?.forEach((handler) => handler(...args));\n },\n };\n}\n\n// Singleton events instance\nconst routerEvents = createRouterEvents();\n\nfunction resolveUrl(url: string | UrlObject): string {\n if (typeof url === \"string\") return url;\n let result = url.pathname ?? \"/\";\n if (url.query) {\n const params = urlQueryToSearchParams(url.query);\n result = appendSearchParamsToUrl(result, params);\n }\n return result;\n}\n\n/**\n * When `as` is provided, use it as the navigation target. This is a\n * simplification: Next.js keeps `url` and `as` as separate values (url for\n * data fetching, as for the browser URL). We collapse them because vinext's\n * navigateClient() fetches HTML from the target URL, so `as` must be a\n * server-resolvable path. Purely decorative `as` values are not supported.\n */\nfunction resolveNavigationTarget(\n url: string | UrlObject,\n as: string | undefined,\n locale: string | undefined,\n): string {\n return applyNavigationLocale(as ?? resolveUrl(url), locale);\n}\n\nfunction getDomainLocales(): readonly DomainLocale[] | undefined {\n return (window.__NEXT_DATA__ as VinextNextData | undefined)?.domainLocales;\n}\n\nfunction getCurrentHostname(): string | undefined {\n return window.location?.hostname;\n}\n\nfunction getDomainLocalePath(url: string, locale: string): string | undefined {\n return getDomainLocaleUrl(url, locale, {\n basePath: __basePath,\n currentHostname: getCurrentHostname(),\n domainItems: getDomainLocales(),\n });\n}\n\n/**\n * Apply locale prefix to a URL for client-side navigation.\n * Same logic as Link's applyLocaleToHref but reads from window globals.\n */\nexport function applyNavigationLocale(url: string, locale?: string): string {\n if (!locale || typeof window === \"undefined\") return url;\n // Absolute and protocol-relative URLs must not be prefixed — locale\n // only applies to local paths.\n if (url.startsWith(\"http://\") || url.startsWith(\"https://\") || url.startsWith(\"//\")) {\n return url;\n }\n\n const domainLocalePath = getDomainLocalePath(url, locale);\n if (domainLocalePath) return domainLocalePath;\n\n return addLocalePrefix(url, locale, window.__VINEXT_DEFAULT_LOCALE__ ?? \"\");\n}\n\n/** Check if a URL is external (any URL scheme per RFC 3986, or protocol-relative) */\nexport function isExternalUrl(url: string): boolean {\n return /^[a-z][a-z0-9+.-]*:/i.test(url) || url.startsWith(\"//\");\n}\n\n/** Resolve a hash URL to a basePath-stripped app URL for event payloads */\nfunction resolveHashUrl(url: string): string {\n if (typeof window === \"undefined\") return url;\n if (url.startsWith(\"#\"))\n return stripBasePath(window.location.pathname, __basePath) + window.location.search + url;\n // Full-path hash URL — strip basePath for consistency with other events\n try {\n const parsed = new URL(url, window.location.href);\n return stripBasePath(parsed.pathname, __basePath) + parsed.search + parsed.hash;\n } catch {\n return url;\n }\n}\n\n/** Check if a href is only a hash change relative to the current URL */\nexport function isHashOnlyChange(href: string): boolean {\n if (href.startsWith(\"#\")) return true;\n if (typeof window === \"undefined\") return false;\n try {\n const current = new URL(window.location.href);\n const next = new URL(href, window.location.href);\n return current.pathname === next.pathname && current.search === next.search && next.hash !== \"\";\n } catch {\n return false;\n }\n}\n\n/** Scroll to hash target element, or top if no hash */\nfunction scrollToHash(hash: string): void {\n if (!hash || hash === \"#\") {\n window.scrollTo(0, 0);\n return;\n }\n const el = document.getElementById(hash.slice(1));\n if (el) el.scrollIntoView({ behavior: \"auto\" });\n}\n\n/** Save current scroll position into history state for back/forward restoration */\nfunction saveScrollPosition(): void {\n const state = window.history.state ?? {};\n window.history.replaceState(\n { ...state, __vinext_scrollX: window.scrollX, __vinext_scrollY: window.scrollY },\n \"\",\n );\n}\n\n/** Restore scroll position from history state */\nfunction restoreScrollPosition(state: unknown): void {\n if (state && typeof state === \"object\" && \"__vinext_scrollY\" in state) {\n const { __vinext_scrollX: x, __vinext_scrollY: y } = state as {\n __vinext_scrollX: number;\n __vinext_scrollY: number;\n };\n requestAnimationFrame(() => window.scrollTo(x, y));\n }\n}\n\n/**\n * SSR context - set by the dev server before rendering each page.\n */\ninterface SSRContext {\n pathname: string;\n query: Record<string, string | string[]>;\n asPath: string;\n locale?: string;\n locales?: string[];\n defaultLocale?: string;\n domainLocales?: VinextNextData[\"domainLocales\"];\n}\n\n// ---------------------------------------------------------------------------\n// Server-side SSR state uses a registration pattern so this module can be\n// bundled for the browser. The ALS-backed implementation lives in\n// router-state.ts (server-only) and registers itself on import.\n// ---------------------------------------------------------------------------\n\nlet _ssrContext: SSRContext | null = null;\n\nlet _getSSRContext = (): SSRContext | null => _ssrContext;\nlet _setSSRContextImpl = (ctx: SSRContext | null): void => {\n _ssrContext = ctx;\n};\n\n/**\n * Register ALS-backed state accessors. Called by router-state.ts on import.\n * @internal\n */\nexport function _registerRouterStateAccessors(accessors: {\n getSSRContext: () => SSRContext | null;\n setSSRContext: (ctx: SSRContext | null) => void;\n}): void {\n _getSSRContext = accessors.getSSRContext;\n _setSSRContextImpl = accessors.setSSRContext;\n}\n\nexport function setSSRContext(ctx: SSRContext | null): void {\n _setSSRContextImpl(ctx);\n}\n\n/**\n * Extract param names from a Next.js route pattern.\n * E.g., \"/posts/[id]\" → [\"id\"], \"/docs/[...slug]\" → [\"slug\"],\n * \"/shop/[[...path]]\" → [\"path\"], \"/blog/[year]/[month]\" → [\"year\", \"month\"]\n * Also handles internal format: \"/posts/:id\" → [\"id\"], \"/docs/:slug+\" → [\"slug\"]\n */\nfunction extractRouteParamNames(pattern: string): string[] {\n const names: string[] = [];\n // Match Next.js bracket format: [id], [...slug], [[...slug]]\n const bracketMatches = pattern.matchAll(/\\[{1,2}(?:\\.\\.\\.)?([\\w-]+)\\]{1,2}/g);\n for (const m of bracketMatches) {\n names.push(m[1]);\n }\n if (names.length > 0) return names;\n // Fallback: match internal :param format\n const colonMatches = pattern.matchAll(/:([\\w-]+)[+*]?/g);\n for (const m of colonMatches) {\n names.push(m[1]);\n }\n return names;\n}\n\nfunction getPathnameAndQuery(): {\n pathname: string;\n query: Record<string, string | string[]>;\n asPath: string;\n} {\n if (typeof window === \"undefined\") {\n const _ssrCtx = _getSSRContext();\n if (_ssrCtx) {\n const query: Record<string, string | string[]> = {};\n for (const [key, value] of Object.entries(_ssrCtx.query)) {\n query[key] = Array.isArray(value) ? [...value] : value;\n }\n return { pathname: _ssrCtx.pathname, query, asPath: _ssrCtx.asPath };\n }\n return { pathname: \"/\", query: {}, asPath: \"/\" };\n }\n const resolvedPath = stripBasePath(window.location.pathname, __basePath);\n // In Next.js, router.pathname is the route pattern (e.g., \"/posts/[id]\"),\n // not the resolved path (\"/posts/42\"). __NEXT_DATA__.page holds the route\n // pattern and is updated by navigateClient() on every client-side navigation.\n const pathname = window.__NEXT_DATA__?.page ?? resolvedPath;\n const routeQuery: Record<string, string | string[]> = {};\n // Include dynamic route params from __NEXT_DATA__ (e.g., { id: \"42\" } from /posts/[id]).\n // Only include keys that are part of the route pattern (not stale query params).\n const nextData = window.__NEXT_DATA__;\n if (nextData && nextData.query && nextData.page) {\n const routeParamNames = extractRouteParamNames(nextData.page);\n for (const key of routeParamNames) {\n const value = nextData.query[key];\n if (typeof value === \"string\") {\n routeQuery[key] = value;\n } else if (Array.isArray(value)) {\n routeQuery[key] = [...value];\n }\n }\n }\n // URL search params always reflect the current URL\n const searchQuery: Record<string, string | string[]> = {};\n const params = new URLSearchParams(window.location.search);\n for (const [key, value] of params) {\n addQueryParam(searchQuery, key, value);\n }\n const query = { ...searchQuery, ...routeQuery };\n // asPath uses the resolved browser path, not the route pattern\n const asPath = resolvedPath + window.location.search + window.location.hash;\n return { pathname, query, asPath };\n}\n\n/**\n * Perform client-side navigation: fetch the target page's HTML,\n * extract __NEXT_DATA__, and re-render the React root.\n */\nlet _navInProgress = false;\nasync function navigateClient(url: string): Promise<void> {\n if (typeof window === \"undefined\") return;\n\n const root = window.__VINEXT_ROOT__;\n if (!root) {\n // No React root yet — fall back to hard navigation\n window.location.href = url;\n return;\n }\n\n // Prevent re-entrant navigation (e.g., double popstate events)\n if (_navInProgress) return;\n _navInProgress = true;\n\n try {\n // Fetch the target page's SSR HTML\n const res = await fetch(url, { headers: { Accept: \"text/html\" } });\n if (!res.ok) {\n window.location.href = url;\n return;\n }\n\n const html = await res.text();\n\n // Extract __NEXT_DATA__ from the HTML\n const match = html.match(/<script>window\\.__NEXT_DATA__\\s*=\\s*(.*?)<\\/script>/);\n if (!match) {\n window.location.href = url;\n return;\n }\n\n const nextData = JSON.parse(match[1]);\n const { pageProps } = nextData.props;\n window.__NEXT_DATA__ = nextData;\n\n // Get the page module URL from __NEXT_DATA__.__vinext (preferred),\n // or fall back to parsing the hydration script\n let pageModuleUrl: string | undefined = nextData.__vinext?.pageModuleUrl;\n\n if (!pageModuleUrl) {\n // Legacy fallback: try to find the module URL in the inline script\n const moduleMatch = html.match(/import\\(\"([^\"]+)\"\\);\\s*\\n\\s*const PageComponent/);\n const altMatch = html.match(/await import\\(\"([^\"]+pages\\/[^\"]+)\"\\)/);\n pageModuleUrl = moduleMatch?.[1] ?? altMatch?.[1] ?? undefined;\n }\n\n if (!pageModuleUrl) {\n window.location.href = url;\n return;\n }\n\n // Validate the module URL before importing — defense-in-depth against\n // unexpected __NEXT_DATA__ or malformed HTML responses\n if (!isValidModulePath(pageModuleUrl)) {\n console.error(\"[vinext] Blocked import of invalid page module path:\", pageModuleUrl);\n window.location.href = url;\n return;\n }\n\n // Dynamically import the new page module\n const pageModule = await import(/* @vite-ignore */ pageModuleUrl);\n const PageComponent = pageModule.default;\n\n if (!PageComponent) {\n window.location.href = url;\n return;\n }\n\n // Import React for createElement\n const React = (await import(\"react\")).default;\n\n // Re-render with the new page, loading _app if needed\n let AppComponent = window.__VINEXT_APP__;\n const appModuleUrl: string | undefined = nextData.__vinext?.appModuleUrl;\n\n if (!AppComponent && appModuleUrl) {\n if (!isValidModulePath(appModuleUrl)) {\n console.error(\"[vinext] Blocked import of invalid app module path:\", appModuleUrl);\n } else {\n try {\n const appModule = await import(/* @vite-ignore */ appModuleUrl);\n AppComponent = appModule.default;\n window.__VINEXT_APP__ = AppComponent;\n } catch {\n // _app not available — continue without it\n }\n }\n }\n\n let element;\n if (AppComponent) {\n element = React.createElement(AppComponent, {\n Component: PageComponent,\n pageProps,\n });\n } else {\n element = React.createElement(PageComponent, pageProps);\n }\n\n // Wrap with RouterContext.Provider so next/compat/router works\n element = wrapWithRouterContext(element);\n\n root.render(element);\n } catch (err) {\n console.error(\"[vinext] Client navigation failed:\", err);\n routerEvents.emit(\"routeChangeError\", err, url, { shallow: false });\n window.location.href = url;\n } finally {\n _navInProgress = false;\n }\n}\n\n/**\n * Build the full router value object from the current pathname, query, asPath,\n * and a set of navigation methods. Shared by useRouter() (which passes\n * hook-derived callbacks) and wrapWithRouterContext() (which passes the Router\n * singleton methods) so the shape stays in sync.\n */\nfunction buildRouterValue(\n pathname: string,\n query: Record<string, string | string[]>,\n asPath: string,\n methods: {\n push: NextRouter[\"push\"];\n replace: NextRouter[\"replace\"];\n back: NextRouter[\"back\"];\n reload: NextRouter[\"reload\"];\n prefetch: NextRouter[\"prefetch\"];\n beforePopState: NextRouter[\"beforePopState\"];\n },\n): NextRouter {\n const _ssrState = _getSSRContext();\n const nextData =\n typeof window !== \"undefined\"\n ? (window.__NEXT_DATA__ as VinextNextData | undefined)\n : undefined;\n const locale = typeof window === \"undefined\" ? _ssrState?.locale : window.__VINEXT_LOCALE__;\n const locales = typeof window === \"undefined\" ? _ssrState?.locales : window.__VINEXT_LOCALES__;\n const defaultLocale =\n typeof window === \"undefined\" ? _ssrState?.defaultLocale : window.__VINEXT_DEFAULT_LOCALE__;\n const domainLocales =\n typeof window === \"undefined\" ? _ssrState?.domainLocales : nextData?.domainLocales;\n\n const route = typeof window !== \"undefined\" ? (nextData?.page ?? pathname) : pathname;\n\n return {\n pathname,\n route,\n query,\n asPath,\n basePath: __basePath,\n locale,\n locales,\n defaultLocale,\n domainLocales,\n isReady: true,\n isPreview: false,\n isFallback: typeof window !== \"undefined\" && nextData?.isFallback === true,\n ...methods,\n events: routerEvents,\n };\n}\n\n/**\n * useRouter hook - Pages Router compatible.\n */\nexport function useRouter(): NextRouter {\n const [{ pathname, query, asPath }, setState] = useState(getPathnameAndQuery);\n\n // Popstate is handled by the module-level listener below so beforePopState()\n // is consistently enforced even when multiple components mount useRouter().\n useEffect(() => {\n const onNavigate = ((_e: CustomEvent) => {\n setState(getPathnameAndQuery());\n }) as EventListener;\n window.addEventListener(\"vinext:navigate\", onNavigate);\n return () => window.removeEventListener(\"vinext:navigate\", onNavigate);\n }, []);\n\n const push = useCallback(\n async (url: string | UrlObject, as?: string, options?: TransitionOptions): Promise<boolean> => {\n let resolved = resolveNavigationTarget(url, as, options?.locale);\n\n // External URLs — delegate to browser (unless same-origin)\n if (isExternalUrl(resolved)) {\n const localPath = toSameOriginAppPath(resolved, __basePath);\n if (localPath == null) {\n window.location.assign(resolved);\n return true;\n }\n resolved = localPath;\n }\n\n const full = toBrowserNavigationHref(resolved, window.location.href, __basePath);\n\n // Hash-only change — no page fetch needed\n if (isHashOnlyChange(resolved)) {\n const eventUrl = resolveHashUrl(resolved);\n routerEvents.emit(\"hashChangeStart\", eventUrl, {\n shallow: options?.shallow ?? false,\n });\n const hash = resolved.includes(\"#\") ? resolved.slice(resolved.indexOf(\"#\")) : \"\";\n window.history.pushState({}, \"\", resolved.startsWith(\"#\") ? resolved : full);\n _lastPathnameAndSearch = window.location.pathname + window.location.search;\n scrollToHash(hash);\n setState(getPathnameAndQuery());\n routerEvents.emit(\"hashChangeComplete\", eventUrl, {\n shallow: options?.shallow ?? false,\n });\n window.dispatchEvent(new CustomEvent(\"vinext:navigate\"));\n return true;\n }\n\n saveScrollPosition();\n routerEvents.emit(\"routeChangeStart\", resolved, { shallow: options?.shallow ?? false });\n routerEvents.emit(\"beforeHistoryChange\", resolved, { shallow: options?.shallow ?? false });\n window.history.pushState({}, \"\", full);\n _lastPathnameAndSearch = window.location.pathname + window.location.search;\n if (!options?.shallow) {\n await navigateClient(full);\n }\n setState(getPathnameAndQuery());\n routerEvents.emit(\"routeChangeComplete\", resolved, { shallow: options?.shallow ?? false });\n\n // Scroll: handle hash target, else scroll to top unless scroll:false\n const hash = resolved.includes(\"#\") ? resolved.slice(resolved.indexOf(\"#\")) : \"\";\n if (hash) {\n scrollToHash(hash);\n } else if (options?.scroll !== false) {\n window.scrollTo(0, 0);\n }\n window.dispatchEvent(new CustomEvent(\"vinext:navigate\"));\n return true;\n },\n [],\n );\n\n const replace = useCallback(\n async (url: string | UrlObject, as?: string, options?: TransitionOptions): Promise<boolean> => {\n let resolved = resolveNavigationTarget(url, as, options?.locale);\n\n // External URLs — delegate to browser (unless same-origin)\n if (isExternalUrl(resolved)) {\n const localPath = toSameOriginAppPath(resolved, __basePath);\n if (localPath == null) {\n window.location.replace(resolved);\n return true;\n }\n resolved = localPath;\n }\n\n const full = toBrowserNavigationHref(resolved, window.location.href, __basePath);\n\n // Hash-only change — no page fetch needed\n if (isHashOnlyChange(resolved)) {\n const eventUrl = resolveHashUrl(resolved);\n routerEvents.emit(\"hashChangeStart\", eventUrl, {\n shallow: options?.shallow ?? false,\n });\n const hash = resolved.includes(\"#\") ? resolved.slice(resolved.indexOf(\"#\")) : \"\";\n window.history.replaceState({}, \"\", resolved.startsWith(\"#\") ? resolved : full);\n _lastPathnameAndSearch = window.location.pathname + window.location.search;\n scrollToHash(hash);\n setState(getPathnameAndQuery());\n routerEvents.emit(\"hashChangeComplete\", eventUrl, {\n shallow: options?.shallow ?? false,\n });\n window.dispatchEvent(new CustomEvent(\"vinext:navigate\"));\n return true;\n }\n\n routerEvents.emit(\"routeChangeStart\", resolved, { shallow: options?.shallow ?? false });\n routerEvents.emit(\"beforeHistoryChange\", resolved, { shallow: options?.shallow ?? false });\n window.history.replaceState({}, \"\", full);\n _lastPathnameAndSearch = window.location.pathname + window.location.search;\n if (!options?.shallow) {\n await navigateClient(full);\n }\n setState(getPathnameAndQuery());\n routerEvents.emit(\"routeChangeComplete\", resolved, { shallow: options?.shallow ?? false });\n\n // Scroll: handle hash target, else scroll to top unless scroll:false\n const hash = resolved.includes(\"#\") ? resolved.slice(resolved.indexOf(\"#\")) : \"\";\n if (hash) {\n scrollToHash(hash);\n } else if (options?.scroll !== false) {\n window.scrollTo(0, 0);\n }\n window.dispatchEvent(new CustomEvent(\"vinext:navigate\"));\n return true;\n },\n [],\n );\n\n const back = useCallback(() => {\n window.history.back();\n }, []);\n\n const reload = useCallback(() => {\n window.location.reload();\n }, []);\n\n const prefetch = useCallback(async (url: string): Promise<void> => {\n // Inject a <link rel=\"prefetch\"> for the target page\n if (typeof document !== \"undefined\") {\n const link = document.createElement(\"link\");\n link.rel = \"prefetch\";\n link.href = url;\n link.as = \"document\";\n document.head.appendChild(link);\n }\n }, []);\n\n const router = useMemo(\n (): NextRouter =>\n buildRouterValue(pathname, query, asPath, {\n push,\n replace,\n back,\n reload,\n prefetch,\n beforePopState: (cb: BeforePopStateCallback) => {\n _beforePopStateCb = cb;\n },\n }),\n [pathname, query, asPath, push, replace, back, reload, prefetch],\n );\n\n return router;\n}\n\n// beforePopState callback: called before handling browser back/forward.\n// If it returns false, the navigation is cancelled.\nlet _beforePopStateCb: BeforePopStateCallback | undefined;\n\n// Track pathname+search for detecting hash-only back/forward in the popstate\n// handler. Updated after every pushState/replaceState so that popstate can\n// compare the previous value with the (already-changed) window.location.\nlet _lastPathnameAndSearch =\n typeof window !== \"undefined\" ? window.location.pathname + window.location.search : \"\";\n\n// Module-level popstate listener: handles browser back/forward by re-rendering\n// the React root with the page at the new URL. This runs regardless of whether\n// any component calls useRouter().\nif (typeof window !== \"undefined\") {\n window.addEventListener(\"popstate\", (e: PopStateEvent) => {\n const browserUrl = window.location.pathname + window.location.search;\n const appUrl = stripBasePath(window.location.pathname, __basePath) + window.location.search;\n\n // Detect hash-only back/forward: pathname+search unchanged, only hash differs.\n const isHashOnly = browserUrl === _lastPathnameAndSearch;\n\n // Check beforePopState callback\n if (_beforePopStateCb !== undefined) {\n const shouldContinue = (_beforePopStateCb as BeforePopStateCallback)({\n url: appUrl,\n as: appUrl,\n options: { shallow: false },\n });\n if (!shouldContinue) return;\n }\n\n // Update tracker only after beforePopState confirms navigation proceeds.\n // If beforePopState cancels, the tracker must retain the previous value\n // so the next popstate compares against the correct baseline.\n _lastPathnameAndSearch = browserUrl;\n\n if (isHashOnly) {\n // Hash-only back/forward — no page fetch needed\n const hashUrl = appUrl + window.location.hash;\n routerEvents.emit(\"hashChangeStart\", hashUrl, { shallow: false });\n scrollToHash(window.location.hash);\n routerEvents.emit(\"hashChangeComplete\", hashUrl, { shallow: false });\n window.dispatchEvent(new CustomEvent(\"vinext:navigate\"));\n return;\n }\n\n const fullAppUrl = appUrl + window.location.hash;\n routerEvents.emit(\"routeChangeStart\", fullAppUrl, { shallow: false });\n // Note: The browser has already updated window.location by the time popstate\n // fires, so this is not truly \"before\" the URL change. In Next.js the popstate\n // handler calls replaceState to store history metadata — beforeHistoryChange\n // precedes that call, not the URL change itself. We emit it here for API\n // compatibility.\n routerEvents.emit(\"beforeHistoryChange\", fullAppUrl, { shallow: false });\n void navigateClient(browserUrl).then(() => {\n routerEvents.emit(\"routeChangeComplete\", fullAppUrl, { shallow: false });\n restoreScrollPosition(e.state);\n window.dispatchEvent(new CustomEvent(\"vinext:navigate\"));\n });\n });\n}\n\n/**\n * Wrap a React element in a RouterContext.Provider so that\n * next/compat/router's useRouter() returns the real Pages Router value.\n *\n * This is a plain function, NOT a React component — it builds the router\n * value object directly from the current SSR context (server) or\n * window.location + Router singleton (client), avoiding duplicate state\n * that a hook-based component would create.\n */\nexport function wrapWithRouterContext(element: ReactElement): ReactElement {\n const { pathname, query, asPath } = getPathnameAndQuery();\n\n const routerValue = buildRouterValue(pathname, query, asPath, {\n push: Router.push,\n replace: Router.replace,\n back: Router.back,\n reload: Router.reload,\n prefetch: Router.prefetch,\n beforePopState: Router.beforePopState,\n });\n\n return createElement(RouterContext.Provider, { value: routerValue }, element) as ReactElement;\n}\n\n// Also export a default Router singleton for `import Router from 'next/router'`\nconst Router = {\n push: async (url: string | UrlObject, as?: string, options?: TransitionOptions) => {\n let resolved = resolveNavigationTarget(url, as, options?.locale);\n\n // External URLs (unless same-origin)\n if (isExternalUrl(resolved)) {\n const localPath = toSameOriginAppPath(resolved, __basePath);\n if (localPath == null) {\n window.location.assign(resolved);\n return true;\n }\n resolved = localPath;\n }\n\n const full = toBrowserNavigationHref(resolved, window.location.href, __basePath);\n\n // Hash-only change\n if (isHashOnlyChange(resolved)) {\n const eventUrl = resolveHashUrl(resolved);\n routerEvents.emit(\"hashChangeStart\", eventUrl, {\n shallow: options?.shallow ?? false,\n });\n const hash = resolved.includes(\"#\") ? resolved.slice(resolved.indexOf(\"#\")) : \"\";\n window.history.pushState({}, \"\", resolved.startsWith(\"#\") ? resolved : full);\n _lastPathnameAndSearch = window.location.pathname + window.location.search;\n scrollToHash(hash);\n routerEvents.emit(\"hashChangeComplete\", eventUrl, {\n shallow: options?.shallow ?? false,\n });\n window.dispatchEvent(new CustomEvent(\"vinext:navigate\"));\n return true;\n }\n\n saveScrollPosition();\n routerEvents.emit(\"routeChangeStart\", resolved, { shallow: options?.shallow ?? false });\n routerEvents.emit(\"beforeHistoryChange\", resolved, { shallow: options?.shallow ?? false });\n window.history.pushState({}, \"\", full);\n _lastPathnameAndSearch = window.location.pathname + window.location.search;\n if (!options?.shallow) {\n await navigateClient(full);\n }\n routerEvents.emit(\"routeChangeComplete\", resolved, { shallow: options?.shallow ?? false });\n\n const hash = resolved.includes(\"#\") ? resolved.slice(resolved.indexOf(\"#\")) : \"\";\n if (hash) {\n scrollToHash(hash);\n } else if (options?.scroll !== false) {\n window.scrollTo(0, 0);\n }\n window.dispatchEvent(new CustomEvent(\"vinext:navigate\"));\n return true;\n },\n replace: async (url: string | UrlObject, as?: string, options?: TransitionOptions) => {\n let resolved = resolveNavigationTarget(url, as, options?.locale);\n\n // External URLs (unless same-origin)\n if (isExternalUrl(resolved)) {\n const localPath = toSameOriginAppPath(resolved, __basePath);\n if (localPath == null) {\n window.location.replace(resolved);\n return true;\n }\n resolved = localPath;\n }\n\n const full = toBrowserNavigationHref(resolved, window.location.href, __basePath);\n\n // Hash-only change\n if (isHashOnlyChange(resolved)) {\n const eventUrl = resolveHashUrl(resolved);\n routerEvents.emit(\"hashChangeStart\", eventUrl, {\n shallow: options?.shallow ?? false,\n });\n const hash = resolved.includes(\"#\") ? resolved.slice(resolved.indexOf(\"#\")) : \"\";\n window.history.replaceState({}, \"\", resolved.startsWith(\"#\") ? resolved : full);\n _lastPathnameAndSearch = window.location.pathname + window.location.search;\n scrollToHash(hash);\n routerEvents.emit(\"hashChangeComplete\", eventUrl, {\n shallow: options?.shallow ?? false,\n });\n window.dispatchEvent(new CustomEvent(\"vinext:navigate\"));\n return true;\n }\n\n routerEvents.emit(\"routeChangeStart\", resolved, { shallow: options?.shallow ?? false });\n routerEvents.emit(\"beforeHistoryChange\", resolved, { shallow: options?.shallow ?? false });\n window.history.replaceState({}, \"\", full);\n _lastPathnameAndSearch = window.location.pathname + window.location.search;\n if (!options?.shallow) {\n await navigateClient(full);\n }\n routerEvents.emit(\"routeChangeComplete\", resolved, { shallow: options?.shallow ?? false });\n\n const hash = resolved.includes(\"#\") ? resolved.slice(resolved.indexOf(\"#\")) : \"\";\n if (hash) {\n scrollToHash(hash);\n } else if (options?.scroll !== false) {\n window.scrollTo(0, 0);\n }\n window.dispatchEvent(new CustomEvent(\"vinext:navigate\"));\n return true;\n },\n back: () => window.history.back(),\n reload: () => window.location.reload(),\n prefetch: async (url: string) => {\n if (typeof document !== \"undefined\") {\n const link = document.createElement(\"link\");\n link.rel = \"prefetch\";\n link.href = url;\n link.as = \"document\";\n document.head.appendChild(link);\n }\n },\n beforePopState: (cb: BeforePopStateCallback) => {\n _beforePopStateCb = cb;\n },\n events: routerEvents,\n};\n\nexport default Router;\n"],"mappings":"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"}
|
package/dist/shims/server.d.ts
CHANGED
|
@@ -12,7 +12,15 @@
|
|
|
12
12
|
declare class NextRequest extends Request {
|
|
13
13
|
private _nextUrl;
|
|
14
14
|
private _cookies;
|
|
15
|
-
constructor(input: URL | RequestInfo, init?: RequestInit
|
|
15
|
+
constructor(input: URL | RequestInfo, init?: RequestInit & {
|
|
16
|
+
nextConfig?: {
|
|
17
|
+
basePath?: string;
|
|
18
|
+
i18n?: {
|
|
19
|
+
locales: string[];
|
|
20
|
+
defaultLocale: string;
|
|
21
|
+
};
|
|
22
|
+
};
|
|
23
|
+
});
|
|
16
24
|
get nextUrl(): NextURL;
|
|
17
25
|
get cookies(): RequestCookies;
|
|
18
26
|
/**
|
|
@@ -61,9 +69,32 @@ declare class NextResponse<_Body = unknown> extends Response {
|
|
|
61
69
|
*/
|
|
62
70
|
static next(init?: MiddlewareResponseInit): NextResponse;
|
|
63
71
|
}
|
|
72
|
+
interface NextURLConfig {
|
|
73
|
+
basePath?: string;
|
|
74
|
+
nextConfig?: {
|
|
75
|
+
i18n?: {
|
|
76
|
+
locales: string[];
|
|
77
|
+
defaultLocale: string;
|
|
78
|
+
};
|
|
79
|
+
};
|
|
80
|
+
}
|
|
64
81
|
declare class NextURL {
|
|
82
|
+
/** Internal URL stores the pathname WITHOUT basePath or locale prefix. */
|
|
65
83
|
private _url;
|
|
66
|
-
|
|
84
|
+
private _basePath;
|
|
85
|
+
private _locale;
|
|
86
|
+
private _defaultLocale;
|
|
87
|
+
private _locales;
|
|
88
|
+
constructor(input: string | URL, base?: string | URL, config?: NextURLConfig);
|
|
89
|
+
/** Strip basePath prefix from the internal pathname. */
|
|
90
|
+
private _stripBasePath;
|
|
91
|
+
/** Extract locale from pathname, stripping it from the internal URL. */
|
|
92
|
+
private _analyzeLocale;
|
|
93
|
+
/**
|
|
94
|
+
* Reconstruct the full pathname with basePath + locale prefix.
|
|
95
|
+
* Mirrors Next.js's internal formatPathname().
|
|
96
|
+
*/
|
|
97
|
+
private _formatPathname;
|
|
67
98
|
get href(): string;
|
|
68
99
|
set href(value: string);
|
|
69
100
|
get origin(): string;
|
|
@@ -79,6 +110,7 @@ declare class NextURL {
|
|
|
79
110
|
set hostname(value: string);
|
|
80
111
|
get port(): string;
|
|
81
112
|
set port(value: string);
|
|
113
|
+
/** Returns the pathname WITHOUT basePath or locale prefix. */
|
|
82
114
|
get pathname(): string;
|
|
83
115
|
set pathname(value: string);
|
|
84
116
|
get search(): string;
|
|
@@ -86,6 +118,12 @@ declare class NextURL {
|
|
|
86
118
|
get searchParams(): URLSearchParams;
|
|
87
119
|
get hash(): string;
|
|
88
120
|
set hash(value: string);
|
|
121
|
+
get basePath(): string;
|
|
122
|
+
set basePath(value: string);
|
|
123
|
+
get locale(): string;
|
|
124
|
+
set locale(value: string | undefined);
|
|
125
|
+
get defaultLocale(): string | undefined;
|
|
126
|
+
get locales(): string[] | undefined;
|
|
89
127
|
clone(): NextURL;
|
|
90
128
|
toString(): string;
|
|
91
129
|
/**
|
|
@@ -216,5 +254,5 @@ declare function connection(): Promise<void>;
|
|
|
216
254
|
*/
|
|
217
255
|
declare const URLPattern: typeof globalThis.URLPattern;
|
|
218
256
|
//#endregion
|
|
219
|
-
export { MiddlewareResponseInit, NextFetchEvent, NextMiddleware, NextMiddlewareResult, NextRequest, NextResponse, NextURL, RequestCookies, ResponseCookies, URLPattern, UserAgent, after, connection, userAgent, userAgentFromString };
|
|
257
|
+
export { MiddlewareResponseInit, NextFetchEvent, NextMiddleware, NextMiddlewareResult, NextRequest, NextResponse, NextURL, NextURLConfig, RequestCookies, ResponseCookies, URLPattern, UserAgent, after, connection, userAgent, userAgentFromString };
|
|
220
258
|
//# sourceMappingURL=server.d.ts.map
|
package/dist/shims/server.js
CHANGED
|
@@ -23,6 +23,7 @@ var NextRequest = class extends Request {
|
|
|
23
23
|
_nextUrl;
|
|
24
24
|
_cookies;
|
|
25
25
|
constructor(input, init) {
|
|
26
|
+
const { nextConfig: _nextConfig, ...requestInit } = init ?? {};
|
|
26
27
|
if (input instanceof Request) {
|
|
27
28
|
const req = input;
|
|
28
29
|
super(req.url, {
|
|
@@ -30,10 +31,13 @@ var NextRequest = class extends Request {
|
|
|
30
31
|
headers: req.headers,
|
|
31
32
|
body: req.body,
|
|
32
33
|
duplex: req.body ? "half" : void 0,
|
|
33
|
-
...
|
|
34
|
+
...requestInit
|
|
34
35
|
});
|
|
35
|
-
} else super(input,
|
|
36
|
-
this._nextUrl = new NextURL(typeof input === "string" ? new URL(input, "http://localhost") : input instanceof URL ? input : new URL(input.url, "http://localhost")
|
|
36
|
+
} else super(input, requestInit);
|
|
37
|
+
this._nextUrl = new NextURL(typeof input === "string" ? new URL(input, "http://localhost") : input instanceof URL ? input : new URL(input.url, "http://localhost"), void 0, _nextConfig ? {
|
|
38
|
+
basePath: _nextConfig.basePath,
|
|
39
|
+
nextConfig: { i18n: _nextConfig.i18n }
|
|
40
|
+
} : void 0);
|
|
37
41
|
this._cookies = new RequestCookies(this.headers);
|
|
38
42
|
}
|
|
39
43
|
get nextUrl() {
|
|
@@ -135,15 +139,61 @@ var NextResponse = class NextResponse extends Response {
|
|
|
135
139
|
}
|
|
136
140
|
};
|
|
137
141
|
var NextURL = class NextURL {
|
|
142
|
+
/** Internal URL stores the pathname WITHOUT basePath or locale prefix. */
|
|
138
143
|
_url;
|
|
139
|
-
|
|
144
|
+
_basePath;
|
|
145
|
+
_locale;
|
|
146
|
+
_defaultLocale;
|
|
147
|
+
_locales;
|
|
148
|
+
constructor(input, base, config) {
|
|
140
149
|
this._url = new URL(input.toString(), base);
|
|
150
|
+
this._basePath = config?.basePath ?? "";
|
|
151
|
+
this._stripBasePath();
|
|
152
|
+
const i18n = config?.nextConfig?.i18n;
|
|
153
|
+
if (i18n) {
|
|
154
|
+
this._locales = [...i18n.locales];
|
|
155
|
+
this._defaultLocale = i18n.defaultLocale;
|
|
156
|
+
this._analyzeLocale(this._locales);
|
|
157
|
+
}
|
|
158
|
+
}
|
|
159
|
+
/** Strip basePath prefix from the internal pathname. */
|
|
160
|
+
_stripBasePath() {
|
|
161
|
+
if (!this._basePath) return;
|
|
162
|
+
const { pathname } = this._url;
|
|
163
|
+
if (pathname === this._basePath || pathname.startsWith(this._basePath + "/")) this._url.pathname = pathname.slice(this._basePath.length) || "/";
|
|
164
|
+
}
|
|
165
|
+
/** Extract locale from pathname, stripping it from the internal URL. */
|
|
166
|
+
_analyzeLocale(locales) {
|
|
167
|
+
const segments = this._url.pathname.split("/");
|
|
168
|
+
const candidate = segments[1]?.toLowerCase();
|
|
169
|
+
const match = locales.find((l) => l.toLowerCase() === candidate);
|
|
170
|
+
if (match) {
|
|
171
|
+
this._locale = match;
|
|
172
|
+
this._url.pathname = "/" + segments.slice(2).join("/");
|
|
173
|
+
} else this._locale = this._defaultLocale;
|
|
174
|
+
}
|
|
175
|
+
/**
|
|
176
|
+
* Reconstruct the full pathname with basePath + locale prefix.
|
|
177
|
+
* Mirrors Next.js's internal formatPathname().
|
|
178
|
+
*/
|
|
179
|
+
_formatPathname() {
|
|
180
|
+
let prefix = this._basePath;
|
|
181
|
+
if (this._locale && this._locale !== this._defaultLocale) prefix += "/" + this._locale;
|
|
182
|
+
if (!prefix) return this._url.pathname;
|
|
183
|
+
const inner = this._url.pathname;
|
|
184
|
+
return inner === "/" ? prefix : prefix + inner;
|
|
141
185
|
}
|
|
142
186
|
get href() {
|
|
143
|
-
|
|
187
|
+
const formatted = this._formatPathname();
|
|
188
|
+
if (formatted === this._url.pathname) return this._url.href;
|
|
189
|
+
const { href, pathname, search, hash } = this._url;
|
|
190
|
+
const baseEnd = href.length - pathname.length - search.length - hash.length;
|
|
191
|
+
return href.slice(0, baseEnd) + formatted + search + hash;
|
|
144
192
|
}
|
|
145
193
|
set href(value) {
|
|
146
194
|
this._url.href = value;
|
|
195
|
+
this._stripBasePath();
|
|
196
|
+
if (this._locales) this._analyzeLocale(this._locales);
|
|
147
197
|
}
|
|
148
198
|
get origin() {
|
|
149
199
|
return this._url.origin;
|
|
@@ -184,6 +234,7 @@ var NextURL = class NextURL {
|
|
|
184
234
|
set port(value) {
|
|
185
235
|
this._url.port = value;
|
|
186
236
|
}
|
|
237
|
+
/** Returns the pathname WITHOUT basePath or locale prefix. */
|
|
187
238
|
get pathname() {
|
|
188
239
|
return this._url.pathname;
|
|
189
240
|
}
|
|
@@ -205,11 +256,43 @@ var NextURL = class NextURL {
|
|
|
205
256
|
set hash(value) {
|
|
206
257
|
this._url.hash = value;
|
|
207
258
|
}
|
|
259
|
+
get basePath() {
|
|
260
|
+
return this._basePath;
|
|
261
|
+
}
|
|
262
|
+
set basePath(value) {
|
|
263
|
+
this._basePath = value === "" ? "" : value.startsWith("/") ? value : "/" + value;
|
|
264
|
+
}
|
|
265
|
+
get locale() {
|
|
266
|
+
return this._locale ?? "";
|
|
267
|
+
}
|
|
268
|
+
set locale(value) {
|
|
269
|
+
if (this._locales) {
|
|
270
|
+
if (!value) {
|
|
271
|
+
this._locale = this._defaultLocale;
|
|
272
|
+
return;
|
|
273
|
+
}
|
|
274
|
+
if (!this._locales.includes(value)) throw new TypeError(`The locale "${value}" is not in the configured locales: ${this._locales.join(", ")}`);
|
|
275
|
+
}
|
|
276
|
+
this._locale = this._locales ? value : this._locale;
|
|
277
|
+
}
|
|
278
|
+
get defaultLocale() {
|
|
279
|
+
return this._defaultLocale;
|
|
280
|
+
}
|
|
281
|
+
get locales() {
|
|
282
|
+
return this._locales ? [...this._locales] : void 0;
|
|
283
|
+
}
|
|
208
284
|
clone() {
|
|
209
|
-
|
|
285
|
+
const config = {
|
|
286
|
+
basePath: this._basePath,
|
|
287
|
+
nextConfig: this._locales ? { i18n: {
|
|
288
|
+
locales: [...this._locales],
|
|
289
|
+
defaultLocale: this._defaultLocale
|
|
290
|
+
} } : void 0
|
|
291
|
+
};
|
|
292
|
+
return new NextURL(this.href, void 0, config);
|
|
210
293
|
}
|
|
211
294
|
toString() {
|
|
212
|
-
return this.
|
|
295
|
+
return this.href;
|
|
213
296
|
}
|
|
214
297
|
/**
|
|
215
298
|
* The build ID of the Next.js application.
|