Skip to content

Commit dabdc5e

Browse files
authored
feat(computed): add readonly flag if no setter is provided (#1654)
1 parent ad199e1 commit dabdc5e

File tree

2 files changed

+24
-1
lines changed

2 files changed

+24
-1
lines changed

packages/reactivity/__tests__/computed.spec.ts

+20-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,8 @@ import {
44
effect,
55
stop,
66
ref,
7-
WritableComputedRef
7+
WritableComputedRef,
8+
isReadonly
89
} from '../src'
910
import { mockWarn } from '@vue/shared'
1011

@@ -177,4 +178,22 @@ describe('reactivity/computed', () => {
177178
'Write operation failed: computed value is readonly'
178179
).toHaveBeenWarnedLast()
179180
})
181+
182+
it('should be readonly', () => {
183+
let a = { a: 1 }
184+
const x = computed(() => a)
185+
expect(isReadonly(x)).toBe(true)
186+
expect(isReadonly(x.value)).toBe(false)
187+
expect(isReadonly(x.value.a)).toBe(false)
188+
const z = computed<typeof a>({
189+
get() {
190+
return a
191+
},
192+
set(v) {
193+
a = v
194+
}
195+
})
196+
expect(isReadonly(z)).toBe(false)
197+
expect(isReadonly(z.value.a)).toBe(false)
198+
})
180199
})

packages/reactivity/src/computed.ts

+4
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@ import { effect, ReactiveEffect, trigger, track } from './effect'
22
import { TriggerOpTypes, TrackOpTypes } from './operations'
33
import { Ref } from './ref'
44
import { isFunction, NOOP } from '@vue/shared'
5+
import { ReactiveFlags } from './reactive'
56

67
export interface ComputedRef<T = any> extends WritableComputedRef<T> {
78
readonly value: T
@@ -56,6 +57,9 @@ export function computed<T>(
5657
})
5758
computed = {
5859
__v_isRef: true,
60+
[ReactiveFlags.IS_READONLY]:
61+
isFunction(getterOrOptions) || !getterOrOptions.set,
62+
5963
// expose effect so computed can be stopped
6064
effect: runner,
6165
get value() {

0 commit comments

Comments
 (0)