-
Notifications
You must be signed in to change notification settings - Fork 344
/
Copy pathcomputed.ts
102 lines (90 loc) · 2.38 KB
/
computed.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
import { getVueConstructor } from '../runtimeContext'
import { createRef, ComputedRef, WritableComputedRef } from '../reactivity'
import {
warn,
noopFn,
defineComponentInstance,
getVueInternalClasses,
isFunction,
} from '../utils'
import { getCurrentScopeVM } from './effectScope'
export type ComputedGetter<T> = (ctx?: any) => T
export type ComputedSetter<T> = (v: T) => void
export interface WritableComputedOptions<T> {
get: ComputedGetter<T>
set: ComputedSetter<T>
}
// read-only
export function computed<T>(getter: ComputedGetter<T>): ComputedRef<T>
// writable
export function computed<T>(
options: WritableComputedOptions<T>
): WritableComputedRef<T>
// implement
export function computed<T>(
getterOrOptions: ComputedGetter<T> | WritableComputedOptions<T>
): ComputedRef<T> | WritableComputedRef<T> {
const vm = getCurrentScopeVM()
let getter: ComputedGetter<T>
let setter: ComputedSetter<T> | undefined
if (isFunction(getterOrOptions)) {
getter = getterOrOptions
} else {
getter = getterOrOptions.get
setter = getterOrOptions.set
}
let computedSetter
let computedGetter
if (vm && !vm.$isServer) {
const { Watcher, Dep } = getVueInternalClasses()
let watcher: any
computedGetter = () => {
if (!watcher) {
watcher = new Watcher(vm, getter, noopFn, { lazy: true })
}
if (watcher.dirty) {
watcher.evaluate()
}
if (Dep.target) {
watcher.depend()
}
return watcher.value
}
computedSetter = (v: T) => {
if (__DEV__ && !setter) {
warn('Write operation failed: computed value is readonly.', vm!)
return
}
if (setter) {
setter(v)
}
}
} else {
// fallback
const computedHost = defineComponentInstance(getVueConstructor(), {
computed: {
$$state: {
get: getter,
set: setter,
},
},
})
vm && vm.$on('hook:destroyed', () => computedHost.$destroy())
computedGetter = () => (computedHost as any).$$state
computedSetter = (v: T) => {
if (__DEV__ && !setter) {
warn('Write operation failed: computed value is readonly.', vm!)
return
}
;(computedHost as any).$$state = v
}
}
return createRef<T>(
{
get: computedGetter,
set: computedSetter,
},
!setter,
true
) as WritableComputedRef<T> | ComputedRef<T>
}