-
Notifications
You must be signed in to change notification settings - Fork 267
/
Copy pathmount.ts
125 lines (109 loc) · 3.69 KB
/
mount.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
import { ComponentPublicInstance, DefineComponent, VNode } from 'vue'
import type {
ComponentExposed,
ComponentProps,
ComponentSlots
} from 'vue-component-type-helpers'
import { createInstance } from './createInstance'
import { MountingOptions } from './types'
import { trackInstance } from './utils/autoUnmount'
import { VueWrapper } from './vueWrapper'
import { createVueWrapper } from './wrapperFactory'
type ShimSlotReturnType<T> = T extends (...args: infer P) => any
? (...args: P) => any
: never
type WithArray<T> = T | T[]
// eslint-disable-next-line @typescript-eslint/no-empty-object-type
type ComponentData<T> = T extends { data?(...args: any): infer D } ? D : {}
export type ComponentMountingOptions<
T,
P extends ComponentProps<T> = ComponentProps<T>
> = Omit<MountingOptions<P, ComponentData<T>>, 'slots'> & {
slots?: {
[K in keyof ComponentSlots<T>]: WithArray<
| ShimSlotReturnType<ComponentSlots<T>[K]>
| string
| VNode
| (new () => any)
| { template: string }
>
}
} & Record<string, unknown>
export function mount<
T,
C = T extends ((...args: any) => any) | (new (...args: any) => any)
? T
: T extends { props?: infer Props }
? DefineComponent<
Props extends Readonly<(infer PropNames)[]> | (infer PropNames)[]
? { [key in PropNames extends string ? PropNames : string]?: any }
: Props
>
: DefineComponent,
P extends ComponentProps<C> = ComponentProps<C>
>(
originalComponent: T,
options?: ComponentMountingOptions<C, P>
): VueWrapper<
ComponentProps<C> & ComponentData<C> & ComponentExposed<C>,
ComponentPublicInstance<
ComponentProps<C>,
ComponentData<C> & ComponentExposed<C> & Omit<P, keyof ComponentProps<C>>
>
>
// implementation
export function mount(
inputComponent: any,
options?: MountingOptions<any> & Record<string, any>
): VueWrapper<any> {
const { app, props, componentRef } = createInstance(inputComponent, options)
const setProps = (newProps: Record<string, unknown>) => {
for (const [k, v] of Object.entries(newProps)) {
props[k] = v
}
return vm.$nextTick()
}
// Workaround for https://github.com/vuejs/core/issues/7020
const originalErrorHandler = app.config.errorHandler
const errorsOnMount: unknown[] = []
app.config.errorHandler = (err, instance, info) => {
errorsOnMount.push(err)
return originalErrorHandler?.(err, instance, info)
}
// mount the app!
const el = document.createElement('div')
if (options?.attachTo) {
let to: Element | null
if (typeof options.attachTo === 'string') {
to = document.querySelector(options.attachTo)
if (!to) {
throw new Error(
`Unable to find the element matching the selector ${options.attachTo} given as the \`attachTo\` option`
)
}
} else {
to = options.attachTo
}
to.appendChild(el)
}
const vm = app.mount(el)
if (errorsOnMount.length) {
// If several errors are thrown during mount, then throw the first one
throw errorsOnMount[0]
}
app.config.errorHandler = originalErrorHandler
const appRef = componentRef.value! as ComponentPublicInstance
// we add `hasOwnProperty` so Jest can spy on the proxied vm without throwing
// note that this is not necessary with Jest v27+ or Vitest, but is kept for compatibility with older Jest versions
if (!app.hasOwnProperty) {
appRef.hasOwnProperty = (property) => {
return Reflect.has(appRef, property)
}
}
const wrapper = createVueWrapper(app, appRef, setProps)
trackInstance(wrapper)
return wrapper
}
export const shallowMount: typeof mount = (component: any, options?: any) => {
return mount(component, { ...options, shallow: true })
}