Skip to content

Commit c944827

Browse files
eddyerburghyyx990803
authored andcommitted
feat: add async option (#8240)
This is an option intended only for testing purposes, and should not be used in application code.
1 parent 833175e commit c944827

File tree

6 files changed

+90
-0
lines changed

6 files changed

+90
-0
lines changed

src/core/config.js

+9
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,9 @@ export type Config = {
2828
getTagNamespace: (x?: string) => string | void;
2929
mustUseProp: (tag: string, type: ?string, name: string) => boolean;
3030

31+
// private
32+
async: boolean;
33+
3134
// legacy
3235
_lifecycleHooks: Array<string>;
3336
};
@@ -114,6 +117,12 @@ export default ({
114117
*/
115118
mustUseProp: no,
116119

120+
/**
121+
* Perform updates asynchronously. Intended to be used by Vue Test Utils
122+
* This will significantly reduce performance if set to false.
123+
*/
124+
async: true,
125+
117126
/**
118127
* Exposed for legacy reasons
119128
*/

src/core/observer/dep.js

+7
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22

33
import type Watcher from './watcher'
44
import { remove } from '../util/index'
5+
import config from '../config'
56

67
let uid = 0
78

@@ -36,6 +37,12 @@ export default class Dep {
3637
notify () {
3738
// stabilize the subscriber list first
3839
const subs = this.subs.slice()
40+
if (process.env.NODE_ENV !== 'production' && !config.async) {
41+
// subs aren't sorted in scheduler if not running async
42+
// we need to sort them now to make sure they fire in correct
43+
// order
44+
subs.sort((a, b) => a.id - b.id)
45+
}
3946
for (let i = 0, l = subs.length; i < l; i++) {
4047
subs[i].update()
4148
}

src/core/observer/scheduler.js

+5
Original file line numberDiff line numberDiff line change
@@ -145,6 +145,11 @@ export function queueWatcher (watcher: Watcher) {
145145
// queue the flush
146146
if (!waiting) {
147147
waiting = true
148+
149+
if (process.env.NODE_ENV !== 'production' && !config.async) {
150+
flushSchedulerQueue()
151+
return
152+
}
148153
nextTick(flushSchedulerQueue)
149154
}
150155
}

test/unit/features/global-api/config.spec.js

+67
Original file line numberDiff line numberDiff line change
@@ -55,4 +55,71 @@ describe('Global config', () => {
5555
Vue.config.ignoredElements = []
5656
})
5757
})
58+
59+
describe('async', () => {
60+
it('does not update synchronously when true', () => {
61+
const spy = jasmine.createSpy()
62+
const vm = new Vue({
63+
template: `<div :class="value"></div>`,
64+
updated: spy,
65+
data: { value: true }
66+
}).$mount()
67+
vm.value = false
68+
expect(spy).not.toHaveBeenCalled()
69+
})
70+
71+
it('updates synchronously when false', () => {
72+
const spy = jasmine.createSpy()
73+
Vue.config.async = false
74+
const vm = new Vue({
75+
template: `<div :class="value"></div>`,
76+
updated: spy,
77+
data: { value: true }
78+
}).$mount()
79+
vm.value = false
80+
expect(spy).toHaveBeenCalled()
81+
Vue.config.async = true
82+
})
83+
84+
it('runs watchers in correct order when false', () => {
85+
Vue.config.async = false
86+
const vm = new Vue({
87+
template: `
88+
<div id="app">
89+
{{ computed }}
90+
</div>`,
91+
props: ['prop'],
92+
propsData: {
93+
'prop': []
94+
},
95+
data: () => ({
96+
data: ''
97+
}),
98+
computed: {
99+
computed () {
100+
return this.prop.join(',')
101+
}
102+
},
103+
watch: {
104+
prop: 'execute'
105+
},
106+
methods: {
107+
execute () {
108+
this.data = this.computed
109+
}
110+
}
111+
}).$mount()
112+
expect(vm.computed).toBe('')
113+
expect(vm.data).toBe('')
114+
115+
vm.prop = [1, 2, 3]
116+
expect(vm.computed).toBe('1,2,3')
117+
expect(vm.data).toBe('1,2,3')
118+
119+
vm.prop.push(4, 5)
120+
expect(vm.computed).toBe('1,2,3,4,5')
121+
expect(vm.data).toBe('1,2,3,4,5')
122+
Vue.config.async = true
123+
})
124+
})
58125
})

types/test/vue-test.ts

+1
Original file line numberDiff line numberDiff line change
@@ -73,6 +73,7 @@ class Test extends Vue {
7373
};
7474
config.keyCodes = { esc: 27 };
7575
config.ignoredElements = ['foo', /^ion-/];
76+
config.async = false
7677
}
7778

7879
static testMethods() {

types/vue.d.ts

+1
Original file line numberDiff line numberDiff line change
@@ -74,6 +74,7 @@ export interface VueConfiguration {
7474
warnHandler(msg: string, vm: Vue, trace: string): void;
7575
ignoredElements: (string | RegExp)[];
7676
keyCodes: { [key: string]: number | number[] };
77+
async: boolean;
7778
}
7879

7980
export interface VueConstructor<V extends Vue = Vue> {

0 commit comments

Comments
 (0)