-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy patheffect.ts
108 lines (89 loc) · 1.99 KB
/
effect.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
103
104
105
106
107
108
import { extend } from '../shared'
let activeEffect
let shouldTrack
export class ReactiveEffect {
private _fn: any // 保存传入的函数
deps = [] // 依赖数组
onStop: any // stop函数的回调
active = true // 是否存在该依赖
constructor(fn, public scheduler) {
this._fn = fn
}
run() {
// 1、收集依赖
// 2、shouldTrack来作区分
if (!this.active) {
return this._fn()
}
shouldTrack = true
activeEffect = this
const result = this._fn()
shouldTrack = false
return result
}
stop() {
if (this.active) {
cleanupEffect(this)
if (this.onStop) {
this.onStop()
}
}
}
}
function cleanupEffect(effect) {
effect.deps.forEach((dep: any) => {
dep.delete(effect)
})
effect.deps.length = 0
}
const targetMap = new WeakMap()
// 收集依赖
export function track(target, key) {
if (!isTracking()) return
let depsMap = targetMap.get(target)
if (!depsMap) {
depsMap = new Map()
targetMap.set(target, depsMap)
}
let dep = depsMap.get(key)
if (!dep) {
dep = new Set()
depsMap.set(key, dep)
}
trackEffects(dep)
}
export function trackEffects(dep) {
if (dep.has(activeEffect)) return
dep.add(activeEffect)
activeEffect.deps.push(dep)
}
export function isTracking() {
return shouldTrack && activeEffect !== undefined
}
// 触发依赖
export function trigger(target, key) {
const depsMap = targetMap.get(target)
const dep = depsMap.get(key)
triggerEffects(dep)
}
export function triggerEffects(dep) {
dep.forEach((effect) => {
if (effect.scheduler) {
effect.scheduler()
} else {
effect.run()
}
})
}
export function effect(fn, options: any = {}) {
const { scheduler, onStop } = options
const _effect: any = new ReactiveEffect(fn, scheduler)
_effect.run()
extend(_effect, options)
const runner: any = _effect.run.bind(_effect)
runner.effect = _effect
return runner
}
export function stop(runner) {
runner.effect.stop()
}