-
Notifications
You must be signed in to change notification settings - Fork 617
Expand file tree
/
Copy pathindex.ts
More file actions
105 lines (87 loc) · 2.46 KB
/
Copy pathindex.ts
File metadata and controls
105 lines (87 loc) · 2.46 KB
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
import { CliRenderer, createCliRenderer, engine, type CliRendererConfig } from "@opentui/core"
import { createTestRenderer, type TestRendererOptions } from "@opentui/core/testing"
import type { JSX } from "./jsx-runtime.js"
import { RendererContext } from "./src/elements/index.js"
import { _render as renderInternal, createComponent } from "./src/reconciler.js"
type DisposeFn = () => void
const mountSolidRoot = (renderer: CliRenderer, node: () => JSX.Element) => {
let dispose: DisposeFn | undefined
let disposeRequested = false
let disposed = false
let mounting = true
let destroyRequested = false
const originalDestroy = renderer.destroy.bind(renderer)
const runDispose = () => {
if (disposed) {
return
}
if (!dispose) {
disposeRequested = true
return
}
disposed = true
dispose()
}
renderer.once("destroy", runDispose)
renderer.destroy = () => {
if (mounting) {
destroyRequested = true
return
}
originalDestroy()
}
try {
dispose = renderInternal(
() =>
createComponent(RendererContext.Provider, {
get value() {
return renderer
},
get children() {
return (createComponent as any)(node, {})
},
}),
renderer.root,
)
} finally {
mounting = false
renderer.destroy = originalDestroy
}
if (disposeRequested) {
runDispose()
}
if (destroyRequested) {
originalDestroy()
}
}
export const render = async (node: () => JSX.Element, rendererOrConfig: CliRenderer | CliRendererConfig = {}) => {
const renderer =
rendererOrConfig instanceof CliRenderer
? rendererOrConfig
: await createCliRenderer({
...rendererOrConfig,
onDestroy: () => {
rendererOrConfig.onDestroy?.()
},
})
engine.attach(renderer)
mountSolidRoot(renderer, node)
}
export const testRender = async (node: () => JSX.Element, renderConfig: TestRendererOptions = {}) => {
const testSetup = await createTestRenderer({
...renderConfig,
onDestroy: () => {
renderConfig.onDestroy?.()
},
})
engine.attach(testSetup.renderer)
mountSolidRoot(testSetup.renderer, node)
return testSetup
}
export * from "./src/reconciler.js"
export * from "./src/elements/index.js"
export * from "./src/scrollback.js"
export * from "./src/time-to-first-draw.js"
export * from "./src/plugins/slot.js"
export * from "./src/types/elements.js"
export { type JSX }