Skip to content

Commit ebca15a

Browse files
committed
fix(router): allow null | undefined for params
null and undefined params are now automatically removed. Empty strings are stil kept because custom regexps can be an empty string
1 parent f1c01ab commit ebca15a

File tree

5 files changed

+36
-7
lines changed

5 files changed

+36
-7
lines changed

__tests__/router.spec.ts

+18
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,7 @@ const routes: RouteRecordRaw[] = [
3434
{ path: '/to-foo-query', redirect: '/foo?a=2#b' },
3535
{ path: '/to-p/:p', redirect: { name: 'Param' } },
3636
{ path: '/p/:p', name: 'Param', component: components.Bar },
37+
{ path: '/optional/:p?', name: 'optional', component: components.Bar },
3738
{ path: '/repeat/:r+', name: 'repeat', component: components.Bar },
3839
{ path: '/to-p/:p', redirect: to => `/p/${to.params.p}` },
3940
{ path: '/redirect-with-param/:p', redirect: () => `/` },
@@ -240,6 +241,23 @@ describe('Router', () => {
240241
expect(router.currentRoute.value).toMatchObject({ params: { p: '0' } })
241242
})
242243

244+
it('casts null/undefined params to empty strings', async () => {
245+
const { router } = await newRouter()
246+
expect(
247+
router.resolve({ name: 'optional', params: { p: undefined } })
248+
).toMatchObject({
249+
params: {},
250+
})
251+
expect(
252+
router.resolve({ name: 'optional', params: { p: null } })
253+
).toMatchObject({
254+
params: {},
255+
})
256+
await router.push({ name: 'optional', params: { p: null } })
257+
expect(router.currentRoute.value).toMatchObject({ params: {} })
258+
await router.push({ name: 'optional', params: {} })
259+
})
260+
243261
it('navigates to same route record but different query', async () => {
244262
const { router } = await newRouter()
245263
await router.push('/?q=1')

src/encoding.ts

+4-3
Original file line numberDiff line numberDiff line change
@@ -120,13 +120,14 @@ export function encodePath(text: string | number): string {
120120
/**
121121
* Encode characters that need to be encoded on the path section of the URL as a
122122
* param. This function encodes everything {@link encodePath} does plus the
123-
* slash (`/`) character.
123+
* slash (`/`) character. If `text` is `null` or `undefined`, returns an empty
124+
* string instead.
124125
*
125126
* @param text - string to encode
126127
* @returns encoded string
127128
*/
128-
export function encodeParam(text: string | number): string {
129-
return encodePath(text).replace(SLASH_RE, '%2F')
129+
export function encodeParam(text: string | number | null | undefined): string {
130+
return text == null ? '' : encodePath(text).replace(SLASH_RE, '%2F')
130131
}
131132

132133
/**

src/router.ts

+11-1
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@ import {
1212
NavigationGuardWithThis,
1313
RouteLocationOptions,
1414
MatcherLocationRaw,
15+
RouteParams,
1516
} from './types'
1617
import { RouterHistory, HistoryState, NavigationType } from './history/common'
1718
import {
@@ -380,7 +381,9 @@ export function createRouter(options: RouterOptions): Router {
380381
paramValue => '' + paramValue
381382
)
382383
const encodeParams = applyToParams.bind(null, encodeParam)
383-
const decodeParams = applyToParams.bind(null, decode)
384+
const decodeParams: (params: RouteParams | undefined) => RouteParams =
385+
// @ts-expect-error: intentionally avoid the type check
386+
applyToParams.bind(null, decode)
384387

385388
function addRoute(
386389
parentOrRoute: RouteRecordName | RouteRecordRaw,
@@ -473,6 +476,13 @@ export function createRouter(options: RouterOptions): Router {
473476
path: parseURL(parseQuery, rawLocation.path, currentLocation.path).path,
474477
})
475478
} else {
479+
// remove any nullish param
480+
const targetParams = assign({}, rawLocation.params)
481+
for (const key in targetParams) {
482+
if (targetParams[key] == null) {
483+
delete targetParams[key]
484+
}
485+
}
476486
// pass encoded values to the matcher so it can produce encoded path and fullPath
477487
matcherLocation = assign({}, rawLocation, {
478488
params: encodeParams(rawLocation.params),

src/types/index.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -31,11 +31,11 @@ export type RouteParamValue = string
3131
/**
3232
* @internal
3333
*/
34-
export type RouteParamValueRaw = RouteParamValue | number
34+
export type RouteParamValueRaw = RouteParamValue | number | null | undefined
3535
export type RouteParams = Record<string, RouteParamValue | RouteParamValue[]>
3636
export type RouteParamsRaw = Record<
3737
string,
38-
RouteParamValueRaw | RouteParamValueRaw[]
38+
RouteParamValueRaw | Exclude<RouteParamValueRaw, null | undefined>[]
3939
>
4040

4141
/**

src/utils/index.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,7 @@ export function isESModule(obj: any): obj is { default: RouteComponent } {
1010
export const assign = Object.assign
1111

1212
export function applyToParams(
13-
fn: (v: string | number) => string,
13+
fn: (v: string | number | null | undefined) => string,
1414
params: RouteParamsRaw | undefined
1515
): RouteParams {
1616
const newParams: RouteParams = {}

0 commit comments

Comments
 (0)