Skip to content

Commit fcd486d

Browse files
committed
refactor(core): consolidate options validation and cleanup into core
1 parent 79a2924 commit fcd486d

12 files changed

+193
-135
lines changed

src/__tests__/render-runes.test-d.ts

+3-2
Original file line numberDiff line numberDiff line change
@@ -29,8 +29,9 @@ describe('types', () => {
2929
test('render result has container and component', () => {
3030
const result = subject.render(Component, { name: 'Alice', count: 42 })
3131

32-
expectTypeOf(result).toMatchTypeOf<{
32+
expectTypeOf(result).toExtend<{
3333
container: HTMLElement
34+
baseElement: HTMLElement
3435
component: { hello: string }
3536
debug: (el?: HTMLElement) => void
3637
rerender: (props: { name?: string; count?: number }) => Promise<void>
@@ -55,7 +56,7 @@ describe('legacy component types', () => {
5556
count: 42,
5657
})
5758

58-
expectTypeOf(component).toMatchTypeOf<{ hello: string }>()
59+
expectTypeOf(component).toExtend<{ hello: string }>()
5960

6061
// @ts-expect-error: Svelte 5 mount does not return `$set`
6162
component.$on('greeting', onGreeting)

src/__tests__/render-utilities.test-d.ts

+7-7
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@ describe('render query and utility types', () => {
88
test('render result has default queries', () => {
99
const result = subject.render(Component, { name: 'Alice' })
1010

11-
expectTypeOf(result.getByRole).parameters.toMatchTypeOf<
11+
expectTypeOf(result.getByRole).parameters.toExtend<
1212
[role: subject.ByRoleMatcher, options?: subject.ByRoleOptions]
1313
>()
1414
})
@@ -27,25 +27,25 @@ describe('render query and utility types', () => {
2727
{ queries: { getByVibes } }
2828
)
2929

30-
expectTypeOf(result.getByVibes).parameters.toMatchTypeOf<[vibes: string]>()
30+
expectTypeOf(result.getByVibes).parameters.toExtend<[vibes: string]>()
3131
})
3232

3333
test('act is an async function', () => {
34-
expectTypeOf(subject.act).toMatchTypeOf<() => Promise<void>>()
34+
expectTypeOf(subject.act).toExtend<() => Promise<void>>()
3535
})
3636

3737
test('act accepts a sync function', () => {
38-
expectTypeOf(subject.act).toMatchTypeOf<(fn: () => void) => Promise<void>>()
38+
expectTypeOf(subject.act).toExtend<(fn: () => void) => Promise<void>>()
3939
})
4040

4141
test('act accepts an async function', () => {
42-
expectTypeOf(subject.act).toMatchTypeOf<
42+
expectTypeOf(subject.act).toExtend<
4343
(fn: () => Promise<void>) => Promise<void>
4444
>()
4545
})
4646

4747
test('fireEvent is an async function', () => {
48-
expectTypeOf(subject.fireEvent).toMatchTypeOf<
48+
expectTypeOf(subject.fireEvent).toExtend<
4949
(
5050
element: Element | Node | Document | Window,
5151
event: Event
@@ -54,7 +54,7 @@ describe('render query and utility types', () => {
5454
})
5555

5656
test('fireEvent[eventName] is an async function', () => {
57-
expectTypeOf(subject.fireEvent.click).toMatchTypeOf<
57+
expectTypeOf(subject.fireEvent.click).toExtend<
5858
(
5959
element: Element | Node | Document | Window,
6060
// eslint-disable-next-line @typescript-eslint/no-empty-object-type

src/__tests__/render.test-d.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -37,8 +37,9 @@ describe('types', () => {
3737
test('render result has container and component', () => {
3838
const result = subject.render(Component, { name: 'Alice', count: 42 })
3939

40-
expectTypeOf(result).toMatchTypeOf<{
40+
expectTypeOf(result).toExtend<{
4141
container: HTMLElement
42+
baseElement: HTMLElement
4243
component: { hello: string }
4344
debug: (el?: HTMLElement) => void
4445
rerender: (props: { name?: string; count?: number }) => Promise<void>

src/core/cleanup.js

+32
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
/** @type {Set<() => void>} */
2+
const cleanupTasks = new Set()
3+
4+
/**
5+
* Register later cleanup task
6+
*
7+
* @param {() => void} onCleanup
8+
*/
9+
const addCleanupTask = (onCleanup) => {
10+
cleanupTasks.add(onCleanup)
11+
return onCleanup
12+
}
13+
14+
/**
15+
* Remove a cleanup task without running it.
16+
*
17+
* @param {() => void} onCleanup
18+
*/
19+
const removeCleanupTask = (onCleanup) => {
20+
cleanupTasks.delete(onCleanup)
21+
}
22+
23+
/** Clean up all components and elements added to the document. */
24+
const cleanup = () => {
25+
for (const handleCleanup of cleanupTasks.values()) {
26+
handleCleanup()
27+
}
28+
29+
cleanupTasks.clear()
30+
}
31+
32+
export { addCleanupTask, cleanup, removeCleanupTask }

src/core/index.js

+3-17
Original file line numberDiff line numberDiff line change
@@ -5,20 +5,6 @@
55
* Will switch to legacy, class-based mounting logic
66
* if it looks like we're in a Svelte <= 4 environment.
77
*/
8-
import * as MountLegacy from './mount-legacy.js'
9-
import * as MountModern from './mount-modern.svelte.js'
10-
import { createValidateOptions, UnknownSvelteOptionsError } from './prepare.js'
11-
12-
const { mount, unmount, updateProps, allowedOptions } =
13-
MountModern.IS_MODERN_SVELTE ? MountModern : MountLegacy
14-
15-
/** Validate component options. */
16-
const validateOptions = createValidateOptions(allowedOptions)
17-
18-
export {
19-
mount,
20-
UnknownSvelteOptionsError,
21-
unmount,
22-
updateProps,
23-
validateOptions,
24-
}
8+
export { cleanup } from './cleanup.js'
9+
export { mount } from './mount.js'
10+
export { prepare, UnknownSvelteOptionsError } from './prepare.js'

src/core/mount-legacy.js

+23-22
Original file line numberDiff line numberDiff line change
@@ -4,8 +4,10 @@
44
* Supports Svelte <= 4.
55
*/
66

7+
import { addCleanupTask, removeCleanupTask } from './cleanup.js'
8+
79
/** Allowed options for the component constructor. */
8-
const allowedOptions = [
10+
const ALLOWED_OPTIONS = [
911
'target',
1012
'accessors',
1113
'anchor',
@@ -15,32 +17,31 @@ const allowedOptions = [
1517
'context',
1618
]
1719

18-
/**
19-
* Mount the component into the DOM.
20-
*
21-
* The `onDestroy` callback is included for strict backwards compatibility
22-
* with previous versions of this library. It's mostly unnecessary logic.
23-
*/
24-
const mount = (Component, options, onDestroy) => {
20+
/** Mount the component into the DOM. */
21+
const mount = (Component, options) => {
2522
const component = new Component(options)
2623

27-
if (typeof onDestroy === 'function') {
28-
component.$$.on_destroy.push(() => {
29-
onDestroy(component)
30-
})
24+
/** Remove the component from the DOM. */
25+
const unmount = () => {
26+
component.$destroy()
27+
removeCleanupTask(unmount)
3128
}
3229

33-
return component
34-
}
30+
/** Update the component's props. */
31+
const rerender = (nextProps) => {
32+
component.$set(nextProps)
33+
}
3534

36-
/** Remove the component from the DOM. */
37-
const unmount = (component) => {
38-
component.$destroy()
39-
}
35+
// This `$$.on_destroy` listener is included for strict backwards compatibility
36+
// with previous versions of `@testing-library/svelte`.
37+
// It's unnecessary and will be removed in a future major version.
38+
component.$$.on_destroy.push(() => {
39+
removeCleanupTask(unmount)
40+
})
41+
42+
addCleanupTask(unmount)
4043

41-
/** Update the component's props. */
42-
const updateProps = (component, nextProps) => {
43-
component.$set(nextProps)
44+
return { component, unmount, rerender }
4445
}
4546

46-
export { allowedOptions, mount, unmount, updateProps }
47+
export { ALLOWED_OPTIONS, mount }

src/core/mount-modern.svelte.js

+15-21
Original file line numberDiff line numberDiff line change
@@ -5,14 +5,13 @@
55
*/
66
import * as Svelte from 'svelte'
77

8-
/** Props signals for each rendered component. */
9-
const propsByComponent = new Map()
8+
import { addCleanupTask, removeCleanupTask } from './cleanup.js'
109

1110
/** Whether we're using Svelte >= 5. */
1211
const IS_MODERN_SVELTE = typeof Svelte.mount === 'function'
1312

1413
/** Allowed options to the `mount` call. */
15-
const allowedOptions = [
14+
const ALLOWED_OPTIONS = [
1615
'target',
1716
'anchor',
1817
'props',
@@ -26,26 +25,21 @@ const mount = (Component, options) => {
2625
const props = $state(options.props ?? {})
2726
const component = Svelte.mount(Component, { ...options, props })
2827

29-
Svelte.flushSync()
30-
propsByComponent.set(component, props)
28+
/** Remove the component from the DOM. */
29+
const unmount = () => {
30+
Svelte.flushSync(() => Svelte.unmount(component))
31+
removeCleanupTask(unmount)
32+
}
3133

32-
return component
33-
}
34+
/** Update the component's props. */
35+
const rerender = (nextProps) => {
36+
Svelte.flushSync(() => Object.assign(props, nextProps))
37+
}
3438

35-
/** Remove the component from the DOM. */
36-
const unmount = (component) => {
37-
propsByComponent.delete(component)
38-
Svelte.flushSync(() => Svelte.unmount(component))
39-
}
39+
addCleanupTask(unmount)
40+
Svelte.flushSync()
4041

41-
/**
42-
* Update the component's props.
43-
*
44-
* Relies on the `$state` signal added in `mount`.
45-
*/
46-
const updateProps = (component, nextProps) => {
47-
const prevProps = propsByComponent.get(component)
48-
Object.assign(prevProps, nextProps)
42+
return { component, unmount, rerender }
4943
}
5044

51-
export { allowedOptions, IS_MODERN_SVELTE, mount, unmount, updateProps }
45+
export { ALLOWED_OPTIONS, IS_MODERN_SVELTE, mount }

src/core/mount.js

+36
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
import { tick } from 'svelte'
2+
3+
import * as MountLegacy from './mount-legacy.js'
4+
import * as MountModern from './mount-modern.svelte.js'
5+
6+
const mountComponent = MountModern.IS_MODERN_SVELTE
7+
? MountModern.mount
8+
: MountLegacy.mount
9+
10+
/**
11+
* Render a Svelte component into the document.
12+
*
13+
* @template {import('./types.js').Component} C
14+
* @param {import('./types.js').ComponentType<C>} Component
15+
* @param {import('./types.js').MountOptions<C>} options
16+
* @returns {{
17+
* component: C
18+
* unmount: () => void
19+
* rerender: (props: Partial<import('./types.js').Props<C>>) => Promise<void>
20+
* }}
21+
*/
22+
const mount = (Component, options = {}) => {
23+
const { component, unmount, rerender } = mountComponent(Component, options)
24+
25+
return {
26+
component,
27+
unmount,
28+
rerender: async (props) => {
29+
rerender(props)
30+
// Await the next tick for Svelte 4, which cannot flush changes synchronously
31+
await tick()
32+
},
33+
}
34+
}
35+
36+
export { mount }

src/core/prepare.js

+48-7
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,18 @@
1+
import { addCleanupTask } from './cleanup.js'
2+
import * as MountLegacy from './mount-legacy.js'
3+
import * as MountModern from './mount-modern.svelte.js'
4+
5+
const ALLOWED_OPTIONS = MountModern.IS_MODERN_SVELTE
6+
? MountModern.ALLOWED_OPTIONS
7+
: MountLegacy.ALLOWED_OPTIONS
8+
9+
/** An error thrown for incorrect options and clashes between props and Svelte options. */
110
class UnknownSvelteOptionsError extends TypeError {
2-
constructor(unknownOptions, allowedOptions) {
11+
constructor(unknownOptions) {
312
super(`Unknown options.
413
514
Unknown: [ ${unknownOptions.join(', ')} ]
6-
Allowed: [ ${allowedOptions.join(', ')} ]
15+
Allowed: [ ${ALLOWED_OPTIONS.join(', ')} ]
716
817
To pass both Svelte options and props to a component,
918
or to use props that share a name with a Svelte option,
@@ -15,9 +24,41 @@ class UnknownSvelteOptionsError extends TypeError {
1524
}
1625
}
1726

18-
const createValidateOptions = (allowedOptions) => (options) => {
27+
/**
28+
* Prepare DOM elements for rendering.
29+
*
30+
* @template {import('./types.js').Component} C
31+
* @param {import('./types.js').PropsOrMountOptions<C>} propsOrOptions
32+
* @param {{ baseElement?: HTMLElement }} renderOptions
33+
* @returns {{
34+
* baseElement: HTMLElement
35+
* target: HTMLElement
36+
* mountOptions: import('./types.js').MountOptions<C>
37+
* }}
38+
*/
39+
const prepare = (propsOrOptions = {}, renderOptions = {}) => {
40+
const mountOptions = validateMountOptions(propsOrOptions)
41+
42+
const baseElement =
43+
renderOptions.baseElement ?? mountOptions.target ?? document.body
44+
45+
const target =
46+
mountOptions.target ??
47+
baseElement.appendChild(document.createElement('div'))
48+
49+
addCleanupTask(() => {
50+
if (target.parentNode === document.body) {
51+
document.body.removeChild(target)
52+
}
53+
})
54+
55+
return { baseElement, target, mountOptions: { ...mountOptions, target } }
56+
}
57+
58+
/** Prevent incorrect options and clashes between props and Svelte options. */
59+
const validateMountOptions = (options) => {
1960
const isProps = !Object.keys(options).some((option) =>
20-
allowedOptions.includes(option)
61+
ALLOWED_OPTIONS.includes(option)
2162
)
2263

2364
if (isProps) {
@@ -26,14 +67,14 @@ const createValidateOptions = (allowedOptions) => (options) => {
2667

2768
// Check if any props and Svelte options were accidentally mixed.
2869
const unknownOptions = Object.keys(options).filter(
29-
(option) => !allowedOptions.includes(option)
70+
(option) => !ALLOWED_OPTIONS.includes(option)
3071
)
3172

3273
if (unknownOptions.length > 0) {
33-
throw new UnknownSvelteOptionsError(unknownOptions, allowedOptions)
74+
throw new UnknownSvelteOptionsError(unknownOptions)
3475
}
3576

3677
return options
3778
}
3879

39-
export { createValidateOptions, UnknownSvelteOptionsError }
80+
export { prepare, UnknownSvelteOptionsError }

src/core/types.d.ts

+5
Original file line numberDiff line numberDiff line change
@@ -59,3 +59,8 @@ export type Exports<C> = IS_MODERN_SVELTE extends true
5959
export type MountOptions<C extends Component> = IS_MODERN_SVELTE extends true
6060
? Parameters<typeof mount<Props<C>, Exports<C>>>[1]
6161
: LegacyConstructorOptions<Props<C>>
62+
63+
/** Component props or partial mount options. */
64+
export type PropsOrMountOptions<C extends Component> =
65+
| Props<C>
66+
| Partial<MountOptions<C>>

0 commit comments

Comments
 (0)