mirror of https://github.com/vuejs/core.git
				
				
				
			
		
			
				
	
	
		
			1330 lines
		
	
	
		
			33 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
			
		
		
	
	
			1330 lines
		
	
	
		
			33 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
| import {
 | |
|   type DebuggerEvent,
 | |
|   type ReactiveEffectRunner,
 | |
|   TrackOpTypes,
 | |
|   TriggerOpTypes,
 | |
|   effect,
 | |
|   markRaw,
 | |
|   reactive,
 | |
|   readonly,
 | |
|   shallowReactive,
 | |
|   stop,
 | |
|   toRaw,
 | |
| } from '../src/index'
 | |
| import { type Dep, ITERATE_KEY, getDepFromReactive } from '../src/dep'
 | |
| import {
 | |
|   computed,
 | |
|   h,
 | |
|   nextTick,
 | |
|   nodeOps,
 | |
|   ref,
 | |
|   render,
 | |
|   serializeInner,
 | |
| } from '@vue/runtime-test'
 | |
| import {
 | |
|   endBatch,
 | |
|   onEffectCleanup,
 | |
|   pauseTracking,
 | |
|   resetTracking,
 | |
|   startBatch,
 | |
| } from '../src/effect'
 | |
| 
 | |
| describe('reactivity/effect', () => {
 | |
|   it('should run the passed function once (wrapped by a effect)', () => {
 | |
|     const fnSpy = vi.fn(() => {})
 | |
|     effect(fnSpy)
 | |
|     expect(fnSpy).toHaveBeenCalledTimes(1)
 | |
|   })
 | |
| 
 | |
|   it('should observe basic properties', () => {
 | |
|     let dummy
 | |
|     const counter = reactive({ num: 0 })
 | |
|     effect(() => (dummy = counter.num))
 | |
| 
 | |
|     expect(dummy).toBe(0)
 | |
|     counter.num = 7
 | |
|     expect(dummy).toBe(7)
 | |
|   })
 | |
| 
 | |
|   it('should observe multiple properties', () => {
 | |
|     let dummy
 | |
|     const counter = reactive({ num1: 0, num2: 0 })
 | |
|     effect(() => (dummy = counter.num1 + counter.num1 + counter.num2))
 | |
| 
 | |
|     expect(dummy).toBe(0)
 | |
|     counter.num1 = counter.num2 = 7
 | |
|     expect(dummy).toBe(21)
 | |
|   })
 | |
| 
 | |
|   it('should handle multiple effects', () => {
 | |
|     let dummy1, dummy2
 | |
|     const counter = reactive({ num: 0 })
 | |
|     effect(() => (dummy1 = counter.num))
 | |
|     effect(() => (dummy2 = counter.num))
 | |
| 
 | |
|     expect(dummy1).toBe(0)
 | |
|     expect(dummy2).toBe(0)
 | |
|     counter.num++
 | |
|     expect(dummy1).toBe(1)
 | |
|     expect(dummy2).toBe(1)
 | |
|   })
 | |
| 
 | |
|   it('should observe nested properties', () => {
 | |
|     let dummy
 | |
|     const counter = reactive({ nested: { num: 0 } })
 | |
|     effect(() => (dummy = counter.nested.num))
 | |
| 
 | |
|     expect(dummy).toBe(0)
 | |
|     counter.nested.num = 8
 | |
|     expect(dummy).toBe(8)
 | |
|   })
 | |
| 
 | |
|   it('should observe delete operations', () => {
 | |
|     let dummy
 | |
|     const obj = reactive<{
 | |
|       prop?: string
 | |
|     }>({ prop: 'value' })
 | |
|     effect(() => (dummy = obj.prop))
 | |
| 
 | |
|     expect(dummy).toBe('value')
 | |
|     delete obj.prop
 | |
|     expect(dummy).toBe(undefined)
 | |
|   })
 | |
| 
 | |
|   it('should observe has operations', () => {
 | |
|     let dummy
 | |
|     const obj = reactive<{ prop?: string | number }>({ prop: 'value' })
 | |
|     effect(() => (dummy = 'prop' in obj))
 | |
| 
 | |
|     expect(dummy).toBe(true)
 | |
|     delete obj.prop
 | |
|     expect(dummy).toBe(false)
 | |
|     obj.prop = 12
 | |
|     expect(dummy).toBe(true)
 | |
|   })
 | |
| 
 | |
|   it('should observe properties on the prototype chain', () => {
 | |
|     let dummy
 | |
|     const counter = reactive<{ num?: number }>({ num: 0 })
 | |
|     const parentCounter = reactive({ num: 2 })
 | |
|     Object.setPrototypeOf(counter, parentCounter)
 | |
|     effect(() => (dummy = counter.num))
 | |
| 
 | |
|     expect(dummy).toBe(0)
 | |
|     delete counter.num
 | |
|     expect(dummy).toBe(2)
 | |
|     parentCounter.num = 4
 | |
|     expect(dummy).toBe(4)
 | |
|     counter.num = 3
 | |
|     expect(dummy).toBe(3)
 | |
|   })
 | |
| 
 | |
|   it('should observe has operations on the prototype chain', () => {
 | |
|     let dummy
 | |
|     const counter = reactive<{ num?: number }>({ num: 0 })
 | |
|     const parentCounter = reactive<{ num?: number }>({ num: 2 })
 | |
|     Object.setPrototypeOf(counter, parentCounter)
 | |
|     effect(() => (dummy = 'num' in counter))
 | |
| 
 | |
|     expect(dummy).toBe(true)
 | |
|     delete counter.num
 | |
|     expect(dummy).toBe(true)
 | |
|     delete parentCounter.num
 | |
|     expect(dummy).toBe(false)
 | |
|     counter.num = 3
 | |
|     expect(dummy).toBe(true)
 | |
|   })
 | |
| 
 | |
|   it('should observe inherited property accessors', () => {
 | |
|     let dummy, parentDummy, hiddenValue: any
 | |
|     const obj = reactive<{ prop?: number }>({})
 | |
|     const parent = reactive({
 | |
|       set prop(value) {
 | |
|         hiddenValue = value
 | |
|       },
 | |
|       get prop() {
 | |
|         return hiddenValue
 | |
|       },
 | |
|     })
 | |
|     Object.setPrototypeOf(obj, parent)
 | |
|     effect(() => (dummy = obj.prop))
 | |
|     effect(() => (parentDummy = parent.prop))
 | |
| 
 | |
|     expect(dummy).toBe(undefined)
 | |
|     expect(parentDummy).toBe(undefined)
 | |
|     obj.prop = 4
 | |
|     expect(dummy).toBe(4)
 | |
|     // this doesn't work, should it?
 | |
|     // expect(parentDummy).toBe(4)
 | |
|     parent.prop = 2
 | |
|     expect(dummy).toBe(2)
 | |
|     expect(parentDummy).toBe(2)
 | |
|   })
 | |
| 
 | |
|   it('should observe function call chains', () => {
 | |
|     let dummy
 | |
|     const counter = reactive({ num: 0 })
 | |
|     effect(() => (dummy = getNum()))
 | |
| 
 | |
|     function getNum() {
 | |
|       return counter.num
 | |
|     }
 | |
| 
 | |
|     expect(dummy).toBe(0)
 | |
|     counter.num = 2
 | |
|     expect(dummy).toBe(2)
 | |
|   })
 | |
| 
 | |
|   it('should observe iteration', () => {
 | |
|     let dummy
 | |
|     const list = reactive(['Hello'])
 | |
|     effect(() => (dummy = list.join(' ')))
 | |
| 
 | |
|     expect(dummy).toBe('Hello')
 | |
|     list.push('World!')
 | |
|     expect(dummy).toBe('Hello World!')
 | |
|     list.shift()
 | |
|     expect(dummy).toBe('World!')
 | |
|   })
 | |
| 
 | |
|   it('should observe implicit array length changes', () => {
 | |
|     let dummy
 | |
|     const list = reactive(['Hello'])
 | |
|     effect(() => (dummy = list.join(' ')))
 | |
| 
 | |
|     expect(dummy).toBe('Hello')
 | |
|     list[1] = 'World!'
 | |
|     expect(dummy).toBe('Hello World!')
 | |
|     list[3] = 'Hello!'
 | |
|     expect(dummy).toBe('Hello World!  Hello!')
 | |
|   })
 | |
| 
 | |
|   it('should observe sparse array mutations', () => {
 | |
|     let dummy
 | |
|     const list = reactive<string[]>([])
 | |
|     list[1] = 'World!'
 | |
|     effect(() => (dummy = list.join(' ')))
 | |
| 
 | |
|     expect(dummy).toBe(' World!')
 | |
|     list[0] = 'Hello'
 | |
|     expect(dummy).toBe('Hello World!')
 | |
|     list.pop()
 | |
|     expect(dummy).toBe('Hello')
 | |
|   })
 | |
| 
 | |
|   it('should observe enumeration', () => {
 | |
|     let dummy = 0
 | |
|     const numbers = reactive<Record<string, number>>({ num1: 3 })
 | |
|     effect(() => {
 | |
|       dummy = 0
 | |
|       for (let key in numbers) {
 | |
|         dummy += numbers[key]
 | |
|       }
 | |
|     })
 | |
| 
 | |
|     expect(dummy).toBe(3)
 | |
|     numbers.num2 = 4
 | |
|     expect(dummy).toBe(7)
 | |
|     delete numbers.num1
 | |
|     expect(dummy).toBe(4)
 | |
|   })
 | |
| 
 | |
|   it('should observe symbol keyed properties', () => {
 | |
|     const key = Symbol('symbol keyed prop')
 | |
|     let dummy, hasDummy
 | |
|     const obj = reactive<{ [key]?: string }>({ [key]: 'value' })
 | |
|     effect(() => (dummy = obj[key]))
 | |
|     effect(() => (hasDummy = key in obj))
 | |
| 
 | |
|     expect(dummy).toBe('value')
 | |
|     expect(hasDummy).toBe(true)
 | |
|     obj[key] = 'newValue'
 | |
|     expect(dummy).toBe('newValue')
 | |
|     delete obj[key]
 | |
|     expect(dummy).toBe(undefined)
 | |
|     expect(hasDummy).toBe(false)
 | |
|   })
 | |
| 
 | |
|   it('should not observe well-known symbol keyed properties', () => {
 | |
|     const key = Symbol.isConcatSpreadable
 | |
|     let dummy
 | |
|     const array: any = reactive([])
 | |
|     effect(() => (dummy = array[key]))
 | |
| 
 | |
|     expect(array[key]).toBe(undefined)
 | |
|     expect(dummy).toBe(undefined)
 | |
|     array[key] = true
 | |
|     expect(array[key]).toBe(true)
 | |
|     expect(dummy).toBe(undefined)
 | |
|   })
 | |
| 
 | |
|   it('should not observe well-known symbol keyed properties in has operation', () => {
 | |
|     const key = Symbol.isConcatSpreadable
 | |
|     const obj = reactive({
 | |
|       [key]: true,
 | |
|     }) as any
 | |
| 
 | |
|     const spy = vi.fn(() => {
 | |
|       key in obj
 | |
|     })
 | |
|     effect(spy)
 | |
|     expect(spy).toHaveBeenCalledTimes(1)
 | |
| 
 | |
|     obj[key] = false
 | |
|     expect(spy).toHaveBeenCalledTimes(1)
 | |
|   })
 | |
| 
 | |
|   it('should support manipulating an array while observing symbol keyed properties', () => {
 | |
|     const key = Symbol()
 | |
|     let dummy
 | |
|     const array: any = reactive([1, 2, 3])
 | |
|     effect(() => (dummy = array[key]))
 | |
| 
 | |
|     expect(dummy).toBe(undefined)
 | |
|     array.pop()
 | |
|     array.shift()
 | |
|     array.splice(0, 1)
 | |
|     expect(dummy).toBe(undefined)
 | |
|     array[key] = 'value'
 | |
|     array.length = 0
 | |
|     expect(dummy).toBe('value')
 | |
|   })
 | |
| 
 | |
|   it('should observe function valued properties', () => {
 | |
|     const oldFunc = () => {}
 | |
|     const newFunc = () => {}
 | |
| 
 | |
|     let dummy
 | |
|     const obj = reactive({ func: oldFunc })
 | |
|     effect(() => (dummy = obj.func))
 | |
| 
 | |
|     expect(dummy).toBe(oldFunc)
 | |
|     obj.func = newFunc
 | |
|     expect(dummy).toBe(newFunc)
 | |
|   })
 | |
| 
 | |
|   it('should observe chained getters relying on this', () => {
 | |
|     const obj = reactive({
 | |
|       a: 1,
 | |
|       get b() {
 | |
|         return this.a
 | |
|       },
 | |
|     })
 | |
| 
 | |
|     let dummy
 | |
|     effect(() => (dummy = obj.b))
 | |
|     expect(dummy).toBe(1)
 | |
|     obj.a++
 | |
|     expect(dummy).toBe(2)
 | |
|   })
 | |
| 
 | |
|   it('should observe methods relying on this', () => {
 | |
|     const obj = reactive({
 | |
|       a: 1,
 | |
|       b() {
 | |
|         return this.a
 | |
|       },
 | |
|     })
 | |
| 
 | |
|     let dummy
 | |
|     effect(() => (dummy = obj.b()))
 | |
|     expect(dummy).toBe(1)
 | |
|     obj.a++
 | |
|     expect(dummy).toBe(2)
 | |
|   })
 | |
| 
 | |
|   it('should not observe set operations without a value change', () => {
 | |
|     let hasDummy, getDummy
 | |
|     const obj = reactive({ prop: 'value' })
 | |
| 
 | |
|     const getSpy = vi.fn(() => (getDummy = obj.prop))
 | |
|     const hasSpy = vi.fn(() => (hasDummy = 'prop' in obj))
 | |
|     effect(getSpy)
 | |
|     effect(hasSpy)
 | |
| 
 | |
|     expect(getDummy).toBe('value')
 | |
|     expect(hasDummy).toBe(true)
 | |
|     obj.prop = 'value'
 | |
|     expect(getSpy).toHaveBeenCalledTimes(1)
 | |
|     expect(hasSpy).toHaveBeenCalledTimes(1)
 | |
|     expect(getDummy).toBe('value')
 | |
|     expect(hasDummy).toBe(true)
 | |
|   })
 | |
| 
 | |
|   it('should not observe raw mutations', () => {
 | |
|     let dummy
 | |
|     const obj = reactive<{ prop?: string }>({})
 | |
|     effect(() => (dummy = toRaw(obj).prop))
 | |
| 
 | |
|     expect(dummy).toBe(undefined)
 | |
|     obj.prop = 'value'
 | |
|     expect(dummy).toBe(undefined)
 | |
|   })
 | |
| 
 | |
|   it('should not be triggered by raw mutations', () => {
 | |
|     let dummy
 | |
|     const obj = reactive<{ prop?: string }>({})
 | |
|     effect(() => (dummy = obj.prop))
 | |
| 
 | |
|     expect(dummy).toBe(undefined)
 | |
|     toRaw(obj).prop = 'value'
 | |
|     expect(dummy).toBe(undefined)
 | |
|   })
 | |
| 
 | |
|   it('should not be triggered by inherited raw setters', () => {
 | |
|     let dummy, parentDummy, hiddenValue: any
 | |
|     const obj = reactive<{ prop?: number }>({})
 | |
|     const parent = reactive({
 | |
|       set prop(value) {
 | |
|         hiddenValue = value
 | |
|       },
 | |
|       get prop() {
 | |
|         return hiddenValue
 | |
|       },
 | |
|     })
 | |
|     Object.setPrototypeOf(obj, parent)
 | |
|     effect(() => (dummy = obj.prop))
 | |
|     effect(() => (parentDummy = parent.prop))
 | |
| 
 | |
|     expect(dummy).toBe(undefined)
 | |
|     expect(parentDummy).toBe(undefined)
 | |
|     toRaw(obj).prop = 4
 | |
|     expect(dummy).toBe(undefined)
 | |
|     expect(parentDummy).toBe(undefined)
 | |
|   })
 | |
| 
 | |
|   it('should avoid implicit infinite recursive loops with itself', () => {
 | |
|     const counter = reactive({ num: 0 })
 | |
| 
 | |
|     const counterSpy = vi.fn(() => counter.num++)
 | |
|     effect(counterSpy)
 | |
|     expect(counter.num).toBe(1)
 | |
|     expect(counterSpy).toHaveBeenCalledTimes(1)
 | |
|     counter.num = 4
 | |
|     expect(counter.num).toBe(5)
 | |
|     expect(counterSpy).toHaveBeenCalledTimes(2)
 | |
|   })
 | |
| 
 | |
|   it('should avoid infinite recursive loops when use Array.prototype.push/unshift/pop/shift', () => {
 | |
|     ;(['push', 'unshift'] as const).forEach(key => {
 | |
|       const arr = reactive<number[]>([])
 | |
|       const counterSpy1 = vi.fn(() => (arr[key] as any)(1))
 | |
|       const counterSpy2 = vi.fn(() => (arr[key] as any)(2))
 | |
|       effect(counterSpy1)
 | |
|       effect(counterSpy2)
 | |
|       expect(arr.length).toBe(2)
 | |
|       expect(counterSpy1).toHaveBeenCalledTimes(1)
 | |
|       expect(counterSpy2).toHaveBeenCalledTimes(1)
 | |
|     })
 | |
|     ;(['pop', 'shift'] as const).forEach(key => {
 | |
|       const arr = reactive<number[]>([1, 2, 3, 4])
 | |
|       const counterSpy1 = vi.fn(() => (arr[key] as any)())
 | |
|       const counterSpy2 = vi.fn(() => (arr[key] as any)())
 | |
|       effect(counterSpy1)
 | |
|       effect(counterSpy2)
 | |
|       expect(arr.length).toBe(2)
 | |
|       expect(counterSpy1).toHaveBeenCalledTimes(1)
 | |
|       expect(counterSpy2).toHaveBeenCalledTimes(1)
 | |
|     })
 | |
|   })
 | |
| 
 | |
|   it('should allow explicitly recursive raw function loops', () => {
 | |
|     const counter = reactive({ num: 0 })
 | |
|     const numSpy = vi.fn(() => {
 | |
|       counter.num++
 | |
|       if (counter.num < 10) {
 | |
|         numSpy()
 | |
|       }
 | |
|     })
 | |
|     effect(numSpy)
 | |
|     expect(counter.num).toEqual(10)
 | |
|     expect(numSpy).toHaveBeenCalledTimes(10)
 | |
|   })
 | |
| 
 | |
|   it('should avoid infinite loops with other effects', () => {
 | |
|     const nums = reactive({ num1: 0, num2: 1 })
 | |
| 
 | |
|     const spy1 = vi.fn(() => (nums.num1 = nums.num2))
 | |
|     const spy2 = vi.fn(() => (nums.num2 = nums.num1))
 | |
|     effect(spy1)
 | |
|     effect(spy2)
 | |
|     expect(nums.num1).toBe(1)
 | |
|     expect(nums.num2).toBe(1)
 | |
|     expect(spy1).toHaveBeenCalledTimes(1)
 | |
|     expect(spy2).toHaveBeenCalledTimes(1)
 | |
|     nums.num2 = 4
 | |
|     expect(nums.num1).toBe(4)
 | |
|     expect(nums.num2).toBe(4)
 | |
|     expect(spy1).toHaveBeenCalledTimes(2)
 | |
|     expect(spy2).toHaveBeenCalledTimes(2)
 | |
|     nums.num1 = 10
 | |
|     expect(nums.num1).toBe(10)
 | |
|     expect(nums.num2).toBe(10)
 | |
|     expect(spy1).toHaveBeenCalledTimes(3)
 | |
|     expect(spy2).toHaveBeenCalledTimes(3)
 | |
|   })
 | |
| 
 | |
|   it('should return a new reactive version of the function', () => {
 | |
|     function greet() {
 | |
|       return 'Hello World'
 | |
|     }
 | |
|     const effect1 = effect(greet)
 | |
|     const effect2 = effect(greet)
 | |
|     expect(typeof effect1).toBe('function')
 | |
|     expect(typeof effect2).toBe('function')
 | |
|     expect(effect1).not.toBe(greet)
 | |
|     expect(effect1).not.toBe(effect2)
 | |
|   })
 | |
| 
 | |
|   it('should discover new branches while running automatically', () => {
 | |
|     let dummy
 | |
|     const obj = reactive({ prop: 'value', run: false })
 | |
| 
 | |
|     const conditionalSpy = vi.fn(() => {
 | |
|       dummy = obj.run ? obj.prop : 'other'
 | |
|     })
 | |
|     effect(conditionalSpy)
 | |
| 
 | |
|     expect(dummy).toBe('other')
 | |
|     expect(conditionalSpy).toHaveBeenCalledTimes(1)
 | |
|     obj.prop = 'Hi'
 | |
|     expect(dummy).toBe('other')
 | |
|     expect(conditionalSpy).toHaveBeenCalledTimes(1)
 | |
|     obj.run = true
 | |
|     expect(dummy).toBe('Hi')
 | |
|     expect(conditionalSpy).toHaveBeenCalledTimes(2)
 | |
|     obj.prop = 'World'
 | |
|     expect(dummy).toBe('World')
 | |
|     expect(conditionalSpy).toHaveBeenCalledTimes(3)
 | |
|   })
 | |
| 
 | |
|   it('should discover new branches when running manually', () => {
 | |
|     let dummy
 | |
|     let run = false
 | |
|     const obj = reactive({ prop: 'value' })
 | |
|     const runner = effect(() => {
 | |
|       dummy = run ? obj.prop : 'other'
 | |
|     })
 | |
| 
 | |
|     expect(dummy).toBe('other')
 | |
|     runner()
 | |
|     expect(dummy).toBe('other')
 | |
|     run = true
 | |
|     runner()
 | |
|     expect(dummy).toBe('value')
 | |
|     obj.prop = 'World'
 | |
|     expect(dummy).toBe('World')
 | |
|   })
 | |
| 
 | |
|   it('should not be triggered by mutating a property, which is used in an inactive branch', () => {
 | |
|     let dummy
 | |
|     const obj = reactive({ prop: 'value', run: true })
 | |
| 
 | |
|     const conditionalSpy = vi.fn(() => {
 | |
|       dummy = obj.run ? obj.prop : 'other'
 | |
|     })
 | |
|     effect(conditionalSpy)
 | |
| 
 | |
|     expect(dummy).toBe('value')
 | |
|     expect(conditionalSpy).toHaveBeenCalledTimes(1)
 | |
|     obj.run = false
 | |
|     expect(dummy).toBe('other')
 | |
|     expect(conditionalSpy).toHaveBeenCalledTimes(2)
 | |
|     obj.prop = 'value2'
 | |
|     expect(dummy).toBe('other')
 | |
|     expect(conditionalSpy).toHaveBeenCalledTimes(2)
 | |
|   })
 | |
| 
 | |
|   it('should handle deep effect recursion using cleanup fallback', () => {
 | |
|     const results = reactive([0])
 | |
|     const effects: { fx: ReactiveEffectRunner; index: number }[] = []
 | |
|     for (let i = 1; i < 40; i++) {
 | |
|       ;(index => {
 | |
|         const fx = effect(() => {
 | |
|           results[index] = results[index - 1] * 2
 | |
|         })
 | |
|         effects.push({ fx, index })
 | |
|       })(i)
 | |
|     }
 | |
| 
 | |
|     expect(results[39]).toBe(0)
 | |
|     results[0] = 1
 | |
|     expect(results[39]).toBe(Math.pow(2, 39))
 | |
|   })
 | |
| 
 | |
|   it('should register deps independently during effect recursion', () => {
 | |
|     const input = reactive({ a: 1, b: 2, c: 0 })
 | |
|     const output = reactive({ fx1: 0, fx2: 0 })
 | |
| 
 | |
|     const fx1Spy = vi.fn(() => {
 | |
|       let result = 0
 | |
|       if (input.c < 2) result += input.a
 | |
|       if (input.c > 1) result += input.b
 | |
|       output.fx1 = result
 | |
|     })
 | |
| 
 | |
|     const fx1 = effect(fx1Spy)
 | |
| 
 | |
|     const fx2Spy = vi.fn(() => {
 | |
|       let result = 0
 | |
|       if (input.c > 1) result += input.a
 | |
|       if (input.c < 3) result += input.b
 | |
|       output.fx2 = result + output.fx1
 | |
|     })
 | |
| 
 | |
|     const fx2 = effect(fx2Spy)
 | |
| 
 | |
|     expect(fx1).not.toBeNull()
 | |
|     expect(fx2).not.toBeNull()
 | |
| 
 | |
|     expect(output.fx1).toBe(1)
 | |
|     expect(output.fx2).toBe(2 + 1)
 | |
|     expect(fx1Spy).toHaveBeenCalledTimes(1)
 | |
|     expect(fx2Spy).toHaveBeenCalledTimes(1)
 | |
| 
 | |
|     fx1Spy.mockClear()
 | |
|     fx2Spy.mockClear()
 | |
|     input.b = 3
 | |
|     expect(output.fx1).toBe(1)
 | |
|     expect(output.fx2).toBe(3 + 1)
 | |
|     expect(fx1Spy).toHaveBeenCalledTimes(0)
 | |
|     expect(fx2Spy).toHaveBeenCalledTimes(1)
 | |
| 
 | |
|     fx1Spy.mockClear()
 | |
|     fx2Spy.mockClear()
 | |
|     input.c = 1
 | |
|     expect(output.fx1).toBe(1)
 | |
|     expect(output.fx2).toBe(3 + 1)
 | |
|     expect(fx1Spy).toHaveBeenCalledTimes(1)
 | |
|     expect(fx2Spy).toHaveBeenCalledTimes(1)
 | |
| 
 | |
|     fx1Spy.mockClear()
 | |
|     fx2Spy.mockClear()
 | |
|     input.c = 2
 | |
|     expect(output.fx1).toBe(3)
 | |
|     expect(output.fx2).toBe(1 + 3 + 3)
 | |
|     expect(fx1Spy).toHaveBeenCalledTimes(1)
 | |
| 
 | |
|     // Invoked due to change of fx1.
 | |
|     expect(fx2Spy).toHaveBeenCalledTimes(1)
 | |
| 
 | |
|     fx1Spy.mockClear()
 | |
|     fx2Spy.mockClear()
 | |
|     input.c = 3
 | |
|     expect(output.fx1).toBe(3)
 | |
|     expect(output.fx2).toBe(1 + 3)
 | |
|     expect(fx1Spy).toHaveBeenCalledTimes(1)
 | |
|     expect(fx2Spy).toHaveBeenCalledTimes(1)
 | |
| 
 | |
|     fx1Spy.mockClear()
 | |
|     fx2Spy.mockClear()
 | |
|     input.a = 10
 | |
|     expect(output.fx1).toBe(3)
 | |
|     expect(output.fx2).toBe(10 + 3)
 | |
|     expect(fx1Spy).toHaveBeenCalledTimes(0)
 | |
|     expect(fx2Spy).toHaveBeenCalledTimes(1)
 | |
|   })
 | |
| 
 | |
|   it('should not double wrap if the passed function is a effect', () => {
 | |
|     const runner = effect(() => {})
 | |
|     const otherRunner = effect(runner)
 | |
|     expect(runner).not.toBe(otherRunner)
 | |
|     expect(runner.effect.fn).toBe(otherRunner.effect.fn)
 | |
|   })
 | |
| 
 | |
|   it('should wrap if the passed function is a fake effect', () => {
 | |
|     const fakeRunner = () => {}
 | |
|     fakeRunner.effect = {}
 | |
|     const runner = effect(fakeRunner)
 | |
|     expect(fakeRunner).not.toBe(runner)
 | |
|     expect(runner.effect.fn).toBe(fakeRunner)
 | |
|   })
 | |
| 
 | |
|   it('should not run multiple times for a single mutation', () => {
 | |
|     let dummy
 | |
|     const obj = reactive<Record<string, number>>({})
 | |
|     const fnSpy = vi.fn(() => {
 | |
|       for (const key in obj) {
 | |
|         dummy = obj[key]
 | |
|       }
 | |
|       dummy = obj.prop
 | |
|     })
 | |
|     effect(fnSpy)
 | |
| 
 | |
|     expect(fnSpy).toHaveBeenCalledTimes(1)
 | |
|     obj.prop = 16
 | |
|     expect(dummy).toBe(16)
 | |
|     expect(fnSpy).toHaveBeenCalledTimes(2)
 | |
|   })
 | |
| 
 | |
|   it('should allow nested effects', () => {
 | |
|     const nums = reactive({ num1: 0, num2: 1, num3: 2 })
 | |
|     const dummy: any = {}
 | |
| 
 | |
|     const childSpy = vi.fn(() => (dummy.num1 = nums.num1))
 | |
|     const childeffect = effect(childSpy)
 | |
|     const parentSpy = vi.fn(() => {
 | |
|       dummy.num2 = nums.num2
 | |
|       childeffect()
 | |
|       dummy.num3 = nums.num3
 | |
|     })
 | |
|     effect(parentSpy)
 | |
| 
 | |
|     expect(dummy).toEqual({ num1: 0, num2: 1, num3: 2 })
 | |
|     expect(parentSpy).toHaveBeenCalledTimes(1)
 | |
|     expect(childSpy).toHaveBeenCalledTimes(2)
 | |
|     // this should only call the childeffect
 | |
|     nums.num1 = 4
 | |
|     expect(dummy).toEqual({ num1: 4, num2: 1, num3: 2 })
 | |
|     expect(parentSpy).toHaveBeenCalledTimes(1)
 | |
|     expect(childSpy).toHaveBeenCalledTimes(3)
 | |
|     // this calls the parenteffect, which calls the childeffect once
 | |
|     nums.num2 = 10
 | |
|     expect(dummy).toEqual({ num1: 4, num2: 10, num3: 2 })
 | |
|     expect(parentSpy).toHaveBeenCalledTimes(2)
 | |
|     expect(childSpy).toHaveBeenCalledTimes(4)
 | |
|     // this calls the parenteffect, which calls the childeffect once
 | |
|     nums.num3 = 7
 | |
|     expect(dummy).toEqual({ num1: 4, num2: 10, num3: 7 })
 | |
|     expect(parentSpy).toHaveBeenCalledTimes(3)
 | |
|     expect(childSpy).toHaveBeenCalledTimes(5)
 | |
|   })
 | |
| 
 | |
|   it('should observe json methods', () => {
 | |
|     let dummy = <Record<string, number>>{}
 | |
|     const obj = reactive<Record<string, number>>({})
 | |
|     effect(() => {
 | |
|       dummy = JSON.parse(JSON.stringify(obj))
 | |
|     })
 | |
|     obj.a = 1
 | |
|     expect(dummy.a).toBe(1)
 | |
|   })
 | |
| 
 | |
|   it('should observe class method invocations', () => {
 | |
|     class Model {
 | |
|       count: number
 | |
|       constructor() {
 | |
|         this.count = 0
 | |
|       }
 | |
|       inc() {
 | |
|         this.count++
 | |
|       }
 | |
|     }
 | |
|     const model = reactive(new Model())
 | |
|     let dummy
 | |
|     effect(() => {
 | |
|       dummy = model.count
 | |
|     })
 | |
|     expect(dummy).toBe(0)
 | |
|     model.inc()
 | |
|     expect(dummy).toBe(1)
 | |
|   })
 | |
| 
 | |
|   it('scheduler', () => {
 | |
|     let dummy
 | |
|     let run: any
 | |
|     const scheduler = vi.fn(() => {
 | |
|       run = runner
 | |
|     })
 | |
|     const obj = reactive({ foo: 1 })
 | |
|     const runner = effect(
 | |
|       () => {
 | |
|         dummy = obj.foo
 | |
|       },
 | |
|       { scheduler },
 | |
|     )
 | |
|     expect(scheduler).not.toHaveBeenCalled()
 | |
|     expect(dummy).toBe(1)
 | |
|     // should be called on first trigger
 | |
|     obj.foo++
 | |
|     expect(scheduler).toHaveBeenCalledTimes(1)
 | |
|     // should not run yet
 | |
|     expect(dummy).toBe(1)
 | |
|     // manually run
 | |
|     run()
 | |
|     // should have run
 | |
|     expect(dummy).toBe(2)
 | |
|   })
 | |
| 
 | |
|   it('events: onTrack', () => {
 | |
|     let events: DebuggerEvent[] = []
 | |
|     let dummy
 | |
|     const onTrack = vi.fn((e: DebuggerEvent) => {
 | |
|       events.push(e)
 | |
|     })
 | |
|     const obj = reactive({ foo: 1, bar: 2 })
 | |
|     const runner = effect(
 | |
|       () => {
 | |
|         dummy = obj.foo
 | |
|         dummy = 'bar' in obj
 | |
|         dummy = Object.keys(obj)
 | |
|       },
 | |
|       { onTrack },
 | |
|     )
 | |
|     expect(dummy).toEqual(['foo', 'bar'])
 | |
|     expect(onTrack).toHaveBeenCalledTimes(3)
 | |
|     expect(events).toEqual([
 | |
|       {
 | |
|         effect: runner.effect,
 | |
|         target: toRaw(obj),
 | |
|         type: TrackOpTypes.GET,
 | |
|         key: 'foo',
 | |
|       },
 | |
|       {
 | |
|         effect: runner.effect,
 | |
|         target: toRaw(obj),
 | |
|         type: TrackOpTypes.HAS,
 | |
|         key: 'bar',
 | |
|       },
 | |
|       {
 | |
|         effect: runner.effect,
 | |
|         target: toRaw(obj),
 | |
|         type: TrackOpTypes.ITERATE,
 | |
|         key: ITERATE_KEY,
 | |
|       },
 | |
|     ])
 | |
|   })
 | |
| 
 | |
|   it('debug: the call sequence of onTrack', () => {
 | |
|     const seq: number[] = []
 | |
|     const s = ref(0)
 | |
| 
 | |
|     const track1 = () => seq.push(1)
 | |
|     const track2 = () => seq.push(2)
 | |
| 
 | |
|     effect(
 | |
|       () => {
 | |
|         s.value
 | |
|       },
 | |
|       {
 | |
|         onTrack: track1,
 | |
|       },
 | |
|     )
 | |
|     effect(
 | |
|       () => {
 | |
|         s.value
 | |
|       },
 | |
|       {
 | |
|         onTrack: track2,
 | |
|       },
 | |
|     )
 | |
|     expect(seq.toString()).toBe('1,2')
 | |
|   })
 | |
| 
 | |
|   it('events: onTrigger', () => {
 | |
|     let events: DebuggerEvent[] = []
 | |
|     let dummy
 | |
|     const onTrigger = vi.fn((e: DebuggerEvent) => {
 | |
|       events.push(e)
 | |
|     })
 | |
|     const obj = reactive<{ foo?: number }>({ foo: 1 })
 | |
|     const runner = effect(
 | |
|       () => {
 | |
|         dummy = obj.foo
 | |
|       },
 | |
|       { onTrigger },
 | |
|     )
 | |
| 
 | |
|     obj.foo!++
 | |
|     expect(dummy).toBe(2)
 | |
|     expect(onTrigger).toHaveBeenCalledTimes(1)
 | |
|     expect(events[0]).toEqual({
 | |
|       effect: runner.effect,
 | |
|       target: toRaw(obj),
 | |
|       type: TriggerOpTypes.SET,
 | |
|       key: 'foo',
 | |
|       oldValue: 1,
 | |
|       newValue: 2,
 | |
|     })
 | |
| 
 | |
|     delete obj.foo
 | |
|     expect(dummy).toBeUndefined()
 | |
|     expect(onTrigger).toHaveBeenCalledTimes(2)
 | |
|     expect(events[1]).toEqual({
 | |
|       effect: runner.effect,
 | |
|       target: toRaw(obj),
 | |
|       type: TriggerOpTypes.DELETE,
 | |
|       key: 'foo',
 | |
|       oldValue: 2,
 | |
|     })
 | |
|   })
 | |
| 
 | |
|   it('debug: the call sequence of onTrigger', () => {
 | |
|     const seq: number[] = []
 | |
|     const s = ref(0)
 | |
| 
 | |
|     const trigger1 = () => seq.push(1)
 | |
|     const trigger2 = () => seq.push(2)
 | |
|     const trigger3 = () => seq.push(3)
 | |
|     const trigger4 = () => seq.push(4)
 | |
| 
 | |
|     effect(
 | |
|       () => {
 | |
|         s.value
 | |
|       },
 | |
|       {
 | |
|         onTrigger: trigger1,
 | |
|       },
 | |
|     )
 | |
|     effect(
 | |
|       () => {
 | |
|         s.value
 | |
|         effect(
 | |
|           () => {
 | |
|             s.value
 | |
|             effect(
 | |
|               () => {
 | |
|                 s.value
 | |
|               },
 | |
|               {
 | |
|                 onTrigger: trigger4,
 | |
|               },
 | |
|             )
 | |
|           },
 | |
|           {
 | |
|             onTrigger: trigger3,
 | |
|           },
 | |
|         )
 | |
|       },
 | |
|       {
 | |
|         onTrigger: trigger2,
 | |
|       },
 | |
|     )
 | |
|     s.value++
 | |
|     expect(seq.toString()).toBe('1,2,3,4')
 | |
|   })
 | |
| 
 | |
|   it('stop', () => {
 | |
|     let dummy
 | |
|     const obj = reactive({ prop: 1 })
 | |
|     const runner = effect(() => {
 | |
|       dummy = obj.prop
 | |
|     })
 | |
|     obj.prop = 2
 | |
|     expect(dummy).toBe(2)
 | |
|     stop(runner)
 | |
|     obj.prop = 3
 | |
|     expect(dummy).toBe(2)
 | |
| 
 | |
|     // stopped effect should still be manually callable
 | |
|     runner()
 | |
|     expect(dummy).toBe(3)
 | |
|   })
 | |
| 
 | |
|   it('stop with multiple dependencies', () => {
 | |
|     let dummy1, dummy2
 | |
|     const obj1 = reactive({ prop: 1 })
 | |
|     const obj2 = reactive({ prop: 1 })
 | |
|     const runner = effect(() => {
 | |
|       dummy1 = obj1.prop
 | |
|       dummy2 = obj2.prop
 | |
|     })
 | |
| 
 | |
|     obj1.prop = 2
 | |
|     expect(dummy1).toBe(2)
 | |
| 
 | |
|     obj2.prop = 3
 | |
|     expect(dummy2).toBe(3)
 | |
| 
 | |
|     stop(runner)
 | |
| 
 | |
|     obj1.prop = 4
 | |
|     obj2.prop = 5
 | |
| 
 | |
|     // Check that both dependencies have been cleared
 | |
|     expect(dummy1).toBe(2)
 | |
|     expect(dummy2).toBe(3)
 | |
|   })
 | |
| 
 | |
|   it('events: onStop', () => {
 | |
|     const onStop = vi.fn()
 | |
|     const runner = effect(() => {}, {
 | |
|       onStop,
 | |
|     })
 | |
| 
 | |
|     stop(runner)
 | |
|     expect(onStop).toHaveBeenCalled()
 | |
|   })
 | |
| 
 | |
|   it('stop: a stopped effect is nested in a normal effect', () => {
 | |
|     let dummy
 | |
|     const obj = reactive({ prop: 1 })
 | |
|     const runner = effect(() => {
 | |
|       dummy = obj.prop
 | |
|     })
 | |
|     stop(runner)
 | |
|     obj.prop = 2
 | |
|     expect(dummy).toBe(1)
 | |
| 
 | |
|     // observed value in inner stopped effect
 | |
|     // will track outer effect as an dependency
 | |
|     effect(() => {
 | |
|       runner()
 | |
|     })
 | |
|     expect(dummy).toBe(2)
 | |
| 
 | |
|     // notify outer effect to run
 | |
|     obj.prop = 3
 | |
|     expect(dummy).toBe(3)
 | |
|   })
 | |
| 
 | |
|   it('markRaw', () => {
 | |
|     const obj = reactive({
 | |
|       foo: markRaw({
 | |
|         prop: 0,
 | |
|       }),
 | |
|     })
 | |
|     let dummy
 | |
|     effect(() => {
 | |
|       dummy = obj.foo.prop
 | |
|     })
 | |
|     expect(dummy).toBe(0)
 | |
|     obj.foo.prop++
 | |
|     expect(dummy).toBe(0)
 | |
|     obj.foo = { prop: 1 }
 | |
|     expect(dummy).toBe(1)
 | |
|   })
 | |
| 
 | |
|   it('should not be triggered when the value and the old value both are NaN', () => {
 | |
|     const obj = reactive({
 | |
|       foo: NaN,
 | |
|     })
 | |
|     const fnSpy = vi.fn(() => obj.foo)
 | |
|     effect(fnSpy)
 | |
|     obj.foo = NaN
 | |
|     expect(fnSpy).toHaveBeenCalledTimes(1)
 | |
|   })
 | |
| 
 | |
|   it('should trigger all effects when array length is set to 0', () => {
 | |
|     const observed: any = reactive([1])
 | |
|     let dummy, record
 | |
|     effect(() => {
 | |
|       dummy = observed.length
 | |
|     })
 | |
|     effect(() => {
 | |
|       record = observed[0]
 | |
|     })
 | |
|     expect(dummy).toBe(1)
 | |
|     expect(record).toBe(1)
 | |
| 
 | |
|     observed[1] = 2
 | |
|     expect(observed[1]).toBe(2)
 | |
| 
 | |
|     observed.unshift(3)
 | |
|     expect(dummy).toBe(3)
 | |
|     expect(record).toBe(3)
 | |
| 
 | |
|     observed.length = 0
 | |
|     expect(dummy).toBe(0)
 | |
|     expect(record).toBeUndefined()
 | |
|   })
 | |
| 
 | |
|   it('should not be triggered when set with the same proxy', () => {
 | |
|     const obj = reactive({ foo: 1 })
 | |
|     const observed: any = reactive({ obj })
 | |
|     const fnSpy = vi.fn(() => observed.obj)
 | |
| 
 | |
|     effect(fnSpy)
 | |
| 
 | |
|     expect(fnSpy).toHaveBeenCalledTimes(1)
 | |
|     observed.obj = obj
 | |
|     expect(fnSpy).toHaveBeenCalledTimes(1)
 | |
| 
 | |
|     const obj2 = reactive({ foo: 1 })
 | |
|     const observed2: any = shallowReactive({ obj2 })
 | |
|     const fnSpy2 = vi.fn(() => observed2.obj2)
 | |
| 
 | |
|     effect(fnSpy2)
 | |
| 
 | |
|     expect(fnSpy2).toHaveBeenCalledTimes(1)
 | |
|     observed2.obj2 = obj2
 | |
|     expect(fnSpy2).toHaveBeenCalledTimes(1)
 | |
|   })
 | |
| 
 | |
|   it('should be triggered when set length with string', () => {
 | |
|     let ret1 = 'idle'
 | |
|     let ret2 = 'idle'
 | |
|     const arr1 = reactive(new Array(11).fill(0))
 | |
|     const arr2 = reactive(new Array(11).fill(0))
 | |
|     effect(() => {
 | |
|       ret1 = arr1[10] === undefined ? 'arr[10] is set to empty' : 'idle'
 | |
|     })
 | |
|     effect(() => {
 | |
|       ret2 = arr2[10] === undefined ? 'arr[10] is set to empty' : 'idle'
 | |
|     })
 | |
|     arr1.length = 2
 | |
|     arr2.length = '2' as any
 | |
|     expect(ret1).toBe(ret2)
 | |
|   })
 | |
| 
 | |
|   describe('readonly + reactive for Map', () => {
 | |
|     test('should work with readonly(reactive(Map))', () => {
 | |
|       const m = reactive(new Map())
 | |
|       const roM = readonly(m)
 | |
|       const fnSpy = vi.fn(() => roM.get(1))
 | |
| 
 | |
|       effect(fnSpy)
 | |
|       expect(fnSpy).toHaveBeenCalledTimes(1)
 | |
|       m.set(1, 1)
 | |
|       expect(fnSpy).toHaveBeenCalledTimes(2)
 | |
|     })
 | |
| 
 | |
|     test('should work with observed value as key', () => {
 | |
|       const key = reactive({})
 | |
|       const m = reactive(new Map())
 | |
|       m.set(key, 1)
 | |
|       const roM = readonly(m)
 | |
|       const fnSpy = vi.fn(() => roM.get(key))
 | |
| 
 | |
|       effect(fnSpy)
 | |
|       expect(fnSpy).toHaveBeenCalledTimes(1)
 | |
|       m.set(key, 1)
 | |
|       expect(fnSpy).toHaveBeenCalledTimes(1)
 | |
|       m.set(key, 2)
 | |
|       expect(fnSpy).toHaveBeenCalledTimes(2)
 | |
|     })
 | |
| 
 | |
|     test('should track hasOwnProperty', () => {
 | |
|       const obj: any = reactive({})
 | |
|       let has = false
 | |
|       const fnSpy = vi.fn()
 | |
| 
 | |
|       effect(() => {
 | |
|         fnSpy()
 | |
|         has = obj.hasOwnProperty('foo')
 | |
|       })
 | |
|       expect(fnSpy).toHaveBeenCalledTimes(1)
 | |
|       expect(has).toBe(false)
 | |
| 
 | |
|       obj.foo = 1
 | |
|       expect(fnSpy).toHaveBeenCalledTimes(2)
 | |
|       expect(has).toBe(true)
 | |
| 
 | |
|       delete obj.foo
 | |
|       expect(fnSpy).toHaveBeenCalledTimes(3)
 | |
|       expect(has).toBe(false)
 | |
| 
 | |
|       // should not trigger on unrelated key
 | |
|       obj.bar = 2
 | |
|       expect(fnSpy).toHaveBeenCalledTimes(3)
 | |
|       expect(has).toBe(false)
 | |
|     })
 | |
|   })
 | |
| 
 | |
|   it('should be triggered once with batching', () => {
 | |
|     const counter = reactive({ num: 0 })
 | |
| 
 | |
|     const counterSpy = vi.fn(() => counter.num)
 | |
|     effect(counterSpy)
 | |
| 
 | |
|     counterSpy.mockClear()
 | |
| 
 | |
|     startBatch()
 | |
|     counter.num++
 | |
|     counter.num++
 | |
|     endBatch()
 | |
|     expect(counterSpy).toHaveBeenCalledTimes(1)
 | |
|   })
 | |
| 
 | |
|   // #10082
 | |
|   it('should set dirtyLevel when effect is allowRecurse and is running', async () => {
 | |
|     const s = ref(0)
 | |
|     const n = computed(() => s.value + 1)
 | |
| 
 | |
|     const Child = {
 | |
|       setup() {
 | |
|         s.value++
 | |
|         return () => n.value
 | |
|       },
 | |
|     }
 | |
| 
 | |
|     const renderSpy = vi.fn()
 | |
|     const Parent = {
 | |
|       setup() {
 | |
|         return () => {
 | |
|           renderSpy()
 | |
|           return [n.value, h(Child)]
 | |
|         }
 | |
|       },
 | |
|     }
 | |
| 
 | |
|     const root = nodeOps.createElement('div')
 | |
|     render(h(Parent), root)
 | |
|     await nextTick()
 | |
|     expect(serializeInner(root)).toBe('22')
 | |
|     expect(renderSpy).toHaveBeenCalledTimes(2)
 | |
|   })
 | |
| 
 | |
|   it('nested effect should force track in untracked zone', () => {
 | |
|     const n = ref(0)
 | |
|     const spy1 = vi.fn()
 | |
|     const spy2 = vi.fn()
 | |
| 
 | |
|     effect(() => {
 | |
|       spy1()
 | |
|       pauseTracking()
 | |
|       n.value
 | |
|       effect(() => {
 | |
|         n.value
 | |
|         spy2()
 | |
|       })
 | |
|       n.value
 | |
|       resetTracking()
 | |
|     })
 | |
| 
 | |
|     expect(spy1).toHaveBeenCalledTimes(1)
 | |
|     expect(spy2).toHaveBeenCalledTimes(1)
 | |
| 
 | |
|     n.value++
 | |
|     // outer effect should not trigger
 | |
|     expect(spy1).toHaveBeenCalledTimes(1)
 | |
|     // inner effect should trigger
 | |
|     expect(spy2).toHaveBeenCalledTimes(2)
 | |
|   })
 | |
| 
 | |
|   describe('dep unsubscribe', () => {
 | |
|     function getSubCount(dep: Dep | undefined) {
 | |
|       let count = 0
 | |
|       let sub = dep!.subs
 | |
|       while (sub) {
 | |
|         count++
 | |
|         sub = sub.prevSub
 | |
|       }
 | |
|       return count
 | |
|     }
 | |
| 
 | |
|     it('should remove the dep when the effect is stopped', () => {
 | |
|       const obj = reactive({ prop: 1 })
 | |
|       const runner = effect(() => obj.prop)
 | |
|       const dep = getDepFromReactive(toRaw(obj), 'prop')
 | |
|       expect(getSubCount(dep)).toBe(1)
 | |
|       obj.prop = 2
 | |
|       expect(getSubCount(dep)).toBe(1)
 | |
|       stop(runner)
 | |
|       expect(getSubCount(dep)).toBe(0)
 | |
|       obj.prop = 3
 | |
|       runner()
 | |
|       expect(getSubCount(dep)).toBe(0)
 | |
|     })
 | |
| 
 | |
|     it('should only remove the dep when the last effect is stopped', () => {
 | |
|       const obj = reactive({ prop: 1 })
 | |
|       const runner1 = effect(() => obj.prop)
 | |
|       const dep = getDepFromReactive(toRaw(obj), 'prop')
 | |
|       expect(getSubCount(dep)).toBe(1)
 | |
|       const runner2 = effect(() => obj.prop)
 | |
|       expect(getSubCount(dep)).toBe(2)
 | |
|       obj.prop = 2
 | |
|       expect(getSubCount(dep)).toBe(2)
 | |
|       stop(runner1)
 | |
|       expect(getSubCount(dep)).toBe(1)
 | |
|       obj.prop = 3
 | |
|       expect(getSubCount(dep)).toBe(1)
 | |
|       stop(runner2)
 | |
|       obj.prop = 4
 | |
|       runner1()
 | |
|       runner2()
 | |
|       expect(getSubCount(dep)).toBe(0)
 | |
|     })
 | |
| 
 | |
|     it('should remove the dep when it is no longer used by the effect', () => {
 | |
|       const obj = reactive<{ a: number; b: number; c: 'a' | 'b' }>({
 | |
|         a: 1,
 | |
|         b: 2,
 | |
|         c: 'a',
 | |
|       })
 | |
|       effect(() => obj[obj.c])
 | |
|       const depC = getDepFromReactive(toRaw(obj), 'c')
 | |
|       expect(getSubCount(getDepFromReactive(toRaw(obj), 'a'))).toBe(1)
 | |
|       expect(getSubCount(depC)).toBe(1)
 | |
|       obj.c = 'b'
 | |
|       obj.a = 4
 | |
|       expect(getSubCount(getDepFromReactive(toRaw(obj), 'b'))).toBe(1)
 | |
|       expect(getDepFromReactive(toRaw(obj), 'c')).toBe(depC)
 | |
|       expect(getSubCount(depC)).toBe(1)
 | |
|     })
 | |
|   })
 | |
| 
 | |
|   describe('onEffectCleanup', () => {
 | |
|     it('should get called correctly', async () => {
 | |
|       const count = ref(0)
 | |
|       const cleanupEffect = vi.fn()
 | |
| 
 | |
|       const e = effect(() => {
 | |
|         onEffectCleanup(cleanupEffect)
 | |
|         count.value
 | |
|       })
 | |
| 
 | |
|       count.value++
 | |
|       await nextTick()
 | |
|       expect(cleanupEffect).toHaveBeenCalledTimes(1)
 | |
| 
 | |
|       count.value++
 | |
|       await nextTick()
 | |
|       expect(cleanupEffect).toHaveBeenCalledTimes(2)
 | |
| 
 | |
|       // call it on stop
 | |
|       e.effect.stop()
 | |
|       expect(cleanupEffect).toHaveBeenCalledTimes(3)
 | |
|     })
 | |
| 
 | |
|     it('should warn if called without active effect', () => {
 | |
|       onEffectCleanup(() => {})
 | |
|       expect(
 | |
|         `onEffectCleanup() was called when there was no active effect`,
 | |
|       ).toHaveBeenWarned()
 | |
|     })
 | |
| 
 | |
|     it('should not warn without active effect when failSilently argument is passed', () => {
 | |
|       onEffectCleanup(() => {}, true)
 | |
|       expect(
 | |
|         `onEffectCleanup() was called when there was no active effect`,
 | |
|       ).not.toHaveBeenWarned()
 | |
|     })
 | |
|   })
 | |
| 
 | |
|   test('should pause/resume effect', () => {
 | |
|     const obj = reactive({ foo: 1 })
 | |
|     const fnSpy = vi.fn(() => obj.foo)
 | |
|     const runner = effect(fnSpy)
 | |
| 
 | |
|     expect(fnSpy).toHaveBeenCalledTimes(1)
 | |
|     expect(obj.foo).toBe(1)
 | |
| 
 | |
|     runner.effect.pause()
 | |
|     obj.foo++
 | |
|     expect(fnSpy).toHaveBeenCalledTimes(1)
 | |
|     expect(obj.foo).toBe(2)
 | |
| 
 | |
|     runner.effect.resume()
 | |
|     expect(fnSpy).toHaveBeenCalledTimes(2)
 | |
|     expect(obj.foo).toBe(2)
 | |
| 
 | |
|     obj.foo++
 | |
|     expect(fnSpy).toHaveBeenCalledTimes(3)
 | |
|     expect(obj.foo).toBe(3)
 | |
|   })
 | |
| 
 | |
|   test('should be executed once immediately when resume is called', () => {
 | |
|     const obj = reactive({ foo: 1 })
 | |
|     const fnSpy = vi.fn(() => obj.foo)
 | |
|     const runner = effect(fnSpy)
 | |
| 
 | |
|     expect(fnSpy).toHaveBeenCalledTimes(1)
 | |
|     expect(obj.foo).toBe(1)
 | |
| 
 | |
|     runner.effect.pause()
 | |
|     obj.foo++
 | |
|     expect(fnSpy).toHaveBeenCalledTimes(1)
 | |
|     expect(obj.foo).toBe(2)
 | |
| 
 | |
|     obj.foo++
 | |
|     expect(fnSpy).toHaveBeenCalledTimes(1)
 | |
|     expect(obj.foo).toBe(3)
 | |
| 
 | |
|     runner.effect.resume()
 | |
|     expect(fnSpy).toHaveBeenCalledTimes(2)
 | |
|     expect(obj.foo).toBe(3)
 | |
|   })
 | |
| })
 |