Rendering our React & Vue apps
The HTML
I will use an HTML select to choose between Vue & React, and re-render upon select change.
html
// src/spa-client-side/index.html
<!doctype html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Model View Demo</title>
<link rel="stylesheet" href="./main.css" />
</head>
<body>
<header>
<label>
App Renderer:
<select id="app-renderer" value="vue">
<option value="react">React</option>
<option value="vue">Vue</option>
<option value="solid" disabled>Solid</option>
</select>
</label>
<span>·</span>
<label>
Api Preflight Delay:
<select id="api-delay">
<option value="0">None</option>
<option value="100">100ms</option>
<option value="500">500ms</option>
<option value="1000">1s</option>
</select>
</label>
</header>
<main>
<div id="root"></div>
</main>
<footer>
<span>Made with ❤️ by</span
><a href="https://github.com/SacDeNoeuds">SacDeNoeuds</a>
</footer>
<script type="module" src="./main.ts"></script>
</body>
</html>
The rendering process
We will create a function for each engine: React & Vue, and another one to switch between the engines.
ts
// src/spa-client-side/render-app-8.ts
import { makeAppModel } from './5-app-model/AppModel'
import { todoFetchApi } from './setup/TodoApi.fetch'
const rootNode = document.getElementById('root') as HTMLElement
if (!rootNode) throw new Error('where is my root??')
const appModel = makeAppModel(todoFetchApi)
export async function renderApp(kind: string) {
// clear the root node
rootNode.replaceChildren()
switch (kind) {
case 'react':
return renderReactApp()
case 'vue':
return renderVueApp()
default:
throw new Error(`unknown app kind "${kind}"`)
}
}
async function renderReactApp() {
const [React, ReactDOM, { App }] = await Promise.all([
import('react'),
import('react-dom/client'),
import('./6-react-app/App'),
])
const root = ReactDOM.createRoot(rootNode)
root.render(React.createElement(App, { model: appModel }))
}
async function renderVueApp() {
const [{ createApp }, { default: App }] = await Promise.all([
import('vue'),
import('./7-vue-app/App.vue'),
])
const app = createApp(App, { model: appModel })
app.mount(rootNode)
}
The entry point – main.ts
Now our main is very simple: we get the value from the select and render the corresponding app.
When the select value changes, we re-render the app.
ts
// src/spa-client-side/main.ts
import { renderApp } from './render-app-11'
import { apiDefaults } from './setup/api-latency'
// import { renderApp } from './render-app-10'
// import { renderApp } from './render-app-9'
// import { renderApp } from './render-app-8'
const selectElement = document.getElementById('app-renderer')
if (!(selectElement instanceof HTMLSelectElement)) throw new Error('mmh…')
const searchParams = new URLSearchParams(window.location.search)
// let’s enforce a default value for testing
selectElement.value = searchParams.get('renderer') ?? 'vue'
selectElement.addEventListener('change', () => {
void renderApp(selectElement.value)
})
const apiDelaySelect = document.getElementById('api-delay') as HTMLSelectElement
apiDelaySelect.value = String(apiDefaults.preflightDelayInMs)
apiDelaySelect.addEventListener('change', () => {
apiDefaults.preflightDelayInMs = parseInt(apiDelaySelect.value)
})
void renderApp(selectElement.value)
Testing the whole thing
sh
npm start
Tada 🎉
Going further: