/**
|
* Virtual DOM patching
|
* Modified from snabbdom https://github.com/snabbdom/snabbdom/blob/master/src/init.ts
|
*
|
* The design has been simplified to focus on the purpose in SVG rendering in SVG.
|
*
|
* Licensed under the MIT License
|
* https://github.com/paldepind/snabbdom/blob/master/LICENSE
|
*/
|
|
import { isArray, isObject } from '../core/util';
|
import { createElement, createVNode, SVGVNode, XMLNS, XML_NAMESPACE, XLINKNS } from './core';
|
import * as api from './domapi';
|
|
const colonChar = 58;
|
const xChar = 120;
|
const emptyNode = createVNode('', '');
|
|
type NonUndefined<T> = T extends undefined ? never : T;
|
|
function isUndef(s: any): boolean {
|
return s === undefined;
|
}
|
|
function isDef<A>(s: A): s is NonUndefined<A> {
|
return s !== undefined;
|
}
|
|
function createKeyToOldIdx(
|
children: SVGVNode[],
|
beginIdx: number,
|
endIdx: number
|
): KeyToIndexMap {
|
const map: KeyToIndexMap = {};
|
for (let i = beginIdx; i <= endIdx; ++i) {
|
const key = children[i].key;
|
if (key !== undefined) {
|
if (process.env.NODE_ENV !== 'production') {
|
if (map[key] != null) {
|
console.error(`Duplicate key ${key}`);
|
}
|
}
|
map[key] = i;
|
}
|
}
|
return map;
|
}
|
|
function sameVnode(vnode1: SVGVNode, vnode2: SVGVNode): boolean {
|
const isSameKey = vnode1.key === vnode2.key;
|
const isSameTag = vnode1.tag === vnode2.tag;
|
|
return isSameTag && isSameKey;
|
}
|
|
type KeyToIndexMap = { [key: string]: number };
|
|
function createElm(vnode: SVGVNode): Node {
|
let i: any;
|
const children = vnode.children;
|
const tag = vnode.tag;
|
// if (tag === '!') {
|
// if (isUndef(vnode.text)) {
|
// vnode.text = '';
|
// }
|
// vnode.elm = api.createComment(vnode.text!);
|
// }
|
// else
|
if (isDef(tag)) {
|
const elm = (vnode.elm = createElement(tag));
|
|
updateAttrs(emptyNode, vnode);
|
|
if (isArray(children)) {
|
for (i = 0; i < children.length; ++i) {
|
const ch = children[i];
|
if (ch != null) {
|
api.appendChild(elm, createElm(ch));
|
}
|
}
|
}
|
else if (isDef(vnode.text) && !isObject(vnode.text)) {
|
api.appendChild(elm, api.createTextNode(vnode.text));
|
}
|
}
|
else {
|
vnode.elm = api.createTextNode(vnode.text!);
|
}
|
return vnode.elm;
|
}
|
|
function addVnodes(
|
parentElm: Node,
|
before: Node | null,
|
vnodes: SVGVNode[],
|
startIdx: number,
|
endIdx: number
|
) {
|
for (; startIdx <= endIdx; ++startIdx) {
|
const ch = vnodes[startIdx];
|
if (ch != null) {
|
api.insertBefore(parentElm, createElm(ch), before);
|
}
|
}
|
}
|
|
function removeVnodes(parentElm: Node, vnodes: SVGVNode[], startIdx: number, endIdx: number): void {
|
for (; startIdx <= endIdx; ++startIdx) {
|
const ch = vnodes[startIdx];
|
if (ch != null) {
|
if (isDef(ch.tag)) {
|
const parent = api.parentNode(ch.elm);
|
api.removeChild(parent, ch.elm);
|
}
|
else {
|
// Text node
|
api.removeChild(parentElm, ch.elm!);
|
}
|
}
|
}
|
}
|
|
export function updateAttrs(oldVnode: SVGVNode, vnode: SVGVNode): void {
|
let key: string;
|
const elm = vnode.elm as Element;
|
const oldAttrs = oldVnode && oldVnode.attrs || {};
|
const attrs = vnode.attrs || {};
|
|
if (oldAttrs === attrs) {
|
return;
|
}
|
|
// update modified attributes, add new attributes
|
// eslint-disable-next-line
|
for (key in attrs) {
|
const cur = attrs[key];
|
const old = oldAttrs[key];
|
if (old !== cur) {
|
if (cur === true) {
|
elm.setAttribute(key, '');
|
}
|
else if (cur === false) {
|
elm.removeAttribute(key);
|
}
|
else {
|
if (key.charCodeAt(0) !== xChar) {
|
elm.setAttribute(key, cur as any);
|
}
|
// TODO
|
else if (key === 'xmlns:xlink' || key === 'xmlns') {
|
elm.setAttributeNS(XMLNS, key, cur as any);
|
}
|
else if (key.charCodeAt(3) === colonChar) {
|
// Assume xml namespace
|
elm.setAttributeNS(XML_NAMESPACE, key, cur as any);
|
}
|
else if (key.charCodeAt(5) === colonChar) {
|
// Assume xlink namespace
|
elm.setAttributeNS(XLINKNS, key, cur as any);
|
}
|
else {
|
elm.setAttribute(key, cur as any);
|
}
|
}
|
}
|
}
|
|
// remove removed attributes
|
// use `in` operator since the previous `for` iteration uses it (.i.e. add even attributes with undefined value)
|
// the other option is to remove all attributes with value == undefined
|
for (key in oldAttrs) {
|
if (!(key in attrs)) {
|
elm.removeAttribute(key);
|
}
|
}
|
}
|
|
|
function updateChildren(parentElm: Node, oldCh: SVGVNode[], newCh: SVGVNode[]) {
|
let oldStartIdx = 0;
|
let newStartIdx = 0;
|
let oldEndIdx = oldCh.length - 1;
|
let oldStartVnode = oldCh[0];
|
let oldEndVnode = oldCh[oldEndIdx];
|
let newEndIdx = newCh.length - 1;
|
let newStartVnode = newCh[0];
|
let newEndVnode = newCh[newEndIdx];
|
let oldKeyToIdx: KeyToIndexMap | undefined;
|
let idxInOld: number;
|
let elmToMove: SVGVNode;
|
let before: any;
|
|
while (oldStartIdx <= oldEndIdx && newStartIdx <= newEndIdx) {
|
if (oldStartVnode == null) {
|
oldStartVnode = oldCh[++oldStartIdx]; // Vnode might have been moved left
|
}
|
else if (oldEndVnode == null) {
|
oldEndVnode = oldCh[--oldEndIdx];
|
}
|
else if (newStartVnode == null) {
|
newStartVnode = newCh[++newStartIdx];
|
}
|
else if (newEndVnode == null) {
|
newEndVnode = newCh[--newEndIdx];
|
}
|
else if (sameVnode(oldStartVnode, newStartVnode)) {
|
patchVnode(oldStartVnode, newStartVnode);
|
oldStartVnode = oldCh[++oldStartIdx];
|
newStartVnode = newCh[++newStartIdx];
|
}
|
else if (sameVnode(oldEndVnode, newEndVnode)) {
|
patchVnode(oldEndVnode, newEndVnode);
|
oldEndVnode = oldCh[--oldEndIdx];
|
newEndVnode = newCh[--newEndIdx];
|
}
|
else if (sameVnode(oldStartVnode, newEndVnode)) {
|
// Vnode moved right
|
patchVnode(oldStartVnode, newEndVnode);
|
api.insertBefore(parentElm, oldStartVnode.elm!, api.nextSibling(oldEndVnode.elm!));
|
oldStartVnode = oldCh[++oldStartIdx];
|
newEndVnode = newCh[--newEndIdx];
|
}
|
else if (sameVnode(oldEndVnode, newStartVnode)) {
|
// Vnode moved left
|
patchVnode(oldEndVnode, newStartVnode);
|
api.insertBefore(parentElm, oldEndVnode.elm!, oldStartVnode.elm!);
|
oldEndVnode = oldCh[--oldEndIdx];
|
newStartVnode = newCh[++newStartIdx];
|
}
|
else {
|
if (isUndef(oldKeyToIdx)) {
|
oldKeyToIdx = createKeyToOldIdx(oldCh, oldStartIdx, oldEndIdx);
|
}
|
idxInOld = oldKeyToIdx[newStartVnode.key];
|
if (isUndef(idxInOld)) {
|
// New element
|
api.insertBefore(parentElm, createElm(newStartVnode), oldStartVnode.elm!);
|
}
|
else {
|
elmToMove = oldCh[idxInOld];
|
if (elmToMove.tag !== newStartVnode.tag) {
|
api.insertBefore(parentElm, createElm(newStartVnode), oldStartVnode.elm!);
|
}
|
else {
|
patchVnode(elmToMove, newStartVnode);
|
oldCh[idxInOld] = undefined;
|
api.insertBefore(parentElm, elmToMove.elm!, oldStartVnode.elm!);
|
}
|
}
|
newStartVnode = newCh[++newStartIdx];
|
}
|
}
|
if (oldStartIdx <= oldEndIdx || newStartIdx <= newEndIdx) {
|
if (oldStartIdx > oldEndIdx) {
|
before = newCh[newEndIdx + 1] == null ? null : newCh[newEndIdx + 1].elm;
|
addVnodes(parentElm, before, newCh, newStartIdx, newEndIdx);
|
}
|
else {
|
removeVnodes(parentElm, oldCh, oldStartIdx, oldEndIdx);
|
}
|
}
|
}
|
|
function patchVnode(oldVnode: SVGVNode, vnode: SVGVNode) {
|
const elm = (vnode.elm = oldVnode.elm)!;
|
const oldCh = oldVnode.children;
|
const ch = vnode.children;
|
if (oldVnode === vnode) {
|
return;
|
}
|
|
updateAttrs(oldVnode, vnode);
|
|
if (isUndef(vnode.text)) {
|
if (isDef(oldCh) && isDef(ch)) {
|
if (oldCh !== ch) {
|
updateChildren(elm, oldCh, ch);
|
}
|
}
|
else if (isDef(ch)) {
|
if (isDef(oldVnode.text)) {
|
api.setTextContent(elm, '');
|
}
|
addVnodes(elm, null, ch, 0, ch.length - 1);
|
}
|
else if (isDef(oldCh)) {
|
removeVnodes(elm, oldCh, 0, oldCh.length - 1);
|
}
|
else if (isDef(oldVnode.text)) {
|
api.setTextContent(elm, '');
|
}
|
}
|
else if (oldVnode.text !== vnode.text) {
|
if (isDef(oldCh)) {
|
removeVnodes(elm, oldCh, 0, oldCh.length - 1);
|
}
|
api.setTextContent(elm, vnode.text!);
|
}
|
}
|
|
export default function patch(oldVnode: SVGVNode, vnode: SVGVNode): SVGVNode {
|
if (sameVnode(oldVnode, vnode)) {
|
patchVnode(oldVnode, vnode);
|
}
|
else {
|
const elm = oldVnode.elm!;
|
const parent = api.parentNode(elm);
|
|
createElm(vnode);
|
|
if (parent !== null) {
|
api.insertBefore(parent, vnode.elm!, api.nextSibling(elm));
|
removeVnodes(parent, [oldVnode], 0, 0);
|
}
|
}
|
|
return vnode;
|
}
|