Skip to content

Commit 85a280b

Browse files
Bundle JSX stop-gap impls.
1 parent 3cafc1e commit 85a280b

File tree

5 files changed

+143
-3
lines changed

5 files changed

+143
-3
lines changed

package-lock.json

+2-2
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
{
22
"name": "@imlib/core",
3-
"version": "3.2.2",
3+
"version": "3.3.0",
44
"main": "out/index.js",
55
"bin": {
66
"imlib": "out/cli.js"

src/index.ts

+1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
export { startDevServer } from './dev-server.js';
22
export { generateFiles } from './file-generator.js';
33
export { File } from './file.js';
4+
export { jsxToString } from './jsx-strings.js';
45
export { Module } from './module.js';
56
export { Runtime } from './runtime.js';
67
export * from './ssp.js';

src/jsx-dom.ts

+80
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,80 @@
1+
const jsx = Symbol.for('jsx');
2+
3+
export const $ = jsxToDOM;
4+
5+
export function jsxToDOM<T extends Node>({ [jsx]: tag, ...attrs }: JSX.Element): T {
6+
if (typeof tag === 'function') {
7+
const result = tag(attrs);
8+
if (result && typeof result === 'object' && jsx in result) {
9+
return jsxToDOM(result);
10+
}
11+
return result;
12+
}
13+
14+
let children = attrs.children;
15+
delete attrs.children;
16+
if (!Array.isArray(children)) children = [children];
17+
18+
if (tag === '') {
19+
const goodChildren = getGoodChildren(children);
20+
if (goodChildren.length === 1) {
21+
return goodChildren[0];
22+
}
23+
const el = document.createDocumentFragment();
24+
pushChildren(el, goodChildren);
25+
return el as unknown as T;
26+
}
27+
28+
const isSvg = svgs.has(tag);
29+
const el = (isSvg
30+
? document.createElementNS('http://www.w3.org/2000/svg', tag)
31+
: document.createElement(tag));
32+
33+
pushChildren(el, getGoodChildren(children));
34+
35+
for (const [key, val] of Object.entries(attrs)) {
36+
if (key.startsWith('data-')) {
37+
el.dataset[key.slice(5)] = val;
38+
}
39+
else if (isSvg) {
40+
const jsKey = key.replace(/-\w/, (s) => `${s.toUpperCase()}`);
41+
el.setAttribute(jsKey, val);
42+
}
43+
else {
44+
const jsKey = replacements[key] ?? key.replace(/-\w/, (s) => `${s.toUpperCase()}`);
45+
if (isSvg || attributes.has(jsKey)) {
46+
el.setAttribute(jsKey, val);
47+
}
48+
else {
49+
(el as any)[jsKey] = val;
50+
}
51+
}
52+
}
53+
return el;
54+
}
55+
56+
function getGoodChildren(children: any) {
57+
return [children].flat(Infinity).filter(child => child !== null && child !== undefined && child !== false);
58+
}
59+
60+
function pushChildren(el: DocumentFragment | HTMLElement | SVGElement, children: any[]) {
61+
for (let child of children) {
62+
if (typeof child === 'object' && jsx in child) {
63+
child = jsxToDOM(child);
64+
}
65+
66+
if (child instanceof HTMLLinkElement || child instanceof HTMLStyleElement) {
67+
document.head.append(child);
68+
}
69+
else {
70+
el.append(child);
71+
}
72+
}
73+
}
74+
75+
const replacements: Record<string, string> = {
76+
"class": 'className',
77+
};
78+
79+
const attributes = new Set(['d', 'viewBox']);
80+
const svgs = new Set(['svg', 'path', 'use']);

src/jsx-strings.ts

+59
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,59 @@
1+
const UNARY = new Set(["img", "br", "hr", "input", "meta", "link"]);
2+
3+
const jsx = Symbol.for('jsx');
4+
5+
export function jsxToString({ [jsx]: tag, ...attrs }: JSX.Element): string {
6+
if (typeof tag === 'function') {
7+
const result = tag(attrs);
8+
if (result && typeof result === 'object' && jsx in result) {
9+
return jsxToString(result);
10+
}
11+
return result;
12+
}
13+
14+
let children = attrs.children;
15+
delete attrs.children;
16+
if (!Array.isArray(children)) children = [children];
17+
18+
const parts: string[] = [];
19+
20+
if (tag === '') {
21+
pushChildren(children, parts);
22+
return parts.join('');
23+
}
24+
25+
parts.push('<', tag);
26+
for (const k in attrs) {
27+
const v = attrs[k];
28+
if (v === true)
29+
parts.push(' ', k);
30+
else if (v)
31+
parts.push(' ', k, '="', v, '"');
32+
}
33+
parts.push('>');
34+
35+
if (!UNARY.has(tag)) {
36+
pushChildren(children, parts);
37+
parts.push('</', tag, '>');
38+
}
39+
40+
return parts.join('');
41+
}
42+
43+
function pushChildren(children: any[], parts: string[]) {
44+
for (const child of children) {
45+
if (child !== null && child !== undefined && child !== false) {
46+
if (Array.isArray(child)) {
47+
pushChildren(child, parts);
48+
}
49+
else {
50+
if (typeof child === 'object' && jsx in child) {
51+
parts.push(jsxToString(child));
52+
}
53+
else {
54+
parts.push(child);
55+
}
56+
}
57+
}
58+
}
59+
}

0 commit comments

Comments
 (0)