Yeap is a React alternative to be the most simplify and small as possible without virtual dom (actualy ~7.5kb ungzip on bundlephobia). Based on reactivity, it is easily usable with web component.
Yeap VS React (or Preact)
React has side effects, when you want to create a component, and you update it, React re-render a lot of components whereas nothing about them ask to be re-rendering. With Yeap this probleme has solved, you call the component only when you need them, no useless re-render.
Yeap VS Solid
Yeap and Solid are in the idea the same, but the main difference is in the compilation, Solid compile with its own compiler, unlike Yeap decide to use the default jsx compiler.
npm install yeap
# or
yarn add yeap
#or
pnpm add yeap
Templates and a Preset for Vite are available.
Yeap use the jsx, and the reactivity system.
The reactivity system is based on the function createReactor
.
createReactor
return a reactor, it's the base of the reactivity system.
import { createReactor } from "yeap/app"
import { render } from "yeap/web"
function App () {
const count = createReactor(0)
return (
<div>
<p>{count}</p>
<button onClick={() => count((count) =>count + 1)}>+</button>
</div>
)
}
render(<App />, document.getElementById("root"))
createReactor
is the main function of the reactivity system, it return a reactor. A reactor is a function. When you call it, it return the current value of the reactor. (it's the getter)
import { createReactor } from "yeap/app"
const a = createReactor(0)
console.log(a()) // 0
And when you want to update the value, it's easy, you just call the reactor with the new value, it will return the old value and save the new value. (it's the setter)
import { createReactor } from "yeap/app"
const a = createReactor(0)
console.log(a(1)) // 0
console.log(a()) // 1
It's possible to use the setter with a function, it will take the old value in parameter and return the new value.
import { createReactor } from "yeap/app"
const a = createReactor(0)
console.log(a((old) => old + 1)) // 0
console.log(a()) // 1
On top of this, a reactor a lot of other methods like subscribe
, freeze
or reader
.
The .subscribe
method is used to subscribe to the reactor, it's a take one parameter, and it's a function that will be called when the reactor is updated. The function will receive the old value and the new value.
import { createReactor } from "yeap/app"
const a = createReactor(0)
a.subscribe((prev, next) => {
console.log(
"a changed",
prev, // 0
next // 1
)
})
a(1)
The method .subscribe
return a function that you can use to unsubscribe.
import { createReactor } from "yeap/app"
const a = createReactor(0)
const unsubscribe = a.subscribe((prev, next) => {
console.log("a changed", prev, next) // never called
})
unsubscribe()
a(1)
Note: .subscribe
is used in internal of yeap/web
for observe the changes of the reactivity system and update the dom in consequence.
.freeze
and .reader
are used to create a ReadOnlyReactor, but the difference between both is the value of the ReadOnlyReactor.
With .freeze
, the value isn't updated over time.
import { createReactor } from "yeap/app"
const a = createReactor(0)
const b = a.freeze()
a(1)
console.log(a()) // 1
console.log(b()) // 0
but with .reader
, the value is updated over time.
import { createReactor } from "yeap/app"
const a = createReactor(0)
const b = a.reader()
a(1)
console.log(a()) //
console.log(b()) // 1
In the same way, you have createRef
, it's a function that return a reactor like createReactor
, but it can be updated only one time (default value isn't an update), after it become a ReadOnlyReactor
.
Example:
import { createRef } from "yeap/app"
const a = createRef(0)
console.log(a(1)) // 0
console.log(a(2)) // 1
console.log(a()) // 1
Example of use:
import { createRef } from "yeap/app"
import { render } from "yeap/web"
function App () {
const paragraph = createRef(null)
return (
<div>
<p ref={paragraph}>Hello World!</p>
<button onClick={() => paragraph().textContent = "Hello Yeap!"}>Update the text of the paragraph</button>
</div>
)
}
render(<App />, document.getElementById("root"))
Yeap has two function to create effect and computed, createEffect
and createComputed
.
createEffect
is a function that take a callback and dependencies, when a dependency is updated, the callback was recalled.
import { createEffect, createReactor } from "yeap/app"
import { render } from "yeap/web"
function App () {
const count = createReactor(0)
createEffect(() => {
console.log(count())
}, count)
return (
<div>
<p>{count}</p>
<button onClick={() => count((count) =>count + 1)}>+</button>
</div>
)
}
render(<App />, document.getElementById("root"))
And in the same way, createComputed
is a function that take a callback, and return a reactor, the function take the same arguments as the function of createEffect
.
import { createComputed, createReactor } from "yeap/app"
import { render } from "yeap/web"
function App () {
const count = createReactor(0)
const double = createComputed(() => count() * 2, count)
return (
<div>
<p>{count}</p>
<p>{double}</p>
<button onClick={() => count((count) =>count + 1)}>+</button>
</div>
)
}
render(<App />, document.getElementById("root"))
Or you can use the method .compute
of the reactor, it's the same as createComputed
, but the callback take the value of the reactor as first argument and that all.
import { createReactor } from "yeap/app"
import { render } from "yeap/web"
function App () {
const count = createReactor(0)
const double = count.compute((count) => count * 2)
return (
<div>
<p>{count}</p>
<p>{double}</p>
<button onClick={() => count((count) =>count + 1)}>+</button>
</div>
)
}
render(<App />, document.getElementById("root"))
When can use createComputed
or .compute
. It's not an obligation, but .compute
it's make to be used in the jsx directly.
createComputed
and createEffect
can be take options before the dependencies.
In yeap you can use condition in the jsx, but you can't use if
, else
, &&
or ||
because it's break the reactivity system, you can use createComputed
or .compute
to create a condition.
import { createReactor } from "yeap/app"
import { render } from "yeap/web"
function App () {
const count = createReactor(0)
return (
<div>
<p>{count}</p>
{count.compute((count) => {
if(count > 0) return <p>Count is positive</p>
else if(count < 0) return <p>Count is negative</p>
else return <p>Count is null</p>
})}
<button onClick={() => count((count) =>count + 1)}>+</button>
</div>
)
}
render(<App />, document.getElementById("root"))
But for a condition if/else you can use the method .when
, it takes two arguments, the truthy value and the falsy value (truthy and falsy value can be a function).
import { createReactor } from "yeap/app"
import { render } from "yeap/web"
function App () {
const show = createReactor(true)
return (
<div>
<button onClick={() => show((show) =>!show)}>{show.when(() =>"To Hide", "To Show")}</button>
</div>
)
}
render(<App />, document.getElementById("root"))
And in the same way, you have the attribute when
and fallback
on html element and component.
import { createReactor } from "yeap/app"
import { render } from "yeap/web"
function App () {
const show = createReactor(true)
return (
<div>
<button onClick={() => show((show) =>!show)}>{show.when("To Hide", "To Show")}</button>
<p when={show} fallback={<p>Bye!</p>}>Hello!</p>
</div>
)
}
render(<App />, document.getElementById("root"))
import { createReactor } from "yeap/app"
import { render } from "yeap/web"
function App () {
const list = createReactor([0, 1, 2, 3, 4])
return (
<div>
<ul>
{list.compute((list) => {
const newList = []
for(const i of list) {
newList.push(<li>{i} ** 2 = {i*i}</li>)
}
return newList
})}
</ul>
<button onClick={() => list((list) =>[...list, list.length])}>+ item</button>
</div>
)
}
render(<App />, document.getElementById("root"))
It's possible, but it's not the best way to do it, you can use the method .map
(or other method on a array according to usage).
import { createReactor } from "yeap/app"
import { render } from "yeap/web"
function App () {
const list = createReactor([0, 1, 2, 3, 4])
return (
<div>
<ul>
{list.map((i) => <li>{i} ** 2 = {i*i}</li>) /*it keep the reactivity*/}
</ul>
<button onClick={() => list((list) =>[...list, list.length])}>+ item</button>
</div>
)
}
render(<App />, document.getElementById("root"))
But, why it works?
If you write in vanilla js that:
const list = [0, 1, 2, 3, 4]
const doubleList = list.map((i) => i * 2)
doubleList
has the value [0, 2, 4, 6, 8]
, but if you push a new item in the list, the value of doubleList stays the same.
With the reactivity, when you want to get a property you can get normally, it just change the value of the property by a reactor and it automatically update when the reactor parent is updated.
With a reactor, the previous example gives:
const list = createReactor([0, 1, 2, 3, 4])
const doubleList = list.map((i) => i * 2)
Now, doubleList()
has the value [0, 2, 4, 6, 8]
, and if you push a new item in the list, the value of doubleList is updated.
Warning:
- if you use a reactor in a array, you can't use the methods
.push
,.pop
or other methods that change the value directly, because it's not possible to observe the changes of the array. - the value of property became a reactor, but the return value of a method is not a reactor but a ReadOnlyReactor.
Example:
import { createReactor } from "yeap/app"
import { render } from "yeap/web"
function App () {
const inputValue = createReactor("")
return (
<div>
<input value={inputValue} placeholder="Type something here" onInput={(e) => inputValue(e.currentTarget.value)}>
the input has a length of {inputValue.length}
</div>
)
}
render(<App />, document.getElementById("root"))
In addition to that, you have on the jsx attribute key
that works like the key
attribute in React.
import { createAsync } from "yeap/app"
import { Fragment } from "yeap/components" // if you use the yeap-vite-preset you can't import Fragment directly because it's already imported with the yeap-vite-preset (same for h in yeap/web)
import { render } from "yeap/web"
function App () {
const {data: count, loading} = createAsync(async () =>{
const res = await fetch("https://api.github.com/repos/yeapjs/yeap/stargazers")
const data = await res.json()
return data.length
}, 0)
return (
<div>
<p>{loading.when("Loading...", <Fragment>The repo yeapjs/yeap has {count} star{count.compute((c) => c > 1 ? "s" : "")}</Fragment>)}</p>
</div>
)
}
render(<App />, document.getElementById("root"))
This is how you can use async/await in yeap, you can use createAsync
to create an async reactor, it takes two arguments, the async function and the default value.
But if you want to refresh the data when a reactor change, you have createAsyncComputed
, it works like that:
const {data, error, loading, refetch} = createAsync(asyncCallback, defaultData)
createComputed(refetch, deps) // or createEffect(refetch, deps)
In yeap you have 3 functions, onMount
, onUnmount
and createEventDispatcher
.
onMount
and onUnmount
are called when the component is mounted and unmounted in the DOM.
And createEventDispatcher
is used to create an event dispatcher, it returns a function that you can call to dispatch the event. (call a function in properties named on
+ name of the event, in camelCase)
import { createReactor, createEventDispatcher } from "yeap/app"
import { render } from "yeap/web"
function App () {
const count = createReactor(0)
const dispatch = createEventDispatcher()
return (
<div>
<p>{count}</p>
<button onClick={() => count((count) =>count + 1)}>+</button>
<button onClick={() => dispatch("count", count)}>Dispatch</button>
</div>
)
}
/**
* @param {CustomEvent} param0
*/
function handleCount ({detail: count}) {
console.log(count)
}
render(<App onCount={handleCount}/>, document.getElementById("root"))
An event can be recieve an array on the format [function, ...args]
In yeap you have createDirective
and createEventModifier
. Directives are made to update html elements and event modifiers as their name indicates, it is to modify the event
object.
event:event-modifier
is the syntax for event modifiers and use:directive
for the directives.
Works exactly like in React.
You have the function createPersistor
, it takes a callback and call it only once, when the app is mounted. But if a component is unmounted and mounted again, the callback is not called, and the value is kept.
There are also 2 aliases, createPersistentCallback
and createPersistentReactor
.
import { createReactor, createPersistor } from "yeap/app"
import { render } from "yeap/web"
function App () {
const show = createReactor(true)
return (
<div>
<button onClick={() => show((show) =>!show)}>{show.when("Hide Counter", "Show Counter")}</button>
<Counter when={show} />
</div>
)
}
function Counter () {
const count = createPersistor(() => createReactor(0))
return (
<div>
<p>{count}</p>
<button onClick={() => count((count) =>count + 1)}>+</button>
</div>
)
}
render(<App />, document.getElementById("root"))
You can create a component with a function, but you have only one difference between a react funtional component and a yeap component, you have 2 arguments, the props and the children.
import { createReactor, createComponent } from "yeap/app"
import { render } from "yeap/web"
function Counter({ start }, children) {
const count = createReactor(start)
return (
<div>
<p>{count}</p>
<button onClick={() => count((count) =>count + 1)}>+</button>
{children}
</div>
)
}
function App () {
return (
<Counter start={0}>
Hello World!
</Counter>
)
}
render(<App />, document.getElementById("root"))
In SolidJS you can create a component like that, but you can't destructure the props because you lose the reactivity, but in Yeap you can.
Yeap have also 3 built-in components, Fragment
, Dynamic
, Portal
and one function for async components, lazy
.
Yeap has a function to create web component, define
in yeap/web
, it's like the customElements.define
of the web component API, it's take a name, a function and options.
import { createReactor } from "yeap/app"
import { define } from "yeap/web"
function MyCounter ({ ref: element }) {
const count = createReactor(0)
console.log(element)
return (
<div>
<p>{count}</p>
<button onClick={() => count((count) =>count + 1)}>+</button>
</div>
)
}
define("my-counter", MyCounter, {
reactiveAttributes: [],
shadowed: "closed" // "closed" | "open" | false (default)
})
The function in define
is a yeap component, it takes the props and the children, and you can use the reactiveAttributes
option to make some attributes reactive, and the shadowed
option to make the component shadowed.
It's possible to use YourComponent.defaultProps
to set the default props of the component.
It's also possible to use YourComponent.attributeTypes
to cast the attributes' value into Number
, BigInt
or Boolean
(for boolean it check if you have the attribute), see the issue #12 for more detail.
The component can be take 2 arguments, the attributes and the child nodes. When an attribute is in the reactiveAttributes
array, it became a reactor. In the attributes, you can find ref
, is the html element.