Skip to content

Commit 99a2e18

Browse files
committed
feat(runtime-core): add watchEffect API
BREAKING CHANGE: replae `watch(fn, options?)` with `watchEffect` The `watch(fn, options?)` signature has been replaced by the new `watchEffect` API, which has the same usage and behavior. `watch` now only supports the `watch(source, cb, options?)` signautre.
1 parent b36a76f commit 99a2e18

File tree

8 files changed

+77
-31
lines changed

8 files changed

+77
-31
lines changed

packages/runtime-core/__tests__/apiSetupContext.spec.ts

+3-3
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@ import {
66
render,
77
serializeInner,
88
nextTick,
9-
watch,
9+
watchEffect,
1010
defineComponent,
1111
triggerEvent,
1212
TestElement
@@ -55,7 +55,7 @@ describe('api: setup context', () => {
5555

5656
const Child = defineComponent({
5757
setup(props: { count: number }) {
58-
watch(() => {
58+
watchEffect(() => {
5959
dummy = props.count
6060
})
6161
return () => h('div', props.count)
@@ -88,7 +88,7 @@ describe('api: setup context', () => {
8888
},
8989

9090
setup(props) {
91-
watch(() => {
91+
watchEffect(() => {
9292
dummy = props.count
9393
})
9494
return () => h('div', props.count)

packages/runtime-core/__tests__/apiWatch.spec.ts

+43-14
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,12 @@
1-
import { watch, reactive, computed, nextTick, ref, h } from '../src/index'
1+
import {
2+
watch,
3+
watchEffect,
4+
reactive,
5+
computed,
6+
nextTick,
7+
ref,
8+
h
9+
} from '../src/index'
210
import { render, nodeOps, serializeInner } from '@vue/runtime-test'
311
import {
412
ITERATE_KEY,
@@ -13,10 +21,10 @@ import { mockWarn } from '@vue/shared'
1321
describe('api: watch', () => {
1422
mockWarn()
1523

16-
it('watch(effect)', async () => {
24+
it('effect', async () => {
1725
const state = reactive({ count: 0 })
1826
let dummy
19-
watch(() => {
27+
watchEffect(() => {
2028
dummy = state.count
2129
})
2230
expect(dummy).toBe(0)
@@ -117,10 +125,10 @@ describe('api: watch', () => {
117125
expect(dummy).toMatchObject([[2, true], [1, false]])
118126
})
119127

120-
it('stopping the watcher', async () => {
128+
it('stopping the watcher (effect)', async () => {
121129
const state = reactive({ count: 0 })
122130
let dummy
123-
const stop = watch(() => {
131+
const stop = watchEffect(() => {
124132
dummy = state.count
125133
})
126134
expect(dummy).toBe(0)
@@ -132,11 +140,32 @@ describe('api: watch', () => {
132140
expect(dummy).toBe(0)
133141
})
134142

143+
it('stopping the watcher (with source)', async () => {
144+
const state = reactive({ count: 0 })
145+
let dummy
146+
const stop = watch(
147+
() => state.count,
148+
count => {
149+
dummy = count
150+
}
151+
)
152+
153+
state.count++
154+
await nextTick()
155+
expect(dummy).toBe(1)
156+
157+
stop()
158+
state.count++
159+
await nextTick()
160+
// should not update
161+
expect(dummy).toBe(1)
162+
})
163+
135164
it('cleanup registration (effect)', async () => {
136165
const state = reactive({ count: 0 })
137166
const cleanup = jest.fn()
138167
let dummy
139-
const stop = watch(onCleanup => {
168+
const stop = watchEffect(onCleanup => {
140169
onCleanup(cleanup)
141170
dummy = state.count
142171
})
@@ -187,7 +216,7 @@ describe('api: watch', () => {
187216

188217
const Comp = {
189218
setup() {
190-
watch(() => {
219+
watchEffect(() => {
191220
assertion(count.value)
192221
})
193222
return () => count.value
@@ -221,7 +250,7 @@ describe('api: watch', () => {
221250

222251
const Comp = {
223252
setup() {
224-
watch(
253+
watchEffect(
225254
() => {
226255
assertion(count.value, count2.value)
227256
},
@@ -263,7 +292,7 @@ describe('api: watch', () => {
263292

264293
const Comp = {
265294
setup() {
266-
watch(
295+
watchEffect(
267296
() => {
268297
assertion(count.value)
269298
},
@@ -363,14 +392,14 @@ describe('api: watch', () => {
363392
expect(spy).toHaveBeenCalledTimes(3)
364393
})
365394

366-
it('warn immediate option when using effect signature', async () => {
395+
it('warn immediate option when using effect', async () => {
367396
const count = ref(0)
368397
let dummy
369-
// @ts-ignore
370-
watch(
398+
watchEffect(
371399
() => {
372400
dummy = count.value
373401
},
402+
// @ts-ignore
374403
{ immediate: false }
375404
)
376405
expect(dummy).toBe(0)
@@ -388,7 +417,7 @@ describe('api: watch', () => {
388417
events.push(e)
389418
})
390419
const obj = reactive({ foo: 1, bar: 2 })
391-
watch(
420+
watchEffect(
392421
() => {
393422
dummy = [obj.foo, 'bar' in obj, Object.keys(obj)]
394423
},
@@ -423,7 +452,7 @@ describe('api: watch', () => {
423452
events.push(e)
424453
})
425454
const obj = reactive({ foo: 1 })
426-
watch(
455+
watchEffect(
427456
() => {
428457
dummy = obj.foo
429458
},

packages/runtime-core/__tests__/components/Suspense.spec.ts

+3-2
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@ import {
99
nextTick,
1010
onMounted,
1111
watch,
12+
watchEffect,
1213
onUnmounted,
1314
onErrorCaptured
1415
} from '@vue/runtime-test'
@@ -163,7 +164,7 @@ describe('Suspense', () => {
163164
// extra tick needed for Node 12+
164165
deps.push(p.then(() => Promise.resolve()))
165166

166-
watch(() => {
167+
watchEffect(() => {
167168
calls.push('immediate effect')
168169
})
169170

@@ -265,7 +266,7 @@ describe('Suspense', () => {
265266
const p = new Promise(r => setTimeout(r, 1))
266267
deps.push(p)
267268

268-
watch(() => {
269+
watchEffect(() => {
269270
calls.push('immediate effect')
270271
})
271272

packages/runtime-core/__tests__/errorHandling.spec.ts

+8-7
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,8 @@ import {
77
watch,
88
ref,
99
nextTick,
10-
defineComponent
10+
defineComponent,
11+
watchEffect
1112
} from '@vue/runtime-test'
1213
import { setErrorRecovery } from '../src/errorHandling'
1314
import { mockWarn } from '@vue/shared'
@@ -241,7 +242,7 @@ describe('error handling', () => {
241242
expect(fn).toHaveBeenCalledWith(err, 'ref function')
242243
})
243244

244-
test('in watch (effect)', () => {
245+
test('in effect', () => {
245246
const err = new Error('foo')
246247
const fn = jest.fn()
247248

@@ -257,7 +258,7 @@ describe('error handling', () => {
257258

258259
const Child = {
259260
setup() {
260-
watch(() => {
261+
watchEffect(() => {
261262
throw err
262263
})
263264
return () => null
@@ -268,7 +269,7 @@ describe('error handling', () => {
268269
expect(fn).toHaveBeenCalledWith(err, 'watcher callback')
269270
})
270271

271-
test('in watch (getter)', () => {
272+
test('in watch getter', () => {
272273
const err = new Error('foo')
273274
const fn = jest.fn()
274275

@@ -298,7 +299,7 @@ describe('error handling', () => {
298299
expect(fn).toHaveBeenCalledWith(err, 'watcher getter')
299300
})
300301

301-
test('in watch (callback)', async () => {
302+
test('in watch callback', async () => {
302303
const err = new Error('foo')
303304
const fn = jest.fn()
304305

@@ -332,7 +333,7 @@ describe('error handling', () => {
332333
expect(fn).toHaveBeenCalledWith(err, 'watcher callback')
333334
})
334335

335-
test('in watch cleanup', async () => {
336+
test('in effect cleanup', async () => {
336337
const err = new Error('foo')
337338
const count = ref(0)
338339
const fn = jest.fn()
@@ -349,7 +350,7 @@ describe('error handling', () => {
349350

350351
const Child = {
351352
setup() {
352-
watch(onCleanup => {
353+
watchEffect(onCleanup => {
353354
count.value
354355
onCleanup(() => {
355356
throw err

packages/runtime-core/src/apiWatch.ts

+15
Original file line numberDiff line numberDiff line change
@@ -71,6 +71,14 @@ export type StopHandle = () => void
7171

7272
const invoke = (fn: Function) => fn()
7373

74+
// Simple effect.
75+
export function watchEffect(
76+
effect: WatchEffect,
77+
options?: BaseWatchOptions
78+
): StopHandle {
79+
return doWatch(effect, null, options)
80+
}
81+
7482
// initial value for watchers to trigger on undefined initial values
7583
const INITIAL_WATCHER_VALUE = {}
7684

@@ -110,6 +118,13 @@ export function watch<T = any>(
110118
// watch(source, cb)
111119
return doWatch(effectOrSource, cbOrOptions, options)
112120
} else {
121+
// TODO remove this in the next release
122+
__DEV__ &&
123+
warn(
124+
`\`watch(fn, options?)\` signature has been moved to a separate API. ` +
125+
`Use \`watchEffect(fn, options?)\` instead. \`watch\` will only ` +
126+
`support \`watch(source, cb, options?) signature in the next release.`
127+
)
113128
// watch(effect)
114129
return doWatch(effectOrSource, null, cbOrOptions)
115130
}

packages/runtime-core/src/index.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,7 @@ export {
1717
markNonReactive
1818
} from '@vue/reactivity'
1919
export { computed } from './apiComputed'
20-
export { watch } from './apiWatch'
20+
export { watch, watchEffect } from './apiWatch'
2121
export {
2222
onBeforeMount,
2323
onMounted,

packages/vue/examples/composition/commits.html

+2-2
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,7 @@ <h1>Latest Vue.js Commits</h1>
2222
</div>
2323

2424
<script>
25-
const { createApp, ref, watch } = Vue
25+
const { createApp, ref, watchEffect } = Vue
2626
const API_URL = `https://api.github.com/repos/vuejs/vue-next/commits?per_page=3&sha=`
2727

2828
const truncate = v => {
@@ -37,7 +37,7 @@ <h1>Latest Vue.js Commits</h1>
3737
const currentBranch = ref('master')
3838
const commits = ref(null)
3939

40-
watch(() => {
40+
watchEffect(() => {
4141
fetch(`${API_URL}${currentBranch.value}`)
4242
.then(res => res.json())
4343
.then(data => {

packages/vue/examples/composition/todomvc.html

+2-2
Original file line numberDiff line numberDiff line change
@@ -53,7 +53,7 @@ <h1>todos</h1>
5353
</div>
5454

5555
<script>
56-
const { createApp, reactive, computed, watch, onMounted, onUnmounted } = Vue
56+
const { createApp, reactive, computed, watchEffect, onMounted, onUnmounted } = Vue
5757

5858
const STORAGE_KEY = 'todos-vuejs-3.x'
5959
const todoStorage = {
@@ -119,7 +119,7 @@ <h1>todos</h1>
119119
})
120120
})
121121

122-
watch(() => {
122+
watchEffect(() => {
123123
todoStorage.save(state.todos)
124124
})
125125

0 commit comments

Comments
 (0)