Source: Shape/Graph.js

import Node from './Node';
import Line from './Line';
import Event from '../Utils/Event';
import Animation from './Animation';
import AnchorLine from './AnchorLine';
const backSvg = require('../back.svg');
/**
 * @class
 * @extends Event
 */
class Graph extends Event {
	constructor(editor) {
		super();
		this.editor = editor;
        /**
         * @property {Node} node
         */
		this.node = new Node(this);
        /**
         * @property {Line} line
         */
		this.line = new Line(this);
         /**
         * @prop {AnchorLine} anchorLine
         */
		this.anchorLine = new AnchorLine(this);
		this.node.linkPointsG.before(this.line.lineG);
        /**
         * @prop {Animation} animation
         */
		this.animation = Animation;

		// 模式:操作、查看模式
		this.mode = editor.config.mode;

		this.listenEvents();
		if (this.editor.config.showBackGrid) {this.addBack();}
	}

	addBack() {
		this.editor.container.select('.mm-editor-back').node.style.backgroundImage = `url(${backSvg})`;
	}
	 
	listenEvents() {
		this.on('node:move', ({ node }) => {
			this.line.updateByNode(node);
		});
		this.editor.svg.attr({
			tabindex: '0'
		});
		this.editor.svg.click(e => {
			if (e.target.tagName === 'svg') {
				this.fire('paper:click', e);
			}
		});

		// 查看模式不能删除节点、线条;如果存在部分可操作则自己在业务中监听处理相关逻辑
		if (this.mode !== 'view') {
			document.addEventListener('keydown', this.onKeyDown);
		}
		
		this.on('line:drag', () => {
			this.linkStatus = 'lineing';
			for (let key in this.node.nodes) {
				const node = this.node.nodes[key];
				node.linkPoints.forEach(point => {
					point.attr({
						display: 'block'
					});
				});
			}
		});
		this.on('line:drop', () => {
			this.linkStatus = 'none';
			for (let key in this.node.nodes) {
				const node = this.node.nodes[key];
				node.linkPoints.forEach(point => {
					point.attr({
						display: 'none'
					});
				});
			}
		});
	}

    //todo:
	onKeyDown = (e) => {
		if (["TEXTAREA", "INPUT"].indexOf(document.activeElement.tagName) > -1 && document.activeElement.getAttribute("contenteditable")!=="false") {
            return;
        }
		if (e.key === 'Backspace') {
			const deleteKeys = [];
			for (let key in this.node.actives) {
				// 不触发事件
				this.node.deleteNode(this.node.actives[key], true);
				delete this.node.actives[key];
				deleteKeys.push(key);
			}
			this.line.activeLine && this.line.deleteLine(this.line.activeLine);
            /**
             * @event Graph#delete
             * @type {Object}
             */
			this.fire('delete', { event: e, deleteKeys });
		}
		if (e.keyCode === 'C'.charCodeAt(0) && (e.metaKey || e.ctrlKey)) {
			/**
             * @event Graph#copy
             * @type {Object}
             */
            this.fire('copy', { event: e });
		}
		if (e.keyCode === 'V'.charCodeAt(0) && (e.metaKey || e.ctrlKey)) {
            /**
             * @event Graph#paste
             * @type {Object}
             */
			this.fire('paste', { event: e });
		}
		if (e.keyCode === 'Z'.charCodeAt(0) && (e.metaKey || e.ctrlKey) && !e.shiftKey) {
			this.editor.schema.undo();
		}
		if (e.keyCode === 'Z'.charCodeAt(0) && (e.metaKey || e.ctrlKey) && e.shiftKey) {
			this.editor.schema.redo();
		}
		e.preventDefault();
		return false;
	}

	/**
	 * 添加链接点事件
	 */
	addLinkHoverEvent() {
		const linkPoints = this.editor.paper.selectAll('.mm-link-points');
		linkPoints.forEach(point => {
			point.mouseover(this.onLinkPointHover);
			point.mouseout(this.onLinkPointOut);
		});
	}

	onLinkPointHover = ele => {
		this.hoverLinkPoint = ele;
	};
	onLinkPointOut = ele => {
		this.hoverLinkPoint = undefined;
	};

	/**
	 * 关闭线hover事件
	 */
	offLinkHoverEvent() {
		const linkPoints = this.editor.paper.selectAll('.mm-link-points');
		linkPoints.forEach(point => {
			point.unmouseover(this.onLinkPointHover);
			point.unmouseout(this.onLinkPointOut);
		});
		this.hoverLinkPoint = undefined;
	}

	/**
	 *
	 * @param {*} data
	 */
	async render(data) {
        /**
         * @event Graph#beforeRender 渲染之前触发
         */
		this.fire('beforeRender')
		this.data = data;
		await this.node.render(data.nodesMap);
		await this.line.render(data.linesMap);
        /**
         * @event Graph#render  渲染后触发
         */
        this.fire('render')
	}

	/**
	 * 清空画布
	 */
	clearGraph() {
		this.line.clear();
		this.node.clear();
	}

	destroy(){
		this.clearGraph();
		this.clear();
		document.removeEventListener('keydown', this.onKeyDown);
	}
}
export default Graph;