Skip to content

Commit 643bd15

Browse files
committed
fix: ignore order of keys in query and params
1 parent 8bbed1b commit 643bd15

File tree

2 files changed

+68
-20
lines changed

2 files changed

+68
-20
lines changed

__tests__/location.spec.ts

+32
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ import {
55
stringifyURL as originalStringifyURL,
66
stripBase,
77
} from '../src/utils/location'
8+
import { isSameLocationObject } from '../src/utils'
89

910
describe('parseURL', () => {
1011
let parseURL = originalParseURL.bind(null, parseQuery)
@@ -160,3 +161,34 @@ describe('stripBase', () => {
160161
expect(stripBase('/base/foo', '/base')).toBe('/foo')
161162
})
162163
})
164+
165+
describe('isSameLocationObject', () => {
166+
it('compare simple values', () => {
167+
expect(isSameLocationObject({ a: '2' }, { a: '2' })).toBe(true)
168+
expect(isSameLocationObject({ a: '3' }, { a: '2' })).toBe(false)
169+
// different order
170+
expect(isSameLocationObject({ a: '2', b: '3' }, { b: '3', a: '2' })).toBe(
171+
true
172+
)
173+
expect(isSameLocationObject({ a: '3', b: '3' }, { b: '3', a: '2' })).toBe(
174+
false
175+
)
176+
})
177+
178+
it('compare array values', () => {
179+
expect(isSameLocationObject({ a: ['2'] }, { a: ['2'] })).toBe(true)
180+
expect(isSameLocationObject({ a: ['3'] }, { a: ['2'] })).toBe(false)
181+
// different order
182+
expect(
183+
isSameLocationObject({ a: ['2'], b: ['3'] }, { b: ['3'], a: ['2'] })
184+
).toBe(true)
185+
expect(
186+
isSameLocationObject({ a: ['3'], b: ['3'] }, { b: ['3'], a: ['2'] })
187+
).toBe(false)
188+
})
189+
190+
it('considers arrays of one item same as the item itself', () => {
191+
expect(isSameLocationObject({ a: ['2'] }, { a: '2' })).toBe(true)
192+
expect(isSameLocationObject({ a: ['3'] }, { a: '2' })).toBe(false)
193+
})
194+
})

src/utils/index.ts

+36-20
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@ import {
33
RouteParams,
44
RouteComponent,
55
RouteLocationNormalizedLoaded,
6+
RouteParamValue,
67
} from '../types'
78
import { guardToPromiseFn } from './guardToPromiseFn'
89
import { RouteRecord, RouteRecordNormalized } from '../matcher/types'
@@ -90,16 +91,10 @@ export function isSameLocationObject(
9091
a: RouteLocationNormalized['query' | 'params'],
9192
b: RouteLocationNormalized['query' | 'params']
9293
): boolean {
93-
const aKeys = Object.keys(a)
94-
const bKeys = Object.keys(b)
95-
if (aKeys.length !== bKeys.length) return false
96-
let i = 0
97-
let key: string
98-
while (i < aKeys.length) {
99-
key = aKeys[i]
100-
if (key !== bKeys[i]) return false
94+
if (Object.keys(a).length !== Object.keys(b).length) return false
95+
96+
for (let key in a) {
10197
if (!isSameLocationObjectValue(a[key], b[key])) return false
102-
i++
10398
}
10499

105100
return true
@@ -109,17 +104,38 @@ function isSameLocationObjectValue(
109104
a: LocationQueryValue | LocationQueryValue[],
110105
b: LocationQueryValue | LocationQueryValue[]
111106
): boolean
112-
function isSameLocationObjectValue(a: RouteParams, b: RouteParams): boolean
113107
function isSameLocationObjectValue(
114-
a: LocationQueryValue | LocationQueryValue[] | RouteParams,
115-
b: LocationQueryValue | LocationQueryValue[] | RouteParams
108+
a: RouteParamValue | RouteParamValue[],
109+
b: RouteParamValue | RouteParamValue[]
110+
): boolean
111+
function isSameLocationObjectValue(
112+
a:
113+
| LocationQueryValue
114+
| LocationQueryValue[]
115+
| RouteParamValue
116+
| RouteParamValue[],
117+
b:
118+
| LocationQueryValue
119+
| LocationQueryValue[]
120+
| RouteParamValue
121+
| RouteParamValue[]
116122
): boolean {
117-
if (typeof a !== typeof b) return false
118-
// both a and b are arrays
119-
if (Array.isArray(a))
120-
return (
121-
a.length === (b as any[]).length &&
122-
a.every((value, i) => value === (b as LocationQueryValue[])[i])
123-
)
124-
return a === b
123+
return Array.isArray(a)
124+
? isEquivalentArray(a, b)
125+
: Array.isArray(b)
126+
? isEquivalentArray(b, a)
127+
: a === b
128+
}
129+
130+
/**
131+
* Check if two arrays are the same or if an array with one single entry is the
132+
* same as another primitive value. Used to check query and parameters
133+
*
134+
* @param a array of values
135+
* @param b array of values or a single value
136+
*/
137+
function isEquivalentArray<T>(a: T[], b: T[] | T): boolean {
138+
return Array.isArray(b)
139+
? a.length === b.length && a.every((value, i) => value === b[i])
140+
: a.length === 1 && a[0] === b
125141
}

0 commit comments

Comments
 (0)