@hyeonqyu/typed-router-core 0.2.5 → 0.2.7

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/dist/index.d.mts CHANGED
@@ -1,10 +1,19 @@
1
1
  import * as react_jsx_runtime from 'react/jsx-runtime';
2
2
  import { ReactNode } from 'react';
3
3
 
4
- type Paths<TObject, TSplitter extends string, TPrev extends string = '', TMaxDepth extends number = 5, TDepth extends unknown[] = []> = TDepth['length'] extends TMaxDepth ? never : TSplitter | {
4
+ type Paths<TObject, TSplitter extends string, TPrev extends string = '', TMaxDepth extends number = 10, TDepth extends unknown[] = []> = TDepth['length'] extends TMaxDepth ? never : TSplitter | {
5
5
  [K in keyof TObject & string]: TObject[K] extends object ? `${TSplitter}${TPrev extends '' ? K : `${TPrev}${TSplitter}${K}`}` | Paths<TObject[K], TSplitter, TPrev extends '' ? K : `${TPrev}${TSplitter}${K}`, TMaxDepth, [...TDepth, 1]> : `${TSplitter}${TPrev extends '' ? K : `${TPrev}${TSplitter}${K}`}`;
6
6
  }[keyof TObject & string];
7
- type PathValue<T, P extends string, S extends string, TMaxDepth extends number = 5, TDepth extends unknown[] = []> = TDepth['length'] extends TMaxDepth ? never : P extends S ? T : P extends `${S}${infer Rest}` ? Rest extends `${infer K}${S}${infer R}` ? K extends keyof T ? PathValue<T[K], `${S}${R}`, S, TMaxDepth, [...TDepth, 1]> : never : Rest extends keyof T ? T[Rest] : never : never;
7
+ type PathValue<T, P extends string, S extends string, TMaxDepth extends number = 10, TDepth extends unknown[] = []> = TDepth['length'] extends TMaxDepth ? never : P extends S ? T : P extends `${S}${infer Rest}` ? Rest extends `${infer K}${S}${infer R}` ? K extends keyof T ? PathValue<T[K], `${S}${R}`, S, TMaxDepth, [...TDepth, 1]> : never : Rest extends keyof T ? T[Rest] : never : never;
8
+
9
+ type QueryValue = string | number | boolean | readonly (string | number | boolean)[];
10
+ type QueryParams = Record<string, QueryValue>;
11
+
12
+ declare const getSafely: <TObject, TSplitter extends string, TPath extends string & Paths<TObject, TSplitter>>(splitter: TSplitter, obj: TObject, path: TPath) => PathValue<TObject, TPath, TSplitter>;
13
+ type QueryStringOptions = {
14
+ includeQuestionMark?: boolean;
15
+ };
16
+ declare const toQueryString: (query: QueryParams, options?: QueryStringOptions) => string;
8
17
 
9
18
  type BaseMetadata = NonNullable<unknown>;
10
19
  type RouteNode<TMetadata extends BaseMetadata, TContext> = {
@@ -29,16 +38,15 @@ type RouteTree<TMetadata extends BaseMetadata, TContext, TSubRouteTree extends P
29
38
  type RouteTreeWithoutMetadata<T> = {
30
39
  [K in keyof T as K extends '_metadata' ? never : K]: T[K] extends object ? RouteTreeWithoutMetadata<T[K]> : T[K];
31
40
  };
32
- type RoutePathname<TMetadata extends BaseMetadata, TContext, TRouteTree extends PartialRouteTree<TMetadata, TContext>> = [
33
- TRouteTree
34
- ] extends [never] ? never : Paths<RouteTreeWithoutMetadata<TRouteTree>, '/', ''>;
41
+ type RoutePathname<TMetadata extends BaseMetadata, TContext, TRouteTree extends PartialRouteTree<TMetadata, TContext>> = Paths<RouteTreeWithoutMetadata<TRouteTree>, '/', ''>;
35
42
  type SimplifyPathname<T> = T extends string ? T : never;
43
+ type ResolvedRouteTree<TMetadata extends BaseMetadata, TContext, TRouteTree extends PartialRouteTree<TMetadata, TContext>> = TRouteTree & RouteTree<TMetadata, TContext, TRouteTree>;
36
44
 
37
45
  declare const createAppRoutes: <TMetadata extends BaseMetadata, TContext>() => <TRouteTree extends PartialRouteTree<TMetadata, TContext>>(appRoutes: TRouteTree & RouteTree<TMetadata, TContext, TRouteTree>) => {
38
46
  AppRoutesProvider: ({ children }: {
39
47
  children: ReactNode;
40
48
  }) => react_jsx_runtime.JSX.Element;
41
- useAppRoutes: () => TRouteTree & RouteTree<TMetadata, TContext, TRouteTree, TRouteTree>;
49
+ useAppRoutes: () => ResolvedRouteTree<TMetadata, TContext, TRouteTree>;
42
50
  useCurrentRouteNode: <TPath extends RoutePathname<TMetadata, TContext, TRouteTree>>(pathname: TPath) => PathValue<TRouteTree, TPath, "/">;
43
51
  _types: {
44
52
  AppRoutesMetadata: TMetadata;
@@ -47,4 +55,4 @@ declare const createAppRoutes: <TMetadata extends BaseMetadata, TContext>() => <
47
55
  };
48
56
  };
49
57
 
50
- export { type BaseMetadata, type PartialRouteTree, type RouteNode, type RouteNodeMetadata, type RoutePathname, type RouteTree, type SimplifyPathname, createAppRoutes };
58
+ export { type BaseMetadata, type PartialRouteTree, type PathValue, type Paths, type QueryParams, type QueryStringOptions, type QueryValue, type ResolvedRouteTree, type RouteNode, type RouteNodeMetadata, type RoutePathname, type RouteTree, type SimplifyPathname, createAppRoutes, getSafely, toQueryString };
package/dist/index.d.ts CHANGED
@@ -1,10 +1,19 @@
1
1
  import * as react_jsx_runtime from 'react/jsx-runtime';
2
2
  import { ReactNode } from 'react';
3
3
 
4
- type Paths<TObject, TSplitter extends string, TPrev extends string = '', TMaxDepth extends number = 5, TDepth extends unknown[] = []> = TDepth['length'] extends TMaxDepth ? never : TSplitter | {
4
+ type Paths<TObject, TSplitter extends string, TPrev extends string = '', TMaxDepth extends number = 10, TDepth extends unknown[] = []> = TDepth['length'] extends TMaxDepth ? never : TSplitter | {
5
5
  [K in keyof TObject & string]: TObject[K] extends object ? `${TSplitter}${TPrev extends '' ? K : `${TPrev}${TSplitter}${K}`}` | Paths<TObject[K], TSplitter, TPrev extends '' ? K : `${TPrev}${TSplitter}${K}`, TMaxDepth, [...TDepth, 1]> : `${TSplitter}${TPrev extends '' ? K : `${TPrev}${TSplitter}${K}`}`;
6
6
  }[keyof TObject & string];
7
- type PathValue<T, P extends string, S extends string, TMaxDepth extends number = 5, TDepth extends unknown[] = []> = TDepth['length'] extends TMaxDepth ? never : P extends S ? T : P extends `${S}${infer Rest}` ? Rest extends `${infer K}${S}${infer R}` ? K extends keyof T ? PathValue<T[K], `${S}${R}`, S, TMaxDepth, [...TDepth, 1]> : never : Rest extends keyof T ? T[Rest] : never : never;
7
+ type PathValue<T, P extends string, S extends string, TMaxDepth extends number = 10, TDepth extends unknown[] = []> = TDepth['length'] extends TMaxDepth ? never : P extends S ? T : P extends `${S}${infer Rest}` ? Rest extends `${infer K}${S}${infer R}` ? K extends keyof T ? PathValue<T[K], `${S}${R}`, S, TMaxDepth, [...TDepth, 1]> : never : Rest extends keyof T ? T[Rest] : never : never;
8
+
9
+ type QueryValue = string | number | boolean | readonly (string | number | boolean)[];
10
+ type QueryParams = Record<string, QueryValue>;
11
+
12
+ declare const getSafely: <TObject, TSplitter extends string, TPath extends string & Paths<TObject, TSplitter>>(splitter: TSplitter, obj: TObject, path: TPath) => PathValue<TObject, TPath, TSplitter>;
13
+ type QueryStringOptions = {
14
+ includeQuestionMark?: boolean;
15
+ };
16
+ declare const toQueryString: (query: QueryParams, options?: QueryStringOptions) => string;
8
17
 
9
18
  type BaseMetadata = NonNullable<unknown>;
10
19
  type RouteNode<TMetadata extends BaseMetadata, TContext> = {
@@ -29,16 +38,15 @@ type RouteTree<TMetadata extends BaseMetadata, TContext, TSubRouteTree extends P
29
38
  type RouteTreeWithoutMetadata<T> = {
30
39
  [K in keyof T as K extends '_metadata' ? never : K]: T[K] extends object ? RouteTreeWithoutMetadata<T[K]> : T[K];
31
40
  };
32
- type RoutePathname<TMetadata extends BaseMetadata, TContext, TRouteTree extends PartialRouteTree<TMetadata, TContext>> = [
33
- TRouteTree
34
- ] extends [never] ? never : Paths<RouteTreeWithoutMetadata<TRouteTree>, '/', ''>;
41
+ type RoutePathname<TMetadata extends BaseMetadata, TContext, TRouteTree extends PartialRouteTree<TMetadata, TContext>> = Paths<RouteTreeWithoutMetadata<TRouteTree>, '/', ''>;
35
42
  type SimplifyPathname<T> = T extends string ? T : never;
43
+ type ResolvedRouteTree<TMetadata extends BaseMetadata, TContext, TRouteTree extends PartialRouteTree<TMetadata, TContext>> = TRouteTree & RouteTree<TMetadata, TContext, TRouteTree>;
36
44
 
37
45
  declare const createAppRoutes: <TMetadata extends BaseMetadata, TContext>() => <TRouteTree extends PartialRouteTree<TMetadata, TContext>>(appRoutes: TRouteTree & RouteTree<TMetadata, TContext, TRouteTree>) => {
38
46
  AppRoutesProvider: ({ children }: {
39
47
  children: ReactNode;
40
48
  }) => react_jsx_runtime.JSX.Element;
41
- useAppRoutes: () => TRouteTree & RouteTree<TMetadata, TContext, TRouteTree, TRouteTree>;
49
+ useAppRoutes: () => ResolvedRouteTree<TMetadata, TContext, TRouteTree>;
42
50
  useCurrentRouteNode: <TPath extends RoutePathname<TMetadata, TContext, TRouteTree>>(pathname: TPath) => PathValue<TRouteTree, TPath, "/">;
43
51
  _types: {
44
52
  AppRoutesMetadata: TMetadata;
@@ -47,4 +55,4 @@ declare const createAppRoutes: <TMetadata extends BaseMetadata, TContext>() => <
47
55
  };
48
56
  };
49
57
 
50
- export { type BaseMetadata, type PartialRouteTree, type RouteNode, type RouteNodeMetadata, type RoutePathname, type RouteTree, type SimplifyPathname, createAppRoutes };
58
+ export { type BaseMetadata, type PartialRouteTree, type PathValue, type Paths, type QueryParams, type QueryStringOptions, type QueryValue, type ResolvedRouteTree, type RouteNode, type RouteNodeMetadata, type RoutePathname, type RouteTree, type SimplifyPathname, createAppRoutes, getSafely, toQueryString };
package/dist/index.js CHANGED
@@ -3,10 +3,11 @@
3
3
  var react = require('react');
4
4
  var jsxRuntime = require('react/jsx-runtime');
5
5
 
6
- // src/routes.utils.tsx
6
+ var __defProp = Object.defineProperty;
7
+ var __name = (target, value) => __defProp(target, "name", { value, configurable: true });
7
8
 
8
9
  // src/object.utils.ts
9
- var getSafely = (splitter, obj, path) => {
10
+ var getSafely = /* @__PURE__ */ __name((splitter, obj, path) => {
10
11
  if (path === "" || path === splitter) return obj;
11
12
  const keys = path.split(splitter);
12
13
  let value = obj;
@@ -18,27 +19,52 @@ var getSafely = (splitter, obj, path) => {
18
19
  }
19
20
  }
20
21
  return value;
21
- };
22
- var createAppRoutes = () => (appRoutes) => {
22
+ }, "getSafely");
23
+ var toQueryString = /* @__PURE__ */ __name((query, options = { includeQuestionMark: true }) => {
24
+ const params = [];
25
+ Object.entries(query).forEach(([key, value]) => {
26
+ if (value === void 0 || value === null) {
27
+ return;
28
+ }
29
+ const encodedKey = encodeURIComponent(key);
30
+ if (Array.isArray(value)) {
31
+ value.forEach((item) => {
32
+ if (item !== void 0 && item !== null) {
33
+ params.push(`${encodedKey}=${encodeURIComponent(String(item))}`);
34
+ }
35
+ });
36
+ } else {
37
+ params.push(`${encodedKey}=${encodeURIComponent(String(value))}`);
38
+ }
39
+ });
40
+ if (params.length === 0) {
41
+ return "";
42
+ }
43
+ const queryString = params.join("&");
44
+ return options.includeQuestionMark ? `?${queryString}` : queryString;
45
+ }, "toQueryString");
46
+ var createAppRoutes = /* @__PURE__ */ __name(() => (appRoutes) => {
23
47
  const Context = react.createContext(appRoutes);
24
- const useAppRoutes = () => {
48
+ const useAppRoutes = /* @__PURE__ */ __name(() => {
25
49
  return react.useContext(Context);
26
- };
27
- const useCurrentRouteNode = (pathname) => {
50
+ }, "useAppRoutes");
51
+ const useCurrentRouteNode = /* @__PURE__ */ __name((pathname) => {
28
52
  const routes = react.useContext(Context);
29
53
  return getSafely("/", routes, pathname);
30
- };
31
- const AppRoutesProvider = ({ children }) => {
54
+ }, "useCurrentRouteNode");
55
+ const AppRoutesProvider = /* @__PURE__ */ __name(({ children }) => {
32
56
  return /* @__PURE__ */ jsxRuntime.jsx(Context.Provider, { value: appRoutes, children });
33
- };
57
+ }, "AppRoutesProvider");
34
58
  return {
35
59
  AppRoutesProvider,
36
60
  useAppRoutes,
37
61
  useCurrentRouteNode,
38
62
  _types: {}
39
63
  };
40
- };
64
+ }, "createAppRoutes");
41
65
 
42
66
  exports.createAppRoutes = createAppRoutes;
67
+ exports.getSafely = getSafely;
68
+ exports.toQueryString = toQueryString;
43
69
  //# sourceMappingURL=index.js.map
44
70
  //# sourceMappingURL=index.js.map
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/object.utils.ts","../src/routes.utils.tsx"],"names":["createContext","useContext"],"mappings":";;;;;;;;AAEO,IAAM,SAAA,GAAY,CACvB,QAAA,EACA,GAAA,EACA,IAAA,KACyC;AACzC,EAAA,IAAI,IAAA,KAAS,EAAA,IAAM,IAAA,KAAS,QAAA,EAAU,OAAO,GAAA;AAE7C,EAAA,MAAM,IAAA,GAAQ,IAAA,CAAgB,KAAA,CAAM,QAAQ,CAAA;AAC5C,EAAA,IAAI,KAAA,GAAiB,GAAA;AAErB,EAAA,KAAA,MAAW,OAAO,IAAA,EAAM;AACtB,IAAA,IAAI,OAAO,KAAA,KAAU,QAAA,IAAY,KAAA,KAAU,IAAA,IAAQ,OAAO,KAAA,EAAO;AAC/D,MAAA,KAAA,GAAS,MAAkC,GAAG,CAAA;AAAA,IAChD,CAAA,MAAO;AACL,MAAA,OAAO,MAAA;AAAA,IACT;AAAA,EACF;AAEA,EAAA,OAAO,KAAA;AACT,CAAA;AChBO,IAAM,eAAA,GACX,MACA,CAA2D,SAAA,KAAuE;AAChI,EAAA,MAAM,OAAA,GAAUA,oBAAuE,SAAS,CAAA;AAEhG,EAAA,MAAM,eAAe,MAAM;AACzB,IAAA,OAAOC,iBAAW,OAAO,CAAA;AAAA,EAC3B,CAAA;AAEA,EAAA,MAAM,mBAAA,GAAsB,CAA+D,QAAA,KAAoB;AAC7G,IAAA,MAAM,MAAA,GAASA,iBAAW,OAAO,CAAA;AACjC,IAAA,OAAO,SAAA,CAAU,GAAA,EAAK,MAAA,EAAQ,QAAQ,CAAA;AAAA,EACxC,CAAA;AAEA,EAAA,MAAM,iBAAA,GAAoB,CAAC,EAAE,QAAA,EAAS,KAA+B;AACnE,IAAA,sCAAQ,OAAA,CAAQ,QAAA,EAAR,EAAiB,KAAA,EAAO,WAAY,QAAA,EAAS,CAAA;AAAA,EACvD,CAAA;AAEA,EAAA,OAAO;AAAA,IACL,iBAAA;AAAA,IACA,YAAA;AAAA,IACA,mBAAA;AAAA,IACA,QAAQ;AAAC,GAKX;AACF","file":"index.js","sourcesContent":["import { Paths, PathValue } from './path.types';\n\nexport const getSafely = <TObject, TSplitter extends string, TPath extends string & Paths<TObject, TSplitter>>(\n splitter: TSplitter,\n obj: TObject,\n path: TPath,\n): PathValue<TObject, TPath, TSplitter> => {\n if (path === '' || path === splitter) return obj as PathValue<TObject, TPath, TSplitter>;\n\n const keys = (path as string).split(splitter);\n let value: unknown = obj;\n\n for (const key of keys) {\n if (typeof value === 'object' && value !== null && key in value) {\n value = (value as Record<string, unknown>)[key];\n } else {\n return undefined as PathValue<TObject, TPath, TSplitter>;\n }\n }\n\n return value as PathValue<TObject, TPath, TSplitter>;\n};\n\nexport type QueryValue = string | number | boolean | readonly (string | number | boolean)[];\nexport type QueryParams = Record<string, QueryValue>;\nexport type ToQueryStringOptions = {\n includeQuestionMark?: boolean;\n};\n\nexport const toQueryString = (query: QueryParams, options: ToQueryStringOptions = { includeQuestionMark: true }): string => {\n const params: string[] = [];\n\n Object.entries(query).forEach(([key, value]) => {\n if (value === undefined || value === null) {\n return;\n }\n\n const encodedKey = encodeURIComponent(key);\n\n if (Array.isArray(value)) {\n value.forEach((item) => {\n if (item !== undefined && item !== null) {\n params.push(`${encodedKey}=${encodeURIComponent(String(item))}`);\n }\n });\n } else {\n params.push(`${encodedKey}=${encodeURIComponent(String(value))}`);\n }\n });\n\n if (params.length === 0) {\n return '';\n }\n\n const queryString = params.join('&');\n return options.includeQuestionMark ? `?${queryString}` : queryString;\n};\n","import { createContext, ReactNode, useContext } from 'react';\nimport { getSafely } from './object.utils';\nimport { PathValue } from './path.types';\nimport { BaseMetadata, PartialRouteTree, RoutePathname, RouteTree } from './routes.types';\n\nexport const createAppRoutes =\n <TMetadata extends BaseMetadata, TContext>() =>\n <TRouteTree extends PartialRouteTree<TMetadata, TContext>>(appRoutes: TRouteTree & RouteTree<TMetadata, TContext, TRouteTree>) => {\n const Context = createContext<TRouteTree & RouteTree<TMetadata, TContext, TRouteTree>>(appRoutes);\n\n const useAppRoutes = () => {\n return useContext(Context);\n };\n\n const useCurrentRouteNode = <TPath extends RoutePathname<TMetadata, TContext, TRouteTree>>(pathname: TPath) => {\n const routes = useContext(Context);\n return getSafely('/', routes, pathname) as PathValue<TRouteTree, TPath, '/'>;\n };\n\n const AppRoutesProvider = ({ children }: { children: ReactNode }) => {\n return <Context.Provider value={appRoutes}>{children}</Context.Provider>;\n };\n\n return {\n AppRoutesProvider,\n useAppRoutes,\n useCurrentRouteNode,\n _types: {} as {\n AppRoutesMetadata: TMetadata;\n AppRoutesContext: TContext;\n AppRoutesPathname: RoutePathname<TMetadata, TContext, TRouteTree>;\n },\n };\n };\n"]}
1
+ {"version":3,"sources":["../src/object.utils.ts","../src/routes.utils.tsx"],"names":["createContext","useContext"],"mappings":"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","file":"index.js","sourcesContent":["import { Paths, PathValue } from './path.types';\nimport { QueryParams } from './query.types';\n\nexport const getSafely = <TObject, TSplitter extends string, TPath extends string & Paths<TObject, TSplitter>>(\n splitter: TSplitter,\n obj: TObject,\n path: TPath,\n): PathValue<TObject, TPath, TSplitter> => {\n if (path === '' || path === splitter) return obj as PathValue<TObject, TPath, TSplitter>;\n\n const keys = (path as string).split(splitter);\n let value: unknown = obj;\n\n for (const key of keys) {\n if (typeof value === 'object' && value !== null && key in value) {\n value = (value as Record<string, unknown>)[key];\n } else {\n return undefined as PathValue<TObject, TPath, TSplitter>;\n }\n }\n\n return value as PathValue<TObject, TPath, TSplitter>;\n};\n\nexport type QueryStringOptions = {\n includeQuestionMark?: boolean;\n};\n\nexport const toQueryString = (query: QueryParams, options: QueryStringOptions = { includeQuestionMark: true }): string => {\n const params: string[] = [];\n\n Object.entries(query).forEach(([key, value]) => {\n if (value === undefined || value === null) {\n return;\n }\n\n const encodedKey = encodeURIComponent(key);\n\n if (Array.isArray(value)) {\n value.forEach((item) => {\n if (item !== undefined && item !== null) {\n params.push(`${encodedKey}=${encodeURIComponent(String(item))}`);\n }\n });\n } else {\n params.push(`${encodedKey}=${encodeURIComponent(String(value))}`);\n }\n });\n\n if (params.length === 0) {\n return '';\n }\n\n const queryString = params.join('&');\n return options.includeQuestionMark ? `?${queryString}` : queryString;\n};\n","import { createContext, ReactNode, useContext } from 'react';\nimport { getSafely } from './object.utils';\nimport { PathValue } from './path.types';\nimport { BaseMetadata, PartialRouteTree, ResolvedRouteTree, RoutePathname, RouteTree } from './routes.types';\n\nexport const createAppRoutes =\n <TMetadata extends BaseMetadata, TContext>() =>\n <TRouteTree extends PartialRouteTree<TMetadata, TContext>>(appRoutes: TRouteTree & RouteTree<TMetadata, TContext, TRouteTree>) => {\n type Routes = ResolvedRouteTree<TMetadata, TContext, TRouteTree>;\n type Pathname = RoutePathname<TMetadata, TContext, TRouteTree>;\n\n const Context = createContext<Routes>(appRoutes as Routes);\n\n const useAppRoutes = (): Routes => {\n return useContext(Context);\n };\n\n const useCurrentRouteNode = <TPath extends Pathname>(pathname: TPath): PathValue<TRouteTree, TPath, '/'> => {\n const routes = useContext(Context);\n return getSafely('/', routes, pathname) as PathValue<TRouteTree, TPath, '/'>;\n };\n\n const AppRoutesProvider = ({ children }: { children: ReactNode }) => {\n return <Context.Provider value={appRoutes as Routes}>{children}</Context.Provider>;\n };\n\n return {\n AppRoutesProvider,\n useAppRoutes,\n useCurrentRouteNode,\n _types: {} as {\n AppRoutesMetadata: TMetadata;\n AppRoutesContext: TContext;\n AppRoutesPathname: Pathname;\n },\n };\n };\n"]}
package/dist/index.mjs CHANGED
@@ -1,10 +1,11 @@
1
1
  import { createContext, useContext } from 'react';
2
2
  import { jsx } from 'react/jsx-runtime';
3
3
 
4
- // src/routes.utils.tsx
4
+ var __defProp = Object.defineProperty;
5
+ var __name = (target, value) => __defProp(target, "name", { value, configurable: true });
5
6
 
6
7
  // src/object.utils.ts
7
- var getSafely = (splitter, obj, path) => {
8
+ var getSafely = /* @__PURE__ */ __name((splitter, obj, path) => {
8
9
  if (path === "" || path === splitter) return obj;
9
10
  const keys = path.split(splitter);
10
11
  let value = obj;
@@ -16,27 +17,50 @@ var getSafely = (splitter, obj, path) => {
16
17
  }
17
18
  }
18
19
  return value;
19
- };
20
- var createAppRoutes = () => (appRoutes) => {
20
+ }, "getSafely");
21
+ var toQueryString = /* @__PURE__ */ __name((query, options = { includeQuestionMark: true }) => {
22
+ const params = [];
23
+ Object.entries(query).forEach(([key, value]) => {
24
+ if (value === void 0 || value === null) {
25
+ return;
26
+ }
27
+ const encodedKey = encodeURIComponent(key);
28
+ if (Array.isArray(value)) {
29
+ value.forEach((item) => {
30
+ if (item !== void 0 && item !== null) {
31
+ params.push(`${encodedKey}=${encodeURIComponent(String(item))}`);
32
+ }
33
+ });
34
+ } else {
35
+ params.push(`${encodedKey}=${encodeURIComponent(String(value))}`);
36
+ }
37
+ });
38
+ if (params.length === 0) {
39
+ return "";
40
+ }
41
+ const queryString = params.join("&");
42
+ return options.includeQuestionMark ? `?${queryString}` : queryString;
43
+ }, "toQueryString");
44
+ var createAppRoutes = /* @__PURE__ */ __name(() => (appRoutes) => {
21
45
  const Context = createContext(appRoutes);
22
- const useAppRoutes = () => {
46
+ const useAppRoutes = /* @__PURE__ */ __name(() => {
23
47
  return useContext(Context);
24
- };
25
- const useCurrentRouteNode = (pathname) => {
48
+ }, "useAppRoutes");
49
+ const useCurrentRouteNode = /* @__PURE__ */ __name((pathname) => {
26
50
  const routes = useContext(Context);
27
51
  return getSafely("/", routes, pathname);
28
- };
29
- const AppRoutesProvider = ({ children }) => {
52
+ }, "useCurrentRouteNode");
53
+ const AppRoutesProvider = /* @__PURE__ */ __name(({ children }) => {
30
54
  return /* @__PURE__ */ jsx(Context.Provider, { value: appRoutes, children });
31
- };
55
+ }, "AppRoutesProvider");
32
56
  return {
33
57
  AppRoutesProvider,
34
58
  useAppRoutes,
35
59
  useCurrentRouteNode,
36
60
  _types: {}
37
61
  };
38
- };
62
+ }, "createAppRoutes");
39
63
 
40
- export { createAppRoutes };
64
+ export { createAppRoutes, getSafely, toQueryString };
41
65
  //# sourceMappingURL=index.mjs.map
42
66
  //# sourceMappingURL=index.mjs.map
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/object.utils.ts","../src/routes.utils.tsx"],"names":[],"mappings":";;;;;;AAEO,IAAM,SAAA,GAAY,CACvB,QAAA,EACA,GAAA,EACA,IAAA,KACyC;AACzC,EAAA,IAAI,IAAA,KAAS,EAAA,IAAM,IAAA,KAAS,QAAA,EAAU,OAAO,GAAA;AAE7C,EAAA,MAAM,IAAA,GAAQ,IAAA,CAAgB,KAAA,CAAM,QAAQ,CAAA;AAC5C,EAAA,IAAI,KAAA,GAAiB,GAAA;AAErB,EAAA,KAAA,MAAW,OAAO,IAAA,EAAM;AACtB,IAAA,IAAI,OAAO,KAAA,KAAU,QAAA,IAAY,KAAA,KAAU,IAAA,IAAQ,OAAO,KAAA,EAAO;AAC/D,MAAA,KAAA,GAAS,MAAkC,GAAG,CAAA;AAAA,IAChD,CAAA,MAAO;AACL,MAAA,OAAO,MAAA;AAAA,IACT;AAAA,EACF;AAEA,EAAA,OAAO,KAAA;AACT,CAAA;AChBO,IAAM,eAAA,GACX,MACA,CAA2D,SAAA,KAAuE;AAChI,EAAA,MAAM,OAAA,GAAU,cAAuE,SAAS,CAAA;AAEhG,EAAA,MAAM,eAAe,MAAM;AACzB,IAAA,OAAO,WAAW,OAAO,CAAA;AAAA,EAC3B,CAAA;AAEA,EAAA,MAAM,mBAAA,GAAsB,CAA+D,QAAA,KAAoB;AAC7G,IAAA,MAAM,MAAA,GAAS,WAAW,OAAO,CAAA;AACjC,IAAA,OAAO,SAAA,CAAU,GAAA,EAAK,MAAA,EAAQ,QAAQ,CAAA;AAAA,EACxC,CAAA;AAEA,EAAA,MAAM,iBAAA,GAAoB,CAAC,EAAE,QAAA,EAAS,KAA+B;AACnE,IAAA,2BAAQ,OAAA,CAAQ,QAAA,EAAR,EAAiB,KAAA,EAAO,WAAY,QAAA,EAAS,CAAA;AAAA,EACvD,CAAA;AAEA,EAAA,OAAO;AAAA,IACL,iBAAA;AAAA,IACA,YAAA;AAAA,IACA,mBAAA;AAAA,IACA,QAAQ;AAAC,GAKX;AACF","file":"index.mjs","sourcesContent":["import { Paths, PathValue } from './path.types';\n\nexport const getSafely = <TObject, TSplitter extends string, TPath extends string & Paths<TObject, TSplitter>>(\n splitter: TSplitter,\n obj: TObject,\n path: TPath,\n): PathValue<TObject, TPath, TSplitter> => {\n if (path === '' || path === splitter) return obj as PathValue<TObject, TPath, TSplitter>;\n\n const keys = (path as string).split(splitter);\n let value: unknown = obj;\n\n for (const key of keys) {\n if (typeof value === 'object' && value !== null && key in value) {\n value = (value as Record<string, unknown>)[key];\n } else {\n return undefined as PathValue<TObject, TPath, TSplitter>;\n }\n }\n\n return value as PathValue<TObject, TPath, TSplitter>;\n};\n\nexport type QueryValue = string | number | boolean | readonly (string | number | boolean)[];\nexport type QueryParams = Record<string, QueryValue>;\nexport type ToQueryStringOptions = {\n includeQuestionMark?: boolean;\n};\n\nexport const toQueryString = (query: QueryParams, options: ToQueryStringOptions = { includeQuestionMark: true }): string => {\n const params: string[] = [];\n\n Object.entries(query).forEach(([key, value]) => {\n if (value === undefined || value === null) {\n return;\n }\n\n const encodedKey = encodeURIComponent(key);\n\n if (Array.isArray(value)) {\n value.forEach((item) => {\n if (item !== undefined && item !== null) {\n params.push(`${encodedKey}=${encodeURIComponent(String(item))}`);\n }\n });\n } else {\n params.push(`${encodedKey}=${encodeURIComponent(String(value))}`);\n }\n });\n\n if (params.length === 0) {\n return '';\n }\n\n const queryString = params.join('&');\n return options.includeQuestionMark ? `?${queryString}` : queryString;\n};\n","import { createContext, ReactNode, useContext } from 'react';\nimport { getSafely } from './object.utils';\nimport { PathValue } from './path.types';\nimport { BaseMetadata, PartialRouteTree, RoutePathname, RouteTree } from './routes.types';\n\nexport const createAppRoutes =\n <TMetadata extends BaseMetadata, TContext>() =>\n <TRouteTree extends PartialRouteTree<TMetadata, TContext>>(appRoutes: TRouteTree & RouteTree<TMetadata, TContext, TRouteTree>) => {\n const Context = createContext<TRouteTree & RouteTree<TMetadata, TContext, TRouteTree>>(appRoutes);\n\n const useAppRoutes = () => {\n return useContext(Context);\n };\n\n const useCurrentRouteNode = <TPath extends RoutePathname<TMetadata, TContext, TRouteTree>>(pathname: TPath) => {\n const routes = useContext(Context);\n return getSafely('/', routes, pathname) as PathValue<TRouteTree, TPath, '/'>;\n };\n\n const AppRoutesProvider = ({ children }: { children: ReactNode }) => {\n return <Context.Provider value={appRoutes}>{children}</Context.Provider>;\n };\n\n return {\n AppRoutesProvider,\n useAppRoutes,\n useCurrentRouteNode,\n _types: {} as {\n AppRoutesMetadata: TMetadata;\n AppRoutesContext: TContext;\n AppRoutesPathname: RoutePathname<TMetadata, TContext, TRouteTree>;\n },\n };\n };\n"]}
1
+ {"version":3,"sources":["../src/object.utils.ts","../src/routes.utils.tsx"],"names":[],"mappings":"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","file":"index.mjs","sourcesContent":["import { Paths, PathValue } from './path.types';\nimport { QueryParams } from './query.types';\n\nexport const getSafely = <TObject, TSplitter extends string, TPath extends string & Paths<TObject, TSplitter>>(\n splitter: TSplitter,\n obj: TObject,\n path: TPath,\n): PathValue<TObject, TPath, TSplitter> => {\n if (path === '' || path === splitter) return obj as PathValue<TObject, TPath, TSplitter>;\n\n const keys = (path as string).split(splitter);\n let value: unknown = obj;\n\n for (const key of keys) {\n if (typeof value === 'object' && value !== null && key in value) {\n value = (value as Record<string, unknown>)[key];\n } else {\n return undefined as PathValue<TObject, TPath, TSplitter>;\n }\n }\n\n return value as PathValue<TObject, TPath, TSplitter>;\n};\n\nexport type QueryStringOptions = {\n includeQuestionMark?: boolean;\n};\n\nexport const toQueryString = (query: QueryParams, options: QueryStringOptions = { includeQuestionMark: true }): string => {\n const params: string[] = [];\n\n Object.entries(query).forEach(([key, value]) => {\n if (value === undefined || value === null) {\n return;\n }\n\n const encodedKey = encodeURIComponent(key);\n\n if (Array.isArray(value)) {\n value.forEach((item) => {\n if (item !== undefined && item !== null) {\n params.push(`${encodedKey}=${encodeURIComponent(String(item))}`);\n }\n });\n } else {\n params.push(`${encodedKey}=${encodeURIComponent(String(value))}`);\n }\n });\n\n if (params.length === 0) {\n return '';\n }\n\n const queryString = params.join('&');\n return options.includeQuestionMark ? `?${queryString}` : queryString;\n};\n","import { createContext, ReactNode, useContext } from 'react';\nimport { getSafely } from './object.utils';\nimport { PathValue } from './path.types';\nimport { BaseMetadata, PartialRouteTree, ResolvedRouteTree, RoutePathname, RouteTree } from './routes.types';\n\nexport const createAppRoutes =\n <TMetadata extends BaseMetadata, TContext>() =>\n <TRouteTree extends PartialRouteTree<TMetadata, TContext>>(appRoutes: TRouteTree & RouteTree<TMetadata, TContext, TRouteTree>) => {\n type Routes = ResolvedRouteTree<TMetadata, TContext, TRouteTree>;\n type Pathname = RoutePathname<TMetadata, TContext, TRouteTree>;\n\n const Context = createContext<Routes>(appRoutes as Routes);\n\n const useAppRoutes = (): Routes => {\n return useContext(Context);\n };\n\n const useCurrentRouteNode = <TPath extends Pathname>(pathname: TPath): PathValue<TRouteTree, TPath, '/'> => {\n const routes = useContext(Context);\n return getSafely('/', routes, pathname) as PathValue<TRouteTree, TPath, '/'>;\n };\n\n const AppRoutesProvider = ({ children }: { children: ReactNode }) => {\n return <Context.Provider value={appRoutes as Routes}>{children}</Context.Provider>;\n };\n\n return {\n AppRoutesProvider,\n useAppRoutes,\n useCurrentRouteNode,\n _types: {} as {\n AppRoutesMetadata: TMetadata;\n AppRoutesContext: TContext;\n AppRoutesPathname: Pathname;\n },\n };\n };\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@hyeonqyu/typed-router-core",
3
- "version": "0.2.5",
3
+ "version": "0.2.7",
4
4
  "description": "Core types and utilities for typed-router",
5
5
  "author": "hyeonQyu <dhk0561@naver.com>",
6
6
  "license": "MIT",