diff --git a/docs/api/index.md b/docs/api/index.md index c5549bda9e..167bcf47aa 100644 --- a/docs/api/index.md +++ b/docs/api/index.md @@ -1829,6 +1829,28 @@ function shallowMount(Component, options?: MountingOptions): VueWrapper `shallowMount` behaves exactly like `mount`, but it stubs all child components by default. Essentially, `shallowMount(Component)` is an alias of `mount(Component, { shallow: true })`. +## enableAutoUnmount + + +**Signature:** +```ts +enableAutoUnmount(hook: Function)); +disableAutoUnmount(): void; +``` + +**Details:** + +`enableAutoUnmount` allows to automatically destroy Vue wrappers. Destroy logic is passed as callback to `hook` Function. +Common usage is to use `enableAutoUnmount` with teardown helper functions provided by your test framework, such as `afterEach`: + +```ts +import { enableAutoUnmount } from '@vue/test-utils' + +enableAutoUnmount(afterEach) +``` + +`disableAutoUnmount` might be useful if you want this behavior only in specific subset of your test suite and you want to explicitly disable this behavior + ## flushPromises **Signature:** diff --git a/src/index.ts b/src/index.ts index 3ed2e7a5b8..1f9e8dc751 100644 --- a/src/index.ts +++ b/src/index.ts @@ -6,10 +6,13 @@ import { DOMWrapper } from './domWrapper' import { createWrapperError } from './errorWrapper' import { config } from './config' import { flushPromises } from './utils/flushPromises' +import { enableAutoUnmount, disableAutoUnmount } from './utils/autoUnmount' export { mount, shallowMount, + enableAutoUnmount, + disableAutoUnmount, RouterLinkStub, VueWrapper, DOMWrapper, diff --git a/src/mount.ts b/src/mount.ts index 8deca19821..5c20a0e9c7 100644 --- a/src/mount.ts +++ b/src/mount.ts @@ -45,6 +45,7 @@ import { isLegacyFunctionalComponent, unwrapLegacyVueExtendComponent } from './utils/vueCompatSupport' +import { trackInstance } from './utils/autoUnmount' // NOTE this should come from `vue` type PublicProps = VNodeProps & AllowedComponentProps & ComponentCustomProps @@ -165,9 +166,11 @@ export function mount< Mixin extends ComponentOptionsMixin = ComponentOptionsMixin, Extends extends ComponentOptionsMixin = ComponentOptionsMixin, EE extends string = string, - Props extends Readonly<{ [key in PropNames]?: any }> = Readonly<{ - [key in PropNames]?: any - }> + Props extends Readonly<{ [key in PropNames]?: any }> = Readonly< + { + [key in PropNames]?: any + } + > >( componentOptions: ComponentOptionsWithArrayProps< PropNames, @@ -452,7 +455,9 @@ export function mount( return Reflect.has(appRef, property) } console.warn = warnSave - return createWrapper(app, appRef, setProps) + const wrapper = createWrapper(app, $vm, setProps) + trackInstance(wrapper) + return wrapper } export const shallowMount: typeof mount = (component: any, options?: any) => { diff --git a/src/utils/autoUnmount.ts b/src/utils/autoUnmount.ts new file mode 100644 index 0000000000..5b81fa998c --- /dev/null +++ b/src/utils/autoUnmount.ts @@ -0,0 +1,32 @@ +import { ComponentPublicInstance } from 'vue' +import type { VueWrapper } from '../vueWrapper' + +let isEnabled = false +const wrapperInstances: VueWrapper[] = [] + +export function disableAutoUnmount() { + isEnabled = false + wrapperInstances.length = 0 +} + +export function enableAutoUnmount(hook: Function) { + if (isEnabled) { + throw new Error('enableAutoUnmount cannot be called more than once') + } + + isEnabled = true + + hook(() => { + wrapperInstances.forEach((wrapper: VueWrapper) => { + wrapper.unmount() + }) + + wrapperInstances.length = 0 + }) +} + +export function trackInstance(wrapper: VueWrapper) { + if (!isEnabled) return + + wrapperInstances.push(wrapper) +} diff --git a/tests/autoUnmount.spec.ts b/tests/autoUnmount.spec.ts new file mode 100644 index 0000000000..b03b404cf2 --- /dev/null +++ b/tests/autoUnmount.spec.ts @@ -0,0 +1,37 @@ +import { mount, enableAutoUnmount, disableAutoUnmount } from '../src' + +describe('enableAutoUnmount', () => { + beforeEach(() => { + disableAutoUnmount() + }) + + it('calls the hook function', () => { + const hookMock = jest.fn() + + enableAutoUnmount(hookMock) + + expect(hookMock).toHaveBeenCalledWith(expect.any(Function)) + }) + + it('uses the hook function to unmount wrappers', () => { + const hookMock = jest.fn() + + enableAutoUnmount(hookMock) + const [unmountFn] = hookMock.mock.calls[0] + + const wrapper = mount({ template: '

test

' }) + jest.spyOn(wrapper, 'unmount') + + unmountFn() + + expect(wrapper.unmount).toHaveBeenCalledTimes(1) + }) + + it('cannot be called twice', () => { + const noop = () => {} + + enableAutoUnmount(noop) + + expect(() => enableAutoUnmount(noop)).toThrow() + }) +})