UNPKG

mostly-dom

Version:
120 lines (90 loc) 3.92 kB
import { HtmlTagNames, SvgTagNames, VNode, VNodeProps } from '../types' import { MostlyVNode, addSvgNamespace } from './VNode' import { isPrimitive, isString } from '../helpers' export const h: HyperscriptFn = function(): VNode { const tagName: string | ComponentFn = arguments[0] // required const childrenOrText: HyperscriptChildren = arguments[2] // optional let props: VNodeProps<Element> = {} let children: ArrayLike<VNode> | undefined let text: string | undefined if (childrenOrText) { props = arguments[1] if (isArrayLike(childrenOrText)) children = flattenArrayLike(childrenOrText) as Array<VNode> else if (isPrimitive(childrenOrText)) text = String(childrenOrText) } else if (arguments[1]) { const childrenOrTextOrProps = arguments[1] if (isArrayLike(childrenOrTextOrProps)) children = flattenArrayLike(childrenOrTextOrProps) as Array<VNode> else if (isPrimitive(childrenOrTextOrProps)) text = String(childrenOrTextOrProps) else props = childrenOrTextOrProps } if (typeof tagName === 'function') { return tagName(props, Array.isArray(children) ? children : text ? [ text ] : []) } const isSvg = tagName === 'svg' const vNode = isSvg ? MostlyVNode.createSvg(tagName, props, undefined, text) : MostlyVNode.create(tagName, props, undefined, text) if (Array.isArray(children)) vNode.children = sanitizeChildren(children, vNode) if (isSvg) addSvgNamespace(vNode) return vNode } function isArrayLike<T>(x: any): x is ArrayLike<T> { const typeOf = typeof x return x && typeof x.length === 'number' && typeOf !== 'function' && typeOf !== 'string' } function flattenArrayLike<A>(arrayLike: ArrayLike<A | ArrayLike<A>>, arr: Array<A> = []): Array<A> { forEach( (x: A | ArrayLike<A>) => (isArrayLike(x) ? flattenArrayLike(x, arr) : arr.push(x)), arrayLike ) return arr } function forEach<A>(fn: (value: A) => void, list: ArrayLike<A>): void { for (let i = 0; i < list.length; ++i) fn(list[i]) } function sanitizeChildren(childrenOrText: Array<VNode>, parent: VNode): Array<VNode> { childrenOrText = childrenOrText.filter(Boolean) // remove possible null values const childCount: number = childrenOrText.length const children: Array<VNode> = Array(childCount) for (let i = 0; i < childCount; ++i) { const vNodeOrText = childrenOrText[i] if (isString(vNodeOrText)) children[i] = MostlyVNode.createText(vNodeOrText) else children[i] = vNodeOrText if (parent.scope && !children[i].scope) children[i].scope = parent.scope children[i].parent = parent as VNode<Element> } return children } export type VNodeChildren = string | number | null | VNode export type HyperscriptChildren = | VNodeChildren | ArrayLike<VNodeChildren> | ArrayLike<VNodeChildren | ArrayLike<VNodeChildren>> | ArrayLike<ArrayLike<VNodeChildren>> export interface ComponentFn { (props: VNodeProps, children: Array<string | null | VNode>): VNode } export type ValidTagNames = HtmlTagNames | SvgTagNames | ComponentFn export interface HyperscriptFn { (tagName: ValidTagNames): VNode (tagName: ValidTagNames, props: VNodeProps<any>): VNode (tagName: ValidTagNames, children: HyperscriptChildren): VNode (tagName: ValidTagNames, props: VNodeProps<any>, children: HyperscriptChildren): VNode <T extends Node, Props extends VNodeProps<Element> = VNodeProps<Element>>( tagName: ValidTagNames ): VNode<T, Props> <T extends Node, Props extends VNodeProps<Element> = VNodeProps<Element>>( tagName: ValidTagNames, props: Props ): VNode<T> <T extends Node, Props extends VNodeProps<Element> = VNodeProps<Element>>( tagName: ValidTagNames, children: HyperscriptChildren ): VNode<T, Props> <T extends Node, Props extends VNodeProps<Element> = VNodeProps<Element>>( tagName: ValidTagNames, props: Props, children: HyperscriptChildren ): VNode<T, Props> }