Skip to content

Commit 4a0d88e

Browse files
committed
fix(reactivity): use WeakMap for proxy/raw checks, compat with non-extensible objects
fix #12799 close #12798
1 parent 27eed82 commit 4a0d88e

File tree

5 files changed

+28
-8
lines changed

5 files changed

+28
-8
lines changed

src/core/observer/index.ts

+3-1
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@ import {
1717
noop
1818
} from '../util/index'
1919
import { isReadonly, isRef, TrackOpTypes, TriggerOpTypes } from '../../v3'
20+
import { rawMap } from '../../v3/reactivity/reactive'
2021

2122
const arrayKeys = Object.getOwnPropertyNames(arrayMethods)
2223

@@ -118,7 +119,8 @@ export function observe(
118119
(ssrMockReactivity || !isServerRendering()) &&
119120
(isArray(value) || isPlainObject(value)) &&
120121
Object.isExtensible(value) &&
121-
!value.__v_skip /* ReactiveFlags.SKIP */
122+
!value.__v_skip /* ReactiveFlags.SKIP */ &&
123+
!rawMap.has(value)
122124
) {
123125
ob = new Observer(value, shallow, ssrMockReactivity)
124126
}

src/v3/reactivity/reactive.ts

+7-2
Original file line numberDiff line numberDiff line change
@@ -5,10 +5,13 @@ import {
55
isPrimitive,
66
warn,
77
toRawType,
8-
isServerRendering
8+
isServerRendering,
9+
isObject
910
} from 'core/util'
1011
import type { Ref, UnwrapRefSimple, RawSymbol } from './ref'
1112

13+
export const rawMap = new WeakMap()
14+
1215
export const enum ReactiveFlags {
1316
SKIP = '__v_skip',
1417
IS_READONLY = '__v_isReadonly',
@@ -119,7 +122,9 @@ export function toRaw<T>(observed: T): T {
119122
export function markRaw<T extends object>(
120123
value: T
121124
): T & { [RawSymbol]?: true } {
122-
def(value, ReactiveFlags.SKIP, true)
125+
if (isObject(value)) {
126+
rawMap.set(value, true)
127+
}
123128
return value
124129
}
125130

src/v3/reactivity/readonly.ts

+5-5
Original file line numberDiff line numberDiff line change
@@ -32,8 +32,8 @@ export type DeepReadonly<T> = T extends Builtin
3232
? { readonly [K in keyof T]: DeepReadonly<T[K]> }
3333
: Readonly<T>
3434

35-
const rawToReadonlyFlag = `__v_rawToReadonly`
36-
const rawToShallowReadonlyFlag = `__v_rawToShallowReadonly`
35+
const rawToReadonlyMap = new WeakMap()
36+
const rawToShallowReadonlyMap = new WeakMap()
3737

3838
export function readonly<T extends object>(
3939
target: T
@@ -63,14 +63,14 @@ function createReadonly(target: any, shallow: boolean) {
6363
}
6464

6565
// already has a readonly proxy
66-
const existingFlag = shallow ? rawToShallowReadonlyFlag : rawToReadonlyFlag
67-
const existingProxy = target[existingFlag]
66+
const map = shallow ? rawToShallowReadonlyMap : rawToReadonlyMap
67+
const existingProxy = map.get(target)
6868
if (existingProxy) {
6969
return existingProxy
7070
}
7171

7272
const proxy = Object.create(Object.getPrototypeOf(target))
73-
def(target, existingFlag, proxy)
73+
map.set(target, proxy)
7474

7575
def(proxy, ReactiveFlags.IS_READONLY, true)
7676
def(proxy, ReactiveFlags.RAW, target)

test/unit/features/v3/reactivity/reactive.spec.ts

+6
Original file line numberDiff line numberDiff line change
@@ -258,6 +258,12 @@ describe('reactivity/reactive', () => {
258258
expect(isReactive(obj.bar)).toBe(false)
259259
})
260260

261+
test('markRaw on non-extensible objects', () => {
262+
const foo = Object.freeze({})
263+
markRaw(foo)
264+
expect(isReactive(reactive(foo))).toBe(false)
265+
})
266+
261267
test('should not observe non-extensible objects', () => {
262268
const obj = reactive({
263269
foo: Object.preventExtensions({ a: 1 }),

test/unit/features/v3/reactivity/readonly.spec.ts

+7
Original file line numberDiff line numberDiff line change
@@ -525,4 +525,11 @@ describe('reactivity/readonly', () => {
525525
expect(readonlyFoo.x).toBe(1)
526526
expect(`et operation on key "x" failed`).toHaveBeenWarned()
527527
})
528+
529+
test('compatibility with non-extensible objects', () => {
530+
const foo = Object.freeze({ a: 1 })
531+
const bar = readonly(foo)
532+
expect(isReadonly(bar)).toBe(true)
533+
expect(readonly(foo)).toBe(bar)
534+
})
528535
})

0 commit comments

Comments
 (0)