what-router 0.5.5 → 0.7.0
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 +1 -1
- package/dist/index.js +505 -0
- package/dist/index.js.map +7 -0
- package/dist/index.min.js +2 -0
- package/dist/index.min.js.map +7 -0
- package/package.json +6 -4
- package/src/index.js +188 -120
package/README.md
CHANGED
package/dist/index.js
ADDED
|
@@ -0,0 +1,505 @@
|
|
|
1
|
+
// packages/router/src/index.js
|
|
2
|
+
import { signal, effect, computed, batch, h, ErrorBoundary } from "what-core";
|
|
3
|
+
function isSafeUrl(url) {
|
|
4
|
+
if (typeof url !== "string") return false;
|
|
5
|
+
const trimmed = url.trim();
|
|
6
|
+
const normalized = trimmed.replace(/[\s\x00-\x1f]/g, "").toLowerCase();
|
|
7
|
+
if (normalized.startsWith("javascript:")) return false;
|
|
8
|
+
if (normalized.startsWith("data:")) return false;
|
|
9
|
+
if (normalized.startsWith("vbscript:")) return false;
|
|
10
|
+
return true;
|
|
11
|
+
}
|
|
12
|
+
var _url = signal(typeof location !== "undefined" ? location.pathname + location.search + location.hash : "/");
|
|
13
|
+
var _params = signal({});
|
|
14
|
+
var _query = signal({});
|
|
15
|
+
var _isNavigating = signal(false);
|
|
16
|
+
var _navigationError = signal(null);
|
|
17
|
+
var route = {
|
|
18
|
+
get url() {
|
|
19
|
+
return _url();
|
|
20
|
+
},
|
|
21
|
+
get path() {
|
|
22
|
+
return _url().split("?")[0].split("#")[0];
|
|
23
|
+
},
|
|
24
|
+
get params() {
|
|
25
|
+
return _params();
|
|
26
|
+
},
|
|
27
|
+
get query() {
|
|
28
|
+
return _query();
|
|
29
|
+
},
|
|
30
|
+
get hash() {
|
|
31
|
+
const h2 = _url().split("#")[1];
|
|
32
|
+
return h2 ? "#" + h2 : "";
|
|
33
|
+
},
|
|
34
|
+
get isNavigating() {
|
|
35
|
+
return _isNavigating();
|
|
36
|
+
},
|
|
37
|
+
get error() {
|
|
38
|
+
return _navigationError();
|
|
39
|
+
}
|
|
40
|
+
};
|
|
41
|
+
async function navigate(to, opts = {}) {
|
|
42
|
+
const { replace = false, state = null, transition = true, _fromPopstate = false } = opts;
|
|
43
|
+
if (!isSafeUrl(to)) {
|
|
44
|
+
if (typeof console !== "undefined") {
|
|
45
|
+
console.warn(`[what-router] Blocked navigation to unsafe URL: ${to}`);
|
|
46
|
+
}
|
|
47
|
+
return;
|
|
48
|
+
}
|
|
49
|
+
if (typeof window !== "undefined" && to.startsWith("#")) {
|
|
50
|
+
const currentUrl = _url();
|
|
51
|
+
const basePath = currentUrl.split("#")[0];
|
|
52
|
+
const newUrl = basePath + to;
|
|
53
|
+
history.replaceState(state, "", newUrl);
|
|
54
|
+
_url.set(newUrl);
|
|
55
|
+
const el = document.querySelector(to);
|
|
56
|
+
if (el) el.scrollIntoView({ behavior: "smooth" });
|
|
57
|
+
return;
|
|
58
|
+
}
|
|
59
|
+
if (to === _url()) return;
|
|
60
|
+
if (_isNavigating.peek()) return;
|
|
61
|
+
_isNavigating.set(true);
|
|
62
|
+
_navigationError.set(null);
|
|
63
|
+
const doNavigation = () => {
|
|
64
|
+
if (!_fromPopstate) {
|
|
65
|
+
if (typeof window !== "undefined") {
|
|
66
|
+
scrollPositions.set(_url(), { x: scrollX, y: scrollY });
|
|
67
|
+
}
|
|
68
|
+
if (replace) {
|
|
69
|
+
history.replaceState(state, "", to);
|
|
70
|
+
} else {
|
|
71
|
+
history.pushState(state, "", to);
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
_url.set(to);
|
|
75
|
+
_isNavigating.set(false);
|
|
76
|
+
};
|
|
77
|
+
if (transition && typeof document !== "undefined" && document.startViewTransition) {
|
|
78
|
+
try {
|
|
79
|
+
await document.startViewTransition(doNavigation).finished;
|
|
80
|
+
} catch (e) {
|
|
81
|
+
}
|
|
82
|
+
} else {
|
|
83
|
+
doNavigation();
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
if (typeof window !== "undefined") {
|
|
87
|
+
window.addEventListener("popstate", () => {
|
|
88
|
+
scrollPositions.set(_url(), { x: scrollX, y: scrollY });
|
|
89
|
+
const newUrl = location.pathname + location.search + location.hash;
|
|
90
|
+
navigate(newUrl, { replace: true, _fromPopstate: true, transition: false }).then(() => {
|
|
91
|
+
const saved = scrollPositions.get(newUrl);
|
|
92
|
+
if (saved) {
|
|
93
|
+
requestAnimationFrame(() => window.scrollTo(saved.x, saved.y));
|
|
94
|
+
}
|
|
95
|
+
});
|
|
96
|
+
});
|
|
97
|
+
}
|
|
98
|
+
function compilePath(path) {
|
|
99
|
+
const normalized = path.replace(/\([\w-]+\)\//g, "").replace(/\[\.\.\.(\w+)\]/g, (_, name) => `*:${name}`).replace(/\[(\w+)\]/g, ":$1");
|
|
100
|
+
const paramNames = [];
|
|
101
|
+
let catchAll = null;
|
|
102
|
+
const regexStr = normalized.split("/").map((segment) => {
|
|
103
|
+
if (segment.startsWith("*:")) {
|
|
104
|
+
catchAll = segment.slice(2);
|
|
105
|
+
paramNames.push(catchAll);
|
|
106
|
+
return "(.+)";
|
|
107
|
+
}
|
|
108
|
+
if (segment === "*") {
|
|
109
|
+
catchAll = "rest";
|
|
110
|
+
paramNames.push("rest");
|
|
111
|
+
return "(.+)";
|
|
112
|
+
}
|
|
113
|
+
if (segment.startsWith(":")) {
|
|
114
|
+
paramNames.push(segment.slice(1));
|
|
115
|
+
return "([^/]+)";
|
|
116
|
+
}
|
|
117
|
+
return segment.replace(/[.*+?^${}()|[\]\\]/g, "\\$&");
|
|
118
|
+
}).join("/");
|
|
119
|
+
const regex = new RegExp(`^${regexStr}$`);
|
|
120
|
+
return { regex, paramNames, catchAll };
|
|
121
|
+
}
|
|
122
|
+
function matchRoute(path, routes) {
|
|
123
|
+
const routable = routes.filter((r) => r.path);
|
|
124
|
+
const sorted = routable.sort((a, b) => {
|
|
125
|
+
const aSpecific = (a.path.match(/:/g) || []).length + (a.path.includes("*") ? 100 : 0);
|
|
126
|
+
const bSpecific = (b.path.match(/:/g) || []).length + (b.path.includes("*") ? 100 : 0);
|
|
127
|
+
return aSpecific - bSpecific;
|
|
128
|
+
});
|
|
129
|
+
for (const route2 of sorted) {
|
|
130
|
+
const { regex, paramNames } = compilePath(route2.path);
|
|
131
|
+
const match = path.match(regex);
|
|
132
|
+
if (match) {
|
|
133
|
+
const params = {};
|
|
134
|
+
paramNames.forEach((name, i) => {
|
|
135
|
+
params[name] = decodeURIComponent(match[i + 1]);
|
|
136
|
+
});
|
|
137
|
+
return { route: route2, params };
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
return null;
|
|
141
|
+
}
|
|
142
|
+
function parseQuery(search) {
|
|
143
|
+
const params = {};
|
|
144
|
+
if (!search) return params;
|
|
145
|
+
const qs = search.startsWith("?") ? search.slice(1) : search;
|
|
146
|
+
for (const pair of qs.split("&")) {
|
|
147
|
+
const [key, val] = pair.split("=");
|
|
148
|
+
if (!key) continue;
|
|
149
|
+
const decodedKey = decodeURIComponent(key);
|
|
150
|
+
const decodedVal = val ? decodeURIComponent(val) : "";
|
|
151
|
+
if (decodedKey in params) {
|
|
152
|
+
if (Array.isArray(params[decodedKey])) {
|
|
153
|
+
params[decodedKey].push(decodedVal);
|
|
154
|
+
} else {
|
|
155
|
+
params[decodedKey] = [params[decodedKey], decodedVal];
|
|
156
|
+
}
|
|
157
|
+
} else {
|
|
158
|
+
params[decodedKey] = decodedVal;
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
return params;
|
|
162
|
+
}
|
|
163
|
+
function buildLayoutChain(route2, routes) {
|
|
164
|
+
const layouts = [];
|
|
165
|
+
if (!route2.path) return layouts;
|
|
166
|
+
const segments = route2.path.split("/").filter(Boolean);
|
|
167
|
+
let currentPath = "";
|
|
168
|
+
for (const segment of segments) {
|
|
169
|
+
currentPath += "/" + segment;
|
|
170
|
+
const layoutRoute = routes.find(
|
|
171
|
+
(r) => r.layout && r.path === currentPath + "/_layout"
|
|
172
|
+
);
|
|
173
|
+
if (layoutRoute) {
|
|
174
|
+
layouts.push(layoutRoute.layout);
|
|
175
|
+
}
|
|
176
|
+
}
|
|
177
|
+
if (route2.layout) {
|
|
178
|
+
layouts.push(route2.layout);
|
|
179
|
+
}
|
|
180
|
+
return layouts;
|
|
181
|
+
}
|
|
182
|
+
var _redirectHistory = [];
|
|
183
|
+
var MAX_REDIRECTS = 10;
|
|
184
|
+
function Router({ routes, fallback, globalLayout }) {
|
|
185
|
+
return () => {
|
|
186
|
+
const currentUrl = _url();
|
|
187
|
+
const path = currentUrl.split("?")[0].split("#")[0];
|
|
188
|
+
const search = currentUrl.split("?")[1]?.split("#")[0] || "";
|
|
189
|
+
const isNavigating = _isNavigating();
|
|
190
|
+
const matched = matchRoute(path, routes);
|
|
191
|
+
if (matched) {
|
|
192
|
+
batch(() => {
|
|
193
|
+
_params.set(matched.params);
|
|
194
|
+
_query.set(parseQuery(search));
|
|
195
|
+
});
|
|
196
|
+
const { route: r, params } = matched;
|
|
197
|
+
const queryObj = parseQuery(search);
|
|
198
|
+
if (r.middleware && r.middleware.length > 0) {
|
|
199
|
+
for (const mw of r.middleware) {
|
|
200
|
+
const result = mw({ path, params, query: queryObj, route: r });
|
|
201
|
+
if (result === false) {
|
|
202
|
+
if (fallback) return h(fallback, {});
|
|
203
|
+
return h("div", { class: "what-403" }, h("h1", null, "403"), h("p", null, "Access denied"));
|
|
204
|
+
}
|
|
205
|
+
if (typeof result === "string") {
|
|
206
|
+
_redirectHistory.push(result);
|
|
207
|
+
if (_redirectHistory.length > MAX_REDIRECTS) {
|
|
208
|
+
const cycle = _redirectHistory.slice(-5).join(" \u2192 ");
|
|
209
|
+
_redirectHistory.length = 0;
|
|
210
|
+
console.error(`[what-router] Redirect loop detected: ${cycle}`);
|
|
211
|
+
_isNavigating.set(false);
|
|
212
|
+
return h(
|
|
213
|
+
"div",
|
|
214
|
+
{ class: "what-redirect-loop" },
|
|
215
|
+
h("h1", null, "Redirect Loop"),
|
|
216
|
+
h("p", null, "Too many redirects. Check your middleware configuration.")
|
|
217
|
+
);
|
|
218
|
+
}
|
|
219
|
+
const seen = /* @__PURE__ */ new Set();
|
|
220
|
+
let hasCycle = false;
|
|
221
|
+
for (const url of _redirectHistory) {
|
|
222
|
+
if (seen.has(url)) {
|
|
223
|
+
hasCycle = true;
|
|
224
|
+
break;
|
|
225
|
+
}
|
|
226
|
+
seen.add(url);
|
|
227
|
+
}
|
|
228
|
+
if (hasCycle) {
|
|
229
|
+
const cycle = _redirectHistory.join(" \u2192 ");
|
|
230
|
+
_redirectHistory.length = 0;
|
|
231
|
+
console.error(`[what-router] Redirect cycle detected: ${cycle}`);
|
|
232
|
+
_isNavigating.set(false);
|
|
233
|
+
return h(
|
|
234
|
+
"div",
|
|
235
|
+
{ class: "what-redirect-loop" },
|
|
236
|
+
h("h1", null, "Redirect Loop"),
|
|
237
|
+
h("p", null, "Circular redirect detected. Check your middleware configuration.")
|
|
238
|
+
);
|
|
239
|
+
}
|
|
240
|
+
navigate(result, { replace: true });
|
|
241
|
+
return null;
|
|
242
|
+
}
|
|
243
|
+
}
|
|
244
|
+
}
|
|
245
|
+
_redirectHistory.length = 0;
|
|
246
|
+
let element;
|
|
247
|
+
if (r.loading && isNavigating) {
|
|
248
|
+
element = h(r.loading, {});
|
|
249
|
+
} else {
|
|
250
|
+
element = h(r.component, {
|
|
251
|
+
params,
|
|
252
|
+
query: queryObj,
|
|
253
|
+
route: r
|
|
254
|
+
});
|
|
255
|
+
}
|
|
256
|
+
if (r.error) {
|
|
257
|
+
element = h(ErrorBoundary, { fallback: r.error }, element);
|
|
258
|
+
}
|
|
259
|
+
const layouts = buildLayoutChain(r, routes);
|
|
260
|
+
for (const Layout of layouts.reverse()) {
|
|
261
|
+
element = h(Layout, { params, query: queryObj }, element);
|
|
262
|
+
}
|
|
263
|
+
if (globalLayout) {
|
|
264
|
+
element = h(globalLayout, {}, element);
|
|
265
|
+
}
|
|
266
|
+
return element;
|
|
267
|
+
}
|
|
268
|
+
if (fallback) return h(fallback, {});
|
|
269
|
+
return h(
|
|
270
|
+
"div",
|
|
271
|
+
{ class: "what-404" },
|
|
272
|
+
h("h1", null, "404"),
|
|
273
|
+
h("p", null, "Page not found")
|
|
274
|
+
);
|
|
275
|
+
};
|
|
276
|
+
}
|
|
277
|
+
function Link({
|
|
278
|
+
href,
|
|
279
|
+
class: cls,
|
|
280
|
+
className,
|
|
281
|
+
children,
|
|
282
|
+
replace: rep,
|
|
283
|
+
prefetch: shouldPrefetch = true,
|
|
284
|
+
activeClass = "active",
|
|
285
|
+
exactActiveClass = "exact-active",
|
|
286
|
+
transition = true,
|
|
287
|
+
...rest
|
|
288
|
+
}) {
|
|
289
|
+
const safeHref = isSafeUrl(href) ? href : "about:blank";
|
|
290
|
+
if (!isSafeUrl(href) && typeof console !== "undefined") {
|
|
291
|
+
console.warn(`[what-router] Link blocked unsafe href: ${href}`);
|
|
292
|
+
}
|
|
293
|
+
const hrefPath = safeHref.split("?")[0].split("#")[0];
|
|
294
|
+
const reactiveClass = () => {
|
|
295
|
+
const currentPath = route.path;
|
|
296
|
+
const isActive = hrefPath === "/" ? currentPath === "/" : currentPath === hrefPath || currentPath.startsWith(hrefPath + "/");
|
|
297
|
+
const isExactActive = currentPath === hrefPath;
|
|
298
|
+
return [
|
|
299
|
+
cls || className,
|
|
300
|
+
isActive && activeClass,
|
|
301
|
+
isExactActive && exactActiveClass
|
|
302
|
+
].filter(Boolean).join(" ") || void 0;
|
|
303
|
+
};
|
|
304
|
+
return h("a", {
|
|
305
|
+
href: safeHref,
|
|
306
|
+
class: reactiveClass,
|
|
307
|
+
onclick: (e) => {
|
|
308
|
+
if (e.ctrlKey || e.metaKey || e.shiftKey || e.altKey || e.button !== 0) return;
|
|
309
|
+
e.preventDefault();
|
|
310
|
+
navigate(safeHref, { replace: rep, transition });
|
|
311
|
+
},
|
|
312
|
+
onmouseenter: shouldPrefetch ? () => prefetch(safeHref) : void 0,
|
|
313
|
+
...rest
|
|
314
|
+
}, ...Array.isArray(children) ? children : [children]);
|
|
315
|
+
}
|
|
316
|
+
function NavLink(props) {
|
|
317
|
+
return Link(props);
|
|
318
|
+
}
|
|
319
|
+
function defineRoutes(config) {
|
|
320
|
+
return Object.entries(config).map(([path, value]) => {
|
|
321
|
+
if (typeof value === "function") {
|
|
322
|
+
return { path, component: value };
|
|
323
|
+
}
|
|
324
|
+
return { path, ...value };
|
|
325
|
+
});
|
|
326
|
+
}
|
|
327
|
+
function nestedRoutes(basePath, children, options = {}) {
|
|
328
|
+
const { layout, loading, error } = options;
|
|
329
|
+
return children.map((child) => ({
|
|
330
|
+
...child,
|
|
331
|
+
path: basePath + child.path,
|
|
332
|
+
layout: child.layout || layout,
|
|
333
|
+
loading: child.loading || loading,
|
|
334
|
+
error: child.error || error
|
|
335
|
+
}));
|
|
336
|
+
}
|
|
337
|
+
function routeGroup(name, routes, options = {}) {
|
|
338
|
+
const { layout, middleware } = options;
|
|
339
|
+
return routes.map((route2) => ({
|
|
340
|
+
...route2,
|
|
341
|
+
_group: name,
|
|
342
|
+
layout: route2.layout || layout,
|
|
343
|
+
middleware: [...route2.middleware || [], ...middleware || []]
|
|
344
|
+
}));
|
|
345
|
+
}
|
|
346
|
+
function Redirect({ to }) {
|
|
347
|
+
navigate(to, { replace: true });
|
|
348
|
+
return null;
|
|
349
|
+
}
|
|
350
|
+
function guard(check, fallback) {
|
|
351
|
+
return (Component) => {
|
|
352
|
+
return function GuardedRoute(props) {
|
|
353
|
+
const result = check(props);
|
|
354
|
+
if (result instanceof Promise) {
|
|
355
|
+
return h("div", { class: "what-guard-loading" }, "Loading...");
|
|
356
|
+
}
|
|
357
|
+
if (result) {
|
|
358
|
+
return h(Component, props);
|
|
359
|
+
}
|
|
360
|
+
if (typeof fallback === "string") {
|
|
361
|
+
navigate(fallback, { replace: true });
|
|
362
|
+
return null;
|
|
363
|
+
}
|
|
364
|
+
return h(fallback, props);
|
|
365
|
+
};
|
|
366
|
+
};
|
|
367
|
+
}
|
|
368
|
+
function asyncGuard(check, options = {}) {
|
|
369
|
+
const { fallback = "/login", loading = null } = options;
|
|
370
|
+
return (Component) => {
|
|
371
|
+
return function AsyncGuardedRoute(props) {
|
|
372
|
+
const status = signal("pending");
|
|
373
|
+
const checkResult = signal(null);
|
|
374
|
+
let cancelled = false;
|
|
375
|
+
effect(() => {
|
|
376
|
+
cancelled = false;
|
|
377
|
+
Promise.resolve(check(props)).then((result) => {
|
|
378
|
+
if (cancelled) return;
|
|
379
|
+
checkResult.set(result);
|
|
380
|
+
status.set(result ? "allowed" : "denied");
|
|
381
|
+
}).catch(() => {
|
|
382
|
+
if (!cancelled) status.set("denied");
|
|
383
|
+
});
|
|
384
|
+
return () => {
|
|
385
|
+
cancelled = true;
|
|
386
|
+
};
|
|
387
|
+
});
|
|
388
|
+
return () => {
|
|
389
|
+
const currentStatus = status();
|
|
390
|
+
if (currentStatus === "pending") {
|
|
391
|
+
return loading ? h(loading, {}) : null;
|
|
392
|
+
}
|
|
393
|
+
if (currentStatus === "allowed") {
|
|
394
|
+
return h(Component, props);
|
|
395
|
+
}
|
|
396
|
+
if (typeof fallback === "string") {
|
|
397
|
+
navigate(fallback, { replace: true });
|
|
398
|
+
return null;
|
|
399
|
+
}
|
|
400
|
+
return h(fallback, props);
|
|
401
|
+
};
|
|
402
|
+
};
|
|
403
|
+
};
|
|
404
|
+
}
|
|
405
|
+
var prefetchedUrls = /* @__PURE__ */ new Set();
|
|
406
|
+
function prefetch(href) {
|
|
407
|
+
if (typeof document === "undefined") return;
|
|
408
|
+
if (prefetchedUrls.has(href)) return;
|
|
409
|
+
prefetchedUrls.add(href);
|
|
410
|
+
const link = document.createElement("link");
|
|
411
|
+
link.rel = "prefetch";
|
|
412
|
+
link.href = href;
|
|
413
|
+
document.head.appendChild(link);
|
|
414
|
+
}
|
|
415
|
+
var scrollPositions = /* @__PURE__ */ new Map();
|
|
416
|
+
function enableScrollRestoration() {
|
|
417
|
+
if (typeof window === "undefined") return;
|
|
418
|
+
window.addEventListener("beforeunload", () => {
|
|
419
|
+
scrollPositions.set(location.pathname, window.scrollY);
|
|
420
|
+
});
|
|
421
|
+
effect(() => {
|
|
422
|
+
const path = route.path;
|
|
423
|
+
const savedPosition = scrollPositions.get(path);
|
|
424
|
+
requestAnimationFrame(() => {
|
|
425
|
+
if (savedPosition !== void 0) {
|
|
426
|
+
window.scrollTo(0, savedPosition);
|
|
427
|
+
} else if (route.hash) {
|
|
428
|
+
const el = document.querySelector(route.hash);
|
|
429
|
+
el?.scrollIntoView();
|
|
430
|
+
} else {
|
|
431
|
+
window.scrollTo(0, 0);
|
|
432
|
+
}
|
|
433
|
+
});
|
|
434
|
+
});
|
|
435
|
+
}
|
|
436
|
+
function viewTransitionName(name) {
|
|
437
|
+
return { style: { viewTransitionName: name } };
|
|
438
|
+
}
|
|
439
|
+
function setViewTransition(type) {
|
|
440
|
+
if (typeof document === "undefined") return;
|
|
441
|
+
document.documentElement.dataset.transition = type;
|
|
442
|
+
}
|
|
443
|
+
function useRoute() {
|
|
444
|
+
return {
|
|
445
|
+
path: computed(() => route.path),
|
|
446
|
+
params: computed(() => route.params),
|
|
447
|
+
query: computed(() => route.query),
|
|
448
|
+
hash: computed(() => route.hash),
|
|
449
|
+
isNavigating: computed(() => route.isNavigating),
|
|
450
|
+
navigate,
|
|
451
|
+
prefetch
|
|
452
|
+
};
|
|
453
|
+
}
|
|
454
|
+
function Outlet({ children }) {
|
|
455
|
+
return children || null;
|
|
456
|
+
}
|
|
457
|
+
function FileRouter({
|
|
458
|
+
routes,
|
|
459
|
+
layout: globalLayout,
|
|
460
|
+
fallback,
|
|
461
|
+
error: globalError
|
|
462
|
+
}) {
|
|
463
|
+
const routerRoutes = routes.map((r) => ({
|
|
464
|
+
path: r.path,
|
|
465
|
+
component: r.component,
|
|
466
|
+
layout: r.layout || void 0,
|
|
467
|
+
// Attach page mode as metadata for build system
|
|
468
|
+
_mode: r.mode || "client"
|
|
469
|
+
}));
|
|
470
|
+
return Router({
|
|
471
|
+
routes: routerRoutes,
|
|
472
|
+
globalLayout,
|
|
473
|
+
fallback: fallback || Default404
|
|
474
|
+
});
|
|
475
|
+
}
|
|
476
|
+
function Default404() {
|
|
477
|
+
return h(
|
|
478
|
+
"div",
|
|
479
|
+
{ style: "text-align:center;padding:60px 20px" },
|
|
480
|
+
h("h1", { style: "font-size:48px;margin-bottom:8px" }, "404"),
|
|
481
|
+
h("p", { style: "color:#64748b" }, "Page not found")
|
|
482
|
+
);
|
|
483
|
+
}
|
|
484
|
+
export {
|
|
485
|
+
FileRouter,
|
|
486
|
+
Link,
|
|
487
|
+
NavLink,
|
|
488
|
+
Outlet,
|
|
489
|
+
Redirect,
|
|
490
|
+
Router,
|
|
491
|
+
asyncGuard,
|
|
492
|
+
defineRoutes,
|
|
493
|
+
enableScrollRestoration,
|
|
494
|
+
guard,
|
|
495
|
+
isSafeUrl,
|
|
496
|
+
navigate,
|
|
497
|
+
nestedRoutes,
|
|
498
|
+
prefetch,
|
|
499
|
+
route,
|
|
500
|
+
routeGroup,
|
|
501
|
+
setViewTransition,
|
|
502
|
+
useRoute,
|
|
503
|
+
viewTransitionName
|
|
504
|
+
};
|
|
505
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../src/index.js"],
|
|
4
|
+
"sourcesContent": ["// What Framework - Router\n// Production-grade file-based routing with nested layouts, loading states,\n// route groups, view transitions, and middleware.\n\nimport { signal, effect, computed, batch, h, ErrorBoundary } from 'what-core';\n\n// --- URL Sanitization ---\n// Rejects javascript:, data:, vbscript: protocols (case-insensitive, trimmed).\n\nexport function isSafeUrl(url) {\n if (typeof url !== 'string') return false;\n const trimmed = url.trim();\n // Check for dangerous protocols (case-insensitive, ignoring whitespace/control chars)\n const normalized = trimmed.replace(/[\\s\\x00-\\x1f]/g, '').toLowerCase();\n if (normalized.startsWith('javascript:')) return false;\n if (normalized.startsWith('data:')) return false;\n if (normalized.startsWith('vbscript:')) return false;\n return true;\n}\n\n// --- Route State (global singleton) ---\n\nconst _url = signal(typeof location !== 'undefined' ? location.pathname + location.search + location.hash : '/');\nconst _params = signal({});\nconst _query = signal({});\nconst _isNavigating = signal(false);\nconst _navigationError = signal(null);\n\nexport const route = {\n get url() { return _url(); },\n get path() { return _url().split('?')[0].split('#')[0]; },\n get params() { return _params(); },\n get query() { return _query(); },\n get hash() {\n const h = _url().split('#')[1];\n return h ? '#' + h : '';\n },\n get isNavigating() { return _isNavigating(); },\n get error() { return _navigationError(); },\n};\n\n// --- Navigation with View Transitions ---\n\nexport async function navigate(to, opts = {}) {\n const { replace = false, state = null, transition = true, _fromPopstate = false } = opts;\n\n // Reject unsafe URLs\n if (!isSafeUrl(to)) {\n if (typeof console !== 'undefined') {\n console.warn(`[what-router] Blocked navigation to unsafe URL: ${to}`);\n }\n return;\n }\n\n // Handle same-page hash links \u2014 use replaceState and scroll directly\n if (typeof window !== 'undefined' && to.startsWith('#')) {\n const currentUrl = _url();\n const basePath = currentUrl.split('#')[0];\n const newUrl = basePath + to;\n history.replaceState(state, '', newUrl);\n _url.set(newUrl);\n const el = document.querySelector(to);\n if (el) el.scrollIntoView({ behavior: 'smooth' });\n return;\n }\n\n // Don't navigate if already on the same URL\n if (to === _url()) return;\n\n // Prevent concurrent navigations \u2014 wait for current to finish\n if (_isNavigating.peek()) return;\n\n _isNavigating.set(true);\n _navigationError.set(null);\n\n const doNavigation = () => {\n // Skip history manipulation on popstate (browser already updated the URL)\n if (!_fromPopstate) {\n // Save scroll position for current URL before navigating away\n if (typeof window !== 'undefined') {\n scrollPositions.set(_url(), { x: scrollX, y: scrollY });\n }\n if (replace) {\n history.replaceState(state, '', to);\n } else {\n history.pushState(state, '', to);\n }\n }\n _url.set(to);\n _isNavigating.set(false);\n };\n\n // Use View Transitions API if available and enabled\n if (transition && typeof document !== 'undefined' && document.startViewTransition) {\n try {\n await document.startViewTransition(doNavigation).finished;\n } catch (e) {\n // Transition failed, navigation still happened\n }\n } else {\n doNavigation();\n }\n}\n\n// Back/forward support \u2014 route through navigate() so middleware runs\nif (typeof window !== 'undefined') {\n window.addEventListener('popstate', () => {\n // Save scroll position for the URL we're leaving\n scrollPositions.set(_url(), { x: scrollX, y: scrollY });\n\n const newUrl = location.pathname + location.search + location.hash;\n // Use _fromPopstate flag so navigate() skips pushState (browser already updated URL)\n navigate(newUrl, { replace: true, _fromPopstate: true, transition: false }).then(() => {\n // Restore saved scroll position for the URL we're arriving at\n const saved = scrollPositions.get(newUrl);\n if (saved) {\n requestAnimationFrame(() => window.scrollTo(saved.x, saved.y));\n }\n });\n });\n}\n\n// --- Route Matching ---\n\nfunction compilePath(path) {\n // /users/:id -> regex + param names\n // /posts/* -> catch-all\n // /[slug] -> dynamic (file-based syntax)\n // (group) -> route group (ignored in URL)\n\n // Remove route groups from path (they don't affect URL matching)\n const normalized = path\n .replace(/\\([\\w-]+\\)\\//g, '') // Remove (group)/ prefixes\n .replace(/\\[\\.\\.\\.(\\w+)\\]/g, (_, name) => `*:${name}`) // Preserve catch-all name\n .replace(/\\[(\\w+)\\]/g, ':$1'); // File-based [param] to :param\n\n const paramNames = [];\n let catchAll = null;\n\n const regexStr = normalized\n .split('/')\n .map(segment => {\n if (segment.startsWith('*:')) {\n catchAll = segment.slice(2);\n paramNames.push(catchAll);\n return '(.+)';\n }\n if (segment === '*') {\n catchAll = 'rest';\n paramNames.push('rest');\n return '(.+)';\n }\n if (segment.startsWith(':')) {\n paramNames.push(segment.slice(1));\n return '([^/]+)';\n }\n return segment.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&');\n })\n .join('/');\n\n const regex = new RegExp(`^${regexStr}$`);\n return { regex, paramNames, catchAll };\n}\n\nfunction matchRoute(path, routes) {\n // Filter out routes without a path (layout-only routes, etc.)\n const routable = routes.filter(r => r.path);\n\n // Sort routes by specificity (more specific first)\n const sorted = routable.sort((a, b) => {\n const aSpecific = (a.path.match(/:/g) || []).length + (a.path.includes('*') ? 100 : 0);\n const bSpecific = (b.path.match(/:/g) || []).length + (b.path.includes('*') ? 100 : 0);\n return aSpecific - bSpecific;\n });\n\n for (const route of sorted) {\n const { regex, paramNames } = compilePath(route.path);\n const match = path.match(regex);\n if (match) {\n const params = {};\n paramNames.forEach((name, i) => {\n params[name] = decodeURIComponent(match[i + 1]);\n });\n return { route, params };\n }\n }\n return null;\n}\n\nfunction parseQuery(search) {\n const params = {};\n if (!search) return params;\n const qs = search.startsWith('?') ? search.slice(1) : search;\n for (const pair of qs.split('&')) {\n const [key, val] = pair.split('=');\n if (!key) continue;\n const decodedKey = decodeURIComponent(key);\n const decodedVal = val ? decodeURIComponent(val) : '';\n if (decodedKey in params) {\n // Collect repeated keys into arrays\n if (Array.isArray(params[decodedKey])) {\n params[decodedKey].push(decodedVal);\n } else {\n params[decodedKey] = [params[decodedKey], decodedVal];\n }\n } else {\n params[decodedKey] = decodedVal;\n }\n }\n return params;\n}\n\n// --- Nested Layouts ---\n\n// Build the layout chain for a route\nfunction buildLayoutChain(route, routes) {\n const layouts = [];\n if (!route.path) return layouts;\n\n // Check for nested layouts based on path segments\n const segments = route.path.split('/').filter(Boolean);\n let currentPath = '';\n\n for (const segment of segments) {\n currentPath += '/' + segment;\n\n // Find layout for this path level\n const layoutRoute = routes.find(r =>\n r.layout && r.path === currentPath + '/_layout'\n );\n if (layoutRoute) {\n layouts.push(layoutRoute.layout);\n }\n }\n\n // Add route's own layout if specified\n if (route.layout) {\n layouts.push(route.layout);\n }\n\n return layouts;\n}\n\n// --- Middleware redirect loop detection ---\nconst _redirectHistory = [];\nconst MAX_REDIRECTS = 10;\n\n// --- Router Component ---\n\nexport function Router({ routes, fallback, globalLayout }) {\n // Return a reactive function child. The Router component runs ONCE,\n // but the returned function re-evaluates whenever _url changes,\n // and the fine-grained runtime updates the DOM accordingly.\n return () => {\n const currentUrl = _url();\n const path = currentUrl.split('?')[0].split('#')[0];\n const search = currentUrl.split('?')[1]?.split('#')[0] || '';\n const isNavigating = _isNavigating();\n\n const matched = matchRoute(path, routes);\n\n if (matched) {\n batch(() => {\n _params.set(matched.params);\n _query.set(parseQuery(search));\n });\n\n const { route: r, params } = matched;\n const queryObj = parseQuery(search);\n\n // Run middleware (sync only \u2014 async middleware should use asyncGuard)\n if (r.middleware && r.middleware.length > 0) {\n for (const mw of r.middleware) {\n const result = mw({ path, params, query: queryObj, route: r });\n if (result === false) {\n // Middleware rejected \u2014 show fallback\n if (fallback) return h(fallback, {});\n return h('div', { class: 'what-403' }, h('h1', null, '403'), h('p', null, 'Access denied'));\n }\n if (typeof result === 'string') {\n // Redirect loop detection\n _redirectHistory.push(result);\n if (_redirectHistory.length > MAX_REDIRECTS) {\n const cycle = _redirectHistory.slice(-5).join(' \u2192 ');\n _redirectHistory.length = 0;\n console.error(`[what-router] Redirect loop detected: ${cycle}`);\n _isNavigating.set(false);\n return h('div', { class: 'what-redirect-loop' },\n h('h1', null, 'Redirect Loop'),\n h('p', null, 'Too many redirects. Check your middleware configuration.')\n );\n }\n // Check for direct cycle (A \u2192 B \u2192 A)\n const seen = new Set();\n let hasCycle = false;\n for (const url of _redirectHistory) {\n if (seen.has(url)) { hasCycle = true; break; }\n seen.add(url);\n }\n if (hasCycle) {\n const cycle = _redirectHistory.join(' \u2192 ');\n _redirectHistory.length = 0;\n console.error(`[what-router] Redirect cycle detected: ${cycle}`);\n _isNavigating.set(false);\n return h('div', { class: 'what-redirect-loop' },\n h('h1', null, 'Redirect Loop'),\n h('p', null, 'Circular redirect detected. Check your middleware configuration.')\n );\n }\n // Middleware returned a redirect path\n navigate(result, { replace: true });\n return null;\n }\n }\n }\n // Successful render \u2014 clear redirect history\n _redirectHistory.length = 0;\n\n // Build element with loading state support\n let element;\n\n if (r.loading && isNavigating) {\n element = h(r.loading, {});\n } else {\n element = h(r.component, {\n params,\n query: queryObj,\n route: r,\n });\n }\n\n // Wrap with per-route error boundary if specified\n if (r.error) {\n element = h(ErrorBoundary, { fallback: r.error }, element);\n }\n\n // Wrap with nested layouts (innermost to outermost)\n const layouts = buildLayoutChain(r, routes);\n for (const Layout of layouts.reverse()) {\n element = h(Layout, { params, query: queryObj }, element);\n }\n\n // Global layout wrapper\n if (globalLayout) {\n element = h(globalLayout, {}, element);\n }\n\n return element;\n }\n\n // 404\n if (fallback) return h(fallback, {});\n return h('div', { class: 'what-404' },\n h('h1', null, '404'),\n h('p', null, 'Page not found')\n );\n };\n}\n\n// --- Link Component ---\n\nexport function Link({\n href,\n class: cls,\n className,\n children,\n replace: rep,\n prefetch: shouldPrefetch = true,\n activeClass = 'active',\n exactActiveClass = 'exact-active',\n transition = true,\n ...rest\n}) {\n // Sanitize href \u2014 reject dangerous protocols\n const safeHref = isSafeUrl(href) ? href : 'about:blank';\n if (!isSafeUrl(href) && typeof console !== 'undefined') {\n console.warn(`[what-router] Link blocked unsafe href: ${href}`);\n }\n\n // Strip query string and hash from href for path comparison\n const hrefPath = safeHref.split('?')[0].split('#')[0];\n\n // Use a reactive function for class so active states update on navigation.\n // In the run-once model, reading route.path directly would snapshot it.\n const reactiveClass = () => {\n const currentPath = route.path;\n const isActive = hrefPath === '/'\n ? currentPath === '/'\n : currentPath === hrefPath || currentPath.startsWith(hrefPath + '/');\n const isExactActive = currentPath === hrefPath;\n\n return [\n cls || className,\n isActive && activeClass,\n isExactActive && exactActiveClass,\n ].filter(Boolean).join(' ') || undefined;\n };\n\n return h('a', {\n href: safeHref,\n class: reactiveClass,\n onclick: (e) => {\n // Only intercept left-clicks without modifiers\n if (e.ctrlKey || e.metaKey || e.shiftKey || e.altKey || e.button !== 0) return;\n e.preventDefault();\n navigate(safeHref, { replace: rep, transition });\n },\n onmouseenter: shouldPrefetch ? () => prefetch(safeHref) : undefined,\n ...rest,\n }, ...(Array.isArray(children) ? children : [children]));\n}\n\n// --- NavLink with active states ---\n\nexport function NavLink(props) {\n return Link(props);\n}\n\n// --- Define Routes Helper ---\n// Creates route config from a flat object for convenience.\n\nexport function defineRoutes(config) {\n return Object.entries(config).map(([path, value]) => {\n if (typeof value === 'function') {\n return { path, component: value };\n }\n // Object form with layout, middleware, loading, error, etc.\n return { path, ...value };\n });\n}\n\n// --- Nested Route Helper ---\n\nexport function nestedRoutes(basePath, children, options = {}) {\n const { layout, loading, error } = options;\n\n return children.map(child => ({\n ...child,\n path: basePath + child.path,\n layout: child.layout || layout,\n loading: child.loading || loading,\n error: child.error || error,\n }));\n}\n\n// --- Route Groups ---\n// Group routes without affecting URL structure\n\nexport function routeGroup(name, routes, options = {}) {\n const { layout, middleware } = options;\n\n return routes.map(route => ({\n ...route,\n _group: name,\n layout: route.layout || layout,\n middleware: [...(route.middleware || []), ...(middleware || [])],\n }));\n}\n\n// --- Redirect ---\n\nexport function Redirect({ to }) {\n navigate(to, { replace: true });\n return null;\n}\n\n// --- Route Guards / Middleware ---\n\nexport function guard(check, fallback) {\n return (Component) => {\n return function GuardedRoute(props) {\n const result = check(props);\n\n // Support async guards\n if (result instanceof Promise) {\n // Return loading while checking\n return h('div', { class: 'what-guard-loading' }, 'Loading...');\n }\n\n if (result) {\n return h(Component, props);\n }\n\n if (typeof fallback === 'string') {\n navigate(fallback, { replace: true });\n return null;\n }\n return h(fallback, props);\n };\n };\n}\n\n// Async guard with suspense\nexport function asyncGuard(check, options = {}) {\n const { fallback = '/login', loading = null } = options;\n\n return (Component) => {\n return function AsyncGuardedRoute(props) {\n const status = signal('pending');\n const checkResult = signal(null);\n let cancelled = false;\n\n effect(() => {\n cancelled = false;\n Promise.resolve(check(props))\n .then(result => {\n if (cancelled) return;\n checkResult.set(result);\n status.set(result ? 'allowed' : 'denied');\n })\n .catch(() => {\n if (!cancelled) status.set('denied');\n });\n return () => { cancelled = true; };\n });\n\n // Return a reactive function child so status changes update the DOM.\n // Components run once, so reading status() outside a reactive wrapper\n // would snapshot the value and never update.\n return () => {\n const currentStatus = status();\n\n if (currentStatus === 'pending') {\n return loading ? h(loading, {}) : null;\n }\n\n if (currentStatus === 'allowed') {\n return h(Component, props);\n }\n\n if (typeof fallback === 'string') {\n navigate(fallback, { replace: true });\n return null;\n }\n return h(fallback, props);\n };\n };\n };\n}\n\n// --- Prefetch ---\n// Hint the browser to prefetch a route's assets.\n\nconst prefetchedUrls = new Set();\n\nexport function prefetch(href) {\n if (typeof document === 'undefined') return;\n if (prefetchedUrls.has(href)) return;\n prefetchedUrls.add(href);\n\n const link = document.createElement('link');\n link.rel = 'prefetch';\n link.href = href;\n document.head.appendChild(link);\n}\n\n// --- Scroll Restoration ---\n\nconst scrollPositions = new Map();\n\nexport function enableScrollRestoration() {\n if (typeof window === 'undefined') return;\n\n // Save scroll position before navigation\n window.addEventListener('beforeunload', () => {\n scrollPositions.set(location.pathname, window.scrollY);\n });\n\n // Restore scroll position after navigation\n effect(() => {\n const path = route.path;\n const savedPosition = scrollPositions.get(path);\n\n requestAnimationFrame(() => {\n if (savedPosition !== undefined) {\n window.scrollTo(0, savedPosition);\n } else if (route.hash) {\n const el = document.querySelector(route.hash);\n el?.scrollIntoView();\n } else {\n window.scrollTo(0, 0);\n }\n });\n });\n}\n\n// --- View Transition Helpers ---\n\nexport function viewTransitionName(name) {\n return { style: { viewTransitionName: name } };\n}\n\n// Configure view transition types\nexport function setViewTransition(type) {\n if (typeof document === 'undefined') return;\n document.documentElement.dataset.transition = type;\n}\n\n// --- useRoute Hook ---\n\nexport function useRoute() {\n return {\n path: computed(() => route.path),\n params: computed(() => route.params),\n query: computed(() => route.query),\n hash: computed(() => route.hash),\n isNavigating: computed(() => route.isNavigating),\n navigate,\n prefetch,\n };\n}\n\n// --- Outlet Component ---\n// For nested route rendering\n\nexport function Outlet({ children }) {\n // Children passed from parent layout\n return children || null;\n}\n\n// --- File-Based Router ---\n// Consumes routes generated by what-compiler's file router (virtual:what-routes).\n// Usage:\n// import { routes } from 'virtual:what-routes';\n// mount(<FileRouter routes={routes} />, '#app');\n\nexport function FileRouter({\n routes,\n layout: globalLayout,\n fallback,\n error: globalError,\n}) {\n // Convert file-router route format to Router's expected format\n const routerRoutes = routes.map(r => ({\n path: r.path,\n component: r.component,\n layout: r.layout || undefined,\n // Attach page mode as metadata for build system\n _mode: r.mode || 'client',\n }));\n\n // Router already returns a reactive function child \u2014 just delegate\n return Router({\n routes: routerRoutes,\n globalLayout,\n fallback: fallback || Default404,\n });\n}\n\nfunction Default404() {\n return h('div', { style: 'text-align:center;padding:60px 20px' },\n h('h1', { style: 'font-size:48px;margin-bottom:8px' }, '404'),\n h('p', { style: 'color:#64748b' }, 'Page not found'),\n );\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
|
+
"names": ["h", "route"]
|
|
7
|
+
}
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
import{signal as w,effect as _,computed as b,batch as $,h as a,ErrorBoundary as W}from"what-core";function k(e){if(typeof e!="string")return!1;let n=e.trim().replace(/[\s\x00-\x1f]/g,"").toLowerCase();return!(n.startsWith("javascript:")||n.startsWith("data:")||n.startsWith("vbscript:"))}var h=w(typeof location<"u"?location.pathname+location.search+location.hash:"/"),L=w({}),N=w({}),g=w(!1),U=w(null),m={get url(){return h()},get path(){return h().split("?")[0].split("#")[0]},get params(){return L()},get query(){return N()},get hash(){let e=h().split("#")[1];return e?"#"+e:""},get isNavigating(){return g()},get error(){return U()}};async function x(e,t={}){let{replace:n=!1,state:s=null,transition:o=!0,_fromPopstate:i=!1}=t;if(!k(e)){typeof console<"u"&&console.warn(`[what-router] Blocked navigation to unsafe URL: ${e}`);return}if(typeof window<"u"&&e.startsWith("#")){let f=h().split("#")[0]+e;history.replaceState(s,"",f),h.set(f);let l=document.querySelector(e);l&&l.scrollIntoView({behavior:"smooth"});return}if(e===h()||g.peek())return;g.set(!0),U.set(null);let r=()=>{i||(typeof window<"u"&&P.set(h(),{x:scrollX,y:scrollY}),n?history.replaceState(s,"",e):history.pushState(s,"",e)),h.set(e),g.set(!1)};if(o&&typeof document<"u"&&document.startViewTransition)try{await document.startViewTransition(r).finished}catch{}else r()}typeof window<"u"&&window.addEventListener("popstate",()=>{P.set(h(),{x:scrollX,y:scrollY});let e=location.pathname+location.search+location.hash;x(e,{replace:!0,_fromPopstate:!0,transition:!1}).then(()=>{let t=P.get(e);t&&requestAnimationFrame(()=>window.scrollTo(t.x,t.y))})});function j(e){let t=e.replace(/\([\w-]+\)\//g,"").replace(/\[\.\.\.(\w+)\]/g,(r,c)=>`*:${c}`).replace(/\[(\w+)\]/g,":$1"),n=[],s=null,o=t.split("/").map(r=>r.startsWith("*:")?(s=r.slice(2),n.push(s),"(.+)"):r==="*"?(s="rest",n.push("rest"),"(.+)"):r.startsWith(":")?(n.push(r.slice(1)),"([^/]+)"):r.replace(/[.*+?^${}()|[\]\\]/g,"\\$&")).join("/");return{regex:new RegExp(`^${o}$`),paramNames:n,catchAll:s}}function I(e,t){let s=t.filter(o=>o.path).sort((o,i)=>{let r=(o.path.match(/:/g)||[]).length+(o.path.includes("*")?100:0),c=(i.path.match(/:/g)||[]).length+(i.path.includes("*")?100:0);return r-c});for(let o of s){let{regex:i,paramNames:r}=j(o.path),c=e.match(i);if(c){let u={};return r.forEach((f,l)=>{u[f]=decodeURIComponent(c[l+1])}),{route:o,params:u}}}return null}function C(e){let t={};if(!e)return t;let n=e.startsWith("?")?e.slice(1):e;for(let s of n.split("&")){let[o,i]=s.split("=");if(!o)continue;let r=decodeURIComponent(o),c=i?decodeURIComponent(i):"";r in t?Array.isArray(t[r])?t[r].push(c):t[r]=[t[r],c]:t[r]=c}return t}function V(e,t){let n=[];if(!e.path)return n;let s=e.path.split("/").filter(Boolean),o="";for(let i of s){o+="/"+i;let r=t.find(c=>c.layout&&c.path===o+"/_layout");r&&n.push(r.layout)}return e.layout&&n.push(e.layout),n}var y=[],K=10;function B({routes:e,fallback:t,globalLayout:n}){return()=>{let s=h(),o=s.split("?")[0].split("#")[0],i=s.split("?")[1]?.split("#")[0]||"",r=g(),c=I(o,e);if(c){$(()=>{L.set(c.params),N.set(C(i))});let{route:u,params:f}=c,l=C(i);if(u.middleware&&u.middleware.length>0)for(let d of u.middleware){let v=d({path:o,params:f,query:l,route:u});if(v===!1)return t?a(t,{}):a("div",{class:"what-403"},a("h1",null,"403"),a("p",null,"Access denied"));if(typeof v=="string"){if(y.push(v),y.length>K){let R=y.slice(-5).join(" \u2192 ");return y.length=0,console.error(`[what-router] Redirect loop detected: ${R}`),g.set(!1),a("div",{class:"what-redirect-loop"},a("h1",null,"Redirect Loop"),a("p",null,"Too many redirects. Check your middleware configuration."))}let S=new Set,A=!1;for(let R of y){if(S.has(R)){A=!0;break}S.add(R)}if(A){let R=y.join(" \u2192 ");return y.length=0,console.error(`[what-router] Redirect cycle detected: ${R}`),g.set(!1),a("div",{class:"what-redirect-loop"},a("h1",null,"Redirect Loop"),a("p",null,"Circular redirect detected. Check your middleware configuration."))}return x(v,{replace:!0}),null}}y.length=0;let p;u.loading&&r?p=a(u.loading,{}):p=a(u.component,{params:f,query:l,route:u}),u.error&&(p=a(W,{fallback:u.error},p));let q=V(u,e);for(let d of q.reverse())p=a(d,{params:f,query:l},p);return n&&(p=a(n,{},p)),p}return t?a(t,{}):a("div",{class:"what-404"},a("h1",null,"404"),a("p",null,"Page not found"))}}function G({href:e,class:t,className:n,children:s,replace:o,prefetch:i=!0,activeClass:r="active",exactActiveClass:c="exact-active",transition:u=!0,...f}){let l=k(e)?e:"about:blank";!k(e)&&typeof console<"u"&&console.warn(`[what-router] Link blocked unsafe href: ${e}`);let p=l.split("?")[0].split("#")[0];return a("a",{href:l,class:()=>{let d=m.path,v=p==="/"?d==="/":d===p||d.startsWith(p+"/");return[t||n,v&&r,d===p&&c].filter(Boolean).join(" ")||void 0},onclick:d=>{d.ctrlKey||d.metaKey||d.shiftKey||d.altKey||d.button!==0||(d.preventDefault(),x(l,{replace:o,transition:u}))},onmouseenter:i?()=>T(l):void 0,...f},...Array.isArray(s)?s:[s])}function F(e){return G(e)}function O(e){return Object.entries(e).map(([t,n])=>typeof n=="function"?{path:t,component:n}:{path:t,...n})}function X(e,t,n={}){let{layout:s,loading:o,error:i}=n;return t.map(r=>({...r,path:e+r.path,layout:r.layout||s,loading:r.loading||o,error:r.error||i}))}function Y(e,t,n={}){let{layout:s,middleware:o}=n;return t.map(i=>({...i,_group:e,layout:i.layout||s,middleware:[...i.middleware||[],...o||[]]}))}function H({to:e}){return x(e,{replace:!0}),null}function M(e,t){return n=>function(o){let i=e(o);return i instanceof Promise?a("div",{class:"what-guard-loading"},"Loading..."):i?a(n,o):typeof t=="string"?(x(t,{replace:!0}),null):a(t,o)}}function Q(e,t={}){let{fallback:n="/login",loading:s=null}=t;return o=>function(r){let c=w("pending"),u=w(null),f=!1;return _(()=>(f=!1,Promise.resolve(e(r)).then(l=>{f||(u.set(l),c.set(l?"allowed":"denied"))}).catch(()=>{f||c.set("denied")}),()=>{f=!0})),()=>{let l=c();return l==="pending"?s?a(s,{}):null:l==="allowed"?a(o,r):typeof n=="string"?(x(n,{replace:!0}),null):a(n,r)}}}var E=new Set;function T(e){if(typeof document>"u"||E.has(e))return;E.add(e);let t=document.createElement("link");t.rel="prefetch",t.href=e,document.head.appendChild(t)}var P=new Map;function J(){typeof window>"u"||(window.addEventListener("beforeunload",()=>{P.set(location.pathname,window.scrollY)}),_(()=>{let e=m.path,t=P.get(e);requestAnimationFrame(()=>{t!==void 0?window.scrollTo(0,t):m.hash?document.querySelector(m.hash)?.scrollIntoView():window.scrollTo(0,0)})}))}function Z(e){return{style:{viewTransitionName:e}}}function ee(e){typeof document>"u"||(document.documentElement.dataset.transition=e)}function te(){return{path:b(()=>m.path),params:b(()=>m.params),query:b(()=>m.query),hash:b(()=>m.hash),isNavigating:b(()=>m.isNavigating),navigate:x,prefetch:T}}function ne({children:e}){return e||null}function re({routes:e,layout:t,fallback:n,error:s}){let o=e.map(i=>({path:i.path,component:i.component,layout:i.layout||void 0,_mode:i.mode||"client"}));return B({routes:o,globalLayout:t,fallback:n||z})}function z(){return a("div",{style:"text-align:center;padding:60px 20px"},a("h1",{style:"font-size:48px;margin-bottom:8px"},"404"),a("p",{style:"color:#64748b"},"Page not found"))}export{re as FileRouter,G as Link,F as NavLink,ne as Outlet,H as Redirect,B as Router,Q as asyncGuard,O as defineRoutes,J as enableScrollRestoration,M as guard,k as isSafeUrl,x as navigate,X as nestedRoutes,T as prefetch,m as route,Y as routeGroup,ee as setViewTransition,te as useRoute,Z as viewTransitionName};
|
|
2
|
+
//# sourceMappingURL=index.min.js.map
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../src/index.js"],
|
|
4
|
+
"sourcesContent": ["// What Framework - Router\n// Production-grade file-based routing with nested layouts, loading states,\n// route groups, view transitions, and middleware.\n\nimport { signal, effect, computed, batch, h, ErrorBoundary } from 'what-core';\n\n// --- URL Sanitization ---\n// Rejects javascript:, data:, vbscript: protocols (case-insensitive, trimmed).\n\nexport function isSafeUrl(url) {\n if (typeof url !== 'string') return false;\n const trimmed = url.trim();\n // Check for dangerous protocols (case-insensitive, ignoring whitespace/control chars)\n const normalized = trimmed.replace(/[\\s\\x00-\\x1f]/g, '').toLowerCase();\n if (normalized.startsWith('javascript:')) return false;\n if (normalized.startsWith('data:')) return false;\n if (normalized.startsWith('vbscript:')) return false;\n return true;\n}\n\n// --- Route State (global singleton) ---\n\nconst _url = signal(typeof location !== 'undefined' ? location.pathname + location.search + location.hash : '/');\nconst _params = signal({});\nconst _query = signal({});\nconst _isNavigating = signal(false);\nconst _navigationError = signal(null);\n\nexport const route = {\n get url() { return _url(); },\n get path() { return _url().split('?')[0].split('#')[0]; },\n get params() { return _params(); },\n get query() { return _query(); },\n get hash() {\n const h = _url().split('#')[1];\n return h ? '#' + h : '';\n },\n get isNavigating() { return _isNavigating(); },\n get error() { return _navigationError(); },\n};\n\n// --- Navigation with View Transitions ---\n\nexport async function navigate(to, opts = {}) {\n const { replace = false, state = null, transition = true, _fromPopstate = false } = opts;\n\n // Reject unsafe URLs\n if (!isSafeUrl(to)) {\n if (typeof console !== 'undefined') {\n console.warn(`[what-router] Blocked navigation to unsafe URL: ${to}`);\n }\n return;\n }\n\n // Handle same-page hash links \u2014 use replaceState and scroll directly\n if (typeof window !== 'undefined' && to.startsWith('#')) {\n const currentUrl = _url();\n const basePath = currentUrl.split('#')[0];\n const newUrl = basePath + to;\n history.replaceState(state, '', newUrl);\n _url.set(newUrl);\n const el = document.querySelector(to);\n if (el) el.scrollIntoView({ behavior: 'smooth' });\n return;\n }\n\n // Don't navigate if already on the same URL\n if (to === _url()) return;\n\n // Prevent concurrent navigations \u2014 wait for current to finish\n if (_isNavigating.peek()) return;\n\n _isNavigating.set(true);\n _navigationError.set(null);\n\n const doNavigation = () => {\n // Skip history manipulation on popstate (browser already updated the URL)\n if (!_fromPopstate) {\n // Save scroll position for current URL before navigating away\n if (typeof window !== 'undefined') {\n scrollPositions.set(_url(), { x: scrollX, y: scrollY });\n }\n if (replace) {\n history.replaceState(state, '', to);\n } else {\n history.pushState(state, '', to);\n }\n }\n _url.set(to);\n _isNavigating.set(false);\n };\n\n // Use View Transitions API if available and enabled\n if (transition && typeof document !== 'undefined' && document.startViewTransition) {\n try {\n await document.startViewTransition(doNavigation).finished;\n } catch (e) {\n // Transition failed, navigation still happened\n }\n } else {\n doNavigation();\n }\n}\n\n// Back/forward support \u2014 route through navigate() so middleware runs\nif (typeof window !== 'undefined') {\n window.addEventListener('popstate', () => {\n // Save scroll position for the URL we're leaving\n scrollPositions.set(_url(), { x: scrollX, y: scrollY });\n\n const newUrl = location.pathname + location.search + location.hash;\n // Use _fromPopstate flag so navigate() skips pushState (browser already updated URL)\n navigate(newUrl, { replace: true, _fromPopstate: true, transition: false }).then(() => {\n // Restore saved scroll position for the URL we're arriving at\n const saved = scrollPositions.get(newUrl);\n if (saved) {\n requestAnimationFrame(() => window.scrollTo(saved.x, saved.y));\n }\n });\n });\n}\n\n// --- Route Matching ---\n\nfunction compilePath(path) {\n // /users/:id -> regex + param names\n // /posts/* -> catch-all\n // /[slug] -> dynamic (file-based syntax)\n // (group) -> route group (ignored in URL)\n\n // Remove route groups from path (they don't affect URL matching)\n const normalized = path\n .replace(/\\([\\w-]+\\)\\//g, '') // Remove (group)/ prefixes\n .replace(/\\[\\.\\.\\.(\\w+)\\]/g, (_, name) => `*:${name}`) // Preserve catch-all name\n .replace(/\\[(\\w+)\\]/g, ':$1'); // File-based [param] to :param\n\n const paramNames = [];\n let catchAll = null;\n\n const regexStr = normalized\n .split('/')\n .map(segment => {\n if (segment.startsWith('*:')) {\n catchAll = segment.slice(2);\n paramNames.push(catchAll);\n return '(.+)';\n }\n if (segment === '*') {\n catchAll = 'rest';\n paramNames.push('rest');\n return '(.+)';\n }\n if (segment.startsWith(':')) {\n paramNames.push(segment.slice(1));\n return '([^/]+)';\n }\n return segment.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&');\n })\n .join('/');\n\n const regex = new RegExp(`^${regexStr}$`);\n return { regex, paramNames, catchAll };\n}\n\nfunction matchRoute(path, routes) {\n // Filter out routes without a path (layout-only routes, etc.)\n const routable = routes.filter(r => r.path);\n\n // Sort routes by specificity (more specific first)\n const sorted = routable.sort((a, b) => {\n const aSpecific = (a.path.match(/:/g) || []).length + (a.path.includes('*') ? 100 : 0);\n const bSpecific = (b.path.match(/:/g) || []).length + (b.path.includes('*') ? 100 : 0);\n return aSpecific - bSpecific;\n });\n\n for (const route of sorted) {\n const { regex, paramNames } = compilePath(route.path);\n const match = path.match(regex);\n if (match) {\n const params = {};\n paramNames.forEach((name, i) => {\n params[name] = decodeURIComponent(match[i + 1]);\n });\n return { route, params };\n }\n }\n return null;\n}\n\nfunction parseQuery(search) {\n const params = {};\n if (!search) return params;\n const qs = search.startsWith('?') ? search.slice(1) : search;\n for (const pair of qs.split('&')) {\n const [key, val] = pair.split('=');\n if (!key) continue;\n const decodedKey = decodeURIComponent(key);\n const decodedVal = val ? decodeURIComponent(val) : '';\n if (decodedKey in params) {\n // Collect repeated keys into arrays\n if (Array.isArray(params[decodedKey])) {\n params[decodedKey].push(decodedVal);\n } else {\n params[decodedKey] = [params[decodedKey], decodedVal];\n }\n } else {\n params[decodedKey] = decodedVal;\n }\n }\n return params;\n}\n\n// --- Nested Layouts ---\n\n// Build the layout chain for a route\nfunction buildLayoutChain(route, routes) {\n const layouts = [];\n if (!route.path) return layouts;\n\n // Check for nested layouts based on path segments\n const segments = route.path.split('/').filter(Boolean);\n let currentPath = '';\n\n for (const segment of segments) {\n currentPath += '/' + segment;\n\n // Find layout for this path level\n const layoutRoute = routes.find(r =>\n r.layout && r.path === currentPath + '/_layout'\n );\n if (layoutRoute) {\n layouts.push(layoutRoute.layout);\n }\n }\n\n // Add route's own layout if specified\n if (route.layout) {\n layouts.push(route.layout);\n }\n\n return layouts;\n}\n\n// --- Middleware redirect loop detection ---\nconst _redirectHistory = [];\nconst MAX_REDIRECTS = 10;\n\n// --- Router Component ---\n\nexport function Router({ routes, fallback, globalLayout }) {\n // Return a reactive function child. The Router component runs ONCE,\n // but the returned function re-evaluates whenever _url changes,\n // and the fine-grained runtime updates the DOM accordingly.\n return () => {\n const currentUrl = _url();\n const path = currentUrl.split('?')[0].split('#')[0];\n const search = currentUrl.split('?')[1]?.split('#')[0] || '';\n const isNavigating = _isNavigating();\n\n const matched = matchRoute(path, routes);\n\n if (matched) {\n batch(() => {\n _params.set(matched.params);\n _query.set(parseQuery(search));\n });\n\n const { route: r, params } = matched;\n const queryObj = parseQuery(search);\n\n // Run middleware (sync only \u2014 async middleware should use asyncGuard)\n if (r.middleware && r.middleware.length > 0) {\n for (const mw of r.middleware) {\n const result = mw({ path, params, query: queryObj, route: r });\n if (result === false) {\n // Middleware rejected \u2014 show fallback\n if (fallback) return h(fallback, {});\n return h('div', { class: 'what-403' }, h('h1', null, '403'), h('p', null, 'Access denied'));\n }\n if (typeof result === 'string') {\n // Redirect loop detection\n _redirectHistory.push(result);\n if (_redirectHistory.length > MAX_REDIRECTS) {\n const cycle = _redirectHistory.slice(-5).join(' \u2192 ');\n _redirectHistory.length = 0;\n console.error(`[what-router] Redirect loop detected: ${cycle}`);\n _isNavigating.set(false);\n return h('div', { class: 'what-redirect-loop' },\n h('h1', null, 'Redirect Loop'),\n h('p', null, 'Too many redirects. Check your middleware configuration.')\n );\n }\n // Check for direct cycle (A \u2192 B \u2192 A)\n const seen = new Set();\n let hasCycle = false;\n for (const url of _redirectHistory) {\n if (seen.has(url)) { hasCycle = true; break; }\n seen.add(url);\n }\n if (hasCycle) {\n const cycle = _redirectHistory.join(' \u2192 ');\n _redirectHistory.length = 0;\n console.error(`[what-router] Redirect cycle detected: ${cycle}`);\n _isNavigating.set(false);\n return h('div', { class: 'what-redirect-loop' },\n h('h1', null, 'Redirect Loop'),\n h('p', null, 'Circular redirect detected. Check your middleware configuration.')\n );\n }\n // Middleware returned a redirect path\n navigate(result, { replace: true });\n return null;\n }\n }\n }\n // Successful render \u2014 clear redirect history\n _redirectHistory.length = 0;\n\n // Build element with loading state support\n let element;\n\n if (r.loading && isNavigating) {\n element = h(r.loading, {});\n } else {\n element = h(r.component, {\n params,\n query: queryObj,\n route: r,\n });\n }\n\n // Wrap with per-route error boundary if specified\n if (r.error) {\n element = h(ErrorBoundary, { fallback: r.error }, element);\n }\n\n // Wrap with nested layouts (innermost to outermost)\n const layouts = buildLayoutChain(r, routes);\n for (const Layout of layouts.reverse()) {\n element = h(Layout, { params, query: queryObj }, element);\n }\n\n // Global layout wrapper\n if (globalLayout) {\n element = h(globalLayout, {}, element);\n }\n\n return element;\n }\n\n // 404\n if (fallback) return h(fallback, {});\n return h('div', { class: 'what-404' },\n h('h1', null, '404'),\n h('p', null, 'Page not found')\n );\n };\n}\n\n// --- Link Component ---\n\nexport function Link({\n href,\n class: cls,\n className,\n children,\n replace: rep,\n prefetch: shouldPrefetch = true,\n activeClass = 'active',\n exactActiveClass = 'exact-active',\n transition = true,\n ...rest\n}) {\n // Sanitize href \u2014 reject dangerous protocols\n const safeHref = isSafeUrl(href) ? href : 'about:blank';\n if (!isSafeUrl(href) && typeof console !== 'undefined') {\n console.warn(`[what-router] Link blocked unsafe href: ${href}`);\n }\n\n // Strip query string and hash from href for path comparison\n const hrefPath = safeHref.split('?')[0].split('#')[0];\n\n // Use a reactive function for class so active states update on navigation.\n // In the run-once model, reading route.path directly would snapshot it.\n const reactiveClass = () => {\n const currentPath = route.path;\n const isActive = hrefPath === '/'\n ? currentPath === '/'\n : currentPath === hrefPath || currentPath.startsWith(hrefPath + '/');\n const isExactActive = currentPath === hrefPath;\n\n return [\n cls || className,\n isActive && activeClass,\n isExactActive && exactActiveClass,\n ].filter(Boolean).join(' ') || undefined;\n };\n\n return h('a', {\n href: safeHref,\n class: reactiveClass,\n onclick: (e) => {\n // Only intercept left-clicks without modifiers\n if (e.ctrlKey || e.metaKey || e.shiftKey || e.altKey || e.button !== 0) return;\n e.preventDefault();\n navigate(safeHref, { replace: rep, transition });\n },\n onmouseenter: shouldPrefetch ? () => prefetch(safeHref) : undefined,\n ...rest,\n }, ...(Array.isArray(children) ? children : [children]));\n}\n\n// --- NavLink with active states ---\n\nexport function NavLink(props) {\n return Link(props);\n}\n\n// --- Define Routes Helper ---\n// Creates route config from a flat object for convenience.\n\nexport function defineRoutes(config) {\n return Object.entries(config).map(([path, value]) => {\n if (typeof value === 'function') {\n return { path, component: value };\n }\n // Object form with layout, middleware, loading, error, etc.\n return { path, ...value };\n });\n}\n\n// --- Nested Route Helper ---\n\nexport function nestedRoutes(basePath, children, options = {}) {\n const { layout, loading, error } = options;\n\n return children.map(child => ({\n ...child,\n path: basePath + child.path,\n layout: child.layout || layout,\n loading: child.loading || loading,\n error: child.error || error,\n }));\n}\n\n// --- Route Groups ---\n// Group routes without affecting URL structure\n\nexport function routeGroup(name, routes, options = {}) {\n const { layout, middleware } = options;\n\n return routes.map(route => ({\n ...route,\n _group: name,\n layout: route.layout || layout,\n middleware: [...(route.middleware || []), ...(middleware || [])],\n }));\n}\n\n// --- Redirect ---\n\nexport function Redirect({ to }) {\n navigate(to, { replace: true });\n return null;\n}\n\n// --- Route Guards / Middleware ---\n\nexport function guard(check, fallback) {\n return (Component) => {\n return function GuardedRoute(props) {\n const result = check(props);\n\n // Support async guards\n if (result instanceof Promise) {\n // Return loading while checking\n return h('div', { class: 'what-guard-loading' }, 'Loading...');\n }\n\n if (result) {\n return h(Component, props);\n }\n\n if (typeof fallback === 'string') {\n navigate(fallback, { replace: true });\n return null;\n }\n return h(fallback, props);\n };\n };\n}\n\n// Async guard with suspense\nexport function asyncGuard(check, options = {}) {\n const { fallback = '/login', loading = null } = options;\n\n return (Component) => {\n return function AsyncGuardedRoute(props) {\n const status = signal('pending');\n const checkResult = signal(null);\n let cancelled = false;\n\n effect(() => {\n cancelled = false;\n Promise.resolve(check(props))\n .then(result => {\n if (cancelled) return;\n checkResult.set(result);\n status.set(result ? 'allowed' : 'denied');\n })\n .catch(() => {\n if (!cancelled) status.set('denied');\n });\n return () => { cancelled = true; };\n });\n\n // Return a reactive function child so status changes update the DOM.\n // Components run once, so reading status() outside a reactive wrapper\n // would snapshot the value and never update.\n return () => {\n const currentStatus = status();\n\n if (currentStatus === 'pending') {\n return loading ? h(loading, {}) : null;\n }\n\n if (currentStatus === 'allowed') {\n return h(Component, props);\n }\n\n if (typeof fallback === 'string') {\n navigate(fallback, { replace: true });\n return null;\n }\n return h(fallback, props);\n };\n };\n };\n}\n\n// --- Prefetch ---\n// Hint the browser to prefetch a route's assets.\n\nconst prefetchedUrls = new Set();\n\nexport function prefetch(href) {\n if (typeof document === 'undefined') return;\n if (prefetchedUrls.has(href)) return;\n prefetchedUrls.add(href);\n\n const link = document.createElement('link');\n link.rel = 'prefetch';\n link.href = href;\n document.head.appendChild(link);\n}\n\n// --- Scroll Restoration ---\n\nconst scrollPositions = new Map();\n\nexport function enableScrollRestoration() {\n if (typeof window === 'undefined') return;\n\n // Save scroll position before navigation\n window.addEventListener('beforeunload', () => {\n scrollPositions.set(location.pathname, window.scrollY);\n });\n\n // Restore scroll position after navigation\n effect(() => {\n const path = route.path;\n const savedPosition = scrollPositions.get(path);\n\n requestAnimationFrame(() => {\n if (savedPosition !== undefined) {\n window.scrollTo(0, savedPosition);\n } else if (route.hash) {\n const el = document.querySelector(route.hash);\n el?.scrollIntoView();\n } else {\n window.scrollTo(0, 0);\n }\n });\n });\n}\n\n// --- View Transition Helpers ---\n\nexport function viewTransitionName(name) {\n return { style: { viewTransitionName: name } };\n}\n\n// Configure view transition types\nexport function setViewTransition(type) {\n if (typeof document === 'undefined') return;\n document.documentElement.dataset.transition = type;\n}\n\n// --- useRoute Hook ---\n\nexport function useRoute() {\n return {\n path: computed(() => route.path),\n params: computed(() => route.params),\n query: computed(() => route.query),\n hash: computed(() => route.hash),\n isNavigating: computed(() => route.isNavigating),\n navigate,\n prefetch,\n };\n}\n\n// --- Outlet Component ---\n// For nested route rendering\n\nexport function Outlet({ children }) {\n // Children passed from parent layout\n return children || null;\n}\n\n// --- File-Based Router ---\n// Consumes routes generated by what-compiler's file router (virtual:what-routes).\n// Usage:\n// import { routes } from 'virtual:what-routes';\n// mount(<FileRouter routes={routes} />, '#app');\n\nexport function FileRouter({\n routes,\n layout: globalLayout,\n fallback,\n error: globalError,\n}) {\n // Convert file-router route format to Router's expected format\n const routerRoutes = routes.map(r => ({\n path: r.path,\n component: r.component,\n layout: r.layout || undefined,\n // Attach page mode as metadata for build system\n _mode: r.mode || 'client',\n }));\n\n // Router already returns a reactive function child \u2014 just delegate\n return Router({\n routes: routerRoutes,\n globalLayout,\n fallback: fallback || Default404,\n });\n}\n\nfunction Default404() {\n return h('div', { style: 'text-align:center;padding:60px 20px' },\n h('h1', { style: 'font-size:48px;margin-bottom:8px' }, '404'),\n h('p', { style: 'color:#64748b' }, 'Page not found'),\n );\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
|
+
"names": ["signal", "effect", "computed", "batch", "h", "ErrorBoundary", "isSafeUrl", "url", "normalized", "_url", "_params", "_query", "_isNavigating", "_navigationError", "route", "navigate", "to", "opts", "replace", "state", "transition", "_fromPopstate", "newUrl", "el", "doNavigation", "scrollPositions", "saved", "compilePath", "path", "_", "name", "paramNames", "catchAll", "regexStr", "segment", "matchRoute", "routes", "sorted", "r", "a", "b", "aSpecific", "bSpecific", "regex", "match", "params", "i", "parseQuery", "search", "qs", "pair", "key", "val", "decodedKey", "decodedVal", "buildLayoutChain", "layouts", "segments", "currentPath", "layoutRoute", "_redirectHistory", "MAX_REDIRECTS", "Router", "fallback", "globalLayout", "currentUrl", "isNavigating", "matched", "queryObj", "mw", "result", "cycle", "seen", "hasCycle", "element", "Layout", "Link", "href", "cls", "className", "children", "rep", "shouldPrefetch", "activeClass", "exactActiveClass", "rest", "safeHref", "hrefPath", "isActive", "e", "prefetch", "NavLink", "props", "defineRoutes", "config", "value", "nestedRoutes", "basePath", "options", "layout", "loading", "error", "child", "routeGroup", "middleware", "Redirect", "guard", "check", "Component", "asyncGuard", "status", "checkResult", "cancelled", "currentStatus", "prefetchedUrls", "link", "enableScrollRestoration", "savedPosition", "viewTransitionName", "setViewTransition", "type", "useRoute", "Outlet", "FileRouter", "globalError", "routerRoutes", "Default404"]
|
|
7
|
+
}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "what-router",
|
|
3
|
-
"version": "0.
|
|
3
|
+
"version": "0.7.0",
|
|
4
4
|
"description": "What Framework - File-based & programmatic router with View Transitions",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"main": "src/index.js",
|
|
@@ -8,11 +8,13 @@
|
|
|
8
8
|
"exports": {
|
|
9
9
|
".": {
|
|
10
10
|
"types": "./index.d.ts",
|
|
11
|
+
"production": "./dist/index.min.js",
|
|
11
12
|
"import": "./src/index.js"
|
|
12
13
|
}
|
|
13
14
|
},
|
|
14
15
|
"files": [
|
|
15
16
|
"src",
|
|
17
|
+
"dist",
|
|
16
18
|
"index.d.ts"
|
|
17
19
|
],
|
|
18
20
|
"sideEffects": false,
|
|
@@ -26,14 +28,14 @@
|
|
|
26
28
|
"author": "ZVN DEV (https://zvndev.com)",
|
|
27
29
|
"license": "MIT",
|
|
28
30
|
"peerDependencies": {
|
|
29
|
-
"what-core": "^0.
|
|
31
|
+
"what-core": "^0.7.0"
|
|
30
32
|
},
|
|
31
33
|
"repository": {
|
|
32
34
|
"type": "git",
|
|
33
|
-
"url": "https://github.com/CelsianJs/
|
|
35
|
+
"url": "https://github.com/CelsianJs/what-framework"
|
|
34
36
|
},
|
|
35
37
|
"bugs": {
|
|
36
|
-
"url": "https://github.com/CelsianJs/
|
|
38
|
+
"url": "https://github.com/CelsianJs/what-framework/issues"
|
|
37
39
|
},
|
|
38
40
|
"homepage": "https://whatfw.com"
|
|
39
41
|
}
|
package/src/index.js
CHANGED
|
@@ -4,6 +4,20 @@
|
|
|
4
4
|
|
|
5
5
|
import { signal, effect, computed, batch, h, ErrorBoundary } from 'what-core';
|
|
6
6
|
|
|
7
|
+
// --- URL Sanitization ---
|
|
8
|
+
// Rejects javascript:, data:, vbscript: protocols (case-insensitive, trimmed).
|
|
9
|
+
|
|
10
|
+
export function isSafeUrl(url) {
|
|
11
|
+
if (typeof url !== 'string') return false;
|
|
12
|
+
const trimmed = url.trim();
|
|
13
|
+
// Check for dangerous protocols (case-insensitive, ignoring whitespace/control chars)
|
|
14
|
+
const normalized = trimmed.replace(/[\s\x00-\x1f]/g, '').toLowerCase();
|
|
15
|
+
if (normalized.startsWith('javascript:')) return false;
|
|
16
|
+
if (normalized.startsWith('data:')) return false;
|
|
17
|
+
if (normalized.startsWith('vbscript:')) return false;
|
|
18
|
+
return true;
|
|
19
|
+
}
|
|
20
|
+
|
|
7
21
|
// --- Route State (global singleton) ---
|
|
8
22
|
|
|
9
23
|
const _url = signal(typeof location !== 'undefined' ? location.pathname + location.search + location.hash : '/');
|
|
@@ -30,6 +44,26 @@ export const route = {
|
|
|
30
44
|
export async function navigate(to, opts = {}) {
|
|
31
45
|
const { replace = false, state = null, transition = true, _fromPopstate = false } = opts;
|
|
32
46
|
|
|
47
|
+
// Reject unsafe URLs
|
|
48
|
+
if (!isSafeUrl(to)) {
|
|
49
|
+
if (typeof console !== 'undefined') {
|
|
50
|
+
console.warn(`[what-router] Blocked navigation to unsafe URL: ${to}`);
|
|
51
|
+
}
|
|
52
|
+
return;
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
// Handle same-page hash links — use replaceState and scroll directly
|
|
56
|
+
if (typeof window !== 'undefined' && to.startsWith('#')) {
|
|
57
|
+
const currentUrl = _url();
|
|
58
|
+
const basePath = currentUrl.split('#')[0];
|
|
59
|
+
const newUrl = basePath + to;
|
|
60
|
+
history.replaceState(state, '', newUrl);
|
|
61
|
+
_url.set(newUrl);
|
|
62
|
+
const el = document.querySelector(to);
|
|
63
|
+
if (el) el.scrollIntoView({ behavior: 'smooth' });
|
|
64
|
+
return;
|
|
65
|
+
}
|
|
66
|
+
|
|
33
67
|
// Don't navigate if already on the same URL
|
|
34
68
|
if (to === _url()) return;
|
|
35
69
|
|
|
@@ -42,6 +76,10 @@ export async function navigate(to, opts = {}) {
|
|
|
42
76
|
const doNavigation = () => {
|
|
43
77
|
// Skip history manipulation on popstate (browser already updated the URL)
|
|
44
78
|
if (!_fromPopstate) {
|
|
79
|
+
// Save scroll position for current URL before navigating away
|
|
80
|
+
if (typeof window !== 'undefined') {
|
|
81
|
+
scrollPositions.set(_url(), { x: scrollX, y: scrollY });
|
|
82
|
+
}
|
|
45
83
|
if (replace) {
|
|
46
84
|
history.replaceState(state, '', to);
|
|
47
85
|
} else {
|
|
@@ -67,9 +105,18 @@ export async function navigate(to, opts = {}) {
|
|
|
67
105
|
// Back/forward support — route through navigate() so middleware runs
|
|
68
106
|
if (typeof window !== 'undefined') {
|
|
69
107
|
window.addEventListener('popstate', () => {
|
|
108
|
+
// Save scroll position for the URL we're leaving
|
|
109
|
+
scrollPositions.set(_url(), { x: scrollX, y: scrollY });
|
|
110
|
+
|
|
70
111
|
const newUrl = location.pathname + location.search + location.hash;
|
|
71
112
|
// Use _fromPopstate flag so navigate() skips pushState (browser already updated URL)
|
|
72
|
-
navigate(newUrl, { replace: true, _fromPopstate: true, transition: false })
|
|
113
|
+
navigate(newUrl, { replace: true, _fromPopstate: true, transition: false }).then(() => {
|
|
114
|
+
// Restore saved scroll position for the URL we're arriving at
|
|
115
|
+
const saved = scrollPositions.get(newUrl);
|
|
116
|
+
if (saved) {
|
|
117
|
+
requestAnimationFrame(() => window.scrollTo(saved.x, saved.y));
|
|
118
|
+
}
|
|
119
|
+
});
|
|
73
120
|
});
|
|
74
121
|
}
|
|
75
122
|
|
|
@@ -201,108 +248,113 @@ const MAX_REDIRECTS = 10;
|
|
|
201
248
|
// --- Router Component ---
|
|
202
249
|
|
|
203
250
|
export function Router({ routes, fallback, globalLayout }) {
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
251
|
+
// Return a reactive function child. The Router component runs ONCE,
|
|
252
|
+
// but the returned function re-evaluates whenever _url changes,
|
|
253
|
+
// and the fine-grained runtime updates the DOM accordingly.
|
|
254
|
+
return () => {
|
|
255
|
+
const currentUrl = _url();
|
|
256
|
+
const path = currentUrl.split('?')[0].split('#')[0];
|
|
257
|
+
const search = currentUrl.split('?')[1]?.split('#')[0] || '';
|
|
258
|
+
const isNavigating = _isNavigating();
|
|
259
|
+
|
|
260
|
+
const matched = matchRoute(path, routes);
|
|
261
|
+
|
|
262
|
+
if (matched) {
|
|
263
|
+
batch(() => {
|
|
264
|
+
_params.set(matched.params);
|
|
265
|
+
_query.set(parseQuery(search));
|
|
266
|
+
});
|
|
216
267
|
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
}
|
|
229
|
-
if (typeof result === 'string') {
|
|
230
|
-
// Redirect loop detection
|
|
231
|
-
_redirectHistory.push(result);
|
|
232
|
-
if (_redirectHistory.length > MAX_REDIRECTS) {
|
|
233
|
-
const cycle = _redirectHistory.slice(-5).join(' → ');
|
|
234
|
-
_redirectHistory.length = 0;
|
|
235
|
-
console.error(`[what-router] Redirect loop detected: ${cycle}`);
|
|
236
|
-
_isNavigating.set(false);
|
|
237
|
-
return h('div', { class: 'what-redirect-loop' },
|
|
238
|
-
h('h1', null, 'Redirect Loop'),
|
|
239
|
-
h('p', null, 'Too many redirects. Check your middleware configuration.')
|
|
240
|
-
);
|
|
268
|
+
const { route: r, params } = matched;
|
|
269
|
+
const queryObj = parseQuery(search);
|
|
270
|
+
|
|
271
|
+
// Run middleware (sync only — async middleware should use asyncGuard)
|
|
272
|
+
if (r.middleware && r.middleware.length > 0) {
|
|
273
|
+
for (const mw of r.middleware) {
|
|
274
|
+
const result = mw({ path, params, query: queryObj, route: r });
|
|
275
|
+
if (result === false) {
|
|
276
|
+
// Middleware rejected — show fallback
|
|
277
|
+
if (fallback) return h(fallback, {});
|
|
278
|
+
return h('div', { class: 'what-403' }, h('h1', null, '403'), h('p', null, 'Access denied'));
|
|
241
279
|
}
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
|
|
245
|
-
|
|
246
|
-
|
|
247
|
-
|
|
280
|
+
if (typeof result === 'string') {
|
|
281
|
+
// Redirect loop detection
|
|
282
|
+
_redirectHistory.push(result);
|
|
283
|
+
if (_redirectHistory.length > MAX_REDIRECTS) {
|
|
284
|
+
const cycle = _redirectHistory.slice(-5).join(' → ');
|
|
285
|
+
_redirectHistory.length = 0;
|
|
286
|
+
console.error(`[what-router] Redirect loop detected: ${cycle}`);
|
|
287
|
+
_isNavigating.set(false);
|
|
288
|
+
return h('div', { class: 'what-redirect-loop' },
|
|
289
|
+
h('h1', null, 'Redirect Loop'),
|
|
290
|
+
h('p', null, 'Too many redirects. Check your middleware configuration.')
|
|
291
|
+
);
|
|
292
|
+
}
|
|
293
|
+
// Check for direct cycle (A → B → A)
|
|
294
|
+
const seen = new Set();
|
|
295
|
+
let hasCycle = false;
|
|
296
|
+
for (const url of _redirectHistory) {
|
|
297
|
+
if (seen.has(url)) { hasCycle = true; break; }
|
|
298
|
+
seen.add(url);
|
|
299
|
+
}
|
|
300
|
+
if (hasCycle) {
|
|
301
|
+
const cycle = _redirectHistory.join(' → ');
|
|
302
|
+
_redirectHistory.length = 0;
|
|
303
|
+
console.error(`[what-router] Redirect cycle detected: ${cycle}`);
|
|
304
|
+
_isNavigating.set(false);
|
|
305
|
+
return h('div', { class: 'what-redirect-loop' },
|
|
306
|
+
h('h1', null, 'Redirect Loop'),
|
|
307
|
+
h('p', null, 'Circular redirect detected. Check your middleware configuration.')
|
|
308
|
+
);
|
|
309
|
+
}
|
|
310
|
+
// Middleware returned a redirect path
|
|
311
|
+
navigate(result, { replace: true });
|
|
312
|
+
return null;
|
|
248
313
|
}
|
|
249
|
-
if (hasCycle) {
|
|
250
|
-
const cycle = _redirectHistory.join(' → ');
|
|
251
|
-
_redirectHistory.length = 0;
|
|
252
|
-
console.error(`[what-router] Redirect cycle detected: ${cycle}`);
|
|
253
|
-
_isNavigating.set(false);
|
|
254
|
-
return h('div', { class: 'what-redirect-loop' },
|
|
255
|
-
h('h1', null, 'Redirect Loop'),
|
|
256
|
-
h('p', null, 'Circular redirect detected. Check your middleware configuration.')
|
|
257
|
-
);
|
|
258
|
-
}
|
|
259
|
-
// Middleware returned a redirect path
|
|
260
|
-
navigate(result, { replace: true });
|
|
261
|
-
return null;
|
|
262
314
|
}
|
|
263
315
|
}
|
|
264
|
-
|
|
265
|
-
|
|
266
|
-
_redirectHistory.length = 0;
|
|
316
|
+
// Successful render — clear redirect history
|
|
317
|
+
_redirectHistory.length = 0;
|
|
267
318
|
|
|
268
|
-
|
|
269
|
-
|
|
319
|
+
// Build element with loading state support
|
|
320
|
+
let element;
|
|
270
321
|
|
|
271
|
-
|
|
272
|
-
|
|
273
|
-
|
|
274
|
-
|
|
275
|
-
|
|
276
|
-
|
|
277
|
-
|
|
278
|
-
|
|
279
|
-
|
|
322
|
+
if (r.loading && isNavigating) {
|
|
323
|
+
element = h(r.loading, {});
|
|
324
|
+
} else {
|
|
325
|
+
element = h(r.component, {
|
|
326
|
+
params,
|
|
327
|
+
query: queryObj,
|
|
328
|
+
route: r,
|
|
329
|
+
});
|
|
330
|
+
}
|
|
280
331
|
|
|
281
|
-
|
|
282
|
-
|
|
283
|
-
|
|
284
|
-
|
|
332
|
+
// Wrap with per-route error boundary if specified
|
|
333
|
+
if (r.error) {
|
|
334
|
+
element = h(ErrorBoundary, { fallback: r.error }, element);
|
|
335
|
+
}
|
|
285
336
|
|
|
286
|
-
|
|
287
|
-
|
|
288
|
-
|
|
289
|
-
|
|
290
|
-
|
|
337
|
+
// Wrap with nested layouts (innermost to outermost)
|
|
338
|
+
const layouts = buildLayoutChain(r, routes);
|
|
339
|
+
for (const Layout of layouts.reverse()) {
|
|
340
|
+
element = h(Layout, { params, query: queryObj }, element);
|
|
341
|
+
}
|
|
291
342
|
|
|
292
|
-
|
|
293
|
-
|
|
294
|
-
|
|
295
|
-
|
|
343
|
+
// Global layout wrapper
|
|
344
|
+
if (globalLayout) {
|
|
345
|
+
element = h(globalLayout, {}, element);
|
|
346
|
+
}
|
|
296
347
|
|
|
297
|
-
|
|
298
|
-
|
|
348
|
+
return element;
|
|
349
|
+
}
|
|
299
350
|
|
|
300
|
-
|
|
301
|
-
|
|
302
|
-
|
|
303
|
-
|
|
304
|
-
|
|
305
|
-
|
|
351
|
+
// 404
|
|
352
|
+
if (fallback) return h(fallback, {});
|
|
353
|
+
return h('div', { class: 'what-404' },
|
|
354
|
+
h('h1', null, '404'),
|
|
355
|
+
h('p', null, 'Page not found')
|
|
356
|
+
);
|
|
357
|
+
};
|
|
306
358
|
}
|
|
307
359
|
|
|
308
360
|
// --- Link Component ---
|
|
@@ -319,31 +371,41 @@ export function Link({
|
|
|
319
371
|
transition = true,
|
|
320
372
|
...rest
|
|
321
373
|
}) {
|
|
322
|
-
|
|
374
|
+
// Sanitize href — reject dangerous protocols
|
|
375
|
+
const safeHref = isSafeUrl(href) ? href : 'about:blank';
|
|
376
|
+
if (!isSafeUrl(href) && typeof console !== 'undefined') {
|
|
377
|
+
console.warn(`[what-router] Link blocked unsafe href: ${href}`);
|
|
378
|
+
}
|
|
379
|
+
|
|
323
380
|
// Strip query string and hash from href for path comparison
|
|
324
|
-
const hrefPath =
|
|
325
|
-
|
|
326
|
-
|
|
327
|
-
|
|
328
|
-
|
|
329
|
-
|
|
330
|
-
|
|
331
|
-
|
|
332
|
-
|
|
333
|
-
|
|
334
|
-
|
|
335
|
-
|
|
381
|
+
const hrefPath = safeHref.split('?')[0].split('#')[0];
|
|
382
|
+
|
|
383
|
+
// Use a reactive function for class so active states update on navigation.
|
|
384
|
+
// In the run-once model, reading route.path directly would snapshot it.
|
|
385
|
+
const reactiveClass = () => {
|
|
386
|
+
const currentPath = route.path;
|
|
387
|
+
const isActive = hrefPath === '/'
|
|
388
|
+
? currentPath === '/'
|
|
389
|
+
: currentPath === hrefPath || currentPath.startsWith(hrefPath + '/');
|
|
390
|
+
const isExactActive = currentPath === hrefPath;
|
|
391
|
+
|
|
392
|
+
return [
|
|
393
|
+
cls || className,
|
|
394
|
+
isActive && activeClass,
|
|
395
|
+
isExactActive && exactActiveClass,
|
|
396
|
+
].filter(Boolean).join(' ') || undefined;
|
|
397
|
+
};
|
|
336
398
|
|
|
337
399
|
return h('a', {
|
|
338
|
-
href,
|
|
339
|
-
class:
|
|
400
|
+
href: safeHref,
|
|
401
|
+
class: reactiveClass,
|
|
340
402
|
onclick: (e) => {
|
|
341
403
|
// Only intercept left-clicks without modifiers
|
|
342
404
|
if (e.ctrlKey || e.metaKey || e.shiftKey || e.altKey || e.button !== 0) return;
|
|
343
405
|
e.preventDefault();
|
|
344
|
-
navigate(
|
|
406
|
+
navigate(safeHref, { replace: rep, transition });
|
|
345
407
|
},
|
|
346
|
-
onmouseenter: shouldPrefetch ? () => prefetch(
|
|
408
|
+
onmouseenter: shouldPrefetch ? () => prefetch(safeHref) : undefined,
|
|
347
409
|
...rest,
|
|
348
410
|
}, ...(Array.isArray(children) ? children : [children]));
|
|
349
411
|
}
|
|
@@ -452,21 +514,26 @@ export function asyncGuard(check, options = {}) {
|
|
|
452
514
|
return () => { cancelled = true; };
|
|
453
515
|
});
|
|
454
516
|
|
|
455
|
-
|
|
517
|
+
// Return a reactive function child so status changes update the DOM.
|
|
518
|
+
// Components run once, so reading status() outside a reactive wrapper
|
|
519
|
+
// would snapshot the value and never update.
|
|
520
|
+
return () => {
|
|
521
|
+
const currentStatus = status();
|
|
456
522
|
|
|
457
|
-
|
|
458
|
-
|
|
459
|
-
|
|
523
|
+
if (currentStatus === 'pending') {
|
|
524
|
+
return loading ? h(loading, {}) : null;
|
|
525
|
+
}
|
|
460
526
|
|
|
461
|
-
|
|
462
|
-
|
|
463
|
-
|
|
527
|
+
if (currentStatus === 'allowed') {
|
|
528
|
+
return h(Component, props);
|
|
529
|
+
}
|
|
464
530
|
|
|
465
|
-
|
|
466
|
-
|
|
467
|
-
|
|
468
|
-
|
|
469
|
-
|
|
531
|
+
if (typeof fallback === 'string') {
|
|
532
|
+
navigate(fallback, { replace: true });
|
|
533
|
+
return null;
|
|
534
|
+
}
|
|
535
|
+
return h(fallback, props);
|
|
536
|
+
};
|
|
470
537
|
};
|
|
471
538
|
};
|
|
472
539
|
}
|
|
@@ -572,6 +639,7 @@ export function FileRouter({
|
|
|
572
639
|
_mode: r.mode || 'client',
|
|
573
640
|
}));
|
|
574
641
|
|
|
642
|
+
// Router already returns a reactive function child — just delegate
|
|
575
643
|
return Router({
|
|
576
644
|
routes: routerRoutes,
|
|
577
645
|
globalLayout,
|