Skip to content

Commit ef0920a

Browse files
pikaxposva
andcommitted
feat: allow numbers as params
Close #206 Co-authored-by: Eduardo San Martin Morote <[email protected]>
1 parent f5b5949 commit ef0920a

File tree

5 files changed

+46
-30
lines changed

5 files changed

+46
-30
lines changed

__tests__/router.spec.ts

+6
Original file line numberDiff line numberDiff line change
@@ -254,6 +254,12 @@ describe('Router', () => {
254254
expect(spy).toHaveBeenCalledTimes(1)
255255
})
256256

257+
it('casts number params to string', async () => {
258+
const { router } = await newRouter()
259+
await router.push({ name: 'Param', params: { p: 0 } })
260+
expect(router.currentRoute.value).toMatchObject({ params: { p: '0' } })
261+
})
262+
257263
it('navigates to same route record but different query', async () => {
258264
const { router } = await newRouter()
259265
await router.push('/?q=1')

src/encoding.ts

+5-5
Original file line numberDiff line numberDiff line change
@@ -84,7 +84,7 @@ export function encodeQueryProperty(text: string | number): string {
8484
* @param text - string to encode
8585
* @returns encoded string
8686
*/
87-
export function encodePath(text: string): string {
87+
export function encodePath(text: string | number): string {
8888
return commonEncode(text).replace(HASH_RE, '%23').replace(IM_RE, '%3F')
8989
}
9090

@@ -96,7 +96,7 @@ export function encodePath(text: string): string {
9696
* @param text - string to encode
9797
* @returns encoded string
9898
*/
99-
export function encodeParam(text: string): string {
99+
export function encodeParam(text: string | number): string {
100100
return encodePath(text).replace(SLASH_RE, '%2F')
101101
}
102102

@@ -107,11 +107,11 @@ export function encodeParam(text: string): string {
107107
* @param text - string to decode
108108
* @returns decoded string
109109
*/
110-
export function decode(text: string): string {
110+
export function decode(text: string | number): string {
111111
try {
112-
return decodeURIComponent(text)
112+
return decodeURIComponent('' + text)
113113
} catch (err) {
114114
__DEV__ && warn(`Error decoding "${text}". Using original value`)
115115
}
116-
return text
116+
return '' + text
117117
}

src/router.ts

+15-12
Original file line numberDiff line numberDiff line change
@@ -5,13 +5,13 @@ import {
55
PostNavigationGuard,
66
START_LOCATION_NORMALIZED,
77
Lazy,
8-
MatcherLocation,
98
RouteLocationNormalizedLoaded,
109
RouteLocation,
1110
RouteRecordName,
1211
isRouteName,
1312
NavigationGuardWithThis,
1413
RouteLocationOptions,
14+
MatcherLocationRaw,
1515
} from './types'
1616
import { RouterHistory, HistoryState } from './history/common'
1717
import {
@@ -180,6 +180,10 @@ export function createRouter(options: RouterOptions): Router {
180180
history.scrollRestoration = 'manual'
181181
}
182182

183+
const normalizeParams = applyToParams.bind(
184+
null,
185+
paramValue => '' + paramValue
186+
)
183187
const encodeParams = applyToParams.bind(null, encodeParam)
184188
const decodeParams = applyToParams.bind(null, decode)
185189

@@ -249,6 +253,8 @@ export function createRouter(options: RouterOptions): Router {
249253
}
250254
}
251255

256+
let matcherLocation: MatcherLocationRaw
257+
252258
// path could be relative in object as well
253259
if ('path' in rawLocation) {
254260
if (
@@ -263,28 +269,25 @@ export function createRouter(options: RouterOptions): Router {
263269
}" was passed with params but they will be ignored. Use a named route alongside params instead.`
264270
)
265271
}
266-
rawLocation = {
272+
matcherLocation = {
267273
...rawLocation,
268274
path: parseURL(parseQuery, rawLocation.path, currentLocation.path).path,
269275
}
276+
} else {
277+
matcherLocation = {
278+
...rawLocation,
279+
params: encodeParams(rawLocation.params),
280+
}
270281
}
271282

272-
let matchedRoute: MatcherLocation = // relative or named location, path is ignored
273-
// for same reason TS thinks rawLocation.params can be undefined
274-
matcher.resolve(
275-
'params' in rawLocation
276-
? { ...rawLocation, params: encodeParams(rawLocation.params) }
277-
: rawLocation,
278-
currentLocation
279-
)
283+
let matchedRoute = matcher.resolve(matcherLocation, currentLocation)
280284

281285
const hash = encodeHash(rawLocation.hash || '')
282286

283287
// put back the unencoded params as given by the user (avoid the cost of decoding them)
284-
// TODO: normalize params if we accept numbers as raw values
285288
matchedRoute.params =
286289
'params' in rawLocation
287-
? rawLocation.params!
290+
? normalizeParams(rawLocation.params)
288291
: decodeParams(matchedRoute.params)
289292

290293
const fullPath = stringifyURL(stringifyQuery, {

src/types/index.ts

+17-10
Original file line numberDiff line numberDiff line change
@@ -20,14 +20,12 @@ export type VueUseOptions<T> = {
2020
export type TODO = any
2121

2222
export type RouteParamValue = string
23-
// TODO: should we allow more values like numbers and normalize them to strings?
24-
// type RouteParamValueRaw = RouteParamValue | number
23+
export type RouteParamValueRaw = RouteParamValue | number
2524
export type RouteParams = Record<string, RouteParamValue | RouteParamValue[]>
26-
export type RouteParamsRaw = RouteParams
27-
// export type RouteParamsRaw = Record<
28-
// string,
29-
// RouteParamValueRaw | RouteParamValueRaw[]
30-
// >
25+
export type RouteParamsRaw = Record<
26+
string,
27+
RouteParamValueRaw | RouteParamValueRaw[]
28+
>
3129

3230
export interface RouteQueryAndHash {
3331
query?: LocationQueryRaw
@@ -37,13 +35,22 @@ export interface LocationAsPath {
3735
path: string
3836
}
3937

38+
export interface LocationAsNameRaw {
39+
name: RouteRecordName
40+
params?: RouteParamsRaw
41+
}
42+
4043
export interface LocationAsName {
4144
name: RouteRecordName
45+
params?: RouteParams
46+
}
47+
48+
export interface LocationAsRelativeRaw {
4249
params?: RouteParamsRaw
4350
}
4451

4552
export interface LocationAsRelative {
46-
params?: RouteParamsRaw
53+
params?: RouteParams
4754
}
4855

4956
export interface RouteLocationOptions {
@@ -67,8 +74,8 @@ export interface RouteLocationOptions {
6774
export type RouteLocationRaw =
6875
| string
6976
| (RouteQueryAndHash & LocationAsPath & RouteLocationOptions)
70-
| (RouteQueryAndHash & LocationAsName & RouteLocationOptions)
71-
| (RouteQueryAndHash & LocationAsRelative & RouteLocationOptions)
77+
| (RouteQueryAndHash & LocationAsNameRaw & RouteLocationOptions)
78+
| (RouteQueryAndHash & LocationAsRelativeRaw & RouteLocationOptions)
7279

7380
export interface RouteLocationMatched extends RouteRecordNormalized {
7481
// components cannot be Lazy<RouteComponent>

src/utils/index.ts

+3-3
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import { RouteParams, RouteComponent } from '../types'
1+
import { RouteParams, RouteComponent, RouteParamsRaw } from '../types'
22
import { hasSymbol } from '../injectionSymbols'
33

44
export * from './env'
@@ -8,8 +8,8 @@ export function isESModule(obj: any): obj is { default: RouteComponent } {
88
}
99

1010
export function applyToParams(
11-
fn: (v: string) => string,
12-
params: RouteParams | undefined
11+
fn: (v: string | number) => string,
12+
params: RouteParamsRaw | undefined
1313
): RouteParams {
1414
const newParams: RouteParams = {}
1515

0 commit comments

Comments
 (0)