Skip to content

Commit aefb7d2

Browse files
committed
fix(reactivity): Array methods relying on identity should work with raw values
1 parent 3919c78 commit aefb7d2

File tree

2 files changed

+35
-1
lines changed

2 files changed

+35
-1
lines changed

packages/reactivity/__tests__/reactive.spec.ts

+21
Original file line numberDiff line numberDiff line change
@@ -44,6 +44,27 @@ describe('reactivity/reactive', () => {
4444
expect(clone[0]).toBe(observed[0])
4545
})
4646

47+
test('Array identity methods should work with raw values', () => {
48+
const raw = {}
49+
const arr = reactive([{}, {}])
50+
arr.push(raw)
51+
expect(arr.indexOf(raw)).toBe(2)
52+
expect(arr.indexOf(raw, 3)).toBe(-1)
53+
expect(arr.includes(raw)).toBe(true)
54+
expect(arr.includes(raw, 3)).toBe(false)
55+
expect(arr.lastIndexOf(raw)).toBe(2)
56+
expect(arr.lastIndexOf(raw, 1)).toBe(-1)
57+
58+
// should work also for the observed version
59+
const observed = arr[2]
60+
expect(arr.indexOf(observed)).toBe(2)
61+
expect(arr.indexOf(observed, 3)).toBe(-1)
62+
expect(arr.includes(observed)).toBe(true)
63+
expect(arr.includes(observed, 3)).toBe(false)
64+
expect(arr.lastIndexOf(observed)).toBe(2)
65+
expect(arr.lastIndexOf(observed, 1)).toBe(-1)
66+
})
67+
4768
test('nested reactives', () => {
4869
const original = {
4970
nested: {

packages/reactivity/src/baseHandlers.ts

+14-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@ import { reactive, readonly, toRaw } from './reactive'
22
import { TrackOpTypes, TriggerOpTypes } from './operations'
33
import { track, trigger, ITERATE_KEY } from './effect'
44
import { LOCKED } from './lock'
5-
import { isObject, hasOwn, isSymbol, hasChanged } from '@vue/shared'
5+
import { isObject, hasOwn, isSymbol, hasChanged, isArray } from '@vue/shared'
66
import { isRef } from './ref'
77

88
const builtInSymbols = new Set(
@@ -15,8 +15,21 @@ const get = /*#__PURE__*/ createGetter()
1515
const readonlyGet = /*#__PURE__*/ createGetter(true)
1616
const shallowReadonlyGet = /*#__PURE__*/ createGetter(true, true)
1717

18+
const arrayIdentityInstrumentations: Record<string, Function> = {}
19+
;['includes', 'indexOf', 'lastIndexOf'].forEach(key => {
20+
arrayIdentityInstrumentations[key] = function(
21+
value: unknown,
22+
...args: any[]
23+
): any {
24+
return toRaw(this)[key](toRaw(value), ...args)
25+
}
26+
})
27+
1828
function createGetter(isReadonly = false, shallow = false) {
1929
return function get(target: object, key: string | symbol, receiver: object) {
30+
if (isArray(target) && hasOwn(arrayIdentityInstrumentations, key)) {
31+
return Reflect.get(arrayIdentityInstrumentations, key, receiver)
32+
}
2033
const res = Reflect.get(target, key, receiver)
2134
if (isSymbol(key) && builtInSymbols.has(key)) {
2235
return res

0 commit comments

Comments
 (0)