import deepmerge from 'deepmerge' import isMobile from 'ismobilejs' import { cleanup, render, RenderOptions } from '@testing-library/vue' import { afterEach, beforeEach, vi } from 'vitest' import { clickaway, droppable, focus } from '@/directives' import { defineComponent, nextTick } from 'vue' import { commonStore, userStore } from '@/stores' import factory from '@/__tests__/factory' declare type Methods = { [K in keyof T]: T[K] extends Closure ? K : never; }[keyof T] & (string | symbol); export default abstract class ComponentTestCase { private backupMethods = new Map() public constructor () { this.beforeEach() this.afterEach() this.test() } protected beforeEach (cb?: Closure) { beforeEach(() => { commonStore.state.allowDownload = true commonStore.state.useiTunes = true cb && cb() }) } protected afterEach (cb?: Closure) { afterEach(() => { cleanup() this.restoreAllMocks() isMobile.any = false cb && cb() }) } protected actingAs (user: User) { userStore.state.current = user return this } protected actingAsAdmin () { return this.actingAs(factory.states('admin')('user')) } protected mock>> (obj: T, methodName: M, implementation?: any) { const mock = vi.fn() if (implementation !== undefined) { mock.mockImplementation(implementation instanceof Function ? implementation : () => implementation) } this.backupMethods.set([obj, methodName], obj[methodName]) // @ts-ignore obj[methodName] = mock return mock } protected restoreAllMocks () { this.backupMethods.forEach((fn, [obj, methodName]) => (obj[methodName] = fn)) this.backupMethods = new Map() } protected render (component: any, options: RenderOptions = {}) { return render(component, deepmerge({ global: { directives: { 'koel-clickaway': clickaway, 'koel-focus': focus, 'koel-droppable': droppable } } }, options)) } protected stub (testId = 'stub') { return defineComponent({ template: `
` }) } protected async tick (count = 1) { for (let i = 0; i < count; ++i) { await nextTick() } } protected abstract test () }