Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/tvler/experimental-react-like-framework

A frontend for React inspired by SwiftUI
https://github.com/tvler/experimental-react-like-framework

dsl front-end frontend javascript react reactjs swiftui typescript

Last synced: 2 months ago
JSON representation

A frontend for React inspired by SwiftUI

Awesome Lists containing this project

README

        

# experimental-react-like-framework

A new, experimental frontend for React inspired by SwiftUI. In development.

Based off of two questions:

① What would React look like if execution-order-based code wasn't used only for hooks, but for everything?

② What would React look like if JSX never existed?

Hello World app:

```ts
const App = () => {
h1("Hello world!");
};
```

The framework is hosted on React itself. Designed to be incrementally adopted inside of a React subtree without interfering with any existing code, with the benefit of all of React's features and ecosystem.

```tsx
import { NewFramework } from "./NewFramework";
import { NewFrameworkApp } from "./NewFrameworkApp";

const App = () => (

{/* 🆕 */}


);
```

## Table of contents

- [Framework write-up](#framework-write-up)
- [What I've built so far](#what-ive-built-so-far)
- [What needs to be built](#what-needs-to-be-built)
- [Further reading](#further-reading)

## Framework write-up

### Bet: JSX is bad

JSX immediately requires a complex build environment. Differences between HTML is small but painful for beginners. Sebastian Markbåge (a lead react maintainer) tweeted that JSX is a bug.

([twitter.com/sebmarkbage/status/1255886278437945344](https://twitter.com/sebmarkbage/status/1255886278437945344))

### Bet: React.createElement is bad

Using React without JSX forces you to use the React.createElement API. Building a complex template with React.createElement ends up as a deeply-nested function call with no breakpoints at the very bottom of your component code. JSX is an attempt to make this format more presentable, but it doesn't solve the underlying issue that your template is inside of a function call, with no access to if-statements, for-loops, etc.

### Bet: Code based off of execution-order is good

Facebook built hooks around execution-order and received major backlash by the engineering community, but resulted in few actual real-world problems.

### Result

An exploration of what react would look like if execution-order-based code wasn’t considered an antipattern, and if JSX never existed.

```tsx
/*
* New syntax
*/
const App = () => {
const [counter, setCounter] = useState(0);
const handleClick = () => {
setCounter(counter + 1);
};

h1("Hello world!");

// A button with an onClick prop
button({ onClick: handleClick }, counter);

// An ordered list counting up from 0 to 9
ol(() => {
for (let i = 0; i < 10; i++) {
li({ key: i }, i);
}
});

// Conditionally rendering a span if counter is odd
if (counter % 2) {
span("counter is an odd number.");
}
};

/*
* Old JSX syntax
*/
const JSXApp = () => {
const [counter, setCounter] = useState(0);
const handleClick = () => {
setCounter(counter + 1);
};

return (
<>

Hello world!

{/* A button with an onClick prop */}
{counter}

{/* An ordered list counting up from 0 to 9 */}


    {Array.from({ length: 10 }, (_, i) => (
  1. {i}

  2. ))}

{/* Conditionally rendering a span if counter is odd */}
{counter % 2 && counter is an odd number.}
>
);
};
```

### Benefits

Simpler code for loops and conditional rendering

Zero distinction between functions and components

Don’t have to learn JSX

No return statement

No need for closing tags

No need for fragments

Easier to comment elements out

### How it works

Each primitive HTML element function pings a global store when called. The order of the pings determines the order in which the actual HTML elements are rendered to the dom. This is the exact same architecture that Facebook has proven successful with react hooks.

This has the potential to work directly within React itself, turning into a series of React.createElement calls on execution. Making this an experimental new frontend for React, with the added benefits of gradual adoption and an already-existing suite of developer tools to build off of.

## What I've built so far

### ✅ Working prototype

```
cd example-create-react-app
yarn
yarn start
```

### ✅ Everything except state

[example-create-react-app/src/app.ts](example-create-react-app/src/app.ts)

```ts
import { button, h1, ol, li } from "./Framework";

const app = () => {
h1("Hello world!");

// A button with an onClick prop
button(
{
onClick: () => {
alert("clicked!");
},
},
"Click"
);

// An ordered list counting up from 0 to 9
ol(() => {
for (let i = 0; i < 10; i++) {
li(i);
}
});
};

export default app;
```

### ✅ Building on top of React

[example-create-react-app/src/index.ts](example-create-react-app/src/index.ts)

```ts
import React from "react";
import ReactDOM from "react-dom";
import { Framework } from "./Framework";
import app from "./app";

ReactDOM.render(
React.createElement(Framework, { root: app }),
document.getElementById("root")
);
```

## What needs to be built

- [ ] State
- [ ] A cool project name ;) https://github.com/tvler/experimental-react-like-framework/issues/4

## Further reading

[Function builders (draft proposal)](https://github.com/apple/swift-evolution/blob/9992cf3c11c2d5e0ea20bee98657d93902d5b174/proposals/XXXX-function-builders.md) by [John McCall](https://github.com/rjmccall) and [Doug Gregor](http://github.com/DougGregor)

[js-dsl](https://venkatperi.github.io/js-dsl/) by [Venkat Peri](https://github.com/venkatperi)

[incremental-dom](http://google.github.io/incremental-dom/) by Google