Skip to content

Commit 29b5f58

Browse files
committed
fix(reactivity): avoid using WeakMap for IE compatibility
Using a WeakMap polyfill isn't ideal because the reason we tried to use WeakMap was to work with non-extensible objects. However, WeakMap polyfill for non-extensible objects are non-weak and could lead to memory leaks. The trade-off is that we remove support for `readonly()` on non-extensible objects, which seems reasonable. close #12837
1 parent d1899ca commit 29b5f58

File tree

5 files changed

+23
-18
lines changed

5 files changed

+23
-18
lines changed

src/core/observer/index.ts

-2
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,6 @@ import {
1616
noop
1717
} from '../util/index'
1818
import { isReadonly, isRef, TrackOpTypes, TriggerOpTypes } from '../../v3'
19-
import { rawMap } from '../../v3/reactivity/reactive'
2019

2120
const arrayKeys = Object.getOwnPropertyNames(arrayMethods)
2221

@@ -116,7 +115,6 @@ export function observe(
116115
(isArray(value) || isPlainObject(value)) &&
117116
Object.isExtensible(value) &&
118117
!value.__v_skip /* ReactiveFlags.SKIP */ &&
119-
!rawMap.has(value) &&
120118
!isRef(value) &&
121119
!(value instanceof VNode)
122120
) {

src/v3/reactivity/reactive.ts

+4-6
Original file line numberDiff line numberDiff line change
@@ -5,13 +5,10 @@ import {
55
isPrimitive,
66
warn,
77
toRawType,
8-
isServerRendering,
9-
isObject
8+
isServerRendering
109
} from 'core/util'
1110
import type { Ref, UnwrapRefSimple, RawSymbol } from './ref'
1211

13-
export const rawMap = new WeakMap()
14-
1512
export const enum ReactiveFlags {
1613
SKIP = '__v_skip',
1714
IS_READONLY = '__v_isReadonly',
@@ -122,8 +119,9 @@ export function toRaw<T>(observed: T): T {
122119
export function markRaw<T extends object>(
123120
value: T
124121
): T & { [RawSymbol]?: true } {
125-
if (isObject(value)) {
126-
rawMap.set(value, true)
122+
// non-extensible objects won't be observed anyway
123+
if (Object.isExtensible(value)) {
124+
def(value, ReactiveFlags.SKIP, true)
127125
}
128126
return value
129127
}

src/v3/reactivity/readonly.ts

+11-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 rawToReadonlyMap = new WeakMap()
36-
const rawToShallowReadonlyMap = new WeakMap()
35+
const rawToReadonlyFlag = `__v_rawToReadonly`
36+
const rawToShallowReadonlyFlag = `__v_rawToShallowReadonly`
3737

3838
export function readonly<T extends object>(
3939
target: T
@@ -57,20 +57,26 @@ function createReadonly(target: any, shallow: boolean) {
5757
return target as any
5858
}
5959

60+
if (__DEV__ && !Object.isExtensible(target)) {
61+
warn(
62+
`Vue 2 does not support creating readonly proxy for non-extensible object.`
63+
)
64+
}
65+
6066
// already a readonly object
6167
if (isReadonly(target)) {
6268
return target as any
6369
}
6470

6571
// already has a readonly proxy
66-
const map = shallow ? rawToShallowReadonlyMap : rawToReadonlyMap
67-
const existingProxy = map.get(target)
72+
const existingFlag = shallow ? rawToShallowReadonlyFlag : rawToReadonlyFlag
73+
const existingProxy = target[existingFlag]
6874
if (existingProxy) {
6975
return existingProxy
7076
}
7177

7278
const proxy = Object.create(Object.getPrototypeOf(target))
73-
map.set(target, proxy)
79+
def(target, existingFlag, proxy)
7480

7581
def(proxy, ReactiveFlags.IS_READONLY, true)
7682
def(proxy, ReactiveFlags.RAW, target)

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

+1-1
Original file line numberDiff line numberDiff line change
@@ -259,7 +259,7 @@ describe('reactivity/reactive', () => {
259259
})
260260

261261
test('markRaw on non-extensible objects', () => {
262-
const foo = Object.freeze({})
262+
const foo = Object.seal({})
263263
markRaw(foo)
264264
expect(isReactive(reactive(foo))).toBe(false)
265265
})

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

+7-4
Original file line numberDiff line numberDiff line change
@@ -526,10 +526,13 @@ describe('reactivity/readonly', () => {
526526
expect(`et operation on key "x" failed`).toHaveBeenWarned()
527527
})
528528

529-
test('compatibility with non-extensible objects', () => {
529+
test('warn non-extensible objects', () => {
530530
const foo = Object.freeze({ a: 1 })
531-
const bar = readonly(foo)
532-
expect(isReadonly(bar)).toBe(true)
533-
expect(readonly(foo)).toBe(bar)
531+
try {
532+
readonly(foo)
533+
} catch (e) {}
534+
expect(
535+
`Vue 2 does not support creating readonly proxy for non-extensible object`
536+
).toHaveBeenWarned()
534537
})
535538
})

0 commit comments

Comments
 (0)