123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321 |
- /**
- * 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 SVGElement;
- 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 === 'style') {
- elm.style.cssText = cur as string;
- }
- 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;
- }
|