feat(watch): support directly watching reactive object with deep default

Also warn invalid watch sources

close #1110
This commit is contained in:
Evan You 2020-05-04 09:27:28 -04:00
parent 64ef7c76bf
commit 6b33cc4229
2 changed files with 70 additions and 27 deletions

View File

@ -103,6 +103,19 @@ describe('api: watch', () => {
expect(dummy).toMatchObject([1, 0]) expect(dummy).toMatchObject([1, 0])
}) })
it('directly watching reactive object (with automatic deep: true)', async () => {
const src = reactive({
count: 0
})
let dummy
watch(src, ({ count }) => {
dummy = count
})
src.count++
await nextTick()
expect(dummy).toBe(1)
})
it('watching multiple sources', async () => { it('watching multiple sources', async () => {
const state = reactive({ count: 1 }) const state = reactive({ count: 1 })
const count = ref(1) const count = ref(1)
@ -142,6 +155,12 @@ describe('api: watch', () => {
expect(dummy).toMatchObject([[2, true], [1, false]]) expect(dummy).toMatchObject([[2, true], [1, false]])
}) })
it('warn invalid watch source', () => {
// @ts-ignore
watch(1, () => {})
expect(`Invalid watch source`).toHaveBeenWarned()
})
it('stopping the watcher (effect)', async () => { it('stopping the watcher (effect)', async () => {
const state = reactive({ count: 0 }) const state = reactive({ count: 0 })
let dummy let dummy

View File

@ -4,7 +4,8 @@ import {
isRef, isRef,
Ref, Ref,
ComputedRef, ComputedRef,
ReactiveEffectOptions ReactiveEffectOptions,
isReactive
} from '@vue/reactivity' } from '@vue/reactivity'
import { queueJob } from './scheduler' import { queueJob } from './scheduler'
import { import {
@ -80,14 +81,7 @@ export function watchEffect(
// initial value for watchers to trigger on undefined initial values // initial value for watchers to trigger on undefined initial values
const INITIAL_WATCHER_VALUE = {} const INITIAL_WATCHER_VALUE = {}
// overload #1: single source + cb // overload #1: array of multiple sources + cb
export function watch<T, Immediate extends Readonly<boolean> = false>(
source: WatchSource<T>,
cb: WatchCallback<T, Immediate extends true ? (T | undefined) : T>,
options?: WatchOptions<Immediate>
): WatchStopHandle
// overload #2: array of multiple sources + cb
// Readonly constraint helps the callback to correctly infer value types based // 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 // on position in the source array. Otherwise the values will get a union type
// of all possible value types. // of all possible value types.
@ -100,6 +94,23 @@ export function watch<
options?: WatchOptions<Immediate> options?: WatchOptions<Immediate>
): WatchStopHandle ): WatchStopHandle
// overload #2: single source + cb
export function watch<T, Immediate extends Readonly<boolean> = false>(
source: WatchSource<T>,
cb: WatchCallback<T, Immediate extends true ? (T | undefined) : T>,
options?: WatchOptions<Immediate>
): WatchStopHandle
// overload #3: watching reactive object w/ cb
export function watch<
T extends object,
Immediate extends Readonly<boolean> = false
>(
source: T,
cb: WatchCallback<T, Immediate extends true ? (T | undefined) : T>,
options?: WatchOptions<Immediate>
): WatchStopHandle
// implementation // implementation
export function watch<T = any>( export function watch<T = any>(
source: WatchSource<T> | WatchSource<T>[], source: WatchSource<T> | WatchSource<T>[],
@ -149,26 +160,39 @@ function doWatch(
) )
} else if (isRef(source)) { } else if (isRef(source)) {
getter = () => source.value getter = () => source.value
} else if (cb) { } else if (isReactive(source)) {
// getter with cb getter = () => source
getter = () => deep = true
callWithErrorHandling(source, instance, ErrorCodes.WATCH_GETTER) } else if (isFunction(source)) {
} else { if (cb) {
// no cb -> simple effect // getter with cb
getter = () => { getter = () =>
if (instance && instance.isUnmounted) { callWithErrorHandling(source, instance, ErrorCodes.WATCH_GETTER)
return } else {
// no cb -> simple effect
getter = () => {
if (instance && instance.isUnmounted) {
return
}
if (cleanup) {
cleanup()
}
return callWithErrorHandling(
source,
instance,
ErrorCodes.WATCH_CALLBACK,
[onInvalidate]
)
} }
if (cleanup) {
cleanup()
}
return callWithErrorHandling(
source,
instance,
ErrorCodes.WATCH_CALLBACK,
[onInvalidate]
)
} }
} else {
getter = NOOP
warn(
`Invalid watch source: `,
source,
`A watch source can only be a getter/effect function, a ref, ` +
`a reactive object, or an array of these types.`
)
} }
if (cb && deep) { if (cb && deep) {