kiru 0.49.2 → 0.50.0-preview.1

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.
Files changed (143) hide show
  1. package/dist/components/errorBoundary.d.ts +7 -0
  2. package/dist/components/errorBoundary.d.ts.map +1 -0
  3. package/dist/components/errorBoundary.js +6 -0
  4. package/dist/components/errorBoundary.js.map +1 -0
  5. package/dist/components/index.d.ts +2 -0
  6. package/dist/components/index.d.ts.map +1 -1
  7. package/dist/components/index.js +2 -0
  8. package/dist/components/index.js.map +1 -1
  9. package/dist/components/lazy.d.ts.map +1 -1
  10. package/dist/components/lazy.js +11 -136
  11. package/dist/components/lazy.js.map +1 -1
  12. package/dist/components/suspense.d.ts +34 -0
  13. package/dist/components/suspense.d.ts.map +1 -0
  14. package/dist/components/suspense.js +110 -0
  15. package/dist/components/suspense.js.map +1 -0
  16. package/dist/constants.d.ts +4 -2
  17. package/dist/constants.d.ts.map +1 -1
  18. package/dist/constants.js +4 -2
  19. package/dist/constants.js.map +1 -1
  20. package/dist/hmr.d.ts +1 -0
  21. package/dist/hmr.d.ts.map +1 -1
  22. package/dist/hmr.js +7 -0
  23. package/dist/hmr.js.map +1 -1
  24. package/dist/index.d.ts +1 -0
  25. package/dist/index.d.ts.map +1 -1
  26. package/dist/index.js +1 -0
  27. package/dist/index.js.map +1 -1
  28. package/dist/recursiveRender.d.ts +6 -0
  29. package/dist/recursiveRender.d.ts.map +1 -0
  30. package/dist/recursiveRender.js +109 -0
  31. package/dist/recursiveRender.js.map +1 -0
  32. package/dist/renderToString.d.ts.map +1 -1
  33. package/dist/renderToString.js +10 -59
  34. package/dist/renderToString.js.map +1 -1
  35. package/dist/router/config.d.ts +3 -0
  36. package/dist/router/config.d.ts.map +1 -0
  37. package/dist/router/config.js +13 -0
  38. package/dist/router/config.js.map +1 -0
  39. package/dist/router/context.d.ts +15 -0
  40. package/dist/router/context.d.ts.map +1 -0
  41. package/dist/router/context.js +11 -0
  42. package/dist/router/context.js.map +1 -0
  43. package/dist/router/errors.d.ts +4 -0
  44. package/dist/router/errors.d.ts.map +1 -0
  45. package/dist/router/errors.js +7 -0
  46. package/dist/router/errors.js.map +1 -0
  47. package/dist/router/fileRouter.d.ts +48 -0
  48. package/dist/router/fileRouter.d.ts.map +1 -0
  49. package/dist/router/fileRouter.js +311 -0
  50. package/dist/router/fileRouter.js.map +1 -0
  51. package/dist/router/globals.d.ts +5 -0
  52. package/dist/router/globals.d.ts.map +1 -0
  53. package/dist/router/globals.js +4 -0
  54. package/dist/router/globals.js.map +1 -0
  55. package/dist/router/index.d.ts +7 -0
  56. package/dist/router/index.d.ts.map +1 -0
  57. package/dist/router/index.js +7 -0
  58. package/dist/router/index.js.map +1 -0
  59. package/dist/router/link.d.ts +8 -0
  60. package/dist/router/link.d.ts.map +1 -0
  61. package/dist/router/link.js +15 -0
  62. package/dist/router/link.js.map +1 -0
  63. package/dist/router/types.d.ts +63 -0
  64. package/dist/router/types.d.ts.map +1 -0
  65. package/dist/router/types.internal.d.ts +12 -0
  66. package/dist/router/types.internal.d.ts.map +1 -0
  67. package/dist/router/types.internal.js +2 -0
  68. package/dist/router/types.internal.js.map +1 -0
  69. package/dist/router/types.js +2 -0
  70. package/dist/router/types.js.map +1 -0
  71. package/dist/scheduler.d.ts.map +1 -1
  72. package/dist/scheduler.js +35 -10
  73. package/dist/scheduler.js.map +1 -1
  74. package/dist/ssr/server.d.ts +4 -1
  75. package/dist/ssr/server.d.ts.map +1 -1
  76. package/dist/ssr/server.js +50 -82
  77. package/dist/ssr/server.js.map +1 -1
  78. package/dist/types.d.ts +10 -2
  79. package/dist/types.d.ts.map +1 -1
  80. package/dist/types.utils.d.ts +6 -7
  81. package/dist/types.utils.d.ts.map +1 -1
  82. package/dist/utils/runtime.d.ts +1 -1
  83. package/dist/utils/runtime.d.ts.map +1 -1
  84. package/dist/utils/runtime.js.map +1 -1
  85. package/dist/utils/vdom.d.ts +3 -1
  86. package/dist/utils/vdom.d.ts.map +1 -1
  87. package/dist/utils/vdom.js +6 -5
  88. package/dist/utils/vdom.js.map +1 -1
  89. package/package.json +4 -8
  90. package/src/components/errorBoundary.ts +16 -0
  91. package/src/components/index.ts +2 -0
  92. package/src/components/lazy.ts +12 -169
  93. package/src/components/suspense.ts +191 -0
  94. package/src/constants.ts +6 -2
  95. package/src/hmr.ts +8 -0
  96. package/src/index.ts +1 -0
  97. package/src/recursiveRender.ts +127 -0
  98. package/src/renderToString.ts +10 -73
  99. package/src/router/config.ts +15 -0
  100. package/src/router/context.ts +23 -0
  101. package/src/router/errors.ts +6 -0
  102. package/src/router/fileRouter.ts +475 -0
  103. package/src/router/globals.ts +5 -0
  104. package/src/router/index.ts +6 -0
  105. package/src/router/link.ts +32 -0
  106. package/src/router/types.internal.ts +14 -0
  107. package/src/router/types.ts +81 -0
  108. package/src/scheduler.ts +45 -12
  109. package/src/ssr/server.ts +58 -95
  110. package/src/types.ts +11 -2
  111. package/src/types.utils.ts +6 -11
  112. package/src/utils/runtime.ts +1 -1
  113. package/src/utils/vdom.ts +11 -4
  114. package/dist/components/router/index.d.ts +0 -3
  115. package/dist/components/router/index.d.ts.map +0 -1
  116. package/dist/components/router/index.js +0 -3
  117. package/dist/components/router/index.js.map +0 -1
  118. package/dist/components/router/route.d.ts +0 -46
  119. package/dist/components/router/route.d.ts.map +0 -1
  120. package/dist/components/router/route.js +0 -8
  121. package/dist/components/router/route.js.map +0 -1
  122. package/dist/components/router/router.d.ts +0 -62
  123. package/dist/components/router/router.d.ts.map +0 -1
  124. package/dist/components/router/router.js +0 -181
  125. package/dist/components/router/router.js.map +0 -1
  126. package/dist/components/router/routerUtils.d.ts +0 -5
  127. package/dist/components/router/routerUtils.d.ts.map +0 -1
  128. package/dist/components/router/routerUtils.js +0 -39
  129. package/dist/components/router/routerUtils.js.map +0 -1
  130. package/dist/ssr/hydrationBoundary.d.ts +0 -27
  131. package/dist/ssr/hydrationBoundary.d.ts.map +0 -1
  132. package/dist/ssr/hydrationBoundary.js +0 -30
  133. package/dist/ssr/hydrationBoundary.js.map +0 -1
  134. package/dist/ssr/index.d.ts +0 -2
  135. package/dist/ssr/index.d.ts.map +0 -1
  136. package/dist/ssr/index.js +0 -2
  137. package/dist/ssr/index.js.map +0 -1
  138. package/src/components/router/index.ts +0 -2
  139. package/src/components/router/route.ts +0 -51
  140. package/src/components/router/router.ts +0 -280
  141. package/src/components/router/routerUtils.ts +0 -49
  142. package/src/ssr/hydrationBoundary.ts +0 -63
  143. package/src/ssr/index.ts +0 -1
@@ -0,0 +1,109 @@
1
+ import { node, hookIndex } from "./globals.js";
2
+ import { isVNode, encodeHtmlEntities, propsToElementAttributes, isExoticType, assertValidElementProps, } from "./utils/index.js";
3
+ import { Signal } from "./signals/base.js";
4
+ import { $ERROR_BOUNDARY, voidElements, $SUSPENSE_THROW } from "./constants.js";
5
+ import { __DEV__ } from "./env.js";
6
+ import { isSuspenseThrowValue } from "./components/suspense.js";
7
+ export function recursiveRender(ctx, el, parent, idx) {
8
+ if (el === null)
9
+ return;
10
+ if (el === undefined)
11
+ return;
12
+ if (typeof el === "boolean")
13
+ return;
14
+ if (typeof el === "string") {
15
+ return ctx.write(encodeHtmlEntities(el));
16
+ }
17
+ if (typeof el === "number" || typeof el === "bigint") {
18
+ return ctx.write(el.toString());
19
+ }
20
+ if (el instanceof Array) {
21
+ return el.forEach((c, i) => recursiveRender(ctx, c, parent, i));
22
+ }
23
+ if (Signal.isSignal(el)) {
24
+ return ctx.write(String(el.peek()));
25
+ }
26
+ if (!isVNode(el)) {
27
+ return ctx.write(String(el));
28
+ }
29
+ el.parent = parent;
30
+ el.depth = (parent?.depth ?? -1) + 1;
31
+ el.index = idx;
32
+ const { type, props = {} } = el;
33
+ if (type === "#text") {
34
+ return ctx.write(encodeHtmlEntities(props.nodeValue ?? ""));
35
+ }
36
+ const children = props.children;
37
+ if (isExoticType(type)) {
38
+ if (type === $ERROR_BOUNDARY) {
39
+ let boundaryBuffer = "";
40
+ const pending = new Set();
41
+ const boundaryCtx = {
42
+ write(chunk) {
43
+ boundaryBuffer += chunk;
44
+ },
45
+ onPending(data) {
46
+ data.forEach((p) => pending.add(p));
47
+ },
48
+ };
49
+ try {
50
+ recursiveRender(boundaryCtx, children, el, idx);
51
+ // flush successful render
52
+ ctx.write(boundaryBuffer);
53
+ ctx.onPending?.([...pending]);
54
+ }
55
+ catch (error) {
56
+ if (isSuspenseThrowValue(error)) {
57
+ throw error;
58
+ }
59
+ const e = error instanceof Error ? error : new Error(String(error));
60
+ const { fallback, onError } = props;
61
+ onError?.(e);
62
+ const fallbackContent = typeof fallback === "function" ? fallback(e) : fallback;
63
+ recursiveRender(ctx, fallbackContent, el, 0);
64
+ }
65
+ return;
66
+ }
67
+ recursiveRender(ctx, children, el, idx);
68
+ return;
69
+ }
70
+ if (typeof type !== "string") {
71
+ try {
72
+ hookIndex.current = 0;
73
+ node.current = el;
74
+ const res = type(props);
75
+ recursiveRender(ctx, res, el, idx);
76
+ return;
77
+ }
78
+ catch (error) {
79
+ if (isSuspenseThrowValue(error)) {
80
+ const { fallback, pending } = error[$SUSPENSE_THROW];
81
+ ctx.onPending?.(pending);
82
+ return recursiveRender(ctx, fallback, el, 0);
83
+ }
84
+ throw error;
85
+ }
86
+ finally {
87
+ node.current = null;
88
+ }
89
+ }
90
+ if (__DEV__)
91
+ assertValidElementProps(el);
92
+ const attrs = propsToElementAttributes(props);
93
+ ctx.write(`<${type}${attrs.length ? ` ${attrs}` : ""}>`);
94
+ if (voidElements.has(type))
95
+ return;
96
+ if ("innerHTML" in props) {
97
+ ctx.write(String(Signal.isSignal(props.innerHTML)
98
+ ? props.innerHTML.peek()
99
+ : props.innerHTML));
100
+ }
101
+ else if (Array.isArray(children)) {
102
+ children.forEach((c, i) => recursiveRender(ctx, c, el, i));
103
+ }
104
+ else {
105
+ recursiveRender(ctx, children, el, 0);
106
+ }
107
+ ctx.write(`</${type}>`);
108
+ }
109
+ //# sourceMappingURL=recursiveRender.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"recursiveRender.js","sourceRoot":"","sources":["../src/recursiveRender.ts"],"names":[],"mappings":"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"}
@@ -1 +1 @@
1
- {"version":3,"file":"renderToString.d.ts","sourceRoot":"","sources":["../src/renderToString.ts"],"names":[],"mappings":"AAcA,wBAAgB,cAAc,CAAC,OAAO,EAAE,GAAG,CAAC,OAAO,UAOlD"}
1
+ {"version":3,"file":"renderToString.d.ts","sourceRoot":"","sources":["../src/renderToString.ts"],"names":[],"mappings":"AAKA,wBAAgB,cAAc,CAAC,OAAO,EAAE,GAAG,CAAC,OAAO,UAYlD"}
@@ -1,66 +1,17 @@
1
- import { node, renderMode } from "./globals.js";
1
+ import { renderMode } from "./globals.js";
2
2
  import { Fragment } from "./element.js";
3
- import { isVNode, encodeHtmlEntities, propsToElementAttributes, isExoticType, assertValidElementProps, } from "./utils/index.js";
4
- import { Signal } from "./signals/base.js";
5
- import { $HYDRATION_BOUNDARY, voidElements } from "./constants.js";
6
- import { HYDRATION_BOUNDARY_MARKER } from "./ssr/hydrationBoundary.js";
7
- import { __DEV__ } from "./env.js";
3
+ import { recursiveRender } from "./recursiveRender.js";
8
4
  export function renderToString(element) {
9
5
  const prev = renderMode.current;
10
6
  renderMode.current = "string";
11
- const rootNode = Fragment({ children: element });
12
- const res = renderToString_internal(rootNode, null, 0);
7
+ let result = "";
8
+ const ctx = {
9
+ write(chunk) {
10
+ result += chunk;
11
+ },
12
+ };
13
+ recursiveRender(ctx, Fragment({ children: element }), null, 0);
13
14
  renderMode.current = prev;
14
- return res;
15
- }
16
- function renderToString_internal(el, parent, idx) {
17
- if (el === null)
18
- return "";
19
- if (el === undefined)
20
- return "";
21
- if (typeof el === "boolean")
22
- return "";
23
- if (typeof el === "string")
24
- return encodeHtmlEntities(el);
25
- if (typeof el === "number" || typeof el === "bigint")
26
- return el.toString();
27
- if (el instanceof Array) {
28
- return el.map((c, i) => renderToString_internal(c, parent, i)).join("");
29
- }
30
- if (Signal.isSignal(el))
31
- return String(el.peek());
32
- if (!isVNode(el))
33
- return String(el);
34
- el.parent = parent;
35
- el.depth = (parent?.depth ?? -1) + 1;
36
- el.index = idx;
37
- const { type, props = {} } = el;
38
- if (type === "#text")
39
- return encodeHtmlEntities(props.nodeValue ?? "");
40
- const children = props.children;
41
- if (isExoticType(type)) {
42
- if (type === $HYDRATION_BOUNDARY) {
43
- return `<!--${HYDRATION_BOUNDARY_MARKER}-->${renderToString_internal(children, el, idx)}<!--/${HYDRATION_BOUNDARY_MARKER}-->`;
44
- }
45
- return renderToString_internal(children, el, idx);
46
- }
47
- if (typeof type !== "string") {
48
- node.current = el;
49
- const res = type(props);
50
- node.current = null;
51
- return renderToString_internal(res, el, idx);
52
- }
53
- if (__DEV__) {
54
- assertValidElementProps(el);
55
- }
56
- const attrs = propsToElementAttributes(props);
57
- const inner = "innerHTML" in props
58
- ? Signal.isSignal(props.innerHTML)
59
- ? props.innerHTML.peek()
60
- : props.innerHTML
61
- : Array.isArray(children)
62
- ? children.map((c, i) => renderToString_internal(c, el, i)).join("")
63
- : renderToString_internal(children, el, 0);
64
- return `<${type}${attrs.length ? ` ${attrs}` : ""}>${voidElements.has(type) ? "" : `${inner}</${type}>`}`;
15
+ return result;
65
16
  }
66
17
  //# sourceMappingURL=renderToString.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"renderToString.js","sourceRoot":"","sources":["../src/renderToString.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"renderToString.js","sourceRoot":"","sources":["../src/renderToString.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,cAAc,CAAA;AACzC,OAAO,EAAE,QAAQ,EAAE,MAAM,cAAc,CAAA;AAEvC,OAAO,EAAE,eAAe,EAA0B,MAAM,sBAAsB,CAAA;AAE9E,MAAM,UAAU,cAAc,CAAC,OAAoB;IACjD,MAAM,IAAI,GAAG,UAAU,CAAC,OAAO,CAAA;IAC/B,UAAU,CAAC,OAAO,GAAG,QAAQ,CAAA;IAC7B,IAAI,MAAM,GAAG,EAAE,CAAA;IACf,MAAM,GAAG,GAA2B;QAClC,KAAK,CAAC,KAAK;YACT,MAAM,IAAI,KAAK,CAAA;QACjB,CAAC;KACF,CAAA;IACD,eAAe,CAAC,GAAG,EAAE,QAAQ,CAAC,EAAE,QAAQ,EAAE,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC,CAAA;IAC9D,UAAU,CAAC,OAAO,GAAG,IAAI,CAAA;IACzB,OAAO,MAAM,CAAA;AACf,CAAC"}
@@ -0,0 +1,3 @@
1
+ import type { PageConfig } from "./types";
2
+ export declare function definePageConfig<T extends PageConfig>(config: T): T;
3
+ //# sourceMappingURL=config.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"config.d.ts","sourceRoot":"","sources":["../../src/router/config.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,SAAS,CAAA;AAEzC,wBAAgB,gBAAgB,CAAC,CAAC,SAAS,UAAU,EAAE,MAAM,EAAE,CAAC,GAAG,CAAC,CAUnE"}
@@ -0,0 +1,13 @@
1
+ import { __DEV__ } from "../env.js";
2
+ import { fileRouterInstance } from "./globals.js";
3
+ export function definePageConfig(config) {
4
+ if (__DEV__) {
5
+ const filePath = window.__kiru?.HMRContext?.getCurrentFilePath();
6
+ const fileRouter = fileRouterInstance.current;
7
+ if (filePath && fileRouter) {
8
+ fileRouter.onPageConfigDefined(filePath, config);
9
+ }
10
+ }
11
+ return config;
12
+ }
13
+ //# sourceMappingURL=config.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"config.js","sourceRoot":"","sources":["../../src/router/config.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAA;AACnC,OAAO,EAAE,kBAAkB,EAAE,MAAM,cAAc,CAAA;AAGjD,MAAM,UAAU,gBAAgB,CAAuB,MAAS;IAC9D,IAAI,OAAO,EAAE,CAAC;QACZ,MAAM,QAAQ,GAAG,MAAM,CAAC,MAAM,EAAE,UAAU,EAAE,kBAAkB,EAAE,CAAA;QAChE,MAAM,UAAU,GAAG,kBAAkB,CAAC,OAAO,CAAA;QAC7C,IAAI,QAAQ,IAAI,UAAU,EAAE,CAAC;YAC3B,UAAU,CAAC,mBAAmB,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAA;QAClD,CAAC;IACH,CAAC;IAED,OAAO,MAAM,CAAA;AACf,CAAC"}
@@ -0,0 +1,15 @@
1
+ import type { RouteQuery, RouterState } from "./types.js";
2
+ export interface FileRouterContextType {
3
+ state: RouterState;
4
+ navigate: (path: string, options?: {
5
+ replace?: boolean;
6
+ transition?: boolean;
7
+ }) => Promise<void>;
8
+ reload: (options?: {
9
+ transition?: boolean;
10
+ }) => Promise<void>;
11
+ setQuery: (query: RouteQuery) => void;
12
+ }
13
+ export declare const RouterContext: Kiru.Context<FileRouterContextType>;
14
+ export declare function useFileRouter(): FileRouterContextType;
15
+ //# sourceMappingURL=context.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"context.d.ts","sourceRoot":"","sources":["../../src/router/context.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,UAAU,EAAE,WAAW,EAAE,MAAM,YAAY,CAAA;AAEzD,MAAM,WAAW,qBAAqB;IACpC,KAAK,EAAE,WAAW,CAAA;IAClB,QAAQ,EAAE,CACR,IAAI,EAAE,MAAM,EACZ,OAAO,CAAC,EAAE;QAAE,OAAO,CAAC,EAAE,OAAO,CAAC;QAAC,UAAU,CAAC,EAAE,OAAO,CAAA;KAAE,KAClD,OAAO,CAAC,IAAI,CAAC,CAAA;IAClB,MAAM,EAAE,CAAC,OAAO,CAAC,EAAE;QAAE,UAAU,CAAC,EAAE,OAAO,CAAA;KAAE,KAAK,OAAO,CAAC,IAAI,CAAC,CAAA;IAC7D,QAAQ,EAAE,CAAC,KAAK,EAAE,UAAU,KAAK,IAAI,CAAA;CACtC;AAED,eAAO,MAAM,aAAa,qCAA8C,CAAA;AAKxE,wBAAgB,aAAa,IAAI,qBAAqB,CAErD"}
@@ -0,0 +1,11 @@
1
+ import { createContext } from "../context.js";
2
+ import { __DEV__ } from "../env.js";
3
+ import { useContext } from "../hooks/index.js";
4
+ export const RouterContext = createContext(null);
5
+ if (__DEV__) {
6
+ RouterContext.displayName = "RouterContext";
7
+ }
8
+ export function useFileRouter() {
9
+ return useContext(RouterContext);
10
+ }
11
+ //# sourceMappingURL=context.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"context.js","sourceRoot":"","sources":["../../src/router/context.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,aAAa,EAAE,MAAM,eAAe,CAAA;AAC7C,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAA;AACnC,OAAO,EAAE,UAAU,EAAE,MAAM,mBAAmB,CAAA;AAa9C,MAAM,CAAC,MAAM,aAAa,GAAG,aAAa,CAAwB,IAAK,CAAC,CAAA;AACxE,IAAI,OAAO,EAAE,CAAC;IACZ,aAAa,CAAC,WAAW,GAAG,eAAe,CAAA;AAC7C,CAAC;AAED,MAAM,UAAU,aAAa;IAC3B,OAAO,UAAU,CAAC,aAAa,CAAC,CAAA;AAClC,CAAC"}
@@ -0,0 +1,4 @@
1
+ export declare class FileRouterDataLoadError extends Error {
2
+ constructor(cause: unknown);
3
+ }
4
+ //# sourceMappingURL=errors.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"errors.d.ts","sourceRoot":"","sources":["../../src/router/errors.ts"],"names":[],"mappings":"AAAA,qBAAa,uBAAwB,SAAQ,KAAK;gBACpC,KAAK,EAAE,OAAO;CAI3B"}
@@ -0,0 +1,7 @@
1
+ export class FileRouterDataLoadError extends Error {
2
+ constructor(cause) {
3
+ super("An error occurred while loading route data");
4
+ this.cause = cause;
5
+ }
6
+ }
7
+ //# sourceMappingURL=errors.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"errors.js","sourceRoot":"","sources":["../../src/router/errors.ts"],"names":[],"mappings":"AAAA,MAAM,OAAO,uBAAwB,SAAQ,KAAK;IAChD,YAAY,KAAc;QACxB,KAAK,CAAC,4CAA4C,CAAC,CAAA;QACnD,IAAI,CAAC,KAAK,GAAG,KAAK,CAAA;IACpB,CAAC;CACF"}
@@ -0,0 +1,48 @@
1
+ import { type FileRouterContextType } from "./context.js";
2
+ import type { FileRouterConfig, PageConfig } from "./types.js";
3
+ export declare class FileRouterController {
4
+ private enableTransitions;
5
+ private pages;
6
+ private layouts;
7
+ private abortController;
8
+ private currentPage;
9
+ private currentPageProps;
10
+ private currentLayouts;
11
+ private loading;
12
+ private state;
13
+ private contextValue;
14
+ private cleanups;
15
+ private filePathToPageRoute;
16
+ private pageRouteToConfig;
17
+ private currentRoute;
18
+ constructor(props: FileRouterProps);
19
+ onPageConfigDefined<T extends PageConfig>(fp: string, config: T): void;
20
+ getContextValue(): FileRouterContextType;
21
+ getChildren(): Kiru.VNode | null;
22
+ dispose(): void;
23
+ private matchRoute;
24
+ private loadRoute;
25
+ private loadRouteData;
26
+ private navigate;
27
+ private setQuery;
28
+ }
29
+ export interface FileRouterProps {
30
+ /**
31
+ * The router configuration
32
+ * @example
33
+ * ```ts
34
+ *<FileRouter
35
+ config={{
36
+ dir: "/fbr-app", // optional, defaults to "/pages"
37
+ baseUrl: "/app", // optional, defaults to "/"
38
+ pages: import.meta.glob("/∗∗/index.tsx"),
39
+ layouts: import.meta.glob("/∗∗/layout.tsx"),
40
+ transition: true
41
+ }}
42
+ />
43
+ * ```
44
+ */
45
+ config: FileRouterConfig;
46
+ }
47
+ export declare function FileRouter(props: FileRouterProps): JSX.Element;
48
+ //# sourceMappingURL=fileRouter.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"fileRouter.d.ts","sourceRoot":"","sources":["../../src/router/fileRouter.ts"],"names":[],"mappings":"AAIA,OAAO,EAAiB,KAAK,qBAAqB,EAAE,MAAM,cAAc,CAAA;AAGxE,OAAO,KAAK,EAEV,gBAAgB,EAChB,UAAU,EAIX,MAAM,YAAY,CAAA;AAOnB,qBAAa,oBAAoB;IAC/B,OAAO,CAAC,iBAAiB,CAAS;IAClC,OAAO,CAAC,KAAK,CAAe;IAC5B,OAAO,CAAC,OAAO,CAAe;IAC9B,OAAO,CAAC,eAAe,CAAiB;IACxC,OAAO,CAAC,WAAW,CAIV;IACT,OAAO,CAAC,gBAAgB,CAA+B;IACvD,OAAO,CAAC,cAAc,CAAmB;IACzC,OAAO,CAAC,OAAO,CAAiB;IAChC,OAAO,CAAC,KAAK,CAAqB;IAClC,OAAO,CAAC,YAAY,CAA+B;IACnD,OAAO,CAAC,QAAQ,CAAqB;IACrC,OAAO,CAAC,mBAAmB,CAM1B;IACD,OAAO,CAAC,iBAAiB,CAAyB;IAClD,OAAO,CAAC,YAAY,CAAe;gBAEvB,KAAK,EAAE,eAAe;IA2D3B,mBAAmB,CAAC,CAAC,SAAS,UAAU,EAAE,EAAE,EAAE,MAAM,EAAE,MAAM,EAAE,CAAC;IA+B/D,eAAe;IAIf,WAAW;IAgBX,OAAO;IAId,OAAO,CAAC,UAAU;YA+BJ,SAAS;YAkGT,aAAa;YAkCb,QAAQ;IActB,OAAO,CAAC,QAAQ;CAQjB;AAED,MAAM,WAAW,eAAe;IAC9B;;;;;;;;;;;;;;OAcG;IACH,MAAM,EAAE,gBAAgB,CAAA;CACzB;AAED,wBAAgB,UAAU,CAAC,KAAK,EAAE,eAAe,GAAG,GAAG,CAAC,OAAO,CAS9D"}
@@ -0,0 +1,311 @@
1
+ import { Signal, computed, flushSync } from "../index.js";
2
+ import { __DEV__ } from "../env.js";
3
+ import { createElement } from "../element.js";
4
+ import { useState, useEffect } from "../hooks/index.js";
5
+ import { RouterContext } from "./context.js";
6
+ import { FileRouterDataLoadError } from "./errors.js";
7
+ import { fileRouterInstance } from "./globals.js";
8
+ export class FileRouterController {
9
+ constructor(props) {
10
+ this.cleanups = [];
11
+ fileRouterInstance.current = this;
12
+ this.pages = {};
13
+ this.layouts = {};
14
+ this.abortController = new AbortController();
15
+ this.currentPage = new Signal(null);
16
+ this.currentPageProps = new Signal({});
17
+ this.currentLayouts = new Signal([]);
18
+ this.loading = new Signal(true);
19
+ this.state = new Signal({
20
+ path: window.location.pathname,
21
+ params: {},
22
+ query: {},
23
+ signal: this.abortController.signal,
24
+ });
25
+ this.contextValue = computed(() => ({
26
+ state: this.state.value,
27
+ navigate: this.navigate.bind(this),
28
+ setQuery: this.setQuery.bind(this),
29
+ reload: (options) => this.loadRoute(void 0, void 0, options?.transition),
30
+ }));
31
+ this.filePathToPageRoute = new Map();
32
+ this.pageRouteToConfig = new Map();
33
+ this.currentRoute = null;
34
+ const { pages, layouts, dir = "/pages", baseUrl = "/", transition, } = props.config;
35
+ this.enableTransitions = !!transition;
36
+ const [normalizedDir, normalizedBaseUrl] = [
37
+ normalizePrefixPath(dir),
38
+ normalizePrefixPath(baseUrl),
39
+ ];
40
+ debugger;
41
+ this.pages = formatViteImportMap(pages, normalizedDir, normalizedBaseUrl);
42
+ this.layouts = formatViteImportMap(layouts, normalizedDir, normalizedBaseUrl);
43
+ this.loadRoute();
44
+ const handlePopState = () => this.loadRoute();
45
+ window.addEventListener("popstate", handlePopState);
46
+ this.cleanups.push(() => window.removeEventListener("popstate", handlePopState));
47
+ }
48
+ onPageConfigDefined(fp, config) {
49
+ const existing = this.filePathToPageRoute.get(fp);
50
+ if (existing === undefined) {
51
+ const route = this.currentRoute;
52
+ if (!route)
53
+ return;
54
+ this.filePathToPageRoute.set(fp, { route, config });
55
+ return;
56
+ }
57
+ const curPage = this.currentPage.value;
58
+ if (curPage?.route === existing.route && config.loader) {
59
+ const p = this.currentPageProps.value;
60
+ let transition = this.enableTransitions;
61
+ if (config.loader.transition !== undefined) {
62
+ transition = config.loader.transition;
63
+ }
64
+ const props = {
65
+ ...p,
66
+ loading: true,
67
+ data: null,
68
+ error: null,
69
+ };
70
+ handleStateTransition(this.state.value.signal, transition, () => {
71
+ this.currentPageProps.value = props;
72
+ });
73
+ this.loadRouteData(config.loader, props, this.state.value, transition);
74
+ }
75
+ this.pageRouteToConfig.set(existing.route, config);
76
+ }
77
+ getContextValue() {
78
+ return this.contextValue.value;
79
+ }
80
+ getChildren() {
81
+ const page = this.currentPage.value, props = this.currentPageProps.value, layouts = this.currentLayouts.value;
82
+ if (page) {
83
+ // Wrap component with layouts (outermost first)
84
+ return layouts.reduceRight((children, Layout) => createElement(Layout, { children }), createElement(page.component, props));
85
+ }
86
+ return null;
87
+ }
88
+ dispose() {
89
+ this.cleanups.forEach((cleanup) => cleanup());
90
+ }
91
+ matchRoute(pathSegments) {
92
+ outer: for (const [route, pageModuleLoader] of Object.entries(this.pages)) {
93
+ const routeSegments = route.split("/").filter(Boolean);
94
+ const pathMatchingSegments = routeSegments.filter((seg) => !seg.startsWith("(") && !seg.endsWith(")"));
95
+ if (pathMatchingSegments.length !== pathSegments.length) {
96
+ continue;
97
+ }
98
+ const params = {};
99
+ for (let i = 0; i < pathMatchingSegments.length; i++) {
100
+ const routeSeg = pathMatchingSegments[i];
101
+ if (routeSeg.startsWith(":")) {
102
+ const key = routeSeg.slice(1);
103
+ params[key] = pathSegments[i];
104
+ continue;
105
+ }
106
+ if (routeSeg !== pathSegments[i]) {
107
+ continue outer;
108
+ }
109
+ }
110
+ return { route, pageModuleLoader, params, routeSegments };
111
+ }
112
+ return null;
113
+ }
114
+ async loadRoute(path = window.location.pathname, props = {}, enableTransition = this.enableTransitions) {
115
+ this.loading.value = true;
116
+ this.abortController?.abort();
117
+ const query = parseQuery(window.location.search);
118
+ const controller = (this.abortController = new AbortController());
119
+ const signal = controller.signal;
120
+ try {
121
+ const pathSegments = path.split("/").filter(Boolean);
122
+ const routeMatch = this.matchRoute(pathSegments);
123
+ if (!routeMatch) {
124
+ const _404 = this.matchRoute(["404"]);
125
+ if (!_404) {
126
+ if (__DEV__) {
127
+ console.error(`No 404 route defined (path: ${path}). See https://kirujs.dev/404 for more information.`);
128
+ }
129
+ return;
130
+ }
131
+ const errorProps = {
132
+ source: { path },
133
+ };
134
+ return this.navigate("/404", { replace: true, props: errorProps });
135
+ }
136
+ const { route, pageModuleLoader, params, routeSegments } = routeMatch;
137
+ this.currentRoute = route;
138
+ const pagePromise = pageModuleLoader();
139
+ const layoutPromises = ["/", ...routeSegments].reduce((acc, _, i) => {
140
+ const layoutPath = "/" + routeSegments.slice(0, i).join("/");
141
+ const layoutLoad = this.layouts[layoutPath];
142
+ if (!layoutLoad) {
143
+ return acc;
144
+ }
145
+ return [...acc, layoutLoad()];
146
+ }, []);
147
+ const [page, ...layouts] = await Promise.all([
148
+ pagePromise,
149
+ ...layoutPromises,
150
+ ]);
151
+ this.currentRoute = null;
152
+ if (controller.signal.aborted)
153
+ return;
154
+ if (typeof page.default !== "function") {
155
+ throw new Error("Route component must be a default exported function");
156
+ }
157
+ const routerState = {
158
+ path,
159
+ params,
160
+ query,
161
+ signal,
162
+ };
163
+ let config = page.config;
164
+ if (this.pageRouteToConfig.has(route)) {
165
+ config = this.pageRouteToConfig.get(route);
166
+ }
167
+ if (config?.loader) {
168
+ props = { ...props, loading: true, data: null, error: null };
169
+ this.loadRouteData(config.loader, props, routerState, enableTransition);
170
+ }
171
+ handleStateTransition(signal, enableTransition, () => {
172
+ this.currentPage.value = {
173
+ component: page.default,
174
+ config,
175
+ route: "/" + routeSegments.join("/"),
176
+ };
177
+ this.state.value = routerState;
178
+ this.currentPageProps.value = props;
179
+ this.currentLayouts.value = layouts
180
+ .filter((m) => typeof m.default === "function")
181
+ .map((m) => m.default);
182
+ });
183
+ }
184
+ catch (error) {
185
+ console.error("Failed to load route component:", error);
186
+ this.currentPage.value = null;
187
+ }
188
+ finally {
189
+ this.loading.value = false;
190
+ }
191
+ }
192
+ async loadRouteData(loader, props, routerState, enableTransition = this.enableTransitions) {
193
+ loader
194
+ .load(routerState)
195
+ .then((data) => ({ data, error: null }), (error) => ({
196
+ data: null,
197
+ error: new FileRouterDataLoadError(error),
198
+ }))
199
+ .then(({ data, error }) => {
200
+ if (routerState.signal.aborted)
201
+ return;
202
+ let transition = enableTransition;
203
+ if (loader.transition !== undefined) {
204
+ transition = loader.transition;
205
+ }
206
+ handleStateTransition(routerState.signal, transition, () => {
207
+ this.currentPageProps.value = {
208
+ ...props,
209
+ loading: false,
210
+ data,
211
+ error,
212
+ };
213
+ });
214
+ });
215
+ }
216
+ async navigate(path, options) {
217
+ const f = options?.replace ? "replaceState" : "pushState";
218
+ window.history[f]({}, "", path);
219
+ window.dispatchEvent(new PopStateEvent("popstate", { state: {} }));
220
+ return this.loadRoute(path, options?.props, options?.transition);
221
+ }
222
+ setQuery(query) {
223
+ const queryString = buildQueryString(query);
224
+ const newUrl = `${this.state.value.path}${queryString ? `?${queryString}` : ""}`;
225
+ window.history.pushState(null, "", newUrl);
226
+ this.state.value = { ...this.state.value, query };
227
+ }
228
+ }
229
+ export function FileRouter(props) {
230
+ const [controller] = useState(() => new FileRouterController(props));
231
+ useEffect(() => () => controller.dispose(), [controller]);
232
+ return createElement(RouterContext.Provider, { value: controller.getContextValue() }, controller.getChildren());
233
+ }
234
+ // Utility functions
235
+ function parseQuery(search) {
236
+ const params = new URLSearchParams(search);
237
+ const query = {};
238
+ for (const [key, value] of params.entries()) {
239
+ if (query[key]) {
240
+ // Convert to array if multiple values
241
+ if (Array.isArray(query[key])) {
242
+ ;
243
+ query[key].push(value);
244
+ }
245
+ else {
246
+ query[key] = [query[key], value];
247
+ }
248
+ }
249
+ else {
250
+ query[key] = value;
251
+ }
252
+ }
253
+ return query;
254
+ }
255
+ function buildQueryString(query) {
256
+ const params = new URLSearchParams();
257
+ for (const [key, value] of Object.entries(query)) {
258
+ if (value !== undefined) {
259
+ if (Array.isArray(value)) {
260
+ value.forEach((v) => params.append(key, v));
261
+ }
262
+ else {
263
+ params.set(key, value);
264
+ }
265
+ }
266
+ }
267
+ return params.toString();
268
+ }
269
+ function formatViteImportMap(map, dir, baseUrl) {
270
+ return Object.keys(map).reduce((acc, key) => {
271
+ let k = key;
272
+ const dirIndex = k.indexOf(dir);
273
+ if (dirIndex === -1) {
274
+ return acc;
275
+ }
276
+ k = k.slice(dirIndex + dir.length);
277
+ while (k.startsWith("/")) {
278
+ k = k.slice(1);
279
+ }
280
+ k = k.split("/").slice(0, -1).join("/"); // remove filename
281
+ k = k.replace(/\[([^\]]+)\]/g, ":$1"); // replace [param] with :param
282
+ return {
283
+ ...acc,
284
+ [baseUrl + k]: map[key],
285
+ };
286
+ }, {});
287
+ }
288
+ function normalizePrefixPath(path) {
289
+ while (path.startsWith(".")) {
290
+ path = path.slice(1);
291
+ }
292
+ path = `/${path}/`;
293
+ while (path.startsWith("//")) {
294
+ path = path.slice(1);
295
+ }
296
+ while (path.endsWith("//")) {
297
+ path = path.slice(0, -1);
298
+ }
299
+ return path;
300
+ }
301
+ function handleStateTransition(signal, enableTransition, callback) {
302
+ if (!enableTransition || typeof document.startViewTransition !== "function") {
303
+ return callback();
304
+ }
305
+ const vt = document.startViewTransition(() => {
306
+ callback();
307
+ flushSync();
308
+ });
309
+ signal.addEventListener("abort", () => vt.skipTransition());
310
+ }
311
+ //# sourceMappingURL=fileRouter.js.map