vue3-yuanma/packages/runtime-core/src/apiWatch.ts

269 lines
6.5 KiB
TypeScript
Raw Normal View History

2019-05-29 15:44:59 +00:00
import {
effect,
stop,
isRef,
Ref,
ComputedRef,
2019-05-29 15:44:59 +00:00
ReactiveEffectOptions
} from '@vue/reactivity'
import { queueJob } from './scheduler'
import {
EMPTY_OBJ,
isObject,
isArray,
isFunction,
isString,
hasChanged
} from '@vue/shared'
2019-08-20 13:38:00 +00:00
import { recordEffect } from './apiReactivity'
import {
currentInstance,
ComponentInternalInstance,
currentSuspense,
Data
} from './component'
import {
2019-09-06 16:58:31 +00:00
ErrorCodes,
callWithErrorHandling,
callWithAsyncErrorHandling
} from './errorHandling'
import { onBeforeUnmount } from './apiLifecycle'
2019-11-02 16:18:35 +00:00
import { queuePostRenderEffect } from './renderer'
import { warn } from './warning'
2019-10-22 15:26:48 +00:00
2019-12-30 16:30:12 +00:00
export type WatchEffect = (onCleanup: CleanupRegistrator) => void
export type WatchSource<T = any> = Ref<T> | ComputedRef<T> | (() => T)
export type WatchCallback<T = any> = (
2019-10-22 15:26:48 +00:00
value: T,
oldValue: T,
onCleanup: CleanupRegistrator
) => any
2019-05-29 15:44:59 +00:00
2019-12-30 16:30:12 +00:00
type MapSources<T> = {
[K in keyof T]: T[K] extends WatchSource<infer V> ? V : never
}
export type CleanupRegistrator = (invalidate: () => void) => void
2019-05-29 15:44:59 +00:00
export interface WatchOptions {
lazy?: boolean
flush?: 'pre' | 'post' | 'sync'
deep?: boolean
onTrack?: ReactiveEffectOptions['onTrack']
onTrigger?: ReactiveEffectOptions['onTrigger']
}
2019-12-30 16:19:57 +00:00
export type StopHandle = () => void
2019-08-19 02:49:08 +00:00
2019-05-29 15:44:59 +00:00
const invoke = (fn: Function) => fn()
// overload #1: simple effect
2019-12-30 16:30:12 +00:00
export function watch(effect: WatchEffect, options?: WatchOptions): StopHandle
2019-08-19 02:49:08 +00:00
// overload #2: single source + cb
2019-05-29 15:44:59 +00:00
export function watch<T>(
2019-12-30 16:30:12 +00:00
source: WatchSource<T>,
cb: WatchCallback<T>,
2019-08-19 02:49:08 +00:00
options?: WatchOptions
): StopHandle
// overload #3: array of multiple sources + cb
// Readonly constraint helps the callback to correctly infer value types based
// on position in the source array. Otherwise the values will get a union type
// of all possible value types.
2019-12-30 16:30:12 +00:00
export function watch<T extends Readonly<WatchSource<unknown>[]>>(
2019-08-19 02:49:08 +00:00
sources: T,
2019-12-30 16:30:12 +00:00
cb: WatchCallback<MapSources<T>>,
2019-08-19 02:49:08 +00:00
options?: WatchOptions
): StopHandle
// implementation
export function watch<T = any>(
2019-12-30 16:30:12 +00:00
effectOrSource: WatchSource<T> | WatchSource<T>[] | WatchEffect,
cbOrOptions?: WatchCallback<T> | WatchOptions,
2019-08-19 02:49:08 +00:00
options?: WatchOptions
): StopHandle {
if (isFunction(cbOrOptions)) {
2019-08-19 02:49:08 +00:00
// effect callback as 2nd argument - this is a source watcher
return doWatch(effectOrSource, cbOrOptions, options)
2019-08-19 02:49:08 +00:00
} else {
// 2nd argument is either missing or an options object
// - this is a simple effect watcher
return doWatch(effectOrSource, null, cbOrOptions)
2019-08-19 02:49:08 +00:00
}
}
function doWatch(
2019-12-30 16:30:12 +00:00
source: WatchSource | WatchSource[] | WatchEffect,
cb: WatchCallback | null,
2019-08-19 02:49:08 +00:00
{ lazy, deep, flush, onTrack, onTrigger }: WatchOptions = EMPTY_OBJ
): StopHandle {
2019-08-31 20:36:36 +00:00
const instance = currentInstance
const suspense = currentSuspense
2019-05-29 15:44:59 +00:00
let getter: () => any
if (isArray(source)) {
getter = () =>
2019-10-22 15:52:29 +00:00
source.map(
s =>
isRef(s)
? s.value
: callWithErrorHandling(s, instance, ErrorCodes.WATCH_GETTER)
)
} else if (isRef(source)) {
getter = () => source.value
} else if (cb) {
// getter with cb
getter = () =>
2019-09-06 16:58:31 +00:00
callWithErrorHandling(source, instance, ErrorCodes.WATCH_GETTER)
} else {
// no cb -> simple effect
getter = () => {
if (instance && instance.isUnmounted) {
return
}
if (cleanup) {
cleanup()
}
return callWithErrorHandling(
source,
instance,
2019-09-06 16:58:31 +00:00
ErrorCodes.WATCH_CALLBACK,
[registerCleanup]
)
}
}
if (deep) {
const baseGetter = getter
getter = () => traverse(baseGetter())
}
let cleanup: Function
2019-08-19 02:49:08 +00:00
const registerCleanup: CleanupRegistrator = (fn: () => void) => {
2019-10-30 15:29:08 +00:00
cleanup = runner.options.onStop = () => {
2019-09-06 16:58:31 +00:00
callWithErrorHandling(fn, instance, ErrorCodes.WATCH_CLEANUP)
}
2019-06-06 07:19:04 +00:00
}
2019-08-27 02:47:38 +00:00
let oldValue = isArray(source) ? [] : undefined
2019-05-29 15:44:59 +00:00
const applyCb = cb
? () => {
if (instance && instance.isUnmounted) {
return
}
2019-05-29 15:44:59 +00:00
const newValue = runner()
if (deep || hasChanged(newValue, oldValue)) {
2019-06-06 07:19:04 +00:00
// cleanup before running cb again
if (cleanup) {
cleanup()
2019-05-29 15:44:59 +00:00
}
2019-09-06 16:58:31 +00:00
callWithAsyncErrorHandling(cb, instance, ErrorCodes.WATCH_CALLBACK, [
2019-08-30 19:15:23 +00:00
newValue,
oldValue,
registerCleanup
])
2019-05-29 15:44:59 +00:00
oldValue = newValue
}
}
: void 0
let scheduler: (job: () => any) => void
if (flush === 'sync') {
scheduler = invoke
} else if (flush === 'pre') {
scheduler = job => {
if (!instance || instance.vnode.el != null) {
queueJob(job)
} else {
// with 'pre' option, the first call must happen before
// the component is mounted so it is called synchronously.
job()
}
}
} else {
scheduler = job => {
queuePostRenderEffect(job, suspense)
}
}
2019-08-19 18:44:52 +00:00
2019-05-29 15:44:59 +00:00
const runner = effect(getter, {
lazy: true,
// so it runs before component update effects in pre flush mode
computed: true,
2019-06-06 07:19:04 +00:00
onTrack,
onTrigger,
2019-08-19 18:44:52 +00:00
scheduler: applyCb ? () => scheduler(applyCb) : scheduler
2019-05-29 15:44:59 +00:00
})
if (lazy && cb) {
oldValue = runner()
} else {
if (__DEV__ && lazy && !cb) {
warn(
`watch() lazy option is only respected when using the ` +
`watch(getter, callback) signature.`
)
}
2019-08-19 18:44:52 +00:00
if (applyCb) {
scheduler(applyCb)
} else {
scheduler(runner)
}
2019-05-29 15:44:59 +00:00
}
recordEffect(runner)
return () => {
stop(runner)
if (instance) {
const effects = instance.effects!
const index = effects.indexOf(runner)
if (index > -1) {
effects.splice(index, 1)
}
}
2019-05-29 15:44:59 +00:00
}
}
2019-09-04 15:36:27 +00:00
// this.$watch
export function instanceWatch(
2019-09-06 16:58:31 +00:00
this: ComponentInternalInstance,
2019-09-04 15:36:27 +00:00
source: string | Function,
cb: Function,
options?: WatchOptions
): StopHandle {
const ctx = this.proxy as Data
2019-09-04 15:36:27 +00:00
const getter = isString(source) ? () => ctx[source] : source.bind(ctx)
const stop = watch(getter, cb.bind(ctx), options)
onBeforeUnmount(stop, this)
2019-09-04 15:36:27 +00:00
return stop
}
2019-10-22 15:26:48 +00:00
function traverse(value: unknown, seen: Set<unknown> = new Set()) {
2019-05-29 15:44:59 +00:00
if (!isObject(value) || seen.has(value)) {
return
}
seen.add(value)
if (isArray(value)) {
for (let i = 0; i < value.length; i++) {
traverse(value[i], seen)
}
2019-08-27 19:01:01 +00:00
} else if (value instanceof Map) {
value.forEach((v, key) => {
2019-08-27 19:01:01 +00:00
// to register mutation dep for existing keys
traverse(value.get(key), seen)
})
} else if (value instanceof Set) {
value.forEach(v => {
2019-05-29 15:44:59 +00:00
traverse(v, seen)
})
} else {
for (const key in value) {
traverse(value[key], seen)
}
}
return value
}