@tanstack/router-core 1.136.4 → 1.136.6

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 (48) hide show
  1. package/dist/cjs/Matches.cjs.map +1 -1
  2. package/dist/cjs/Matches.d.cts +2 -0
  3. package/dist/cjs/index.cjs +0 -5
  4. package/dist/cjs/index.cjs.map +1 -1
  5. package/dist/cjs/index.d.cts +1 -4
  6. package/dist/cjs/lru-cache.cjs +5 -0
  7. package/dist/cjs/lru-cache.cjs.map +1 -1
  8. package/dist/cjs/lru-cache.d.cts +1 -0
  9. package/dist/cjs/new-process-route-tree.cjs +655 -0
  10. package/dist/cjs/new-process-route-tree.cjs.map +1 -0
  11. package/dist/cjs/new-process-route-tree.d.cts +177 -0
  12. package/dist/cjs/path.cjs +133 -434
  13. package/dist/cjs/path.cjs.map +1 -1
  14. package/dist/cjs/path.d.cts +3 -39
  15. package/dist/cjs/router.cjs +47 -98
  16. package/dist/cjs/router.cjs.map +1 -1
  17. package/dist/cjs/router.d.cts +6 -11
  18. package/dist/esm/Matches.d.ts +2 -0
  19. package/dist/esm/Matches.js.map +1 -1
  20. package/dist/esm/index.d.ts +1 -4
  21. package/dist/esm/index.js +1 -6
  22. package/dist/esm/index.js.map +1 -1
  23. package/dist/esm/lru-cache.d.ts +1 -0
  24. package/dist/esm/lru-cache.js +5 -0
  25. package/dist/esm/lru-cache.js.map +1 -1
  26. package/dist/esm/new-process-route-tree.d.ts +177 -0
  27. package/dist/esm/new-process-route-tree.js +655 -0
  28. package/dist/esm/new-process-route-tree.js.map +1 -0
  29. package/dist/esm/path.d.ts +3 -39
  30. package/dist/esm/path.js +133 -434
  31. package/dist/esm/path.js.map +1 -1
  32. package/dist/esm/router.d.ts +6 -11
  33. package/dist/esm/router.js +48 -99
  34. package/dist/esm/router.js.map +1 -1
  35. package/package.json +1 -1
  36. package/src/Matches.ts +2 -0
  37. package/src/index.ts +0 -6
  38. package/src/lru-cache.ts +6 -0
  39. package/src/new-process-route-tree.ts +1036 -0
  40. package/src/path.ts +168 -639
  41. package/src/router.ts +57 -126
  42. package/dist/cjs/process-route-tree.cjs +0 -144
  43. package/dist/cjs/process-route-tree.cjs.map +0 -1
  44. package/dist/cjs/process-route-tree.d.cts +0 -18
  45. package/dist/esm/process-route-tree.d.ts +0 -18
  46. package/dist/esm/process-route-tree.js +0 -144
  47. package/dist/esm/process-route-tree.js.map +0 -1
  48. package/src/process-route-tree.ts +0 -241
@@ -1 +0,0 @@
1
- {"version":3,"file":"process-route-tree.js","sources":["../../src/process-route-tree.ts"],"sourcesContent":["import invariant from 'tiny-invariant'\nimport {\n SEGMENT_TYPE_OPTIONAL_PARAM,\n SEGMENT_TYPE_PARAM,\n SEGMENT_TYPE_PATHNAME,\n parsePathname,\n trimPathLeft,\n trimPathRight,\n} from './path'\nimport type { Segment } from './path'\nimport type { RouteLike } from './route'\n\nconst SLASH_SCORE = 0.75\nconst STATIC_SEGMENT_SCORE = 1\nconst REQUIRED_PARAM_BASE_SCORE = 0.5\nconst OPTIONAL_PARAM_BASE_SCORE = 0.4\nconst WILDCARD_PARAM_BASE_SCORE = 0.25\nconst STATIC_AFTER_DYNAMIC_BONUS_SCORE = 0.2\nconst BOTH_PRESENCE_BASE_SCORE = 0.05\nconst PREFIX_PRESENCE_BASE_SCORE = 0.02\nconst SUFFIX_PRESENCE_BASE_SCORE = 0.01\nconst PREFIX_LENGTH_SCORE_MULTIPLIER = 0.0002\nconst SUFFIX_LENGTH_SCORE_MULTIPLIER = 0.0001\n\nfunction handleParam(segment: Segment, baseScore: number) {\n if (segment.prefixSegment && segment.suffixSegment) {\n return (\n baseScore +\n BOTH_PRESENCE_BASE_SCORE +\n PREFIX_LENGTH_SCORE_MULTIPLIER * segment.prefixSegment.length +\n SUFFIX_LENGTH_SCORE_MULTIPLIER * segment.suffixSegment.length\n )\n }\n\n if (segment.prefixSegment) {\n return (\n baseScore +\n PREFIX_PRESENCE_BASE_SCORE +\n PREFIX_LENGTH_SCORE_MULTIPLIER * segment.prefixSegment.length\n )\n }\n\n if (segment.suffixSegment) {\n return (\n baseScore +\n SUFFIX_PRESENCE_BASE_SCORE +\n SUFFIX_LENGTH_SCORE_MULTIPLIER * segment.suffixSegment.length\n )\n }\n\n return baseScore\n}\n\nfunction sortRoutes<TRouteLike extends RouteLike>(\n routes: ReadonlyArray<TRouteLike>,\n): Array<TRouteLike> {\n const scoredRoutes: Array<{\n child: TRouteLike\n trimmed: string\n parsed: ReadonlyArray<Segment>\n index: number\n scores: Array<number>\n hasStaticAfter: boolean\n optionalParamCount: number\n }> = []\n\n routes.forEach((d, i) => {\n if (d.isRoot || !d.path) {\n return\n }\n\n const trimmed = trimPathLeft(d.fullPath)\n let parsed = parsePathname(trimmed)\n\n // Removes the leading slash if it is not the only remaining segment\n let skip = 0\n while (parsed.length > skip + 1 && parsed[skip]?.value === '/') {\n skip++\n }\n if (skip > 0) parsed = parsed.slice(skip)\n\n let optionalParamCount = 0\n let hasStaticAfter = false\n const scores = parsed.map((segment, index) => {\n if (segment.value === '/') {\n return SLASH_SCORE\n }\n\n if (segment.type === SEGMENT_TYPE_PATHNAME) {\n return STATIC_SEGMENT_SCORE\n }\n\n let baseScore: number | undefined = undefined\n if (segment.type === SEGMENT_TYPE_PARAM) {\n baseScore = REQUIRED_PARAM_BASE_SCORE\n } else if (segment.type === SEGMENT_TYPE_OPTIONAL_PARAM) {\n baseScore = OPTIONAL_PARAM_BASE_SCORE\n optionalParamCount++\n } else {\n baseScore = WILDCARD_PARAM_BASE_SCORE\n }\n\n // if there is any static segment (that is not an index) after a required / optional param,\n // we will boost this param so it ranks higher than a required/optional param without a static segment after it\n // JUST FOR SORTING, NOT FOR MATCHING\n for (let i = index + 1; i < parsed.length; i++) {\n const nextSegment = parsed[i]!\n if (\n nextSegment.type === SEGMENT_TYPE_PATHNAME &&\n nextSegment.value !== '/'\n ) {\n hasStaticAfter = true\n return handleParam(\n segment,\n baseScore + STATIC_AFTER_DYNAMIC_BONUS_SCORE,\n )\n }\n }\n\n return handleParam(segment, baseScore)\n })\n\n scoredRoutes.push({\n child: d,\n trimmed,\n parsed,\n index: i,\n scores,\n optionalParamCount,\n hasStaticAfter,\n })\n })\n\n const flatRoutes = scoredRoutes\n .sort((a, b) => {\n const minLength = Math.min(a.scores.length, b.scores.length)\n\n // Sort by segment-by-segment score comparison ONLY for the common prefix\n for (let i = 0; i < minLength; i++) {\n if (a.scores[i] !== b.scores[i]) {\n return b.scores[i]! - a.scores[i]!\n }\n }\n\n // If all common segments have equal scores, then consider length and specificity\n if (a.scores.length !== b.scores.length) {\n // If different number of optional parameters, fewer optional parameters wins (more specific)\n // only if both or none of the routes has static segments after the params\n if (a.optionalParamCount !== b.optionalParamCount) {\n if (a.hasStaticAfter === b.hasStaticAfter) {\n return a.optionalParamCount - b.optionalParamCount\n } else if (a.hasStaticAfter && !b.hasStaticAfter) {\n return -1\n } else if (!a.hasStaticAfter && b.hasStaticAfter) {\n return 1\n }\n }\n\n // If same number of optional parameters, longer path wins (for static segments)\n return b.scores.length - a.scores.length\n }\n\n // Sort by min available parsed value for alphabetical ordering\n for (let i = 0; i < minLength; i++) {\n if (a.parsed[i]!.value !== b.parsed[i]!.value) {\n return a.parsed[i]!.value > b.parsed[i]!.value ? 1 : -1\n }\n }\n\n // Sort by original index\n return a.index - b.index\n })\n .map((d, i) => {\n d.child.rank = i\n return d.child\n })\n\n return flatRoutes\n}\n\nexport type ProcessRouteTreeResult<TRouteLike extends RouteLike> = {\n routesById: Record<string, TRouteLike>\n routesByPath: Record<string, TRouteLike>\n flatRoutes: Array<TRouteLike>\n}\n\n/**\n * Build lookup maps and a specificity-sorted flat list from a route tree.\n * Returns `routesById`, `routesByPath`, and `flatRoutes`.\n */\n/**\n * Build lookup maps and a specificity-sorted flat list from a route tree.\n * Returns `routesById`, `routesByPath`, and `flatRoutes`.\n */\nexport function processRouteTree<TRouteLike extends RouteLike>({\n routeTree,\n initRoute,\n}: {\n routeTree: TRouteLike\n initRoute?: (route: TRouteLike, index: number) => void\n}): ProcessRouteTreeResult<TRouteLike> {\n const routesById = {} as Record<string, TRouteLike>\n const routesByPath = {} as Record<string, TRouteLike>\n\n const recurseRoutes = (childRoutes: Array<TRouteLike>) => {\n childRoutes.forEach((childRoute, i) => {\n initRoute?.(childRoute, i)\n\n const existingRoute = routesById[childRoute.id]\n\n invariant(\n !existingRoute,\n `Duplicate routes found with id: ${String(childRoute.id)}`,\n )\n\n routesById[childRoute.id] = childRoute\n\n if (!childRoute.isRoot && childRoute.path) {\n const trimmedFullPath = trimPathRight(childRoute.fullPath)\n if (\n !routesByPath[trimmedFullPath] ||\n childRoute.fullPath.endsWith('/')\n ) {\n routesByPath[trimmedFullPath] = childRoute\n }\n }\n\n const children = childRoute.children as Array<TRouteLike>\n\n if (children?.length) {\n recurseRoutes(children)\n }\n })\n }\n\n recurseRoutes([routeTree])\n\n const flatRoutes = sortRoutes(Object.values(routesById))\n\n return { routesById, routesByPath, flatRoutes }\n}\n"],"names":["i"],"mappings":"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"}
@@ -1,241 +0,0 @@
1
- import invariant from 'tiny-invariant'
2
- import {
3
- SEGMENT_TYPE_OPTIONAL_PARAM,
4
- SEGMENT_TYPE_PARAM,
5
- SEGMENT_TYPE_PATHNAME,
6
- parsePathname,
7
- trimPathLeft,
8
- trimPathRight,
9
- } from './path'
10
- import type { Segment } from './path'
11
- import type { RouteLike } from './route'
12
-
13
- const SLASH_SCORE = 0.75
14
- const STATIC_SEGMENT_SCORE = 1
15
- const REQUIRED_PARAM_BASE_SCORE = 0.5
16
- const OPTIONAL_PARAM_BASE_SCORE = 0.4
17
- const WILDCARD_PARAM_BASE_SCORE = 0.25
18
- const STATIC_AFTER_DYNAMIC_BONUS_SCORE = 0.2
19
- const BOTH_PRESENCE_BASE_SCORE = 0.05
20
- const PREFIX_PRESENCE_BASE_SCORE = 0.02
21
- const SUFFIX_PRESENCE_BASE_SCORE = 0.01
22
- const PREFIX_LENGTH_SCORE_MULTIPLIER = 0.0002
23
- const SUFFIX_LENGTH_SCORE_MULTIPLIER = 0.0001
24
-
25
- function handleParam(segment: Segment, baseScore: number) {
26
- if (segment.prefixSegment && segment.suffixSegment) {
27
- return (
28
- baseScore +
29
- BOTH_PRESENCE_BASE_SCORE +
30
- PREFIX_LENGTH_SCORE_MULTIPLIER * segment.prefixSegment.length +
31
- SUFFIX_LENGTH_SCORE_MULTIPLIER * segment.suffixSegment.length
32
- )
33
- }
34
-
35
- if (segment.prefixSegment) {
36
- return (
37
- baseScore +
38
- PREFIX_PRESENCE_BASE_SCORE +
39
- PREFIX_LENGTH_SCORE_MULTIPLIER * segment.prefixSegment.length
40
- )
41
- }
42
-
43
- if (segment.suffixSegment) {
44
- return (
45
- baseScore +
46
- SUFFIX_PRESENCE_BASE_SCORE +
47
- SUFFIX_LENGTH_SCORE_MULTIPLIER * segment.suffixSegment.length
48
- )
49
- }
50
-
51
- return baseScore
52
- }
53
-
54
- function sortRoutes<TRouteLike extends RouteLike>(
55
- routes: ReadonlyArray<TRouteLike>,
56
- ): Array<TRouteLike> {
57
- const scoredRoutes: Array<{
58
- child: TRouteLike
59
- trimmed: string
60
- parsed: ReadonlyArray<Segment>
61
- index: number
62
- scores: Array<number>
63
- hasStaticAfter: boolean
64
- optionalParamCount: number
65
- }> = []
66
-
67
- routes.forEach((d, i) => {
68
- if (d.isRoot || !d.path) {
69
- return
70
- }
71
-
72
- const trimmed = trimPathLeft(d.fullPath)
73
- let parsed = parsePathname(trimmed)
74
-
75
- // Removes the leading slash if it is not the only remaining segment
76
- let skip = 0
77
- while (parsed.length > skip + 1 && parsed[skip]?.value === '/') {
78
- skip++
79
- }
80
- if (skip > 0) parsed = parsed.slice(skip)
81
-
82
- let optionalParamCount = 0
83
- let hasStaticAfter = false
84
- const scores = parsed.map((segment, index) => {
85
- if (segment.value === '/') {
86
- return SLASH_SCORE
87
- }
88
-
89
- if (segment.type === SEGMENT_TYPE_PATHNAME) {
90
- return STATIC_SEGMENT_SCORE
91
- }
92
-
93
- let baseScore: number | undefined = undefined
94
- if (segment.type === SEGMENT_TYPE_PARAM) {
95
- baseScore = REQUIRED_PARAM_BASE_SCORE
96
- } else if (segment.type === SEGMENT_TYPE_OPTIONAL_PARAM) {
97
- baseScore = OPTIONAL_PARAM_BASE_SCORE
98
- optionalParamCount++
99
- } else {
100
- baseScore = WILDCARD_PARAM_BASE_SCORE
101
- }
102
-
103
- // if there is any static segment (that is not an index) after a required / optional param,
104
- // we will boost this param so it ranks higher than a required/optional param without a static segment after it
105
- // JUST FOR SORTING, NOT FOR MATCHING
106
- for (let i = index + 1; i < parsed.length; i++) {
107
- const nextSegment = parsed[i]!
108
- if (
109
- nextSegment.type === SEGMENT_TYPE_PATHNAME &&
110
- nextSegment.value !== '/'
111
- ) {
112
- hasStaticAfter = true
113
- return handleParam(
114
- segment,
115
- baseScore + STATIC_AFTER_DYNAMIC_BONUS_SCORE,
116
- )
117
- }
118
- }
119
-
120
- return handleParam(segment, baseScore)
121
- })
122
-
123
- scoredRoutes.push({
124
- child: d,
125
- trimmed,
126
- parsed,
127
- index: i,
128
- scores,
129
- optionalParamCount,
130
- hasStaticAfter,
131
- })
132
- })
133
-
134
- const flatRoutes = scoredRoutes
135
- .sort((a, b) => {
136
- const minLength = Math.min(a.scores.length, b.scores.length)
137
-
138
- // Sort by segment-by-segment score comparison ONLY for the common prefix
139
- for (let i = 0; i < minLength; i++) {
140
- if (a.scores[i] !== b.scores[i]) {
141
- return b.scores[i]! - a.scores[i]!
142
- }
143
- }
144
-
145
- // If all common segments have equal scores, then consider length and specificity
146
- if (a.scores.length !== b.scores.length) {
147
- // If different number of optional parameters, fewer optional parameters wins (more specific)
148
- // only if both or none of the routes has static segments after the params
149
- if (a.optionalParamCount !== b.optionalParamCount) {
150
- if (a.hasStaticAfter === b.hasStaticAfter) {
151
- return a.optionalParamCount - b.optionalParamCount
152
- } else if (a.hasStaticAfter && !b.hasStaticAfter) {
153
- return -1
154
- } else if (!a.hasStaticAfter && b.hasStaticAfter) {
155
- return 1
156
- }
157
- }
158
-
159
- // If same number of optional parameters, longer path wins (for static segments)
160
- return b.scores.length - a.scores.length
161
- }
162
-
163
- // Sort by min available parsed value for alphabetical ordering
164
- for (let i = 0; i < minLength; i++) {
165
- if (a.parsed[i]!.value !== b.parsed[i]!.value) {
166
- return a.parsed[i]!.value > b.parsed[i]!.value ? 1 : -1
167
- }
168
- }
169
-
170
- // Sort by original index
171
- return a.index - b.index
172
- })
173
- .map((d, i) => {
174
- d.child.rank = i
175
- return d.child
176
- })
177
-
178
- return flatRoutes
179
- }
180
-
181
- export type ProcessRouteTreeResult<TRouteLike extends RouteLike> = {
182
- routesById: Record<string, TRouteLike>
183
- routesByPath: Record<string, TRouteLike>
184
- flatRoutes: Array<TRouteLike>
185
- }
186
-
187
- /**
188
- * Build lookup maps and a specificity-sorted flat list from a route tree.
189
- * Returns `routesById`, `routesByPath`, and `flatRoutes`.
190
- */
191
- /**
192
- * Build lookup maps and a specificity-sorted flat list from a route tree.
193
- * Returns `routesById`, `routesByPath`, and `flatRoutes`.
194
- */
195
- export function processRouteTree<TRouteLike extends RouteLike>({
196
- routeTree,
197
- initRoute,
198
- }: {
199
- routeTree: TRouteLike
200
- initRoute?: (route: TRouteLike, index: number) => void
201
- }): ProcessRouteTreeResult<TRouteLike> {
202
- const routesById = {} as Record<string, TRouteLike>
203
- const routesByPath = {} as Record<string, TRouteLike>
204
-
205
- const recurseRoutes = (childRoutes: Array<TRouteLike>) => {
206
- childRoutes.forEach((childRoute, i) => {
207
- initRoute?.(childRoute, i)
208
-
209
- const existingRoute = routesById[childRoute.id]
210
-
211
- invariant(
212
- !existingRoute,
213
- `Duplicate routes found with id: ${String(childRoute.id)}`,
214
- )
215
-
216
- routesById[childRoute.id] = childRoute
217
-
218
- if (!childRoute.isRoot && childRoute.path) {
219
- const trimmedFullPath = trimPathRight(childRoute.fullPath)
220
- if (
221
- !routesByPath[trimmedFullPath] ||
222
- childRoute.fullPath.endsWith('/')
223
- ) {
224
- routesByPath[trimmedFullPath] = childRoute
225
- }
226
- }
227
-
228
- const children = childRoute.children as Array<TRouteLike>
229
-
230
- if (children?.length) {
231
- recurseRoutes(children)
232
- }
233
- })
234
- }
235
-
236
- recurseRoutes([routeTree])
237
-
238
- const flatRoutes = sortRoutes(Object.values(routesById))
239
-
240
- return { routesById, routesByPath, flatRoutes }
241
- }