Skip to content

Commit a8ac129

Browse files
committed
fix: ensure looseEqual is not dependant on key enumeration order
close #5908
1 parent 9b4dbba commit a8ac129

File tree

2 files changed

+48
-4
lines changed

2 files changed

+48
-4
lines changed

src/shared/util.js

+20-4
Original file line numberDiff line numberDiff line change
@@ -248,15 +248,31 @@ export function genStaticKeys (modules: Array<ModuleOptions>): string {
248248
* Check if two values are loosely equal - that is,
249249
* if they are plain objects, do they have the same shape?
250250
*/
251-
export function looseEqual (a: mixed, b: mixed): boolean {
251+
export function looseEqual (a: any, b: any): boolean {
252+
if (a === b) return true
252253
const isObjectA = isObject(a)
253254
const isObjectB = isObject(b)
254255
if (isObjectA && isObjectB) {
255256
try {
256-
return JSON.stringify(a) === JSON.stringify(b)
257+
const isArrayA = Array.isArray(a)
258+
const isArrayB = Array.isArray(b)
259+
if (isArrayA && isArrayB) {
260+
return a.length === b.length && a.every((e, i) => {
261+
return looseEqual(e, b[i])
262+
})
263+
} else if (!isArrayA && !isArrayB) {
264+
const keysA = Object.keys(a)
265+
const keysB = Object.keys(b)
266+
return keysA.length === keysB.length && keysA.every(key => {
267+
return looseEqual(a[key], b[key])
268+
})
269+
} else {
270+
/* istanbul ignore next */
271+
return false
272+
}
257273
} catch (e) {
258-
// possible circular reference
259-
return a === b
274+
/* istanbul ignore next */
275+
return false
260276
}
261277
} else if (!isObjectA && !isObjectB) {
262278
return String(a) === String(b)

test/unit/features/directives/model-select.spec.js

+28
Original file line numberDiff line numberDiff line change
@@ -119,6 +119,34 @@ describe('Directive v-model select', () => {
119119
}).then(done)
120120
})
121121

122+
it('should work with value bindings (Array loose equal)', done => {
123+
const vm = new Vue({
124+
data: {
125+
test: [{ a: 2 }]
126+
},
127+
template:
128+
'<select v-model="test">' +
129+
'<option value="1">a</option>' +
130+
'<option :value="[{ a: 2 }]">b</option>' +
131+
'<option :value="[{ a: 3 }]">c</option>' +
132+
'</select>'
133+
}).$mount()
134+
document.body.appendChild(vm.$el)
135+
expect(vm.$el.childNodes[1].selected).toBe(true)
136+
vm.test = [{ a: 3 }]
137+
waitForUpdate(function () {
138+
expect(vm.$el.childNodes[2].selected).toBe(true)
139+
140+
updateSelect(vm.$el, '1')
141+
triggerEvent(vm.$el, 'change')
142+
expect(vm.test).toBe('1')
143+
144+
updateSelect(vm.$el, [{ a: 2 }])
145+
triggerEvent(vm.$el, 'change')
146+
expect(vm.test).toEqual([{ a: 2 }])
147+
}).then(done)
148+
})
149+
122150
it('should work with v-for', done => {
123151
const vm = new Vue({
124152
data: {

0 commit comments

Comments
 (0)