Compare commits
No commits in common. "c15f5f57f96d87d8074d6f0fa651bf58f935b5f7" and "8f488f48bd85f803128eda66c6dab00834a94aeb" have entirely different histories.
c15f5f57f9
...
8f488f48bd
3 changed files with 71 additions and 373 deletions
|
@ -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 [
|
||||||
|
|
|
@ -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);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
409
src/js/SuiWeb.js
409
src/js/SuiWeb.js
|
@ -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
|
|
||||||
* =====================================================================
|
|
||||||
*/
|
|
||||||
|
|
Loading…
Reference in a new issue