toiljs 0.0.15 → 0.0.19
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/.babelrc +13 -13
- package/.gitattributes +2 -2
- package/.github/ISSUE_TEMPLATE/bug_report.md +38 -38
- package/.github/ISSUE_TEMPLATE/bug_report.yml +90 -90
- package/.github/ISSUE_TEMPLATE/config.yml +8 -8
- package/.github/ISSUE_TEMPLATE/feature_request.md +20 -20
- package/.github/PULL_REQUEST_TEMPLATE.md +43 -43
- package/.github/changelog-config.json +45 -45
- package/.github/dependabot.yml +27 -27
- package/.github/workflows/ci.yml +191 -191
- package/.prettierrc.json +11 -11
- package/.vscode/settings.json +9 -9
- package/CHANGELOG.md +116 -5
- package/LICENSE +187 -187
- package/README.md +524 -315
- package/as-pect.asconfig.json +34 -34
- package/as-pect.config.js +65 -65
- package/assets/logo.svg +36 -36
- package/build/backend/.tsbuildinfo +1 -1
- package/build/backend/index.d.ts +1 -0
- package/build/backend/index.js +20 -1
- package/build/cli/.tsbuildinfo +1 -1
- package/build/cli/index.js +1320 -696
- package/build/client/.tsbuildinfo +1 -1
- package/build/client/dev/devtools.d.ts +6 -0
- package/build/client/dev/devtools.js +479 -0
- package/build/client/dev/error-overlay.d.ts +9 -0
- package/build/client/dev/error-overlay.js +19 -4
- package/build/client/errors.d.ts +1 -0
- package/build/client/errors.js +3 -0
- package/build/client/index.d.ts +2 -0
- package/build/client/index.js +2 -0
- package/build/client/navigation/prefetch.d.ts +1 -0
- package/build/client/navigation/prefetch.js +35 -0
- package/build/client/routing/Router.js +1 -1
- package/build/client/routing/hooks.js +6 -2
- package/build/client/routing/loader.d.ts +23 -0
- package/build/client/routing/loader.js +53 -7
- package/build/client/routing/mount.js +4 -3
- package/build/client/rpc.d.ts +1 -0
- package/build/client/rpc.js +37 -0
- package/build/compiler/.tsbuildinfo +1 -1
- package/build/compiler/config.d.ts +16 -0
- package/build/compiler/config.js +9 -0
- package/build/compiler/docs.js +78 -21
- package/build/compiler/generate.js +5 -4
- package/build/compiler/index.d.ts +3 -2
- package/build/compiler/index.js +2 -2
- package/build/compiler/plugin.js +228 -0
- package/build/compiler/prerender.d.ts +1 -0
- package/build/compiler/prerender.js +1 -1
- package/build/compiler/seo.d.ts +1 -1
- package/build/compiler/seo.js +20 -5
- package/build/compiler/ssg.js +39 -2
- package/build/compiler/vite.js +25 -0
- package/build/io/.tsbuildinfo +1 -1
- package/build/io/codec.d.ts +54 -0
- package/build/io/codec.js +143 -0
- package/build/io/index.d.ts +1 -2
- package/build/io/index.js +1 -2
- package/build/logger/.tsbuildinfo +1 -1
- package/build/shared/.tsbuildinfo +1 -1
- package/eslint.config.js +48 -48
- package/examples/basic/client/404.tsx +11 -11
- package/examples/basic/client/components/.gitkeep +1 -1
- package/examples/basic/client/global-error.tsx +13 -13
- package/examples/basic/client/layout.tsx +25 -25
- package/examples/basic/client/public/images/.gitkeep +1 -1
- package/examples/basic/client/public/images/logo.svg +36 -36
- package/examples/basic/client/public/robots.txt +2 -2
- package/examples/basic/client/routes/docs/[...slug].tsx +12 -12
- package/examples/basic/client/routes/features/error/error.tsx +16 -16
- package/examples/basic/client/routes/features/index.tsx +1 -1
- package/examples/basic/client/routes/features/template/b.tsx +14 -14
- package/examples/basic/client/routes/files/[[...slug]].tsx +21 -21
- package/examples/basic/client/routes/gallery/layout.tsx +13 -13
- package/examples/basic/client/routes/io.tsx +23 -24
- package/examples/basic/client/routes/loader-demo/loading.tsx +13 -13
- package/examples/basic/client/routes/rest.tsx +74 -0
- package/examples/basic/client/routes/rpc.tsx +43 -0
- package/examples/basic/client/routes/search.tsx +61 -61
- package/examples/basic/client/toil.tsx +5 -5
- package/package.json +167 -148
- package/presets/eslint.js +88 -88
- package/presets/no-uint8array-tostring.js +200 -200
- package/presets/prettier-plugin.js +51 -0
- package/presets/prettier.json +19 -18
- package/presets/tsconfig.json +37 -37
- package/server/runtime/README.md +97 -0
- package/server/runtime/abort/abort.ts +27 -0
- package/server/runtime/env/Server.ts +61 -0
- package/server/runtime/envelope.ts +191 -0
- package/server/runtime/exports/index.ts +52 -0
- package/server/runtime/handlers/ToilHandler.ts +34 -0
- package/server/runtime/index.ts +26 -0
- package/server/runtime/lang/Potential.ts +5 -0
- package/server/runtime/memory.ts +81 -0
- package/server/runtime/request.ts +55 -0
- package/server/runtime/response.ts +86 -0
- package/server/runtime/rest/Rest.ts +39 -0
- package/server/runtime/rest/RestHandler.ts +20 -0
- package/server/runtime/rest/RouteContext.ts +82 -0
- package/server/runtime/rest/match.ts +48 -0
- package/server/runtime/tsconfig.json +7 -0
- package/src/backend/index.ts +202 -160
- package/src/cli/create.ts +15 -5
- package/src/cli/diagnostics.ts +81 -0
- package/src/cli/doctor.ts +384 -7
- package/src/cli/index.ts +11 -2
- package/src/cli/proc.ts +50 -50
- package/src/cli/updates.ts +69 -69
- package/src/cli/validate.ts +31 -31
- package/src/client/channel/channel.ts +146 -146
- package/src/client/components/Form.tsx +65 -65
- package/src/client/components/Script.tsx +113 -113
- package/src/client/components/Slot.tsx +21 -21
- package/src/client/dev/devtools.tsx +1018 -0
- package/src/client/dev/error-overlay.tsx +30 -4
- package/src/client/errors.ts +11 -0
- package/src/client/head/head.ts +167 -167
- package/src/client/head/metadata.ts +112 -112
- package/src/client/index.ts +91 -89
- package/src/client/navigation/NavLink.tsx +86 -86
- package/src/client/navigation/navigation.ts +235 -235
- package/src/client/navigation/prefetch.ts +169 -130
- package/src/client/navigation/scroll.ts +53 -53
- package/src/client/routing/Router.tsx +8 -2
- package/src/client/routing/action.ts +122 -122
- package/src/client/routing/error-boundary.tsx +43 -43
- package/src/client/routing/hooks.ts +21 -6
- package/src/client/routing/loader.ts +325 -235
- package/src/client/routing/match.ts +47 -47
- package/src/client/routing/mount.tsx +54 -52
- package/src/client/routing/params-context.ts +10 -10
- package/src/client/routing/slot-context.ts +7 -7
- package/src/client/rpc.ts +64 -0
- package/src/client/search/search.ts +189 -189
- package/src/client/search/use-page-search.ts +73 -73
- package/src/client/types.ts +73 -73
- package/src/compiler/config.ts +221 -182
- package/src/compiler/docs.ts +285 -228
- package/src/compiler/generate.ts +395 -394
- package/src/compiler/index.ts +66 -57
- package/src/compiler/pages.ts +70 -70
- package/src/compiler/plugin.ts +258 -2
- package/src/compiler/prerender.ts +156 -156
- package/src/compiler/seo.ts +417 -390
- package/src/compiler/ssg.ts +171 -126
- package/src/compiler/vite.ts +34 -0
- package/src/io/FastMap.ts +151 -127
- package/src/io/FastSet.ts +15 -1
- package/src/io/codec.ts +217 -0
- package/src/io/index.ts +10 -11
- package/src/io/lengths.ts +14 -14
- package/src/io/types.ts +19 -18
- package/src/logger/index.ts +22 -22
- package/src/shared/index.ts +10 -10
- package/std/client/index.d.ts +15 -15
- package/std/client/package.json +3 -3
- package/test/assembly/example.spec.ts +17 -7
- package/test/channel.test.ts +21 -21
- package/test/doctor.test.ts +65 -0
- package/test/dom/Link.test.tsx +47 -47
- package/test/dom/NavLink.test.tsx +37 -37
- package/test/dom/error-overlay.test.tsx +44 -44
- package/test/dom/loader.test.tsx +121 -121
- package/test/dom/navigation.test.ts +59 -59
- package/test/dom/revalidate.test.tsx +38 -38
- package/test/dom/route-head.test.tsx +78 -78
- package/test/dom/router-loading.test.tsx +44 -44
- package/test/dom/scroll.test.ts +56 -56
- package/test/dom/use-metadata.test.tsx +58 -58
- package/test/errors.test.ts +21 -0
- package/test/io.test.ts +117 -93
- package/test/navlink.test.ts +28 -28
- package/test/placeholder.test.ts +9 -9
- package/test/prettier-plugin.test.ts +46 -0
- package/test/routes.test.ts +76 -76
- package/test/rpc.test.ts +50 -0
- package/test/seo.test.ts +175 -164
- package/test/slot-layouts.test.ts +69 -69
- package/test/ssg.test.ts +36 -36
- package/test/update.test.ts +44 -44
- package/test/validate.test.ts +42 -42
- package/tests/data-parity/generated-parity.ts +99 -0
- package/tests/data-parity/parity.ts +80 -0
- package/tests/data-parity/spec.ts +46 -0
- package/toil-routes.d.ts +7 -0
- package/tsconfig.backend.json +13 -13
- package/tsconfig.base.json +35 -35
- package/tsconfig.cli.json +13 -13
- package/tsconfig.client.json +14 -14
- package/tsconfig.compiler.json +13 -13
- package/tsconfig.io.json +12 -12
- package/tsconfig.json +22 -22
- package/tsconfig.logger.json +12 -12
- package/tsconfig.server.json +10 -10
- package/tsconfig.shared.json +12 -12
- package/vitest.config.ts +26 -26
- package/.idea/codeStyles/Project.xml +0 -54
- package/.idea/codeStyles/codeStyleConfig.xml +0 -5
- package/.idea/inspectionProfiles/Project_Default.xml +0 -6
- package/.idea/modules.xml +0 -8
- package/.idea/prettier.xml +0 -7
- package/.idea/toiljs.iml +0 -8
- package/.idea/vcs.xml +0 -6
- package/.toil/entry.tsx +0 -9
- package/.toil/index.html +0 -12
- package/.toil/routes.ts +0 -9
- package/build/cli/configure.d.ts +0 -16
- package/build/cli/configure.js +0 -272
- package/build/cli/create.d.ts +0 -16
- package/build/cli/create.js +0 -420
- package/build/cli/diagnostics.d.ts +0 -55
- package/build/cli/diagnostics.js +0 -333
- package/build/cli/doctor.d.ts +0 -6
- package/build/cli/doctor.js +0 -249
- package/build/cli/features.d.ts +0 -25
- package/build/cli/features.js +0 -107
- package/build/cli/index.d.ts +0 -2
- package/build/cli/proc.d.ts +0 -6
- package/build/cli/proc.js +0 -31
- package/build/cli/ui.d.ts +0 -9
- package/build/cli/ui.js +0 -75
- package/build/cli/update.d.ts +0 -7
- package/build/cli/update.js +0 -117
- package/build/cli/updates.d.ts +0 -10
- package/build/cli/updates.js +0 -45
- package/build/cli/validate.d.ts +0 -4
- package/build/cli/validate.js +0 -19
- package/build/client/Link.d.ts +0 -8
- package/build/client/Link.js +0 -44
- package/build/client/NavLink.d.ts +0 -14
- package/build/client/NavLink.js +0 -37
- package/build/client/Router.d.ts +0 -7
- package/build/client/Router.js +0 -55
- package/build/client/channel.d.ts +0 -23
- package/build/client/channel.js +0 -94
- package/build/client/error-boundary.d.ts +0 -16
- package/build/client/error-boundary.js +0 -19
- package/build/client/head.d.ts +0 -26
- package/build/client/head.js +0 -87
- package/build/client/hooks.d.ts +0 -17
- package/build/client/hooks.js +0 -48
- package/build/client/lazy.d.ts +0 -16
- package/build/client/lazy.js +0 -53
- package/build/client/match.d.ts +0 -2
- package/build/client/match.js +0 -32
- package/build/client/mount.d.ts +0 -2
- package/build/client/mount.js +0 -13
- package/build/client/navigation.d.ts +0 -13
- package/build/client/navigation.js +0 -97
- package/build/client/params-context.d.ts +0 -2
- package/build/client/params-context.js +0 -2
- package/build/client/prefetch.d.ts +0 -11
- package/build/client/prefetch.js +0 -100
- package/build/client/runtime.d.ts +0 -31
- package/build/client/runtime.js +0 -112
- package/build/client/scroll.d.ts +0 -8
- package/build/client/scroll.js +0 -36
- package/build/io/BinaryReader.d.ts +0 -44
- package/build/io/BinaryReader.js +0 -244
- package/build/io/BinaryWriter.d.ts +0 -44
- package/build/io/BinaryWriter.js +0 -297
- package/build/server/release.wasm +0 -0
- package/build/server/release.wat +0 -9
- package/src/io/BinaryReader.ts +0 -340
- package/src/io/BinaryWriter.ts +0 -385
- package/src/server/index.ts +0 -10
- package/src/server/main.ts +0 -13
- package/src/server/tsconfig.json +0 -4
- package/toil-env.d.ts +0 -16
- package/toilconfig.json +0 -30
|
@@ -1,235 +1,235 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* History-based navigation core. Owns the location subscribers, the single `popstate` handler, and
|
|
3
|
-
* the per-entry history keys used for scroll restoration. Consumed by `useLocation` (to re-render),
|
|
4
|
-
* `Link` / `navigate` (to change location), and `Router` (which calls `applyScroll` after commit).
|
|
5
|
-
*/
|
|
6
|
-
import { startTransition } from 'react';
|
|
7
|
-
import { flushSync } from 'react-dom';
|
|
8
|
-
|
|
9
|
-
import { enableManualScrollRestoration, planScroll, rememberScroll } from './scroll.js';
|
|
10
|
-
import type { Href } from '../types.js';
|
|
11
|
-
|
|
12
|
-
/**
|
|
13
|
-
* Asserts a runtime-computed string is a valid {@link Href}, the escape hatch for hrefs built from
|
|
14
|
-
* data (e.g. `` `/${category}/${slug}` ``) that can't be checked against the static route union.
|
|
15
|
-
* Use at the call site: `<Link href={href(path)} />`, `navigate(href(path))`.
|
|
16
|
-
*/
|
|
17
|
-
export const href = (path: string): Href => path as Href;
|
|
18
|
-
|
|
19
|
-
const listeners = new Set<() => void>();
|
|
20
|
-
let popstateBound = false;
|
|
21
|
-
|
|
22
|
-
/** `document.startViewTransition`, present only where the View Transitions API is supported. */
|
|
23
|
-
interface ViewTransitionDocument {
|
|
24
|
-
startViewTransition?: (callback: () => void) => unknown;
|
|
25
|
-
}
|
|
26
|
-
let viewTransitions = false;
|
|
27
|
-
|
|
28
|
-
/** Enables animated View Transitions for navigation. Called once by `mount` from `client.viewTransitions`. */
|
|
29
|
-
export function setViewTransitions(enabled: boolean): void {
|
|
30
|
-
viewTransitions = enabled;
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
// Whether to wrap navigations in a React transition. Off by default: a navigation commits eagerly, so
|
|
34
|
-
// a route's `loading.tsx` shows immediately while its loader runs. On, the current page is kept
|
|
35
|
-
// visible until the next route is ready (smoother, but no loading state). Set from `client.transitions`.
|
|
36
|
-
let navTransitions = false;
|
|
37
|
-
|
|
38
|
-
/** Enables React-transition (keep-current-page) navigation. Called once by `mount` from `client.transitions`. */
|
|
39
|
-
export function setTransitions(enabled: boolean): void {
|
|
40
|
-
navTransitions = enabled;
|
|
41
|
-
}
|
|
42
|
-
|
|
43
|
-
/** Whether the current navigation should animate via the View Transitions API. */
|
|
44
|
-
function shouldViewTransition(): boolean {
|
|
45
|
-
if (!viewTransitions || typeof document === 'undefined' || typeof window === 'undefined') {
|
|
46
|
-
return false;
|
|
47
|
-
}
|
|
48
|
-
if (typeof (document as ViewTransitionDocument).startViewTransition !== 'function')
|
|
49
|
-
return false;
|
|
50
|
-
return !window.matchMedia('(prefers-reduced-motion: reduce)').matches;
|
|
51
|
-
}
|
|
52
|
-
|
|
53
|
-
interface ToilHistoryState {
|
|
54
|
-
__toilKey?: string;
|
|
55
|
-
}
|
|
56
|
-
let keyCounter = 0;
|
|
57
|
-
let currentKey = 'initial';
|
|
58
|
-
function nextKey(): string {
|
|
59
|
-
keyCounter += 1;
|
|
60
|
-
return `t${String(keyCounter)}`;
|
|
61
|
-
}
|
|
62
|
-
|
|
63
|
-
function runListeners(): void {
|
|
64
|
-
for (const listener of listeners) listener();
|
|
65
|
-
}
|
|
66
|
-
|
|
67
|
-
/**
|
|
68
|
-
* Re-renders subscribers for a location change. When View Transitions are enabled and supported, the
|
|
69
|
-
* commit runs synchronously inside `document.startViewTransition` so the browser animates the old and
|
|
70
|
-
* new DOM. Otherwise, with `client.transitions` on, it's wrapped in `startTransition` (the current
|
|
71
|
-
* page stays while the next route loads); by default it commits eagerly, so a route's `loading.tsx`
|
|
72
|
-
* shows right away instead of holding the previous page.
|
|
73
|
-
*/
|
|
74
|
-
function notify(): void {
|
|
75
|
-
if (shouldViewTransition()) {
|
|
76
|
-
(document as ViewTransitionDocument).startViewTransition?.(() => {
|
|
77
|
-
flushSync(runListeners);
|
|
78
|
-
});
|
|
79
|
-
} else if (navTransitions) {
|
|
80
|
-
startTransition(runListeners);
|
|
81
|
-
} else {
|
|
82
|
-
runListeners();
|
|
83
|
-
}
|
|
84
|
-
}
|
|
85
|
-
|
|
86
|
-
// Soft vs hard navigation, for intercepting routes. The initial page load (and any full refresh) is
|
|
87
|
-
// "hard"; client navigations (`navigate` / back / forward) are "soft". `previousPath` is the path we
|
|
88
|
-
// were on before the latest soft navigation, the route the main view keeps showing while an
|
|
89
|
-
// intercepting route fills a slot (the modal overlay).
|
|
90
|
-
let softNav = false;
|
|
91
|
-
let currentPath = typeof window === 'undefined' ? '/' : window.location.pathname;
|
|
92
|
-
let previousPath = currentPath;
|
|
93
|
-
|
|
94
|
-
/** Records a transition to the live location; `soft` is false only for the initial load. */
|
|
95
|
-
function recordTransition(soft: boolean): void {
|
|
96
|
-
previousPath = currentPath;
|
|
97
|
-
currentPath = typeof window === 'undefined' ? '/' : window.location.pathname;
|
|
98
|
-
softNav = soft;
|
|
99
|
-
}
|
|
100
|
-
|
|
101
|
-
/** Whether the current location was reached by a client navigation (not an initial load / refresh). */
|
|
102
|
-
export function isSoftNavigation(): boolean {
|
|
103
|
-
return softNav;
|
|
104
|
-
}
|
|
105
|
-
|
|
106
|
-
/** The path the app was on before the latest navigation (what the main view keeps during an intercept). */
|
|
107
|
-
export function previousPathname(): string {
|
|
108
|
-
return previousPath;
|
|
109
|
-
}
|
|
110
|
-
|
|
111
|
-
// Navigation-pending tracking: a navigation is "pending" from when it starts until the new route
|
|
112
|
-
// commits. Drives useNavigationPending() (e.g. a top loading bar).
|
|
113
|
-
let startedTick = 0;
|
|
114
|
-
let committedTick = 0;
|
|
115
|
-
const pendingListeners = new Set<() => void>();
|
|
116
|
-
function emitPending(): void {
|
|
117
|
-
for (const listener of pendingListeners) listener();
|
|
118
|
-
}
|
|
119
|
-
function beginNavigation(): void {
|
|
120
|
-
startedTick += 1;
|
|
121
|
-
emitPending();
|
|
122
|
-
}
|
|
123
|
-
|
|
124
|
-
/** Marks the in-flight navigation as committed. Called by `Router` after each commit. */
|
|
125
|
-
export function settleNavigation(): void {
|
|
126
|
-
if (committedTick !== startedTick) {
|
|
127
|
-
committedTick = startedTick;
|
|
128
|
-
emitPending();
|
|
129
|
-
}
|
|
130
|
-
}
|
|
131
|
-
|
|
132
|
-
/** Whether a navigation is in flight (started but not yet committed). */
|
|
133
|
-
export function isNavigationPending(): boolean {
|
|
134
|
-
return startedTick !== committedTick;
|
|
135
|
-
}
|
|
136
|
-
|
|
137
|
-
/** Monotonic id incremented on each navigation, used to key/revalidate per-navigation route data. */
|
|
138
|
-
export function navigationEpoch(): number {
|
|
139
|
-
return startedTick;
|
|
140
|
-
}
|
|
141
|
-
|
|
142
|
-
/** Subscribes to navigation-pending changes; returns an unsubscribe function. */
|
|
143
|
-
export function subscribePending(listener: () => void): () => void {
|
|
144
|
-
pendingListeners.add(listener);
|
|
145
|
-
return () => {
|
|
146
|
-
pendingListeners.delete(listener);
|
|
147
|
-
};
|
|
148
|
-
}
|
|
149
|
-
|
|
150
|
-
/** Options for {@link navigate}. */
|
|
151
|
-
export interface NavigateOptions {
|
|
152
|
-
/** Replace the current history entry instead of pushing a new one. Default `false`. */
|
|
153
|
-
readonly replace?: boolean;
|
|
154
|
-
/** Scroll to the top of the page after navigating. Default `true`. */
|
|
155
|
-
readonly scroll?: boolean;
|
|
156
|
-
}
|
|
157
|
-
|
|
158
|
-
/** Initializes manual scroll restoration and the initial history key. Called once by `mount`. */
|
|
159
|
-
export function initNavigation(): void {
|
|
160
|
-
enableManualScrollRestoration();
|
|
161
|
-
const state = window.history.state as ToilHistoryState | null;
|
|
162
|
-
if (state?.__toilKey) {
|
|
163
|
-
currentKey = state.__toilKey;
|
|
164
|
-
} else {
|
|
165
|
-
currentKey = nextKey();
|
|
166
|
-
window.history.replaceState({ ...state, __toilKey: currentKey }, '');
|
|
167
|
-
}
|
|
168
|
-
}
|
|
169
|
-
|
|
170
|
-
/** Navigates to `href` without a full page reload (history push/replace + subscriber re-render). */
|
|
171
|
-
export function navigate(href: Href, options?: NavigateOptions): void {
|
|
172
|
-
beginNavigation();
|
|
173
|
-
rememberScroll(currentKey);
|
|
174
|
-
let hash = '';
|
|
175
|
-
try {
|
|
176
|
-
hash = new URL(href, window.location.href).hash;
|
|
177
|
-
} catch {
|
|
178
|
-
hash = '';
|
|
179
|
-
}
|
|
180
|
-
if (options?.replace) {
|
|
181
|
-
window.history.replaceState({ __toilKey: currentKey }, '', href);
|
|
182
|
-
} else {
|
|
183
|
-
currentKey = nextKey();
|
|
184
|
-
window.history.pushState({ __toilKey: currentKey }, '', href);
|
|
185
|
-
}
|
|
186
|
-
recordTransition(true);
|
|
187
|
-
planScroll({ hash, toTop: options?.scroll !== false });
|
|
188
|
-
notify();
|
|
189
|
-
}
|
|
190
|
-
|
|
191
|
-
/** Goes back one entry in history (fires `popstate`, which notifies subscribers). */
|
|
192
|
-
export function back(): void {
|
|
193
|
-
window.history.back();
|
|
194
|
-
}
|
|
195
|
-
|
|
196
|
-
/** Goes forward one entry in history. */
|
|
197
|
-
export function forward(): void {
|
|
198
|
-
window.history.forward();
|
|
199
|
-
}
|
|
200
|
-
|
|
201
|
-
/**
|
|
202
|
-
* Re-renders the current route, bumping the navigation epoch so a revalidation of the *same* URL
|
|
203
|
-
* re-keys its Suspense boundary (its `loading.tsx` shows while the loader re-runs) and
|
|
204
|
-
* `useNavigationPending` reports the in-flight refetch, instead of silently freezing the old page.
|
|
205
|
-
*/
|
|
206
|
-
export function refresh(): void {
|
|
207
|
-
beginNavigation();
|
|
208
|
-
notify();
|
|
209
|
-
}
|
|
210
|
-
|
|
211
|
-
/** Handles browser back/forward: restores the saved scroll for the target entry, then re-renders. */
|
|
212
|
-
function handlePopState(event: PopStateEvent): void {
|
|
213
|
-
beginNavigation();
|
|
214
|
-
rememberScroll(currentKey);
|
|
215
|
-
const state = event.state as ToilHistoryState | null;
|
|
216
|
-
currentKey = state?.__toilKey ?? 'initial';
|
|
217
|
-
recordTransition(true);
|
|
218
|
-
planScroll({ restoreKey: currentKey, hash: window.location.hash, toTop: false });
|
|
219
|
-
notify();
|
|
220
|
-
}
|
|
221
|
-
|
|
222
|
-
/**
|
|
223
|
-
* Subscribes `listener` to location changes and returns an unsubscribe function. Browser
|
|
224
|
-
* back/forward is wired once, on the first subscription, via a shared `popstate` handler.
|
|
225
|
-
*/
|
|
226
|
-
export function subscribeLocation(listener: () => void): () => void {
|
|
227
|
-
if (!popstateBound) {
|
|
228
|
-
window.addEventListener('popstate', handlePopState);
|
|
229
|
-
popstateBound = true;
|
|
230
|
-
}
|
|
231
|
-
listeners.add(listener);
|
|
232
|
-
return () => {
|
|
233
|
-
listeners.delete(listener);
|
|
234
|
-
};
|
|
235
|
-
}
|
|
1
|
+
/**
|
|
2
|
+
* History-based navigation core. Owns the location subscribers, the single `popstate` handler, and
|
|
3
|
+
* the per-entry history keys used for scroll restoration. Consumed by `useLocation` (to re-render),
|
|
4
|
+
* `Link` / `navigate` (to change location), and `Router` (which calls `applyScroll` after commit).
|
|
5
|
+
*/
|
|
6
|
+
import { startTransition } from 'react';
|
|
7
|
+
import { flushSync } from 'react-dom';
|
|
8
|
+
|
|
9
|
+
import { enableManualScrollRestoration, planScroll, rememberScroll } from './scroll.js';
|
|
10
|
+
import type { Href } from '../types.js';
|
|
11
|
+
|
|
12
|
+
/**
|
|
13
|
+
* Asserts a runtime-computed string is a valid {@link Href}, the escape hatch for hrefs built from
|
|
14
|
+
* data (e.g. `` `/${category}/${slug}` ``) that can't be checked against the static route union.
|
|
15
|
+
* Use at the call site: `<Link href={href(path)} />`, `navigate(href(path))`.
|
|
16
|
+
*/
|
|
17
|
+
export const href = (path: string): Href => path as Href;
|
|
18
|
+
|
|
19
|
+
const listeners = new Set<() => void>();
|
|
20
|
+
let popstateBound = false;
|
|
21
|
+
|
|
22
|
+
/** `document.startViewTransition`, present only where the View Transitions API is supported. */
|
|
23
|
+
interface ViewTransitionDocument {
|
|
24
|
+
startViewTransition?: (callback: () => void) => unknown;
|
|
25
|
+
}
|
|
26
|
+
let viewTransitions = false;
|
|
27
|
+
|
|
28
|
+
/** Enables animated View Transitions for navigation. Called once by `mount` from `client.viewTransitions`. */
|
|
29
|
+
export function setViewTransitions(enabled: boolean): void {
|
|
30
|
+
viewTransitions = enabled;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
// Whether to wrap navigations in a React transition. Off by default: a navigation commits eagerly, so
|
|
34
|
+
// a route's `loading.tsx` shows immediately while its loader runs. On, the current page is kept
|
|
35
|
+
// visible until the next route is ready (smoother, but no loading state). Set from `client.transitions`.
|
|
36
|
+
let navTransitions = false;
|
|
37
|
+
|
|
38
|
+
/** Enables React-transition (keep-current-page) navigation. Called once by `mount` from `client.transitions`. */
|
|
39
|
+
export function setTransitions(enabled: boolean): void {
|
|
40
|
+
navTransitions = enabled;
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
/** Whether the current navigation should animate via the View Transitions API. */
|
|
44
|
+
function shouldViewTransition(): boolean {
|
|
45
|
+
if (!viewTransitions || typeof document === 'undefined' || typeof window === 'undefined') {
|
|
46
|
+
return false;
|
|
47
|
+
}
|
|
48
|
+
if (typeof (document as ViewTransitionDocument).startViewTransition !== 'function')
|
|
49
|
+
return false;
|
|
50
|
+
return !window.matchMedia('(prefers-reduced-motion: reduce)').matches;
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
interface ToilHistoryState {
|
|
54
|
+
__toilKey?: string;
|
|
55
|
+
}
|
|
56
|
+
let keyCounter = 0;
|
|
57
|
+
let currentKey = 'initial';
|
|
58
|
+
function nextKey(): string {
|
|
59
|
+
keyCounter += 1;
|
|
60
|
+
return `t${String(keyCounter)}`;
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
function runListeners(): void {
|
|
64
|
+
for (const listener of listeners) listener();
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
/**
|
|
68
|
+
* Re-renders subscribers for a location change. When View Transitions are enabled and supported, the
|
|
69
|
+
* commit runs synchronously inside `document.startViewTransition` so the browser animates the old and
|
|
70
|
+
* new DOM. Otherwise, with `client.transitions` on, it's wrapped in `startTransition` (the current
|
|
71
|
+
* page stays while the next route loads); by default it commits eagerly, so a route's `loading.tsx`
|
|
72
|
+
* shows right away instead of holding the previous page.
|
|
73
|
+
*/
|
|
74
|
+
function notify(): void {
|
|
75
|
+
if (shouldViewTransition()) {
|
|
76
|
+
(document as ViewTransitionDocument).startViewTransition?.(() => {
|
|
77
|
+
flushSync(runListeners);
|
|
78
|
+
});
|
|
79
|
+
} else if (navTransitions) {
|
|
80
|
+
startTransition(runListeners);
|
|
81
|
+
} else {
|
|
82
|
+
runListeners();
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
// Soft vs hard navigation, for intercepting routes. The initial page load (and any full refresh) is
|
|
87
|
+
// "hard"; client navigations (`navigate` / back / forward) are "soft". `previousPath` is the path we
|
|
88
|
+
// were on before the latest soft navigation, the route the main view keeps showing while an
|
|
89
|
+
// intercepting route fills a slot (the modal overlay).
|
|
90
|
+
let softNav = false;
|
|
91
|
+
let currentPath = typeof window === 'undefined' ? '/' : window.location.pathname;
|
|
92
|
+
let previousPath = currentPath;
|
|
93
|
+
|
|
94
|
+
/** Records a transition to the live location; `soft` is false only for the initial load. */
|
|
95
|
+
function recordTransition(soft: boolean): void {
|
|
96
|
+
previousPath = currentPath;
|
|
97
|
+
currentPath = typeof window === 'undefined' ? '/' : window.location.pathname;
|
|
98
|
+
softNav = soft;
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
/** Whether the current location was reached by a client navigation (not an initial load / refresh). */
|
|
102
|
+
export function isSoftNavigation(): boolean {
|
|
103
|
+
return softNav;
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
/** The path the app was on before the latest navigation (what the main view keeps during an intercept). */
|
|
107
|
+
export function previousPathname(): string {
|
|
108
|
+
return previousPath;
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
// Navigation-pending tracking: a navigation is "pending" from when it starts until the new route
|
|
112
|
+
// commits. Drives useNavigationPending() (e.g. a top loading bar).
|
|
113
|
+
let startedTick = 0;
|
|
114
|
+
let committedTick = 0;
|
|
115
|
+
const pendingListeners = new Set<() => void>();
|
|
116
|
+
function emitPending(): void {
|
|
117
|
+
for (const listener of pendingListeners) listener();
|
|
118
|
+
}
|
|
119
|
+
function beginNavigation(): void {
|
|
120
|
+
startedTick += 1;
|
|
121
|
+
emitPending();
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
/** Marks the in-flight navigation as committed. Called by `Router` after each commit. */
|
|
125
|
+
export function settleNavigation(): void {
|
|
126
|
+
if (committedTick !== startedTick) {
|
|
127
|
+
committedTick = startedTick;
|
|
128
|
+
emitPending();
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
/** Whether a navigation is in flight (started but not yet committed). */
|
|
133
|
+
export function isNavigationPending(): boolean {
|
|
134
|
+
return startedTick !== committedTick;
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
/** Monotonic id incremented on each navigation, used to key/revalidate per-navigation route data. */
|
|
138
|
+
export function navigationEpoch(): number {
|
|
139
|
+
return startedTick;
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
/** Subscribes to navigation-pending changes; returns an unsubscribe function. */
|
|
143
|
+
export function subscribePending(listener: () => void): () => void {
|
|
144
|
+
pendingListeners.add(listener);
|
|
145
|
+
return () => {
|
|
146
|
+
pendingListeners.delete(listener);
|
|
147
|
+
};
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
/** Options for {@link navigate}. */
|
|
151
|
+
export interface NavigateOptions {
|
|
152
|
+
/** Replace the current history entry instead of pushing a new one. Default `false`. */
|
|
153
|
+
readonly replace?: boolean;
|
|
154
|
+
/** Scroll to the top of the page after navigating. Default `true`. */
|
|
155
|
+
readonly scroll?: boolean;
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
/** Initializes manual scroll restoration and the initial history key. Called once by `mount`. */
|
|
159
|
+
export function initNavigation(): void {
|
|
160
|
+
enableManualScrollRestoration();
|
|
161
|
+
const state = window.history.state as ToilHistoryState | null;
|
|
162
|
+
if (state?.__toilKey) {
|
|
163
|
+
currentKey = state.__toilKey;
|
|
164
|
+
} else {
|
|
165
|
+
currentKey = nextKey();
|
|
166
|
+
window.history.replaceState({ ...state, __toilKey: currentKey }, '');
|
|
167
|
+
}
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
/** Navigates to `href` without a full page reload (history push/replace + subscriber re-render). */
|
|
171
|
+
export function navigate(href: Href, options?: NavigateOptions): void {
|
|
172
|
+
beginNavigation();
|
|
173
|
+
rememberScroll(currentKey);
|
|
174
|
+
let hash = '';
|
|
175
|
+
try {
|
|
176
|
+
hash = new URL(href, window.location.href).hash;
|
|
177
|
+
} catch {
|
|
178
|
+
hash = '';
|
|
179
|
+
}
|
|
180
|
+
if (options?.replace) {
|
|
181
|
+
window.history.replaceState({ __toilKey: currentKey }, '', href);
|
|
182
|
+
} else {
|
|
183
|
+
currentKey = nextKey();
|
|
184
|
+
window.history.pushState({ __toilKey: currentKey }, '', href);
|
|
185
|
+
}
|
|
186
|
+
recordTransition(true);
|
|
187
|
+
planScroll({ hash, toTop: options?.scroll !== false });
|
|
188
|
+
notify();
|
|
189
|
+
}
|
|
190
|
+
|
|
191
|
+
/** Goes back one entry in history (fires `popstate`, which notifies subscribers). */
|
|
192
|
+
export function back(): void {
|
|
193
|
+
window.history.back();
|
|
194
|
+
}
|
|
195
|
+
|
|
196
|
+
/** Goes forward one entry in history. */
|
|
197
|
+
export function forward(): void {
|
|
198
|
+
window.history.forward();
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
/**
|
|
202
|
+
* Re-renders the current route, bumping the navigation epoch so a revalidation of the *same* URL
|
|
203
|
+
* re-keys its Suspense boundary (its `loading.tsx` shows while the loader re-runs) and
|
|
204
|
+
* `useNavigationPending` reports the in-flight refetch, instead of silently freezing the old page.
|
|
205
|
+
*/
|
|
206
|
+
export function refresh(): void {
|
|
207
|
+
beginNavigation();
|
|
208
|
+
notify();
|
|
209
|
+
}
|
|
210
|
+
|
|
211
|
+
/** Handles browser back/forward: restores the saved scroll for the target entry, then re-renders. */
|
|
212
|
+
function handlePopState(event: PopStateEvent): void {
|
|
213
|
+
beginNavigation();
|
|
214
|
+
rememberScroll(currentKey);
|
|
215
|
+
const state = event.state as ToilHistoryState | null;
|
|
216
|
+
currentKey = state?.__toilKey ?? 'initial';
|
|
217
|
+
recordTransition(true);
|
|
218
|
+
planScroll({ restoreKey: currentKey, hash: window.location.hash, toTop: false });
|
|
219
|
+
notify();
|
|
220
|
+
}
|
|
221
|
+
|
|
222
|
+
/**
|
|
223
|
+
* Subscribes `listener` to location changes and returns an unsubscribe function. Browser
|
|
224
|
+
* back/forward is wired once, on the first subscription, via a shared `popstate` handler.
|
|
225
|
+
*/
|
|
226
|
+
export function subscribeLocation(listener: () => void): () => void {
|
|
227
|
+
if (!popstateBound) {
|
|
228
|
+
window.addEventListener('popstate', handlePopState);
|
|
229
|
+
popstateBound = true;
|
|
230
|
+
}
|
|
231
|
+
listeners.add(listener);
|
|
232
|
+
return () => {
|
|
233
|
+
listeners.delete(listener);
|
|
234
|
+
};
|
|
235
|
+
}
|