From 8aa6426b048133994072d0b3bf0fb17c38664d1e Mon Sep 17 00:00:00 2001 From: MinatoHikari <35342316+MinatoHikari@users.noreply.github.com> Date: Tue, 26 Apr 2022 05:03:04 +0800 Subject: [PATCH 1/2] fix(type): align watch types with vue3 --- src/apis/watch.ts | 26 +++++++++++++++++++------- test-dts/watch.test-d.tsx | 4 ++-- test/v3/runtime-core/apiWatch.spec.ts | 2 +- 3 files changed, 22 insertions(+), 10 deletions(-) diff --git a/src/apis/watch.ts b/src/apis/watch.ts index b3e0984c..48c2740e 100644 --- a/src/apis/watch.ts +++ b/src/apis/watch.ts @@ -44,6 +44,8 @@ type MapOldSources = { : never } +type MultiWatchSources = (WatchSource | object)[] + export interface WatchOptionsBase { flush?: FlushMode // onTrack?: ReactiveEffectOptions['onTrack']; @@ -416,12 +418,22 @@ export function watchSyncEffect(effect: WatchEffect) { return watchEffect(effect, { flush: 'sync' }) } -// overload #1: 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. +// overload #1 + #2: array of multiple sources + cb + +// overload #1: Readonly constraint helps the callback to correctly infer value types based +// on position in the source array. +export function watch< + T extends Readonly, + Immediate extends Readonly = false +>( + sources: T, + cb: WatchCallback, MapOldSources>, + options?: WatchOptions +): WatchStopHandle + +// overload #2: the values will get a union type of all possible value types export function watch< - T extends Readonly[]>, + T extends MultiWatchSources, Immediate extends Readonly = false >( sources: [...T], @@ -429,14 +441,14 @@ export function watch< options?: WatchOptions ): WatchStopHandle -// overload #2: single source + cb +// overload #3: single source + cb export function watch = false>( source: WatchSource, cb: WatchCallback, options?: WatchOptions ): WatchStopHandle -// overload #3: watching reactive object w/ cb +// overload #4: watching reactive object w/ cb export function watch< T extends object, Immediate extends Readonly = false diff --git a/test-dts/watch.test-d.tsx b/test-dts/watch.test-d.tsx index 2811743c..946e2f92 100644 --- a/test-dts/watch.test-d.tsx +++ b/test-dts/watch.test-d.tsx @@ -16,7 +16,7 @@ watch([source, source2, source3], (values, oldValues) => { }) // const array -watch([source, source2, source3], (values, oldValues) => { +watch([source, source2, source3] as const, (values, oldValues) => { expectType>(values) expectType>(oldValues) }) @@ -42,7 +42,7 @@ watch( // const array watch( - [source, source2, source3], + [source, source2, source3] as const, (values, oldValues) => { expectType>(values) expectType< diff --git a/test/v3/runtime-core/apiWatch.spec.ts b/test/v3/runtime-core/apiWatch.spec.ts index b6825e0f..57dc6335 100644 --- a/test/v3/runtime-core/apiWatch.spec.ts +++ b/test/v3/runtime-core/apiWatch.spec.ts @@ -143,7 +143,7 @@ describe('api: watch', () => { const status = ref(false) let dummy - watch([() => state.count, status], (vals, oldVals) => { + watch([() => state.count, status] as const, (vals, oldVals) => { dummy = [vals, oldVals] const [count] = vals const [, oldStatus] = oldVals From 688c99867381b0610d0495a233387b984483e413 Mon Sep 17 00:00:00 2001 From: MinatoHikari <35342316+MinatoHikari@users.noreply.github.com> Date: Wed, 27 Apr 2022 21:16:20 +0800 Subject: [PATCH 2/2] fix(type): better watch type - add another overload for spread readonly array, change their sequence. - merge MapSources and OldMapSources --- src/apis/watch.ts | 36 +++++++++++++++++++++--------------- test-dts/watch.test-d.tsx | 35 +++++++++++++++++++++++++++++++---- 2 files changed, 52 insertions(+), 19 deletions(-) diff --git a/src/apis/watch.ts b/src/apis/watch.ts index 48c2740e..00788d52 100644 --- a/src/apis/watch.ts +++ b/src/apis/watch.ts @@ -32,11 +32,7 @@ export type WatchCallback = ( onInvalidate: InvalidateCbRegistrator ) => any -type MapSources = { - [K in keyof T]: T[K] extends WatchSource ? V : never -} - -type MapOldSources = { +declare type MapSources = { [K in keyof T]: T[K] extends WatchSource ? Immediate extends true ? V | undefined @@ -206,8 +202,8 @@ function patchWatcherTeardown(watcher: VueWatcher, runCleanup: () => void) { function createWatcher( vm: ComponentInstance, - source: WatchSource | WatchSource[] | WatchEffect, - cb: WatchCallback | null, + source: WatchSource | WatchSource[] | WatchEffect, + cb: WatchCallback | null, options: WatchOptions ): () => void { if (__DEV__ && !cb) { @@ -418,37 +414,47 @@ export function watchSyncEffect(effect: WatchEffect) { return watchEffect(effect, { flush: 'sync' }) } -// overload #1 + #2: array of multiple sources + cb +// overload #1 + #2 + #3: array of multiple sources + cb + +// overload #1: In readonly case the first overload must be spread tuple type. +// In otherwise members in tuple can not get the correct types. +export function watch< + T extends Readonly, + Immediate extends Readonly = false +>( + sources: [...T], + cb: WatchCallback, MapSources>, + options?: WatchOptions +): WatchStopHandle -// overload #1: Readonly constraint helps the callback to correctly infer value types based -// on position in the source array. +// overload #2: for not spread readonly tuple export function watch< T extends Readonly, Immediate extends Readonly = false >( sources: T, - cb: WatchCallback, MapOldSources>, + cb: WatchCallback, MapSources>, options?: WatchOptions ): WatchStopHandle -// overload #2: the values will get a union type of all possible value types +// overload #3: for not readonly multiSources export function watch< T extends MultiWatchSources, Immediate extends Readonly = false >( sources: [...T], - cb: WatchCallback, MapOldSources>, + cb: WatchCallback, MapSources>, options?: WatchOptions ): WatchStopHandle -// overload #3: single source + cb +// overload #4: single source + cb export function watch = false>( source: WatchSource, cb: WatchCallback, options?: WatchOptions ): WatchStopHandle -// overload #4: watching reactive object w/ cb +// overload #5: watching reactive object w/ cb export function watch< T extends object, Immediate extends Readonly = false diff --git a/test-dts/watch.test-d.tsx b/test-dts/watch.test-d.tsx index 946e2f92..581faa2a 100644 --- a/test-dts/watch.test-d.tsx +++ b/test-dts/watch.test-d.tsx @@ -10,17 +10,44 @@ watch(source, (value, oldValue) => { expectType(oldValue) }) -watch([source, source2, source3], (values, oldValues) => { - expectType<(string | number)[]>(values) - expectType<(string | number)[]>(oldValues) -}) +// spread array +watch( + [source, source2, source3], + ([source1, source2, source3], [oldSource1, oldSource2, oldSource3]) => { + expectType(source1) + expectType(source2) + expectType(source3) + expectType(oldSource1) + expectType(oldSource2) + expectType(oldSource3) + } +) // const array watch([source, source2, source3] as const, (values, oldValues) => { expectType>(values) expectType>(oldValues) + expectType(values[0]) + expectType(values[1]) + expectType(values[2]) + expectType(oldValues[0]) + expectType(oldValues[1]) + expectType(oldValues[2]) }) +// const spread array +watch( + [source, source2, source3] as const, + ([source1, source2, source3], [oldSource1, oldSource2, oldSource3]) => { + expectType(source1) + expectType(source2) + expectType(source3) + expectType(oldSource1) + expectType(oldSource2) + expectType(oldSource3) + } +) + // immediate watcher's oldValue will be undefined on first run. watch( source,