Move to own JSX implementation
Some checks failed
continuous-integration/drone/push Build is failing
Some checks failed
continuous-integration/drone/push Build is failing
This commit is contained in:
parent
7c1166bf87
commit
2af5d4f823
16
jsx-html/.github/workflows/ci.yml
vendored
16
jsx-html/.github/workflows/ci.yml
vendored
@ -1,16 +0,0 @@
|
|||||||
name: ci
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
branches:
|
|
||||||
- master
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
test:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v1
|
|
||||||
- uses: denolib/setup-deno@master
|
|
||||||
- run: deno --version
|
|
||||||
- run: deno test examples/01.tsx
|
|
||||||
- run: deno test examples/03-async.tsx
|
|
3
jsx-html/.gitignore
vendored
3
jsx-html/.gitignore
vendored
@ -1,3 +0,0 @@
|
|||||||
/dist
|
|
||||||
node_modules
|
|
||||||
tmp
|
|
@ -1,5 +0,0 @@
|
|||||||
{
|
|
||||||
"trailingComma": "all",
|
|
||||||
"tabWidth": 4,
|
|
||||||
"singleQuote": true
|
|
||||||
}
|
|
4
jsx-html/.vscode/settings.json
vendored
4
jsx-html/.vscode/settings.json
vendored
@ -1,4 +0,0 @@
|
|||||||
{
|
|
||||||
"deno.enable": true,
|
|
||||||
"deno.unstable": true
|
|
||||||
}
|
|
@ -1,21 +0,0 @@
|
|||||||
The MIT License (MIT)
|
|
||||||
|
|
||||||
Copyright (c) 2020 Alexandre Piel
|
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in
|
|
||||||
all copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
||||||
THE SOFTWARE.
|
|
@ -1,103 +0,0 @@
|
|||||||
# jsx-html
|
|
||||||
|
|
||||||
`jsx-html` render JSX template to HTML asynchronously. Compatible with Deno, NodeJs and can also run in browser.
|
|
||||||
|
|
||||||
Try with runkit: https://runkit.com/apiel/jsx-html-example
|
|
||||||
|
|
||||||
## NodeJs
|
|
||||||
|
|
||||||
```sh
|
|
||||||
yarn add async-jsx-html
|
|
||||||
# or
|
|
||||||
npm install async-jsx-html
|
|
||||||
```
|
|
||||||
|
|
||||||
```tsx
|
|
||||||
import { React } from 'async-jsx-html';
|
|
||||||
|
|
||||||
const View = () => <div>Hello</div>;
|
|
||||||
// render return a Promise
|
|
||||||
(<View />).render().then((html: string) => console.log(html));
|
|
||||||
```
|
|
||||||
|
|
||||||
## Deno
|
|
||||||
|
|
||||||
```tsx
|
|
||||||
/// <reference path="https://raw.githubusercontent.com/apiel/jsx-html/master/jsx.d.ts" />
|
|
||||||
|
|
||||||
import { React } from 'https://raw.githubusercontent.com/apiel/jsx-html/master/mod.ts';
|
|
||||||
|
|
||||||
const View = () => <div>Hello</div>;
|
|
||||||
// render return a Promise
|
|
||||||
(<View />).render().then((html: string) => console.log(html));
|
|
||||||
```
|
|
||||||
|
|
||||||
```sh
|
|
||||||
deno run https://raw.githubusercontent.com/apiel/jsx-html/master/examples/00.tsx
|
|
||||||
```
|
|
||||||
|
|
||||||
## TsConfig
|
|
||||||
|
|
||||||
As you would do with React, you need to import `React` from `jsx-html` for the transpiler. If you are not feeling confortable with using `React` as import since it is not React, you can import `jsx` from `jsx-html` but you would have to update your tsconfig file: https://github.com/denoland/deno/issues/3572
|
|
||||||
|
|
||||||
```json
|
|
||||||
{
|
|
||||||
"compilerOptions": {
|
|
||||||
"jsx": "react",
|
|
||||||
"jsxFactory": "jsx"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
```tsx
|
|
||||||
/// <reference path="https://raw.githubusercontent.com/apiel/jsx-html/master/jsx.d.ts" />
|
|
||||||
|
|
||||||
import { jsx } from 'https://raw.githubusercontent.com/apiel/jsx-html/master/mod.ts';
|
|
||||||
|
|
||||||
const View = () => <div>Hello</div>;
|
|
||||||
(<View />).render().then(console.log);
|
|
||||||
```
|
|
||||||
|
|
||||||
> **Note:** prefer using sermver tags version instead of master to avoid conflict with caching, e.g:
|
|
||||||
> `import { jsx } from 'https://raw.githubusercontent.com/apiel/jsx-html/1.0.0/mod.ts';`.
|
|
||||||
|
|
||||||
## Async component
|
|
||||||
|
|
||||||
Unlike React, components can be asynchrone, so you can fetch for data without to handle states.
|
|
||||||
|
|
||||||
```tsx
|
|
||||||
import { React } from 'https://raw.githubusercontent.com/apiel/jsx-html/master/mod.ts';
|
|
||||||
|
|
||||||
const Data = async () => {
|
|
||||||
const res = await fetch('http://example.com/some/api');
|
|
||||||
const content = new Uint8Array(await res.arrayBuffer());
|
|
||||||
return <div>{content}</div>;
|
|
||||||
};
|
|
||||||
|
|
||||||
const View = () => (
|
|
||||||
<div>
|
|
||||||
<Data />
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
|
|
||||||
(<View />).render().then(console.log);
|
|
||||||
```
|
|
||||||
|
|
||||||
# InnerHTML
|
|
||||||
|
|
||||||
The Element property innerHTML sets the HTML or XML markup contained within the property.
|
|
||||||
|
|
||||||
In general, setting HTML from code is risky because it’s easy to inadvertently expose your users to a cross-site scripting (XSS) attack.
|
|
||||||
|
|
||||||
```tsx
|
|
||||||
/// <reference path="https://raw.githubusercontent.com/apiel/jsx-html/master/jsx.d.ts" />
|
|
||||||
|
|
||||||
import { jsx } from 'https://raw.githubusercontent.com/apiel/jsx-html/master/mod.ts';
|
|
||||||
|
|
||||||
const View = () => <div innerHTML="<b>hello</b> world" />;
|
|
||||||
(<View />).render().then(console.log); // will output <div><b>hello</b> world</div>
|
|
||||||
```
|
|
||||||
|
|
||||||
## Browser
|
|
||||||
|
|
||||||
`jsx-html` can also be used directly in browser. Find an example with webpack [here](https://github.com/apiel/jsx-html/tree/master/examples/browser).
|
|
@ -1,26 +0,0 @@
|
|||||||
'use strict';
|
|
||||||
|
|
||||||
var regExp = /\.(ts|tsx|js|jsx)$/i;
|
|
||||||
|
|
||||||
module.exports = function () {
|
|
||||||
return {
|
|
||||||
visitor: {
|
|
||||||
ImportDeclaration: function ImportDeclaration(path) {
|
|
||||||
var source = path.node.source;
|
|
||||||
if (!source.value.match(regExp)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
source.value = source.value.replace(regExp, '');
|
|
||||||
},
|
|
||||||
ExportDeclaration: function ExportDeclaration(path) {
|
|
||||||
var source = path.node.source;
|
|
||||||
if (source) {
|
|
||||||
if (!source.value.match(regExp)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
source.value = source.value.replace(regExp, '');
|
|
||||||
}
|
|
||||||
},
|
|
||||||
},
|
|
||||||
};
|
|
||||||
};
|
|
@ -1,6 +0,0 @@
|
|||||||
export enum NODE_TYPE {
|
|
||||||
ELEMENT = 'element',
|
|
||||||
TEXT = 'text',
|
|
||||||
COMPONENT = 'component',
|
|
||||||
FRAGMENT = 'fragment',
|
|
||||||
};
|
|
3496
jsx-html/deno.d.ts
vendored
3496
jsx-html/deno.d.ts
vendored
File diff suppressed because it is too large
Load Diff
@ -1,17 +0,0 @@
|
|||||||
{
|
|
||||||
"cmds": {
|
|
||||||
"start": ["den 00", "den 01", "den 02", "den 03", "den 04", "den 05", "den 06"],
|
|
||||||
"test": ["den 01:test", "den 03:test", "den 05:test", "den 06:test"],
|
|
||||||
"00": "deno run examples/00.tsx",
|
|
||||||
"01": "deno run examples/01.tsx",
|
|
||||||
"01:test": "deno test examples/01.tsx",
|
|
||||||
"02": "deno run -c examples/02/tsconfig.json examples/02/02.tsx",
|
|
||||||
"03": "deno run examples/03-async.tsx",
|
|
||||||
"03:test": "deno test examples/03-async.tsx",
|
|
||||||
"04": "deno run examples/04.tsx",
|
|
||||||
"05": "deno run examples/05.tsx",
|
|
||||||
"05:test": "deno test examples/05.tsx",
|
|
||||||
"06": "deno run examples/06.tsx",
|
|
||||||
"06:test": "deno test examples/06.tsx"
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,101 +0,0 @@
|
|||||||
const fs = require('fs');
|
|
||||||
const { parse } = require('@babel/parser');
|
|
||||||
const { default: generator } = require('@babel/generator');
|
|
||||||
const { default: traverse } = require('@babel/traverse');
|
|
||||||
const { resolve, extname, join, dirname } = require('path');
|
|
||||||
const { tmpdir } = require('os');
|
|
||||||
const { readdir } = require('fs').promises;
|
|
||||||
const { cwd } = require('process');
|
|
||||||
|
|
||||||
const exts = ['.ts'];
|
|
||||||
const excludes = [
|
|
||||||
'/node_modules/',
|
|
||||||
'/examples/',
|
|
||||||
'/dist/',
|
|
||||||
'/jsx.d.ts',
|
|
||||||
'/mod.d.ts',
|
|
||||||
'/deno.d.ts',
|
|
||||||
];
|
|
||||||
|
|
||||||
const regExpRemoveExts = /\.(ts|tsx|js|jsx)$/i;
|
|
||||||
|
|
||||||
const tsconfig = {
|
|
||||||
compilerOptions: {
|
|
||||||
types: ['node'],
|
|
||||||
module: 'commonjs',
|
|
||||||
declaration: true,
|
|
||||||
removeComments: true,
|
|
||||||
emitDecoratorMetadata: true,
|
|
||||||
experimentalDecorators: true,
|
|
||||||
allowSyntheticDefaultImports: true,
|
|
||||||
target: 'es6',
|
|
||||||
sourceMap: true,
|
|
||||||
outDir: join(cwd(), 'nodejs'),
|
|
||||||
baseUrl: './',
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
// const distFolder = join(tmpdir(), `deno2nodejs-${+new Date()}`);
|
|
||||||
const distFolder = join(cwd(), `tmp`);
|
|
||||||
console.log('distFolder:', distFolder);
|
|
||||||
fs.mkdirSync(distFolder);
|
|
||||||
fs.writeFileSync(join(distFolder, 'tsconfig.json'), JSON.stringify(tsconfig));
|
|
||||||
|
|
||||||
async function getFiles(dir) {
|
|
||||||
const dirents = await readdir(dir, { withFileTypes: true });
|
|
||||||
const files = await Promise.all(
|
|
||||||
dirents.map((dirent) => {
|
|
||||||
const res = resolve(dir, dirent.name);
|
|
||||||
return dirent.isDirectory() ? getFiles(res) : res;
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
return Array.prototype.concat(...files);
|
|
||||||
}
|
|
||||||
getFiles('./').then((files) => {
|
|
||||||
const tsFiles = files.filter(
|
|
||||||
(f) =>
|
|
||||||
exts.includes(extname(f)) &&
|
|
||||||
!excludes.some((val) => f.includes(val)),
|
|
||||||
);
|
|
||||||
tsFiles.forEach((file) => {
|
|
||||||
const code = deno2nodejs(file);
|
|
||||||
const dist = join(distFolder, file.substr(cwd().length));
|
|
||||||
const ensureDir = dirname(dist);
|
|
||||||
fs.mkdirSync(ensureDir, { recursive: true });
|
|
||||||
fs.writeFileSync(dist, code);
|
|
||||||
// console.log({ file, dist, ensureDir });
|
|
||||||
});
|
|
||||||
});
|
|
||||||
// we might want to execute `tsc -p ./tmp/tsconfig.json` in here
|
|
||||||
|
|
||||||
function deno2nodejs(file) {
|
|
||||||
const source = fs.readFileSync(file).toString();
|
|
||||||
|
|
||||||
const ast = parse(source, {
|
|
||||||
sourceType: 'module',
|
|
||||||
plugins: ['typescript', 'classProperties'],
|
|
||||||
});
|
|
||||||
|
|
||||||
traverse(ast, {
|
|
||||||
ImportDeclaration: function ImportDeclaration(path) {
|
|
||||||
var source = path.node.source;
|
|
||||||
if (!source.value.match(regExpRemoveExts)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
source.value = source.value.replace(regExpRemoveExts, '');
|
|
||||||
},
|
|
||||||
ExportDeclaration: function ExportDeclaration(path) {
|
|
||||||
var source = path.node.source;
|
|
||||||
if (source) {
|
|
||||||
if (!source.value.match(regExpRemoveExts)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
source.value = source.value.replace(regExpRemoveExts, '');
|
|
||||||
}
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
const { code } = generator(ast);
|
|
||||||
|
|
||||||
return code;
|
|
||||||
}
|
|
@ -1,6 +0,0 @@
|
|||||||
/// <reference path="https://raw.githubusercontent.com/apiel/jsx-html/latest/jsx.d.ts" />
|
|
||||||
|
|
||||||
import { React } from 'https://raw.githubusercontent.com/apiel/jsx-html/latest/mod.ts';
|
|
||||||
|
|
||||||
const View = () => <div>Hello</div>;
|
|
||||||
console.log((<View />).render());
|
|
@ -1,62 +0,0 @@
|
|||||||
/// <reference path="../jsx.d.ts" />
|
|
||||||
|
|
||||||
import { assertEquals } from 'https://deno.land/std/testing/asserts.ts';
|
|
||||||
import { React, Fragment } from '../mod.ts';
|
|
||||||
|
|
||||||
const Title = () => <h1>title</h1>;
|
|
||||||
const Value = ({ val }: { val: string }) => <p>value: {val}</p>;
|
|
||||||
const Numeric = ({ num }: { num: number }) => <p>num: {num}</p>;
|
|
||||||
|
|
||||||
const View = () => (
|
|
||||||
<div class="deno">
|
|
||||||
<Title />
|
|
||||||
<p onclick={() => 'lol'} valid checked={true} select="">
|
|
||||||
land
|
|
||||||
</p>
|
|
||||||
<br />
|
|
||||||
<hr />
|
|
||||||
<Fragment>
|
|
||||||
<Value val="hello" />
|
|
||||||
<Numeric num={23} />
|
|
||||||
</Fragment>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
|
|
||||||
if (import.meta.main) {
|
|
||||||
(<View />).render().then(console.log);
|
|
||||||
} else {
|
|
||||||
// Run test
|
|
||||||
|
|
||||||
Deno.test('render title', async() => {
|
|
||||||
assertEquals(await (<Title />).render(), '<h1>title</h1>');
|
|
||||||
});
|
|
||||||
|
|
||||||
Deno.test('render value', async() => {
|
|
||||||
const val = 'hello';
|
|
||||||
assertEquals(await (<Value val={val} />).render(), `<p>value: ${val}</p>`);
|
|
||||||
});
|
|
||||||
|
|
||||||
Deno.test('render numeric', async() => {
|
|
||||||
const num = 123;
|
|
||||||
assertEquals(await (<Numeric num={num} />).render(), `<p>num: ${num}</p>`);
|
|
||||||
});
|
|
||||||
|
|
||||||
Deno.test('render view', async() => {
|
|
||||||
assertEquals(
|
|
||||||
await (<View />).render(),
|
|
||||||
'<div class="deno"><h1>title</h1><p valid checked select>land</p><br /><hr /><p>value: hello</p><p>num: 23</p></div>',
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
Deno.test('render empty', async ()=>{
|
|
||||||
assertEquals(
|
|
||||||
await (<div/>).render(),
|
|
||||||
`<div></div>`
|
|
||||||
)
|
|
||||||
|
|
||||||
assertEquals(
|
|
||||||
await (<img/>).render(),
|
|
||||||
`<img />`
|
|
||||||
)
|
|
||||||
})
|
|
||||||
}
|
|
@ -1,6 +0,0 @@
|
|||||||
/// <reference path="../../jsx.d.ts" />
|
|
||||||
|
|
||||||
import { jsx } from '../../mod.ts';
|
|
||||||
|
|
||||||
const View = () => <div>Hello</div>;
|
|
||||||
(<View />).render().then(console.log);
|
|
@ -1,6 +0,0 @@
|
|||||||
{
|
|
||||||
"compilerOptions": {
|
|
||||||
"jsx": "react",
|
|
||||||
"jsxFactory": "jsx"
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,30 +0,0 @@
|
|||||||
/// <reference path="../jsx.d.ts" />
|
|
||||||
|
|
||||||
import { assertEquals } from 'https://deno.land/std/testing/asserts.ts';
|
|
||||||
import { delay } from 'https://deno.land/std/async/delay.ts';
|
|
||||||
import { React } from '../mod.ts';
|
|
||||||
|
|
||||||
const Title = async () => {
|
|
||||||
await delay(100);
|
|
||||||
return <h1>title{ await delay(100) }</h1>;
|
|
||||||
};
|
|
||||||
|
|
||||||
const View = () => (
|
|
||||||
<div>
|
|
||||||
<Title />
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
|
|
||||||
if (import.meta.main) {
|
|
||||||
(<View />).render().then(console.log);
|
|
||||||
} else {
|
|
||||||
// Run test
|
|
||||||
|
|
||||||
Deno.test('render title', async () => {
|
|
||||||
assertEquals(await (<Title />).render(), '<h1>title</h1>');
|
|
||||||
});
|
|
||||||
|
|
||||||
Deno.test('render view', async () => {
|
|
||||||
assertEquals(await (<View />).render(), '<div><h1>title</h1></div>');
|
|
||||||
});
|
|
||||||
}
|
|
@ -1,9 +0,0 @@
|
|||||||
/// <reference path="../jsx.d.ts" />
|
|
||||||
|
|
||||||
import { React } from '../mod.ts';
|
|
||||||
|
|
||||||
const View = () => <div>Hello</div>;
|
|
||||||
|
|
||||||
if (import.meta.main) {
|
|
||||||
(<View />).render().then(console.log);
|
|
||||||
}
|
|
@ -1,27 +0,0 @@
|
|||||||
/// <reference path="../jsx.d.ts" />
|
|
||||||
import { assertEquals } from 'https://deno.land/std/testing/asserts.ts';
|
|
||||||
|
|
||||||
import { React } from '../mod.ts';
|
|
||||||
|
|
||||||
const View = () => {
|
|
||||||
const techs = ['NodeJS', 'React Native', 'Next'];
|
|
||||||
|
|
||||||
return (
|
|
||||||
<ul>
|
|
||||||
{techs.map((tech: any) => (
|
|
||||||
<li>{tech}</li>
|
|
||||||
))}
|
|
||||||
</ul>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
if (import.meta.main) {
|
|
||||||
(<View />).render().then(console.log);
|
|
||||||
} else {
|
|
||||||
Deno.test('render with array', async () => {
|
|
||||||
assertEquals(
|
|
||||||
await (<View />).render(),
|
|
||||||
'<ul><li>NodeJS</li><li>React Native</li><li>Next</li></ul>',
|
|
||||||
);
|
|
||||||
});
|
|
||||||
}
|
|
@ -1,19 +0,0 @@
|
|||||||
/// <reference path="../jsx.d.ts" />
|
|
||||||
import { assertEquals } from 'https://deno.land/std/testing/asserts.ts';
|
|
||||||
|
|
||||||
import { React } from '../mod.ts';
|
|
||||||
|
|
||||||
const View = () => {
|
|
||||||
return <div innerHTML="<b>hello</b> world" />;
|
|
||||||
};
|
|
||||||
|
|
||||||
if (import.meta.main) {
|
|
||||||
(<View />).render().then(console.log);
|
|
||||||
} else {
|
|
||||||
Deno.test('render with array', async () => {
|
|
||||||
assertEquals(
|
|
||||||
await (<View />).render(),
|
|
||||||
'<div><b>hello</b> world</div>',
|
|
||||||
);
|
|
||||||
});
|
|
||||||
}
|
|
19
jsx-html/examples/browser/dist/index.html
vendored
19
jsx-html/examples/browser/dist/index.html
vendored
@ -1,19 +0,0 @@
|
|||||||
<!DOCTYPE html>
|
|
||||||
<html>
|
|
||||||
<head>
|
|
||||||
<title>Page Title</title>
|
|
||||||
</head>
|
|
||||||
<body>
|
|
||||||
|
|
||||||
<script src="index.js"></script>
|
|
||||||
|
|
||||||
<div id="div-container" />
|
|
||||||
|
|
||||||
<script>
|
|
||||||
demo.Test().render('#div-container').then((html) => {
|
|
||||||
document.getElementById('div-container').innerHTML = html;
|
|
||||||
});
|
|
||||||
</script>
|
|
||||||
|
|
||||||
</body>
|
|
||||||
</html>
|
|
502
jsx-html/examples/browser/dist/index.js
vendored
502
jsx-html/examples/browser/dist/index.js
vendored
File diff suppressed because one or more lines are too long
@ -1,25 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "jsx-to-html-testing",
|
|
||||||
"version": "1.0.0",
|
|
||||||
"description": "jsx-to-html-testing",
|
|
||||||
"main": "index.js",
|
|
||||||
"scripts": {
|
|
||||||
"test": "echo \"Error: no test specified\" && exit 1",
|
|
||||||
"watch": "webpack --watch",
|
|
||||||
"start": "webpack-dev-server --open",
|
|
||||||
"build": "webpack"
|
|
||||||
},
|
|
||||||
"keywords": [],
|
|
||||||
"author": "Alex",
|
|
||||||
"license": "ISC",
|
|
||||||
"devDependencies": {
|
|
||||||
"ts-loader": "^8.0.1",
|
|
||||||
"typescript": "^3.9.7",
|
|
||||||
"webpack": "^4.44.1",
|
|
||||||
"webpack-cli": "^3.3.12",
|
|
||||||
"webpack-dev-server": "^3.11.0"
|
|
||||||
},
|
|
||||||
"dependencies": {
|
|
||||||
"async-jsx-html": "^1.2.1"
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,5 +0,0 @@
|
|||||||
import { jsx, ElementNode } from 'async-jsx-html';
|
|
||||||
|
|
||||||
export function Test(): ElementNode {
|
|
||||||
return <div>Hello World</div>;
|
|
||||||
}
|
|
@ -1,15 +0,0 @@
|
|||||||
{
|
|
||||||
"compilerOptions": {
|
|
||||||
"outDir": "./dist/",
|
|
||||||
"module": "commonjs",
|
|
||||||
"jsx": "react",
|
|
||||||
"jsxFactory": "jsx",
|
|
||||||
"esModuleInterop": true,
|
|
||||||
"sourceMap": true,
|
|
||||||
"allowJs": true,
|
|
||||||
"lib": [
|
|
||||||
"es6",
|
|
||||||
"dom"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,33 +0,0 @@
|
|||||||
const path = require('path');
|
|
||||||
const MODULE_NAME = 'demo';
|
|
||||||
|
|
||||||
module.exports = {
|
|
||||||
mode: 'development',
|
|
||||||
entry: {
|
|
||||||
index: './src/index.tsx'
|
|
||||||
},
|
|
||||||
devtool: 'inline-source-map',
|
|
||||||
devServer: {
|
|
||||||
contentBase: './dist'
|
|
||||||
},
|
|
||||||
module: {
|
|
||||||
rules: [
|
|
||||||
{
|
|
||||||
test: /\.tsx?$/,
|
|
||||||
use: 'ts-loader',
|
|
||||||
exclude: /node_modules/,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
},
|
|
||||||
resolve: {
|
|
||||||
extensions: [ '.tsx', '.ts', '.js' ],
|
|
||||||
},
|
|
||||||
output: {
|
|
||||||
filename: '[name].js',
|
|
||||||
path: path.resolve(__dirname, 'dist'),
|
|
||||||
libraryTarget: 'umd',
|
|
||||||
globalObject: 'this',
|
|
||||||
// libraryExport: 'default',
|
|
||||||
library: MODULE_NAME
|
|
||||||
},
|
|
||||||
};
|
|
File diff suppressed because it is too large
Load Diff
@ -1,30 +0,0 @@
|
|||||||
let {
|
|
||||||
React,
|
|
||||||
Fragment
|
|
||||||
} = require('../nodejs/mod');
|
|
||||||
|
|
||||||
const Title = () => /*#__PURE__*/React.createElement("h1", null, "title");
|
|
||||||
|
|
||||||
const Value = ({
|
|
||||||
val
|
|
||||||
}) => /*#__PURE__*/React.createElement("p", null, "value: ", val);
|
|
||||||
|
|
||||||
const Numeric = ({
|
|
||||||
num
|
|
||||||
}) => /*#__PURE__*/React.createElement("p", null, "num: ", num);
|
|
||||||
|
|
||||||
const View = () => /*#__PURE__*/React.createElement("div", {
|
|
||||||
class: "deno"
|
|
||||||
}, /*#__PURE__*/React.createElement(Title, null), /*#__PURE__*/React.createElement("p", {
|
|
||||||
onclick: () => 'lol',
|
|
||||||
valid: true,
|
|
||||||
checked: true,
|
|
||||||
select: ""
|
|
||||||
}, "land"), /*#__PURE__*/React.createElement("br", null), /*#__PURE__*/React.createElement("hr", null), /*#__PURE__*/React.createElement(Fragment, null, /*#__PURE__*/React.createElement(Value, {
|
|
||||||
val: "hello"
|
|
||||||
}), /*#__PURE__*/React.createElement(Numeric, {
|
|
||||||
num: 23
|
|
||||||
})));
|
|
||||||
|
|
||||||
/*#__PURE__*/
|
|
||||||
React.createElement(View, null).render().then(console.log);
|
|
@ -1,22 +0,0 @@
|
|||||||
let { React, Fragment } = require('../nodejs/mod');
|
|
||||||
|
|
||||||
const Title = () => <h1>title</h1>;
|
|
||||||
const Value = ({ val }) => <p>value: {val}</p>;
|
|
||||||
const Numeric = ({ num }) => <p>num: {num}</p>;
|
|
||||||
|
|
||||||
const View = () => (
|
|
||||||
<div class="deno">
|
|
||||||
<Title />
|
|
||||||
<p onclick={() => 'lol'} valid checked={true} select="">
|
|
||||||
land
|
|
||||||
</p>
|
|
||||||
<br />
|
|
||||||
<hr />
|
|
||||||
<Fragment>
|
|
||||||
<Value val="hello" />
|
|
||||||
<Numeric num={23} />
|
|
||||||
</Fragment>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
|
|
||||||
(<View />).render().then(console.log);
|
|
5
jsx-html/jsx.d.ts
vendored
5
jsx-html/jsx.d.ts
vendored
@ -1,5 +0,0 @@
|
|||||||
declare namespace JSX {
|
|
||||||
interface IntrinsicElements {
|
|
||||||
[elemName: string]: any;
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,33 +0,0 @@
|
|||||||
import type {
|
|
||||||
NodePropsType,
|
|
||||||
ComponentFunctionType,
|
|
||||||
NullableChildType,
|
|
||||||
ChildType,
|
|
||||||
} from './types.ts';
|
|
||||||
|
|
||||||
import { ElementNode } from './node/ElementNode.ts';
|
|
||||||
import { ComponentNode } from './node/ComponentNode.ts';
|
|
||||||
|
|
||||||
export const jsx = <P extends NodePropsType = NodePropsType>(
|
|
||||||
element: string | ComponentFunctionType,
|
|
||||||
props: P | null,
|
|
||||||
...children: NullableChildType[]
|
|
||||||
) => {
|
|
||||||
const nodeProps = props || {};
|
|
||||||
|
|
||||||
if (typeof element === 'string') {
|
|
||||||
return new ElementNode(element, nodeProps, children);
|
|
||||||
}
|
|
||||||
if (typeof element === 'function') {
|
|
||||||
return new ComponentNode(element, nodeProps, children);
|
|
||||||
}
|
|
||||||
|
|
||||||
throw new TypeError(`Expected jsx element to be a string or a function`);
|
|
||||||
};
|
|
||||||
|
|
||||||
export const Fragment = (
|
|
||||||
props: NodePropsType,
|
|
||||||
children: ChildType,
|
|
||||||
): NullableChildType => {
|
|
||||||
return children;
|
|
||||||
};
|
|
@ -1,12 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "jsx-html",
|
|
||||||
"version": "0.0.1",
|
|
||||||
"description": "",
|
|
||||||
"author": "Fabian Stamm <dev@fabianstamm.de>",
|
|
||||||
"contributors": [],
|
|
||||||
"files": [
|
|
||||||
"**/*.ts",
|
|
||||||
"**/*.js",
|
|
||||||
"README.md"
|
|
||||||
]
|
|
||||||
}
|
|
@ -1,27 +0,0 @@
|
|||||||
import { jsx, Fragment } from './jsx.ts';
|
|
||||||
import type {
|
|
||||||
ComponentFunctionType,
|
|
||||||
NodePropsType,
|
|
||||||
NullableChildType,
|
|
||||||
} from './types.ts';
|
|
||||||
|
|
||||||
export { ElementNode } from './node/ElementNode.ts';
|
|
||||||
export { ComponentNode } from './node/ComponentNode.ts';
|
|
||||||
export type {
|
|
||||||
jsx,
|
|
||||||
Fragment,
|
|
||||||
ComponentFunctionType,
|
|
||||||
NullableChildType,
|
|
||||||
NodePropsType,
|
|
||||||
};
|
|
||||||
|
|
||||||
export const React = {
|
|
||||||
Fragment,
|
|
||||||
createElement<P extends NodePropsType = NodePropsType>(
|
|
||||||
element: string | ComponentFunctionType,
|
|
||||||
props: P | null,
|
|
||||||
...children: NullableChildType[]
|
|
||||||
) {
|
|
||||||
return jsx(element, props, ...children);
|
|
||||||
},
|
|
||||||
};
|
|
@ -1,37 +0,0 @@
|
|||||||
import type {
|
|
||||||
NodePropsType,
|
|
||||||
ComponentFunctionType,
|
|
||||||
NullableChildType,
|
|
||||||
} from '../types.ts';
|
|
||||||
import { NODE_TYPE } from '../constants.ts';
|
|
||||||
import { FragmentNode } from './FragmentNode.ts';
|
|
||||||
import { Node } from './Node.ts';
|
|
||||||
import { normalizeChildren } from './utils/normalizeChildren.ts';
|
|
||||||
|
|
||||||
export class ComponentNode extends Node {
|
|
||||||
type = NODE_TYPE.COMPONENT;
|
|
||||||
|
|
||||||
constructor(
|
|
||||||
public component: ComponentFunctionType,
|
|
||||||
public props: NodePropsType,
|
|
||||||
children: NullableChildType[],
|
|
||||||
) {
|
|
||||||
super(children);
|
|
||||||
}
|
|
||||||
|
|
||||||
async render(): Promise<string | any[]> {
|
|
||||||
return [].concat((await this.renderComponent()) as any).join('');
|
|
||||||
}
|
|
||||||
|
|
||||||
async renderComponent() {
|
|
||||||
const child = await this.component(this.props, this.children);
|
|
||||||
const children = normalizeChildren(
|
|
||||||
Array.isArray(child) ? child : [child],
|
|
||||||
);
|
|
||||||
if (children.length === 1) {
|
|
||||||
return children[0].render();
|
|
||||||
} else if (children.length > 1) {
|
|
||||||
return new FragmentNode(children).render();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,91 +0,0 @@
|
|||||||
import { NODE_TYPE } from '../constants.ts';
|
|
||||||
import type { NodePropsType, NullableChildType } from '../types.ts';
|
|
||||||
import { Node } from './Node.ts';
|
|
||||||
import { doubleQuoteEncode } from './utils/htmlEncode.ts';
|
|
||||||
|
|
||||||
const ELEMENT_PROP = {
|
|
||||||
INNER_HTML: 'innerHTML',
|
|
||||||
};
|
|
||||||
|
|
||||||
// List taken from http://w3c.github.io/html-reference/syntax.html
|
|
||||||
const VOID_ELEMENTS = new Set<string>([
|
|
||||||
'area',
|
|
||||||
'base',
|
|
||||||
'br',
|
|
||||||
'col',
|
|
||||||
'command',
|
|
||||||
'embed',
|
|
||||||
'hr',
|
|
||||||
'img',
|
|
||||||
'input',
|
|
||||||
'keygen',
|
|
||||||
'link',
|
|
||||||
'meta',
|
|
||||||
'param',
|
|
||||||
'source',
|
|
||||||
'track',
|
|
||||||
'wbr',
|
|
||||||
]);
|
|
||||||
|
|
||||||
export class ElementNode extends Node {
|
|
||||||
type = NODE_TYPE.ELEMENT;
|
|
||||||
|
|
||||||
constructor(
|
|
||||||
public name: string,
|
|
||||||
public props: NodePropsType,
|
|
||||||
children: NullableChildType[],
|
|
||||||
) {
|
|
||||||
super(children);
|
|
||||||
}
|
|
||||||
|
|
||||||
async render(): Promise<string | any[]> {
|
|
||||||
const renderedProps = this.propsToHTML();
|
|
||||||
|
|
||||||
const renderedChildren =
|
|
||||||
typeof this.props[ELEMENT_PROP.INNER_HTML] === 'string'
|
|
||||||
? this.props[ELEMENT_PROP.INNER_HTML]
|
|
||||||
: (await this.renderChildren()).join('');
|
|
||||||
|
|
||||||
return renderedChildren || !VOID_ELEMENTS.has(this.name)
|
|
||||||
? `<${this.name}${renderedProps}>${renderedChildren || ''}</${
|
|
||||||
this.name
|
|
||||||
}>`
|
|
||||||
: `<${this.name}${renderedProps} />`;
|
|
||||||
}
|
|
||||||
|
|
||||||
private getValidProps() {
|
|
||||||
const props = this.props;
|
|
||||||
return Object.keys(this.props).filter((key) => {
|
|
||||||
if (key === ELEMENT_PROP.INNER_HTML) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
const val = props[key];
|
|
||||||
return (
|
|
||||||
typeof val === 'string' ||
|
|
||||||
typeof val === 'number' ||
|
|
||||||
val === true
|
|
||||||
);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
private propsToHTML(): string {
|
|
||||||
const keys = this.getValidProps();
|
|
||||||
if (!keys.length) {
|
|
||||||
return '';
|
|
||||||
}
|
|
||||||
|
|
||||||
const props = this.props;
|
|
||||||
const pairs = keys.map((key) => {
|
|
||||||
if (!/^[a-zA-Z0-9-:\._]+$/.test(key)) {
|
|
||||||
throw new Error(`Invalid attribute name format ${key}`);
|
|
||||||
}
|
|
||||||
const val = props[key];
|
|
||||||
// https://html.spec.whatwg.org/multipage/dom.html#attributes
|
|
||||||
return val === true || val === ''
|
|
||||||
? key
|
|
||||||
: `${key}="${doubleQuoteEncode(val.toString())}"`;
|
|
||||||
});
|
|
||||||
|
|
||||||
return ` ${pairs.join(' ')}`;
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,15 +0,0 @@
|
|||||||
import { NODE_TYPE } from '../constants.ts';
|
|
||||||
import type { ChildNodeType } from '../types.ts';
|
|
||||||
import { Node } from './Node.ts';
|
|
||||||
|
|
||||||
export class FragmentNode extends Node {
|
|
||||||
type = NODE_TYPE.FRAGMENT;
|
|
||||||
|
|
||||||
constructor(children: ChildNodeType[]) {
|
|
||||||
super(children);
|
|
||||||
}
|
|
||||||
|
|
||||||
render() {
|
|
||||||
return this.renderChildren();
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,29 +0,0 @@
|
|||||||
import type { NODE_TYPE } from '../constants.ts';
|
|
||||||
import type { NullableChildType } from '../types.ts';
|
|
||||||
import { normalizeChildren } from './utils/normalizeChildren.ts';
|
|
||||||
|
|
||||||
export abstract class Node {
|
|
||||||
abstract type: NODE_TYPE;
|
|
||||||
|
|
||||||
constructor(public children: NullableChildType[]) {}
|
|
||||||
|
|
||||||
abstract async render(): Promise<string | any[]>;
|
|
||||||
|
|
||||||
async renderChildren() {
|
|
||||||
const result: string[] = [];
|
|
||||||
const children = normalizeChildren(this.children);
|
|
||||||
for (const child of children) {
|
|
||||||
const renderedChild = await child.render();
|
|
||||||
if (renderedChild) {
|
|
||||||
if (Array.isArray(renderedChild)) {
|
|
||||||
renderedChild.forEach(
|
|
||||||
(subchild) => subchild && result.push(subchild),
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
result.push(renderedChild);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,12 +0,0 @@
|
|||||||
import { NODE_TYPE } from '../constants.ts';
|
|
||||||
import { htmlEncode } from './utils/htmlEncode.ts';
|
|
||||||
|
|
||||||
export class TextNode {
|
|
||||||
type = NODE_TYPE.TEXT;
|
|
||||||
|
|
||||||
constructor(public text: string) {}
|
|
||||||
|
|
||||||
async render(): Promise<string | any[]> {
|
|
||||||
return htmlEncode(this.text);
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,13 +0,0 @@
|
|||||||
export function doubleQuoteEncode(text: string): string {
|
|
||||||
return text
|
|
||||||
.replace(/"/g, '"')
|
|
||||||
}
|
|
||||||
|
|
||||||
export function htmlEncode(text: string): string {
|
|
||||||
return doubleQuoteEncode(text
|
|
||||||
.replace(/&/g, '&')
|
|
||||||
.replace(/\//g, '/')
|
|
||||||
.replace(/</g, '<')
|
|
||||||
.replace(/>/g, '>')
|
|
||||||
.replace(/'/g, '''));
|
|
||||||
}
|
|
@ -1,31 +0,0 @@
|
|||||||
import type { NullableChildType, ChildNodeType } from '../../types.ts';
|
|
||||||
import { TextNode } from '../TextNode.ts';
|
|
||||||
import { NODE_TYPE } from '../../constants.ts';
|
|
||||||
|
|
||||||
export function normalizeChildren(
|
|
||||||
children: NullableChildType[],
|
|
||||||
): ChildNodeType[] {
|
|
||||||
const result: any[] = [];
|
|
||||||
|
|
||||||
for (const child of children) {
|
|
||||||
if (child && typeof child !== 'boolean') {
|
|
||||||
if (typeof child === 'string' || typeof child === 'number') {
|
|
||||||
result.push(new TextNode(`${child}`));
|
|
||||||
} else if (Array.isArray(child)) {
|
|
||||||
normalizeChildren(child).forEach((normalized) =>
|
|
||||||
result.push(normalized),
|
|
||||||
);
|
|
||||||
} else if (
|
|
||||||
child.type === NODE_TYPE.ELEMENT ||
|
|
||||||
child.type === NODE_TYPE.TEXT ||
|
|
||||||
child.type === NODE_TYPE.COMPONENT
|
|
||||||
) {
|
|
||||||
result.push(child);
|
|
||||||
} else {
|
|
||||||
throw new TypeError(`Unrecognized node type: ${typeof child}`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
|
6
jsx-html/nodejs/constants.d.ts
vendored
6
jsx-html/nodejs/constants.d.ts
vendored
@ -1,6 +0,0 @@
|
|||||||
export declare enum NODE_TYPE {
|
|
||||||
ELEMENT = "element",
|
|
||||||
TEXT = "text",
|
|
||||||
COMPONENT = "component",
|
|
||||||
FRAGMENT = "fragment"
|
|
||||||
}
|
|
@ -1,12 +0,0 @@
|
|||||||
"use strict";
|
|
||||||
Object.defineProperty(exports, "__esModule", { value: true });
|
|
||||||
exports.NODE_TYPE = void 0;
|
|
||||||
var NODE_TYPE;
|
|
||||||
(function (NODE_TYPE) {
|
|
||||||
NODE_TYPE["ELEMENT"] = "element";
|
|
||||||
NODE_TYPE["TEXT"] = "text";
|
|
||||||
NODE_TYPE["COMPONENT"] = "component";
|
|
||||||
NODE_TYPE["FRAGMENT"] = "fragment";
|
|
||||||
})(NODE_TYPE = exports.NODE_TYPE || (exports.NODE_TYPE = {}));
|
|
||||||
;
|
|
||||||
//# sourceMappingURL=constants.js.map
|
|
@ -1 +0,0 @@
|
|||||||
{"version":3,"file":"constants.js","sourceRoot":"","sources":["../tmp/constants.ts"],"names":[],"mappings":";;;AAAA,IAAY,SAKX;AALD,WAAY,SAAS;IACnB,gCAAmB,CAAA;IACnB,0BAAa,CAAA;IACb,oCAAuB,CAAA;IACvB,kCAAqB,CAAA;AACvB,CAAC,EALW,SAAS,GAAT,iBAAS,KAAT,iBAAS,QAKpB;AACD,CAAC"}
|
|
5
jsx-html/nodejs/jsx.d.ts
vendored
5
jsx-html/nodejs/jsx.d.ts
vendored
@ -1,5 +0,0 @@
|
|||||||
import { NodePropsType, ComponentFunctionType, NullableChildType, ChildType } from "./types";
|
|
||||||
import { ElementNode } from "./node/ElementNode";
|
|
||||||
import { ComponentNode } from "./node/ComponentNode";
|
|
||||||
export declare const jsx: <P extends NodePropsType = NodePropsType>(element: string | ComponentFunctionType, props: P, ...children: NullableChildType[]) => ElementNode | ComponentNode;
|
|
||||||
export declare const Fragment: (props: NodePropsType, children: ChildType) => NullableChildType;
|
|
@ -1,19 +0,0 @@
|
|||||||
"use strict";
|
|
||||||
Object.defineProperty(exports, "__esModule", { value: true });
|
|
||||||
exports.Fragment = exports.jsx = void 0;
|
|
||||||
const ElementNode_1 = require("./node/ElementNode");
|
|
||||||
const ComponentNode_1 = require("./node/ComponentNode");
|
|
||||||
exports.jsx = (element, props, ...children) => {
|
|
||||||
const nodeProps = props || {};
|
|
||||||
if (typeof element === 'string') {
|
|
||||||
return new ElementNode_1.ElementNode(element, nodeProps, children);
|
|
||||||
}
|
|
||||||
if (typeof element === 'function') {
|
|
||||||
return new ComponentNode_1.ComponentNode(element, nodeProps, children);
|
|
||||||
}
|
|
||||||
throw new TypeError(`Expected jsx element to be a string or a function`);
|
|
||||||
};
|
|
||||||
exports.Fragment = (props, children) => {
|
|
||||||
return children;
|
|
||||||
};
|
|
||||||
//# sourceMappingURL=jsx.js.map
|
|
@ -1 +0,0 @@
|
|||||||
{"version":3,"file":"jsx.js","sourceRoot":"","sources":["../tmp/jsx.ts"],"names":[],"mappings":";;;AACA,oDAAiD;AACjD,wDAAqD;AACxC,QAAA,GAAG,GAAG,CAA0C,OAAuC,EAAE,KAAe,EAAE,GAAG,QAA6B,EAAE,EAAE;IACzJ,MAAM,SAAS,GAAG,KAAK,IAAI,EAAE,CAAC;IAE9B,IAAI,OAAO,OAAO,KAAK,QAAQ,EAAE;QAC/B,OAAO,IAAI,yBAAW,CAAC,OAAO,EAAE,SAAS,EAAE,QAAQ,CAAC,CAAC;KACtD;IAED,IAAI,OAAO,OAAO,KAAK,UAAU,EAAE;QACjC,OAAO,IAAI,6BAAa,CAAC,OAAO,EAAE,SAAS,EAAE,QAAQ,CAAC,CAAC;KACxD;IAED,MAAM,IAAI,SAAS,CAAC,mDAAmD,CAAC,CAAC;AAC3E,CAAC,CAAC;AACW,QAAA,QAAQ,GAAG,CAAC,KAAoB,EAAE,QAAmB,EAAqB,EAAE;IACvF,OAAO,QAAQ,CAAC;AAClB,CAAC,CAAC"}
|
|
9
jsx-html/nodejs/mod.d.ts
vendored
9
jsx-html/nodejs/mod.d.ts
vendored
@ -1,9 +0,0 @@
|
|||||||
import { jsx, Fragment } from "./jsx";
|
|
||||||
import { ComponentFunctionType, NodePropsType, NullableChildType } from "./types";
|
|
||||||
export { ElementNode } from "./node/ElementNode";
|
|
||||||
export { ComponentNode } from "./node/ComponentNode";
|
|
||||||
export { jsx, Fragment, ComponentFunctionType, NullableChildType, NodePropsType };
|
|
||||||
export declare const React: {
|
|
||||||
Fragment: (props: NodePropsType, children: import("./types").ChildType) => NullableChildType;
|
|
||||||
createElement<P extends NodePropsType = NodePropsType>(element: string | ComponentFunctionType, props: P, ...children: NullableChildType[]): import("./node/ElementNode").ElementNode | import("./node/ComponentNode").ComponentNode;
|
|
||||||
};
|
|
@ -1,17 +0,0 @@
|
|||||||
"use strict";
|
|
||||||
Object.defineProperty(exports, "__esModule", { value: true });
|
|
||||||
exports.React = exports.Fragment = exports.jsx = void 0;
|
|
||||||
const jsx_1 = require("./jsx");
|
|
||||||
Object.defineProperty(exports, "jsx", { enumerable: true, get: function () { return jsx_1.jsx; } });
|
|
||||||
Object.defineProperty(exports, "Fragment", { enumerable: true, get: function () { return jsx_1.Fragment; } });
|
|
||||||
var ElementNode_1 = require("./node/ElementNode");
|
|
||||||
Object.defineProperty(exports, "ElementNode", { enumerable: true, get: function () { return ElementNode_1.ElementNode; } });
|
|
||||||
var ComponentNode_1 = require("./node/ComponentNode");
|
|
||||||
Object.defineProperty(exports, "ComponentNode", { enumerable: true, get: function () { return ComponentNode_1.ComponentNode; } });
|
|
||||||
exports.React = {
|
|
||||||
Fragment: jsx_1.Fragment,
|
|
||||||
createElement(element, props, ...children) {
|
|
||||||
return jsx_1.jsx(element, props, ...children);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
//# sourceMappingURL=mod.js.map
|
|
@ -1 +0,0 @@
|
|||||||
{"version":3,"file":"mod.js","sourceRoot":"","sources":["../tmp/mod.ts"],"names":[],"mappings":";;;AAAA,+BAAsC;AAI7B,oFAJA,SAAG,OAIA;AAAE,yFAJA,cAAQ,OAIA;AAFtB,kDAAiD;AAAxC,0GAAA,WAAW,OAAA;AACpB,sDAAqD;AAA5C,8GAAA,aAAa,OAAA;AAET,QAAA,KAAK,GAAG;IACnB,QAAQ,EAAR,cAAQ;IAER,aAAa,CAA0C,OAAuC,EAAE,KAAe,EAAE,GAAG,QAA6B;QAC/I,OAAO,SAAG,CAAC,OAAO,EAAE,KAAK,EAAE,GAAG,QAAQ,CAAC,CAAC;IAC1C,CAAC;CAEF,CAAC"}
|
|
11
jsx-html/nodejs/node/ComponentNode.d.ts
vendored
11
jsx-html/nodejs/node/ComponentNode.d.ts
vendored
@ -1,11 +0,0 @@
|
|||||||
import { NodePropsType, ComponentFunctionType, NullableChildType } from "../types";
|
|
||||||
import { NODE_TYPE } from "../constants";
|
|
||||||
import { Node } from "./Node";
|
|
||||||
export declare class ComponentNode extends Node {
|
|
||||||
component: ComponentFunctionType;
|
|
||||||
props: NodePropsType;
|
|
||||||
type: NODE_TYPE;
|
|
||||||
constructor(component: ComponentFunctionType, props: NodePropsType, children: NullableChildType[]);
|
|
||||||
render(): Promise<string | any[]>;
|
|
||||||
renderComponent(): Promise<string | any[]>;
|
|
||||||
}
|
|
@ -1,43 +0,0 @@
|
|||||||
"use strict";
|
|
||||||
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
|
|
||||||
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
|
||||||
return new (P || (P = Promise))(function (resolve, reject) {
|
|
||||||
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
||||||
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
||||||
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
|
||||||
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
||||||
});
|
|
||||||
};
|
|
||||||
Object.defineProperty(exports, "__esModule", { value: true });
|
|
||||||
exports.ComponentNode = void 0;
|
|
||||||
const constants_1 = require("../constants");
|
|
||||||
const FragmentNode_1 = require("./FragmentNode");
|
|
||||||
const Node_1 = require("./Node");
|
|
||||||
const normalizeChildren_1 = require("./utils/normalizeChildren");
|
|
||||||
class ComponentNode extends Node_1.Node {
|
|
||||||
constructor(component, props, children) {
|
|
||||||
super(children);
|
|
||||||
this.component = component;
|
|
||||||
this.props = props;
|
|
||||||
this.type = constants_1.NODE_TYPE.COMPONENT;
|
|
||||||
}
|
|
||||||
render() {
|
|
||||||
return __awaiter(this, void 0, void 0, function* () {
|
|
||||||
return [].concat(yield this.renderComponent()).join('');
|
|
||||||
});
|
|
||||||
}
|
|
||||||
renderComponent() {
|
|
||||||
return __awaiter(this, void 0, void 0, function* () {
|
|
||||||
const child = yield this.component(this.props, this.children);
|
|
||||||
const children = normalizeChildren_1.normalizeChildren(Array.isArray(child) ? child : [child]);
|
|
||||||
if (children.length === 1) {
|
|
||||||
return children[0].render();
|
|
||||||
}
|
|
||||||
else if (children.length > 1) {
|
|
||||||
return new FragmentNode_1.FragmentNode(children).render();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
exports.ComponentNode = ComponentNode;
|
|
||||||
//# sourceMappingURL=ComponentNode.js.map
|
|
@ -1 +0,0 @@
|
|||||||
{"version":3,"file":"ComponentNode.js","sourceRoot":"","sources":["../../tmp/node/ComponentNode.ts"],"names":[],"mappings":";;;;;;;;;;;;AACA,4CAAyC;AACzC,iDAA8C;AAC9C,iCAA8B;AAC9B,iEAA8D;AAC9D,MAAa,aAAc,SAAQ,WAAI;IAGrC,YAAmB,SAAgC,EAAS,KAAoB,EAAE,QAA6B;QAC7G,KAAK,CAAC,QAAQ,CAAC,CAAC;QADC,cAAS,GAAT,SAAS,CAAuB;QAAS,UAAK,GAAL,KAAK,CAAe;QAFhF,SAAI,GAAG,qBAAS,CAAC,SAAS,CAAC;IAI3B,CAAC;IAEK,MAAM;;YACV,OAAO,EAAE,CAAC,MAAM,CAAE,MAAM,IAAI,CAAC,eAAe,EAAU,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACnE,CAAC;KAAA;IAEK,eAAe;;YACnB,MAAM,KAAK,GAAG,MAAM,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,KAAK,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC;YAC9D,MAAM,QAAQ,GAAG,qCAAiB,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC;YAE3E,IAAI,QAAQ,CAAC,MAAM,KAAK,CAAC,EAAE;gBACzB,OAAO,QAAQ,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC;aAC7B;iBAAM,IAAI,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE;gBAC9B,OAAO,IAAI,2BAAY,CAAC,QAAQ,CAAC,CAAC,MAAM,EAAE,CAAC;aAC5C;QACH,CAAC;KAAA;CAEF;AAtBD,sCAsBC"}
|
|
12
jsx-html/nodejs/node/ElementNode.d.ts
vendored
12
jsx-html/nodejs/node/ElementNode.d.ts
vendored
@ -1,12 +0,0 @@
|
|||||||
import { NODE_TYPE } from "../constants";
|
|
||||||
import { NodePropsType, NullableChildType } from "../types";
|
|
||||||
import { Node } from "./Node";
|
|
||||||
export declare class ElementNode extends Node {
|
|
||||||
name: string;
|
|
||||||
props: NodePropsType;
|
|
||||||
type: NODE_TYPE;
|
|
||||||
constructor(name: string, props: NodePropsType, children: NullableChildType[]);
|
|
||||||
render(): Promise<string | any[]>;
|
|
||||||
private getValidProps;
|
|
||||||
private propsToHTML;
|
|
||||||
}
|
|
@ -1,60 +0,0 @@
|
|||||||
"use strict";
|
|
||||||
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
|
|
||||||
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
|
||||||
return new (P || (P = Promise))(function (resolve, reject) {
|
|
||||||
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
||||||
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
||||||
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
|
||||||
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
||||||
});
|
|
||||||
};
|
|
||||||
Object.defineProperty(exports, "__esModule", { value: true });
|
|
||||||
exports.ElementNode = void 0;
|
|
||||||
const constants_1 = require("../constants");
|
|
||||||
const Node_1 = require("./Node");
|
|
||||||
const htmlEncode_1 = require("./utils/htmlEncode");
|
|
||||||
const ELEMENT_PROP = {
|
|
||||||
INNER_HTML: 'innerHTML'
|
|
||||||
};
|
|
||||||
class ElementNode extends Node_1.Node {
|
|
||||||
constructor(name, props, children) {
|
|
||||||
super(children);
|
|
||||||
this.name = name;
|
|
||||||
this.props = props;
|
|
||||||
this.type = constants_1.NODE_TYPE.ELEMENT;
|
|
||||||
}
|
|
||||||
render() {
|
|
||||||
return __awaiter(this, void 0, void 0, function* () {
|
|
||||||
const renderedProps = this.propsToHTML();
|
|
||||||
const renderedChildren = typeof this.props[ELEMENT_PROP.INNER_HTML] === 'string' ? this.props[ELEMENT_PROP.INNER_HTML] : (yield this.renderChildren()).join('');
|
|
||||||
return renderedChildren ? `<${this.name}${renderedProps}>${renderedChildren}</${this.name}>` : `<${this.name}${renderedProps} />`;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
getValidProps() {
|
|
||||||
const props = this.props;
|
|
||||||
return Object.keys(this.props).filter(key => {
|
|
||||||
if (key === ELEMENT_PROP.INNER_HTML) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
const val = props[key];
|
|
||||||
return typeof val === 'string' || typeof val === 'number' || val === true;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
propsToHTML() {
|
|
||||||
const keys = this.getValidProps();
|
|
||||||
if (!keys.length) {
|
|
||||||
return '';
|
|
||||||
}
|
|
||||||
const props = this.props;
|
|
||||||
const pairs = keys.map(key => {
|
|
||||||
if (!/^[a-zA-Z0-9-:\._]+$/.test(key)) {
|
|
||||||
throw new Error(`Invalid attribute name format ${key}`);
|
|
||||||
}
|
|
||||||
const val = props[key];
|
|
||||||
return val === true || val === '' ? key : `${key}="${htmlEncode_1.doubleQuoteEncode(val.toString())}"`;
|
|
||||||
});
|
|
||||||
return ` ${pairs.join(' ')}`;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
exports.ElementNode = ElementNode;
|
|
||||||
//# sourceMappingURL=ElementNode.js.map
|
|
@ -1 +0,0 @@
|
|||||||
{"version":3,"file":"ElementNode.js","sourceRoot":"","sources":["../../tmp/node/ElementNode.ts"],"names":[],"mappings":";;;;;;;;;;;;AAAA,4CAAyC;AAEzC,iCAA8B;AAC9B,mDAAuD;AACvD,MAAM,YAAY,GAAG;IACnB,UAAU,EAAE,WAAW;CACxB,CAAC;AACF,MAAa,WAAY,SAAQ,WAAI;IAGnC,YAAmB,IAAY,EAAS,KAAoB,EAAE,QAA6B;QACzF,KAAK,CAAC,QAAQ,CAAC,CAAC;QADC,SAAI,GAAJ,IAAI,CAAQ;QAAS,UAAK,GAAL,KAAK,CAAe;QAF5D,SAAI,GAAG,qBAAS,CAAC,OAAO,CAAC;IAIzB,CAAC;IAEK,MAAM;;YACV,MAAM,aAAa,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;YACzC,MAAM,gBAAgB,GAAG,OAAO,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,UAAU,CAAC,KAAK,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,CAAC,MAAM,IAAI,CAAC,cAAc,EAAE,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YAChK,OAAO,gBAAgB,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC,IAAI,GAAG,aAAa,IAAI,gBAAgB,KAAK,IAAI,CAAC,IAAI,GAAG,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC,IAAI,GAAG,aAAa,KAAK,CAAC;QACpI,CAAC;KAAA;IAEO,aAAa;QACnB,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC;QACzB,OAAO,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE;YAC1C,IAAI,GAAG,KAAK,YAAY,CAAC,UAAU,EAAE;gBACnC,OAAO,KAAK,CAAC;aACd;YAED,MAAM,GAAG,GAAG,KAAK,CAAC,GAAG,CAAC,CAAC;YACvB,OAAO,OAAO,GAAG,KAAK,QAAQ,IAAI,OAAO,GAAG,KAAK,QAAQ,IAAI,GAAG,KAAK,IAAI,CAAC;QAC5E,CAAC,CAAC,CAAC;IACL,CAAC;IAEO,WAAW;QACjB,MAAM,IAAI,GAAG,IAAI,CAAC,aAAa,EAAE,CAAC;QAElC,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE;YAChB,OAAO,EAAE,CAAC;SACX;QAED,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC;QACzB,MAAM,KAAK,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE;YAC3B,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE;gBACpC,MAAM,IAAI,KAAK,CAAC,iCAAiC,GAAG,EAAE,CAAC,CAAC;aACzD;YAED,MAAM,GAAG,GAAG,KAAK,CAAC,GAAG,CAAC,CAAC;YAEvB,OAAO,GAAG,KAAK,IAAI,IAAI,GAAG,KAAK,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,KAAK,8BAAiB,CAAC,GAAG,CAAC,QAAQ,EAAE,CAAC,GAAG,CAAC;QAC5F,CAAC,CAAC,CAAC;QACH,OAAO,IAAI,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC;IAC/B,CAAC;CAEF;AA7CD,kCA6CC"}
|
|
8
jsx-html/nodejs/node/FragmentNode.d.ts
vendored
8
jsx-html/nodejs/node/FragmentNode.d.ts
vendored
@ -1,8 +0,0 @@
|
|||||||
import { NODE_TYPE } from "../constants";
|
|
||||||
import { ChildNodeType } from "../types";
|
|
||||||
import { Node } from "./Node";
|
|
||||||
export declare class FragmentNode extends Node {
|
|
||||||
type: NODE_TYPE;
|
|
||||||
constructor(children: ChildNodeType[]);
|
|
||||||
render(): Promise<string[]>;
|
|
||||||
}
|
|
@ -1,16 +0,0 @@
|
|||||||
"use strict";
|
|
||||||
Object.defineProperty(exports, "__esModule", { value: true });
|
|
||||||
exports.FragmentNode = void 0;
|
|
||||||
const constants_1 = require("../constants");
|
|
||||||
const Node_1 = require("./Node");
|
|
||||||
class FragmentNode extends Node_1.Node {
|
|
||||||
constructor(children) {
|
|
||||||
super(children);
|
|
||||||
this.type = constants_1.NODE_TYPE.FRAGMENT;
|
|
||||||
}
|
|
||||||
render() {
|
|
||||||
return this.renderChildren();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
exports.FragmentNode = FragmentNode;
|
|
||||||
//# sourceMappingURL=FragmentNode.js.map
|
|
@ -1 +0,0 @@
|
|||||||
{"version":3,"file":"FragmentNode.js","sourceRoot":"","sources":["../../tmp/node/FragmentNode.ts"],"names":[],"mappings":";;;AAAA,4CAAyC;AAEzC,iCAA8B;AAC9B,MAAa,YAAa,SAAQ,WAAI;IAGpC,YAAY,QAAyB;QACnC,KAAK,CAAC,QAAQ,CAAC,CAAC;QAHlB,SAAI,GAAG,qBAAS,CAAC,QAAQ,CAAC;IAI1B,CAAC;IAED,MAAM;QACJ,OAAO,IAAI,CAAC,cAAc,EAAE,CAAC;IAC/B,CAAC;CAEF;AAXD,oCAWC"}
|
|
9
jsx-html/nodejs/node/Node.d.ts
vendored
9
jsx-html/nodejs/node/Node.d.ts
vendored
@ -1,9 +0,0 @@
|
|||||||
import { NODE_TYPE } from "../constants";
|
|
||||||
import { NullableChildType } from "../types";
|
|
||||||
export declare abstract class Node {
|
|
||||||
children: NullableChildType[];
|
|
||||||
abstract type: NODE_TYPE;
|
|
||||||
constructor(children: NullableChildType[]);
|
|
||||||
abstract render(): Promise<string | any[]>;
|
|
||||||
renderChildren(): Promise<string[]>;
|
|
||||||
}
|
|
@ -1,38 +0,0 @@
|
|||||||
"use strict";
|
|
||||||
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
|
|
||||||
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
|
||||||
return new (P || (P = Promise))(function (resolve, reject) {
|
|
||||||
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
||||||
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
||||||
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
|
||||||
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
||||||
});
|
|
||||||
};
|
|
||||||
Object.defineProperty(exports, "__esModule", { value: true });
|
|
||||||
exports.Node = void 0;
|
|
||||||
const normalizeChildren_1 = require("./utils/normalizeChildren");
|
|
||||||
class Node {
|
|
||||||
constructor(children) {
|
|
||||||
this.children = children;
|
|
||||||
}
|
|
||||||
renderChildren() {
|
|
||||||
return __awaiter(this, void 0, void 0, function* () {
|
|
||||||
const result = [];
|
|
||||||
const children = normalizeChildren_1.normalizeChildren(this.children);
|
|
||||||
for (const child of children) {
|
|
||||||
const renderedChild = yield child.render();
|
|
||||||
if (renderedChild) {
|
|
||||||
if (Array.isArray(renderedChild)) {
|
|
||||||
renderedChild.forEach(subchild => subchild && result.push(subchild));
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
result.push(renderedChild);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
exports.Node = Node;
|
|
||||||
//# sourceMappingURL=Node.js.map
|
|
@ -1 +0,0 @@
|
|||||||
{"version":3,"file":"Node.js","sourceRoot":"","sources":["../../tmp/node/Node.ts"],"names":[],"mappings":";;;;;;;;;;;;AAEA,iEAA8D;AAC9D,MAAsB,IAAI;IAGxB,YAAmB,QAA6B;QAA7B,aAAQ,GAAR,QAAQ,CAAqB;IAAG,CAAC;IAI9C,cAAc;;YAClB,MAAM,MAAM,GAAa,EAAE,CAAC;YAC5B,MAAM,QAAQ,GAAG,qCAAiB,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;YAElD,KAAK,MAAM,KAAK,IAAI,QAAQ,EAAE;gBAC5B,MAAM,aAAa,GAAG,MAAM,KAAK,CAAC,MAAM,EAAE,CAAC;gBAE3C,IAAI,aAAa,EAAE;oBACjB,IAAI,KAAK,CAAC,OAAO,CAAC,aAAa,CAAC,EAAE;wBAChC,aAAa,CAAC,OAAO,CAAC,QAAQ,CAAC,EAAE,CAAC,QAAQ,IAAI,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC;qBACtE;yBAAM;wBACL,MAAM,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;qBAC5B;iBACF;aACF;YAED,OAAO,MAAM,CAAC;QAChB,CAAC;KAAA;CAEF;AA1BD,oBA0BC"}
|
|
7
jsx-html/nodejs/node/TextNode.d.ts
vendored
7
jsx-html/nodejs/node/TextNode.d.ts
vendored
@ -1,7 +0,0 @@
|
|||||||
import { NODE_TYPE } from "../constants";
|
|
||||||
export declare class TextNode {
|
|
||||||
text: string;
|
|
||||||
type: NODE_TYPE;
|
|
||||||
constructor(text: string);
|
|
||||||
render(): Promise<string | any[]>;
|
|
||||||
}
|
|
@ -1,27 +0,0 @@
|
|||||||
"use strict";
|
|
||||||
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
|
|
||||||
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
|
||||||
return new (P || (P = Promise))(function (resolve, reject) {
|
|
||||||
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
||||||
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
||||||
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
|
||||||
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
||||||
});
|
|
||||||
};
|
|
||||||
Object.defineProperty(exports, "__esModule", { value: true });
|
|
||||||
exports.TextNode = void 0;
|
|
||||||
const constants_1 = require("../constants");
|
|
||||||
const htmlEncode_1 = require("./utils/htmlEncode");
|
|
||||||
class TextNode {
|
|
||||||
constructor(text) {
|
|
||||||
this.text = text;
|
|
||||||
this.type = constants_1.NODE_TYPE.TEXT;
|
|
||||||
}
|
|
||||||
render() {
|
|
||||||
return __awaiter(this, void 0, void 0, function* () {
|
|
||||||
return htmlEncode_1.htmlEncode(this.text);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
exports.TextNode = TextNode;
|
|
||||||
//# sourceMappingURL=TextNode.js.map
|
|
@ -1 +0,0 @@
|
|||||||
{"version":3,"file":"TextNode.js","sourceRoot":"","sources":["../../tmp/node/TextNode.ts"],"names":[],"mappings":";;;;;;;;;;;;AAAA,4CAAyC;AACzC,mDAAgD;AAChD,MAAa,QAAQ;IAGnB,YAAmB,IAAY;QAAZ,SAAI,GAAJ,IAAI,CAAQ;QAF/B,SAAI,GAAG,qBAAS,CAAC,IAAI,CAAC;IAEY,CAAC;IAE7B,MAAM;;YACV,OAAO,uBAAU,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAC/B,CAAC;KAAA;CAEF;AATD,4BASC"}
|
|
2
jsx-html/nodejs/node/utils/htmlEncode.d.ts
vendored
2
jsx-html/nodejs/node/utils/htmlEncode.d.ts
vendored
@ -1,2 +0,0 @@
|
|||||||
export declare function doubleQuoteEncode(text: string): string;
|
|
||||||
export declare function htmlEncode(text: string): string;
|
|
@ -1,12 +0,0 @@
|
|||||||
"use strict";
|
|
||||||
Object.defineProperty(exports, "__esModule", { value: true });
|
|
||||||
exports.htmlEncode = exports.doubleQuoteEncode = void 0;
|
|
||||||
function doubleQuoteEncode(text) {
|
|
||||||
return text.replace(/"/g, '"');
|
|
||||||
}
|
|
||||||
exports.doubleQuoteEncode = doubleQuoteEncode;
|
|
||||||
function htmlEncode(text) {
|
|
||||||
return doubleQuoteEncode(text.replace(/&/g, '&').replace(/\//g, '/').replace(/</g, '<').replace(/>/g, '>').replace(/'/g, '''));
|
|
||||||
}
|
|
||||||
exports.htmlEncode = htmlEncode;
|
|
||||||
//# sourceMappingURL=htmlEncode.js.map
|
|
@ -1 +0,0 @@
|
|||||||
{"version":3,"file":"htmlEncode.js","sourceRoot":"","sources":["../../../tmp/node/utils/htmlEncode.ts"],"names":[],"mappings":";;;AAAA,SAAgB,iBAAiB,CAAC,IAAY;IAC5C,OAAO,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,QAAQ,CAAC,CAAC;AACtC,CAAC;AAFD,8CAEC;AACD,SAAgB,UAAU,CAAC,IAAY;IACrC,OAAO,iBAAiB,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC,OAAO,CAAC,KAAK,EAAE,QAAQ,CAAC,CAAC,OAAO,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC,OAAO,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC,OAAO,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC,CAAC;AACpJ,CAAC;AAFD,gCAEC"}
|
|
@ -1,2 +0,0 @@
|
|||||||
import { NullableChildType, ChildNodeType } from "../../types";
|
|
||||||
export declare function normalizeChildren(children: NullableChildType[]): ChildNodeType[];
|
|
@ -1,27 +0,0 @@
|
|||||||
"use strict";
|
|
||||||
Object.defineProperty(exports, "__esModule", { value: true });
|
|
||||||
exports.normalizeChildren = void 0;
|
|
||||||
const TextNode_1 = require("../TextNode");
|
|
||||||
const constants_1 = require("../../constants");
|
|
||||||
function normalizeChildren(children) {
|
|
||||||
const result = [];
|
|
||||||
for (const child of children) {
|
|
||||||
if (child && typeof child !== 'boolean') {
|
|
||||||
if (typeof child === 'string' || typeof child === 'number') {
|
|
||||||
result.push(new TextNode_1.TextNode(`${child}`));
|
|
||||||
}
|
|
||||||
else if (Array.isArray(child)) {
|
|
||||||
normalizeChildren(child).forEach(result.push);
|
|
||||||
}
|
|
||||||
else if (child.type === constants_1.NODE_TYPE.ELEMENT || child.type === constants_1.NODE_TYPE.TEXT || child.type === constants_1.NODE_TYPE.COMPONENT) {
|
|
||||||
result.push(child);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
throw new TypeError(`Unrecognized node type: ${typeof child}`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
exports.normalizeChildren = normalizeChildren;
|
|
||||||
//# sourceMappingURL=normalizeChildren.js.map
|
|
@ -1 +0,0 @@
|
|||||||
{"version":3,"file":"normalizeChildren.js","sourceRoot":"","sources":["../../../tmp/node/utils/normalizeChildren.ts"],"names":[],"mappings":";;;AACA,0CAAuC;AACvC,+CAA4C;AAC5C,SAAgB,iBAAiB,CAAC,QAA6B;IAC7D,MAAM,MAAM,GAAU,EAAE,CAAC;IAEzB,KAAK,MAAM,KAAK,IAAI,QAAQ,EAAE;QAC5B,IAAI,KAAK,IAAI,OAAO,KAAK,KAAK,SAAS,EAAE;YACvC,IAAI,OAAO,KAAK,KAAK,QAAQ,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE;gBAC1D,MAAM,CAAC,IAAI,CAAC,IAAI,mBAAQ,CAAC,GAAG,KAAK,EAAE,CAAC,CAAC,CAAC;aACvC;iBAAM,IAAI,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE;gBAC/B,iBAAiB,CAAC,KAAK,CAAC,CAAC,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;aAC/C;iBAAM,IAAI,KAAK,CAAC,IAAI,KAAK,qBAAS,CAAC,OAAO,IAAI,KAAK,CAAC,IAAI,KAAK,qBAAS,CAAC,IAAI,IAAI,KAAK,CAAC,IAAI,KAAK,qBAAS,CAAC,SAAS,EAAE;gBAClH,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;aACpB;iBAAM;gBACL,MAAM,IAAI,SAAS,CAAC,2BAA2B,OAAO,KAAK,EAAE,CAAC,CAAC;aAChE;SACF;KACF;IAED,OAAO,MAAM,CAAC;AAChB,CAAC;AAlBD,8CAkBC"}
|
|
15
jsx-html/nodejs/types.d.ts
vendored
15
jsx-html/nodejs/types.d.ts
vendored
@ -1,15 +0,0 @@
|
|||||||
import { ElementNode } from "./node/ElementNode";
|
|
||||||
import { TextNode } from "./node/TextNode";
|
|
||||||
import { ComponentNode } from "./node/ComponentNode";
|
|
||||||
import { FragmentNode } from "./node/FragmentNode";
|
|
||||||
export declare type NodePropsType = {
|
|
||||||
[key: string]: any;
|
|
||||||
};
|
|
||||||
declare type Primitive = string | boolean | number;
|
|
||||||
declare type NullablePrimitive = Primitive | null | void;
|
|
||||||
export declare type ChildNodeType = ElementNode | TextNode | ComponentNode;
|
|
||||||
export declare type NodeType = ChildNodeType | FragmentNode;
|
|
||||||
export declare type ChildType = ChildNodeType | Primitive;
|
|
||||||
export declare type NullableChildType = ChildType | ChildNodeType | NullablePrimitive;
|
|
||||||
export declare type ComponentFunctionType = (props: NodePropsType, child?: NullableChildType[]) => NullableChildType | Promise<NullableChildType>;
|
|
||||||
export {};
|
|
@ -1,3 +0,0 @@
|
|||||||
"use strict";
|
|
||||||
Object.defineProperty(exports, "__esModule", { value: true });
|
|
||||||
//# sourceMappingURL=types.js.map
|
|
@ -1 +0,0 @@
|
|||||||
{"version":3,"file":"types.js","sourceRoot":"","sources":["../tmp/types.ts"],"names":[],"mappings":""}
|
|
@ -1,16 +0,0 @@
|
|||||||
deno test examples/01.tsx
|
|
||||||
|
|
||||||
deno run examples/00.tsx
|
|
||||||
deno run examples/01.tsx
|
|
||||||
deno run -c examples/02/tsconfig.json examples/02/02.tsx
|
|
||||||
|
|
||||||
https://deno.land/manual/contributing/style_guide
|
|
||||||
|
|
||||||
|
|
||||||
- would be great to make it as well node compatible:
|
|
||||||
tsc examples/01.tsx --jsx react --outDir dist && node dist/examples/01.js
|
|
||||||
|
|
||||||
git tag --delete latest
|
|
||||||
git push --delete origin latest
|
|
||||||
git tag latest
|
|
||||||
git push --tags
|
|
@ -1,25 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "async-jsx-html",
|
|
||||||
"version": "1.2.1",
|
|
||||||
"main": "nodejs/mod.js",
|
|
||||||
"types": "nodejs/mod.d.ts",
|
|
||||||
"repository": "git@github.com:apiel/jsx-html.git",
|
|
||||||
"author": "Alexandre Piel <alexandre.piel@gmail.com>",
|
|
||||||
"license": "MIT",
|
|
||||||
"scripts": {
|
|
||||||
"clean": "rm -rf nodejs/ && rm -rf tmp/",
|
|
||||||
"build": "yarn clean && node deno2nodejs.js && tsc -p ./tmp/tsconfig.json",
|
|
||||||
"transpile": "babel --no-babelrc --plugins @babel/plugin-transform-react-jsx ./examples/node_01.jsx -o ./examples/node_01.js",
|
|
||||||
"start": "yarn transpile && node ./examples/node_01.js"
|
|
||||||
},
|
|
||||||
"devDependencies": {
|
|
||||||
"@babel/cli": "^7.10.3",
|
|
||||||
"@babel/core": "^7.10.3",
|
|
||||||
"@babel/generator": "^7.10.3",
|
|
||||||
"@babel/parser": "^7.10.3",
|
|
||||||
"@babel/plugin-transform-react-jsx": "^7.10.3",
|
|
||||||
"@babel/traverse": "^7.10.3",
|
|
||||||
"@types/node": "^14.0.13",
|
|
||||||
"typescript": "^3.9.5"
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,23 +0,0 @@
|
|||||||
import type { ElementNode } from './node/ElementNode.ts';
|
|
||||||
import type { TextNode } from './node/TextNode.ts';
|
|
||||||
import type { ComponentNode } from './node/ComponentNode.ts';
|
|
||||||
import type { FragmentNode } from './node/FragmentNode.ts';
|
|
||||||
|
|
||||||
export type NodePropsType = {
|
|
||||||
[key: string]: any;
|
|
||||||
};
|
|
||||||
|
|
||||||
type Primitive = string | boolean | number;
|
|
||||||
type NullablePrimitive = Primitive | null | void;
|
|
||||||
|
|
||||||
export type ChildNodeType = ElementNode | TextNode | ComponentNode;
|
|
||||||
|
|
||||||
export type NodeType = ChildNodeType | FragmentNode;
|
|
||||||
|
|
||||||
export type ChildType = ChildNodeType | Primitive;
|
|
||||||
export type NullableChildType = ChildType | ChildNodeType | NullablePrimitive;
|
|
||||||
|
|
||||||
export type ComponentFunctionType = (
|
|
||||||
props: NodePropsType,
|
|
||||||
child?: NullableChildType[],
|
|
||||||
) => NullableChildType | Promise<NullableChildType>;
|
|
1267
jsx-html/yarn.lock
1267
jsx-html/yarn.lock
File diff suppressed because it is too large
Load Diff
@ -1,5 +0,0 @@
|
|||||||
Denreg JSX renderer
|
|
||||||
|
|
||||||
**deprecated**
|
|
||||||
|
|
||||||
**DO NOT USE**
|
|
@ -1,13 +1,9 @@
|
|||||||
{
|
{
|
||||||
"name": "@denreg-jsx",
|
"name": "@denreg-jsx",
|
||||||
"version": "0.0.3",
|
"version": "0.1.0",
|
||||||
"description": "Denreg JSX renderer",
|
"description": "Denreg JSX renderer",
|
||||||
"author": "Fabian Stamm <dev@fabianstamm.de>",
|
"author": "Fabian Stamm <dev@fabianstamm.de>",
|
||||||
"contributors": [],
|
"contributors": [],
|
||||||
"deprecated": true,
|
"deprecated": false,
|
||||||
"files": [
|
"files": ["**/*.ts", "**/*.js", "tsconfig.json", "README.md"]
|
||||||
"**/*.ts",
|
}
|
||||||
"**/*.js",
|
|
||||||
"README.md"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
|
133
jsx/mod.ts
Normal file
133
jsx/mod.ts
Normal file
@ -0,0 +1,133 @@
|
|||||||
|
import "./types.ts";
|
||||||
|
|
||||||
|
const Fragment = Symbol("fragment");
|
||||||
|
|
||||||
|
declare namespace JSX {
|
||||||
|
interface Element {}
|
||||||
|
interface IntrinsicElements {
|
||||||
|
div: any;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export { Fragment };
|
||||||
|
|
||||||
|
export type Element = {
|
||||||
|
component: Component | string;
|
||||||
|
props: any;
|
||||||
|
children: any[];
|
||||||
|
};
|
||||||
|
export type ComponentRetElm = Element | Element[];
|
||||||
|
export type Component = (
|
||||||
|
props: any,
|
||||||
|
children: any
|
||||||
|
) => ComponentRetElm | Promise<ComponentRetElm>;
|
||||||
|
|
||||||
|
export function h(
|
||||||
|
component: string | Component,
|
||||||
|
props: any,
|
||||||
|
...children: Element[]
|
||||||
|
): Element {
|
||||||
|
return {
|
||||||
|
component,
|
||||||
|
props,
|
||||||
|
children,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function renderSSR(element: Element | string): Promise<string> {
|
||||||
|
if (typeof element === "string") return element;
|
||||||
|
else if (typeof element.component === "string")
|
||||||
|
return await renderHTML(element as Element);
|
||||||
|
else if (typeof element.component === "function")
|
||||||
|
return await renderCustom(element as Element);
|
||||||
|
|
||||||
|
console.warn("renderSSR: invalid element", element);
|
||||||
|
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
const selfClosing = new Set([
|
||||||
|
"area",
|
||||||
|
"base",
|
||||||
|
"br",
|
||||||
|
"col",
|
||||||
|
"embed",
|
||||||
|
"hr",
|
||||||
|
"img",
|
||||||
|
"input",
|
||||||
|
"link",
|
||||||
|
"meta",
|
||||||
|
"param",
|
||||||
|
"source",
|
||||||
|
"track",
|
||||||
|
"wbr",
|
||||||
|
]);
|
||||||
|
|
||||||
|
function flatDeep(arr: any, d = Infinity): any[] {
|
||||||
|
if (Array.isArray(arr) && d >= 0) {
|
||||||
|
let res = [];
|
||||||
|
for (const val of arr) {
|
||||||
|
const v = flatDeep(val, d - 1);
|
||||||
|
res.push(...v);
|
||||||
|
}
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
return [arr];
|
||||||
|
}
|
||||||
|
|
||||||
|
function cleanChildren(children: any) {
|
||||||
|
return flatDeep(children).filter((e) => !!e);
|
||||||
|
}
|
||||||
|
|
||||||
|
async function renderHTML(element: Element) {
|
||||||
|
if (typeof element.component !== "string")
|
||||||
|
throw new Error("Internal consistency error");
|
||||||
|
|
||||||
|
console.log("Element:", element.component);
|
||||||
|
let props = "";
|
||||||
|
|
||||||
|
for (const key in element.props) {
|
||||||
|
if (key == "innerHTML") continue;
|
||||||
|
props += `${key}="${element.props[key] || ""}" `;
|
||||||
|
}
|
||||||
|
|
||||||
|
const tag = element.component;
|
||||||
|
|
||||||
|
if (selfClosing.has(element.component)) {
|
||||||
|
return `<${tag} ${props}/>`;
|
||||||
|
} else {
|
||||||
|
let inner = "";
|
||||||
|
if (element.props && element.props["innerHTML"]) {
|
||||||
|
inner = element.props["innerHTML"];
|
||||||
|
} else {
|
||||||
|
const children = cleanChildren(element.children);
|
||||||
|
if (tag == "body") console.log(element.children, children);
|
||||||
|
inner = (
|
||||||
|
await Promise.all(children.map((child) => renderSSR(child)))
|
||||||
|
).join("");
|
||||||
|
}
|
||||||
|
return `<${tag} ${props}>${inner || ""}</${tag}>`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function renderCustom(element: Element) {
|
||||||
|
if (typeof element.component === "string")
|
||||||
|
throw new Error("Internal consistency error");
|
||||||
|
|
||||||
|
console.log("Component:", element.component);
|
||||||
|
const res = await Promise.resolve(
|
||||||
|
element.component(
|
||||||
|
{
|
||||||
|
...element.props,
|
||||||
|
children: element.children,
|
||||||
|
},
|
||||||
|
element.children
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
const ch = (
|
||||||
|
await Promise.all(cleanChildren(res).map((child) => renderSSR(child)))
|
||||||
|
).join("");
|
||||||
|
|
||||||
|
return ch;
|
||||||
|
}
|
7
jsx/tsconfig.json
Normal file
7
jsx/tsconfig.json
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
{
|
||||||
|
"compilerOptions": {
|
||||||
|
"lib": ["dom", "dom.iterable", "esnext", "deno.ns", "deno.unstable"],
|
||||||
|
"jsxFactory": "h",
|
||||||
|
"strictPropertyInitialization": false
|
||||||
|
}
|
||||||
|
}
|
11
jsx/types.ts
Normal file
11
jsx/types.ts
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
declare namespace JSX {
|
||||||
|
interface IntrinsicElements {
|
||||||
|
[elemName: string]: any;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
declare namespace JSX {
|
||||||
|
interface ElementClass {
|
||||||
|
render: any;
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user