Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/virtualstate/focus
JSX implementation
https://github.com/virtualstate/focus
deno jsx jsx-reader jsx-transform jsxgraph typescript
Last synced: 7 days ago
JSON representation
JSX implementation
- Host: GitHub
- URL: https://github.com/virtualstate/focus
- Owner: virtualstate
- License: mit
- Created: 2022-02-28T11:11:04.000Z (almost 3 years ago)
- Default Branch: main
- Last Pushed: 2022-10-14T08:30:39.000Z (over 2 years ago)
- Last Synced: 2024-12-19T06:42:48.185Z (about 1 month ago)
- Topics: deno, jsx, jsx-reader, jsx-transform, jsxgraph, typescript
- Language: TypeScript
- Homepage:
- Size: 938 KB
- Stars: 1
- Watchers: 2
- Forks: 0
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- Contributing: CONTRIBUTING.md
- License: LICENSE.md
- Code of conduct: CODE-OF-CONDUCT.md
Awesome Lists containing this project
README
# `@virtualstate/focus`
WIP implementation of [@virtualstate/fringe](https://github.com/virtualstate/x/blob/main/packages/fringe)
This project is in semver alpha stage
[//]: # (badges)
### Support
![Node.js supported](https://img.shields.io/badge/node-%3E%3D16.0.0-blue) ![Deno supported](https://img.shields.io/badge/deno-%3E%3D1.17.0-blue) ![Bun supported](https://img.shields.io/badge/bun-%3E%3D0.1.11-blue)
### Test Coverage
![94.33%25 lines covered](https://img.shields.io/badge/lines-94.33%25-brightgreen) ![94.33%25 statements covered](https://img.shields.io/badge/statements-94.33%25-brightgreen) ![91.92%25 functions covered](https://img.shields.io/badge/functions-91.92%25-brightgreen) ![84.7%25 branches covered](https://img.shields.io/badge/branches-84.7%25-brightgreen)
[//]: # (badges)
[//]: # (src/trying-to-write-documentation/accessors-how.tsx)
# How to read JSX trees
# The actual JSX node
A single JSX node is something we can inspect directly without any additional resolution (that may require async)
This can tell us things like, what kind of node it is, for example through its tagName, type property, or some other
identifying property, this lets us tell forms from inputs, identify fragment nodes, or identify an associated class or
function's name.## name
Through focus, you can use the `name` accessor
```typescript jsx
const { name } = await import("@virtualstate/focus");node = ;
ok(name(node) === "named");
```## properties
JSX nodes can have key value pairs associated to them, these can be used for attributes, options, or properties,
depending on what the node will be used for!You can get an object with these values using the `properties` accessor
```typescript jsx
const { properties } = await import("@virtualstate/focus");node = ;
object = properties(node);ok(object);
ok(object.key === "value");
ok(object.type === "text");
```# Async + Other Accessors
## children
JSX nodes can have children nodes associated with them, this lets us create trees and graphs with JSX
Children may be resolvable completely synchronously, or async, either way, we use an async interface to our
JSX node's children to ensure a consistent API across all nodes, and allowing this
resolution to be freely swapped behind the scenes without changing dependent implementation.One way we can read this tree is as if each node was a parent to some children, if we need to access
those same children's children, we just repeat the same process again, until we have explored the entire tree.To access a JSX node's children, use the `children` accessor
```typescript jsx
const { children } = await import("@virtualstate/focus");tree = (
);snapshot = await children(tree);
ok>(Array.isArray(snapshot));
ok(name(snapshot[0]) === "first");
ok(name(snapshot[1]) === "second");
```Sometimes a JSX node may have multiple representations of what it's child state looks like,
instead of using `await` with the result of `children`, we can instead use `for await`This may be useful where there is some delay in one of the child node's from resolving
```typescript jsx
async function Component() {
await new Promise((resolve) => setTimeout(resolve, 10));
return ;
}tree = (
);for await (snapshot of children(tree)) {
ok(name(snapshot[0]) === "first");
ok(!snapshot[1] || name(snapshot[1]) === "second");
}
```## childrenSettled
Some children may also throw errors, meaning you may want some way to observe the state of each individual
child node, using children that are fulfilled, and deciding what to do with children that rejectedFor this you can use the `childrenSettled` accessor, this has both the `await` and `for await` functionality
like `children````typescript jsx
const { childrenSettled } = await import("@virtualstate/focus");async function ComponentThrows() {
await new Promise((resolve) => setTimeout(resolve, 10));
throw new Error("Some error!");
}tree = (
);snapshot = await childrenSettled(tree);
ok(snapshot[0].status === "fulfilled");
ok(snapshot[1].status === "rejected");
``````typescript jsx
for await (snapshot of childrenSettled(tree)) {
ok(snapshot[0].status === "fulfilled");
ok(!snapshot[1] || snapshot[1].status === "rejected");
}
```## descendants
Instead of just the direct children of a single JSX node, you may want to find out all descendants that you can reach
For this, you can use the `descendants` accessor
```typescript jsx
const { descendants } = await import("@virtualstate/focus");tree = (
);
snapshot = await descendants(tree);
ok(name(snapshot[0]) === "first");
ok(name(snapshot[1]) === "second");
ok(name(snapshot[2]) === "third");
```As with `children`, `descendants` to can be accessed through `for await`
```typescript jsx
for await (snapshot of descendants(tree)) {
ok(name(snapshot[0]) === "first");
ok(name(snapshot[1]) === "second");
ok(!snapshot[2] || name(snapshot[2]) === "third");
}
```## descendantsSettled
You may want to be able to observe the resolution state of all descendants, as with `childrenSettled`
For this you can use the `descendantsSettled` accessor
```typescript jsx
const { descendantsSettled } = await import("@virtualstate/focus");tree = (
);snapshot = await descendantsSettled(tree);
ok(snapshot[0].status === "fulfilled");
ok(snapshot[1].status === "fulfilled");
ok(snapshot[2].status === "rejected");
``````typescript jsx
for await (snapshot of descendantsSettled(tree)) {
ok(snapshot[0].status === "fulfilled");
ok(snapshot[1].status === "fulfilled");
ok(!snapshot[2] || snapshot[2].status === "rejected");
}
```Because we can expose a bit more information with `descendantsSettled`, you can also
access the parent of an individual status object.This is helpful when re-creating a tree from these descendants, or creating associations between them.
```typescript jsx
snapshot = await descendantsSettled(tree);
ok(snapshot[1].status === "fulfilled");
ok(snapshot[2].parent === snapshot[1].value);
```This can also be accessed using through the `for await` pattern:
```typescript jsx
for await (snapshot of descendantsSettled(tree)) {
ok(snapshot[1].status === "fulfilled");
ok(!snapshot[2] || snapshot[2].parent === snapshot[1].value);
}
```## raw
The value you have access to, returned from the JSX node creation process, may not be the original representation
used for the JSX node. This original raw representation may be helpful when creating new accessor functions, or
what to inspect source definitions of component functions.If there is another representation available, the `raw` accessor can be used to access it, if there
is no raw representation, then that means the passed node is a raw representation itself, which is returned by default.```typescript jsx
const { raw } = await import("@virtualstate/focus");node = ;
rawNode = raw(node);
ok(name(rawNode) === "named");
```## proxy
To provide a new interface for a JSX node, or provide a different object completely, while still providing support
for the above JSX accessors, the `proxy` function can be used```typescript jsx
const { proxy } = await import("@virtualstate/focus");node = ;
api = { name };
proxied = proxy(node, api);
ok(proxied.name === "named");
``````typescript jsx
api = { someAccessorName: name };
proxied = proxy(node, api);
ok(proxied.someAccessorName === "named");
```If a `instance` accessor is provided, then this can be used to provide a new object completely
```typescript jsx
api = {
instance() {
return { name: Math.random(), source: Math.random() };
},
};
proxied = proxy(node, api);ok(typeof proxied.name === "number");
ok(typeof proxied.source === "number");
ok(name(proxied) === "named");
```When instance is used, `raw` becomes useful:
```typescript jsx
rawNode = raw(proxied);
ok(name(rawNode) === "named");
ok(typeof rawNode.name !== "number");
ok(rawNode === raw(node));
```If you need to access the original instance that is being proxied, you can use
the `instance` accessor.```typescript jsx
const { instance } = await import("@virtualstate/focus");staticInstance = { something: 1 };
api = {
instance() {
return staticInstance;
},
};
proxied = proxy(node, api);
nodeInstance = instance(proxied);
ok(typeof proxied.something === "number");
ok(nodeInstance === staticInstance);
ok(nodeInstance !== proxied);
```[//]: # (src/trying-to-write-documentation/accessors-how.tsx)