forked from mystiq/hydrogen-web
143 lines
4.9 KiB
TypeScript
143 lines
4.9 KiB
TypeScript
/*
|
|
Copyright 2020 Bruno Windels <bruno@windels.cloud>
|
|
Copyright 2021 Daniel Fedorin <danila.fedorin@gmail.com>
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
*/
|
|
|
|
// DOM helper functions
|
|
|
|
import {ViewNode} from "./types";
|
|
|
|
export type ClassNames<T> = { [className: string]: boolean | ((value: T) => boolean) }
|
|
export type BasicAttributes<T> = { [attribute: string]: ClassNames<T> | boolean | string }
|
|
export type Child = string | Text | ViewNode;
|
|
|
|
export function isChildren(children: object | Child | Child[]): children is Child | Child[] {
|
|
// children should be an not-object (that's the attributes), or a domnode, or an array
|
|
return typeof children !== "object" || "nodeType" in children || Array.isArray(children);
|
|
}
|
|
|
|
export function classNames<T>(obj: ClassNames<T>, value: T): string {
|
|
return Object.entries(obj).reduce((cn, [name, enabled]) => {
|
|
if (typeof enabled === "function") {
|
|
enabled = enabled(value);
|
|
}
|
|
if (enabled) {
|
|
return cn + (cn.length ? " " : "") + name;
|
|
} else {
|
|
return cn;
|
|
}
|
|
}, "");
|
|
}
|
|
|
|
export function setAttribute(el: Element, name: string, value: string | boolean): void {
|
|
if (name === "className") {
|
|
name = "class";
|
|
}
|
|
if (value === false) {
|
|
el.removeAttribute(name);
|
|
} else {
|
|
if (value === true) {
|
|
value = name;
|
|
}
|
|
el.setAttribute(name, value);
|
|
}
|
|
}
|
|
|
|
export function el(elementName: string, attributes?: BasicAttributes<never> | Child | Child[], children?: Child | Child[]): Element {
|
|
return elNS(HTML_NS, elementName, attributes, children);
|
|
}
|
|
|
|
export function elNS(ns: string, elementName: string, attributes?: BasicAttributes<never> | Child | Child[], children?: Child | Child[]): Element {
|
|
//console.log('html elNS', new Error().stack);
|
|
if (attributes && isChildren(attributes)) {
|
|
children = attributes;
|
|
attributes = undefined;
|
|
}
|
|
|
|
const e = document.createElementNS(ns, elementName);
|
|
|
|
const attrMap = {};
|
|
if (attributes) {
|
|
for (let [name, value] of Object.entries(attributes)) {
|
|
if (typeof value === "object") {
|
|
// Only className should ever be an object; be careful
|
|
// here anyway and ignore object-valued non-className attributes.
|
|
value = (value !== null && name === "className") ? classNames(value, undefined) : false;
|
|
}
|
|
attrMap[name] = value;
|
|
}
|
|
}
|
|
|
|
const attrString = Object.keys(attrMap)
|
|
.map((attrKey) => {
|
|
return `${attrKey}="${attrMap[attrKey]}"`;
|
|
})
|
|
.join(' ');
|
|
|
|
return `<${elementName} ${attrString}>${[].concat(children).join('')}</${elementName}>`;
|
|
|
|
|
|
if (attributes) {
|
|
for (let [name, value] of Object.entries(attributes)) {
|
|
if (typeof value === "object") {
|
|
// Only className should ever be an object; be careful
|
|
// here anyway and ignore object-valued non-className attributes.
|
|
value = (value !== null && name === "className") ? classNames(value, undefined) : false;
|
|
}
|
|
setAttribute(e, name, value);
|
|
}
|
|
}
|
|
|
|
if (children) {
|
|
if (!Array.isArray(children)) {
|
|
children = [children];
|
|
}
|
|
for (let c of children) {
|
|
if (typeof c === "string") {
|
|
c = text(c);
|
|
}
|
|
e.appendChild(c);
|
|
}
|
|
}
|
|
return e;
|
|
}
|
|
|
|
export function text(str: string): Text {
|
|
return str;
|
|
//return document.createTextNode(str);
|
|
}
|
|
|
|
export const HTML_NS: string = "http://www.w3.org/1999/xhtml";
|
|
export const SVG_NS: string = "http://www.w3.org/2000/svg";
|
|
|
|
export const TAG_NAMES = {
|
|
[HTML_NS]: [
|
|
"br", "a", "ol", "ul", "li", "div", "h1", "h2", "h3", "h4", "h5", "h6",
|
|
"p", "strong", "em", "span", "img", "section", "main", "article", "aside", "del", "blockquote",
|
|
"table", "thead", "tbody", "tr", "th", "td", "hr",
|
|
"pre", "code", "button", "time", "input", "textarea", "label", "form", "progress", "output", "video"],
|
|
[SVG_NS]: ["svg", "circle"]
|
|
} as const;
|
|
|
|
export const tag: { [tagName in typeof TAG_NAMES[string][number]]: (attributes?: BasicAttributes<never> | Child | Child[], children?: Child | Child[]) => Element } = {} as any;
|
|
|
|
for (const [ns, tags] of Object.entries(TAG_NAMES)) {
|
|
for (const tagName of tags) {
|
|
tag[tagName] = function(attributes, children) {
|
|
return elNS(ns, tagName, attributes, children);
|
|
}
|
|
}
|
|
}
|