Compare commits

..

No commits in common. "c15f5f57f96d87d8074d6f0fa651bf58f935b5f7" and "8f488f48bd85f803128eda66c6dab00834a94aeb" have entirely different histories.

3 changed files with 71 additions and 373 deletions

View file

@ -9,13 +9,13 @@ const undoClass = "undo-game";
/** /**
* Gets a component which represents the specified {@link game `game`}. * Gets a component which represents the specified {@link game `game`}.
* *
* @param {{game: import("./Game.js").Game }} game * @param {import("./Game.js").Game} game
* The game represented in this app. * The game represented in this app.
* *
* @returns {NodeDescriptor} * @returns {NodeDescriptor}
* The rendered node. * The rendered node.
*/ */
export function App({ game }) export function App(game)
{ {
return [ return [
"div", "div",
@ -91,7 +91,6 @@ export function App({ game })
*/ */
export function Board({ board }) export function Board({ board })
{ {
console.log(board);
let fields = board.flatMap((row) => row); let fields = board.flatMap((row) => row);
return [ return [

View file

@ -1,6 +1,6 @@
import { App } from "./Components.js"; import { App } from "./Components.js";
import { State } from "./State.js"; import { State } from "./State.js";
import { render } from "./SuiWeb.js"; import { SuiWeb } from "./SuiWeb.js";
/** /**
* Represents a game. * Represents a game.
@ -148,12 +148,6 @@ export class Game
load(data) load(data)
{ {
this.setState([], data); this.setState([], data);
if (!this.#state)
{
this.reset();
}
this.draw(); this.draw();
} }
@ -198,7 +192,7 @@ export class Game
{ {
let container = document.getElementById(this.id); let container = document.getElementById(this.id);
container.innerHTML = ""; container.innerHTML = "";
render([App, { game: this }], container); SuiWeb.render([App, this], container);
} }
/** /**

View file

@ -1,376 +1,81 @@
// @ts-nocheck
/* eslint-disable eslint-comments/no-unlimited-disable */
/* eslint-disable */
/** /**
* SuiWeb * Provides component for rendering elements written in SJDON notation.
* Simple User Interface Tool for Web Exercises
*
* @author bkrt
* @version 0.3.4
* @date 11.12.2021
*
* 0.3.4 - only null or undefined qualify as uninitialized state
* 0.3.3 - parseSJDON rewritten to use createElement
* - flatten children array for props.children in JSX
* 0.3.2 - save and restore of active element improved
* 0.3.1 - parseSJDON rewritten
* 0.3.0 - style property
* 0.2.3 - ES6 modules
* 0.2.2 - component state and hooks
* 0.2.1 - parse SJDON rewritten, function components
* 0.2.0 - render single SJDON structures to DOM
*
* Based on ideas and code from
* Rodrigo Pombo: Build your own React
* https://pomb.us/build-your-own-react/
*
* Thanks to Rodrigo Pombo for a great tutorial and for sharing the
* code of the Didact library. Didact is a much more sophisticated
* re-implementtation of React's basics than the simple SuiWeb.
*/ */
export class SuiWeb
/* =====================================================================
* SJDON - Conversion
* =====================================================================
*/
// parseSJDON: convert SJDON to createElement calls
//
// note: this function can also help to use React with SJDON syntax
//
// to simplify calls add something like this to your components:
// let s = (data) => reactFromSJDON(data, React.createElement)
//
function parseSJDON([type, ...rest], create = createElement)
{ {
const isObj = (obj) => typeof (obj) === 'object' && !Array.isArray(obj) /**
const children = rest.filter(item => !isObj(item)) * Renders the specified {@link data `data`} and appends it to the specified {@link element `element`}.
const repr = create(type, *
Object.assign({}, ...rest.filter(isObj)), * @param {NodeDescriptor} data
...children.map(ch => Array.isArray(ch) ? parseSJDON(ch, create) : ch) * The node to render written in SJDON notation.
) *
repr.sjdon = children * @param {HTMLElement} element
return repr * The element to add the rendered node to.
} *
* @returns {Node}
* The resulting element.
// create an element representation */
// static render(data, element)
function createElement(type, props, ...children)
{
return {
type,
props: {
...props,
children: children.flat().map(child =>
typeof child === "object"
? child
: createTextElement(child)
),
},
}
}
// create a text element representation
//
function createTextElement(text)
{
return {
type: "TEXT_ELEMENT",
props: {
nodeValue: text,
children: [],
},
}
}
/* =====================================================================
* Render node tree to DOM
* =====================================================================
*/
// global context
let contextStore = createStore();
// remove children and render new subtree
//
function render(element, container)
{
while (container.firstChild)
{ {
container.removeChild(container.lastChild); if (Array.isArray(data))
}
renderInit(element, container, 0);
}
// render subtree
// and call effects after rendering
//
function renderInit(element, container, n, childIndex)
{
contextStore("effects", []);
// save focus and cursor position of active element
let [focusSelector, position] = getFocusInput();
// ** render the element **
renderElem(element, container, n, childIndex);
// restore focus and cursor position of active element
setFocusInput(focusSelector, position);
// run effects
contextStore("effects").forEach(fun => fun());
contextStore("effects", []);
}
// render an element
// - if it is in SJDON form: parse first
// - render function or host component
//
function renderElem(element, container, n, childIndex)
{
if (Array.isArray(element))
{ {
element = parseSJDON(element); let descriptor = data[0];
let args = data.slice(1);
if (typeof descriptor === "function" || typeof descriptor === "string")
{
/**
* @type {HTMLElement}
*/
let result;
if (typeof descriptor === "function")
{
/**
* @type {any[]}
*/
let arg = [];
if (typeof args[0] === "object" && !Array.isArray(args[0]))
{
arg = [args[0]];
args = args.slice(1);
} }
if (element.type instanceof Function) result = /** @type {HTMLElement} */ (SuiWeb.render(descriptor(...arg), element));
{
updateFunctionComponent(element, container, n, childIndex);
} else
{
updateHostComponent(element, container, childIndex);
} }
} else
// function component
// - run function to get child node
// - render child node
//
function updateFunctionComponent(element, container, n, childIndex)
{
// save re-render function to context
contextStore("re-render", () => renderInit(element, container, n, n));
let children = element.sjdon ?? element.props.children;
let node = element.type({ ...element.props, children });
renderElem(node, container, n, childIndex);
}
// host component
// - create dom node
// - assign properties
// - render child nodes
// - add host to dom
//
function updateHostComponent(element, container, childIndex)
{
// create DOM node
const dom =
element.type == "TEXT_ELEMENT"
? document.createTextNode("")
: document.createElement(element.type)
// assign the element props
const isProperty = key => key !== "children"
Object.keys(element.props)
.filter(isProperty)
.forEach(name =>
{ {
if (name == "style") result = element.ownerDocument.createElement(descriptor);
{ element.appendChild(result);
updateStyleAttribute(dom, element.props.style);
} else
{
dom[name] = element.props[name];
}
})
// render children
element.props.children.forEach((child, index) =>
{
renderElem(child, dom, index);
});
if (typeof (childIndex) == 'number')
{
// re-render: replace node
container.replaceChild(dom, container.childNodes[childIndex]);
} else
{
// add node to container
container.appendChild(dom);
}
}
// update style attribute, value can be:
// - a CSS string: set to style attribute
// - an object: merged to style attribute
// - an array of objects: merged to style attribute
//
function updateStyleAttribute(dom, styles)
{
if (typeof (styles) == "string")
{
dom.style = styles;
} else if (Array.isArray(styles))
{
Object.assign(dom.style, ...styles);
} else if (typeof (styles) == "object")
{
Object.assign(dom.style, styles);
}
}
/* =====================================================================
* Handling state
* =====================================================================
*/
// element state
let stateHooks = createStore();
// state hook
// - access state via id and key
// - return state and update function
//
function useState(id, key, init)
{
let idKey = "id:" + id + "-key:" + key;
// prepare render function
let renderFunc = contextStore("re-render");
// define function to update state
function updateValue(updateFun, rerender = true)
{
stateHooks(idKey, updateFun(stateHooks(idKey)));
if (rerender) renderFunc();
} }
// new state: set initial value for (let arg of args)
if ([undefined, null].includes(stateHooks(idKey)))
{ {
stateHooks(idKey, init); if (typeof arg === "object" && !Array.isArray(arg))
{
Object.assign(result, arg);
} }
else
return [stateHooks(idKey), updateValue];
}
// effect hook
// add function to effects array
//
function useEffect(fun)
{
contextStore("effects", [...contextStore("effects"), fun]);
}
// create a key-value-store
// return accessor function
//
function createStore()
{
let data = {};
function access(key, ...value)
{ {
if (value.length === 0) SuiWeb.render(arg, result);
{
return data[key];
} else
{
data[key] = value[0];
return value[0];
} }
} }
return access;
}
return result;
/* =====================================================================
* Get and set focus and position in certain elements
* Note: this is a quick&dirty solution that probably fails when
* elements are added or removed
* =====================================================================
*/
// create a CSS selector for a given node
//
function getSelectorOf(node)
{
let selector = ""
while (node != document.body)
{
if (selector != "") selector = ">" + selector
if (node.id)
{
selector = "#" + node.id + selector
break
} }
else
let index = Array.from(node.parentNode.children) {
.filter(item => item.tagName == node.tagName) throw new SyntaxError();
.findIndex(item => item == node)
selector = node.tagName + ":nth-of-type(" + (index + 1) + ")" + selector
node = node.parentNode
} }
return selector }
} else if (typeof data === "string")
// find a selector for the element that has focus and the cursor
// position in the element
//
function getFocusInput()
{
const active = document.activeElement;
let sel = active ? getSelectorOf(active) : undefined
let position = active ? active.selectionStart : undefined;
return [sel, position];
}
// set focus to an element in a list of elements matching a
// selector and position cursor in the element
//
function setFocusInput(selector, position)
{
if (selector && typeof (selector) == 'string')
{ {
console.log("Sel:" + selector) return element.appendChild(element.ownerDocument.createTextNode(data));
let el = document.querySelector(selector); }
if (el) el.focus(); else
if (el && "selectionStart" in el
&& "selectionEnd" in el
&& position !== undefined)
{ {
el.selectionStart = position; throw new SyntaxError();
el.selectionEnd = position;
} }
} }
} }
/* =====================================================================
* Module export
* =====================================================================
*/
export { render, createElement, useState, useEffect };
/* =====================================================================
* EOF
* =====================================================================
*/