Source: Utils/Controller.js

import Event from "./Event";
import { Snap } from "../MMEditor";
/**
 * 控制器
 * @class
 * @extends Event
 */
class Controller extends Event {
	constructor(editor) {
		super();
		this.editor = editor;
		this.paper = editor.paper;
		this.svg = editor.svg;
		/**
		 * 缩放比例系数
		 * @type {number}
		 */
		this.scaleRatio = 0.01;
		this.scale = 1;
		this.x = 0;
		this.y = 0;
		/**
		 * 所有吸附节点
		 */
		this.achors = [];
		this.listenEvents();
	}

	/**
	 * 自适应,支持
	 */
	autoFit(center=true,vertical=true) {
		const data = this.editor.schema.getData();
		
		this.x = 0;
		this.y = 0;
		this.paper.transform(`scale(${this.scale})`);

		const width = this.editor.dom.node.clientWidth;
		const height = this.editor.dom.node.clientHeight;
		const bbox = this.paper.getBBox();
		const dx = ((width - bbox.width) / 2 - bbox.x)/this.scale;
		const dy = ((height - bbox.height) / 2 - bbox.y)/this.scale;
		data.nodes.forEach(node=>{
            if(center)node.x +=  dx;
			if(vertical)node.y += dy;
		});
		this.editor.schema.setData(data);
		this.editor.fire("autofit",{data});
		setTimeout(() => {
			this.paper.node.style.transition = null;
		}, 200);
	}


	listenEvents() {
		this.svg.mousedown(this.panStart);
		this.svg.mouseup(this.panStop);
		this.svg.node.addEventListener('mouseleave', this.panStop);
		this.svg.node.addEventListener("wheel", this.onWheel);
	}

	clear() {
		this.svg.unmousedown(this.panStart);
		this.svg.node.removeEventListener('mouseleave', this.panStop);
		this.svg.node.removeEventListener("wheel", this.onWheel);
	}

	/**
	 * 禁用滚轮缩放
	 */
	disableWheel() {
		this.svg.node.removeEventListener("wheel", this.onWheel);
	}

	/**
	 * 禁用鼠标移动
	 */
	disablePan() {
		this.svg.unmousedown(this.panStart);
	}
	/**
	 * 移动到指定位置
	 * @param  {number} x
	 * @param  {number} y
	 */
	pan(x, y) {
		this.x += x;
		this.y += y;
		this.update();
		this.editor.fire("panning")
	}

    /**
     * 移动到指定位置
     * @param {*} x 
     * @param {*} y 
     */
	moveTo(x,y){
		this.x = x;
		this.y = y;
		this.update();
	}

	onWheel = e => {
		e.preventDefault();
		if (e.ctrlKey) {// 双指
			const newScale = Math.min((1 - e.deltaY * this.scaleRatio),0.1);
            
			this.zoom(newScale, e.offsetX, e.offsetY);

		} else {
			this.pan(-e.deltaX, -e.deltaY)
		}
	};

	panStart = ev => {
		ev.preventDefault();
		if (ev.target.tagName !== "svg") {
			return;
		}
		this.startPosition = { x: ev.clientX, y: ev.clientY };
		this.matrix = this.svg.mousemove(this.panning);
	};

	panStop = ev => {
		ev.preventDefault();
		this.svg.unmousemove(this.panning);
		// this.svg.unmouseup(this.panStop);
		this.editor.fire("panEnd", { event: ev });
	};
	/**
	 * 缩放
	 * @param  {} scale 当前基础上 缩放多少
	 * @param  {number} cx=0 zoom 缩放中心点x
	 * @param  {number} cy=0 zoom 缩放中心点y
	 */
	zoom = (newScale, cx = 0, cy = 0) => {
		this.scale *= newScale;
		const dis = [(cx - this.x)*(newScale - 1),(cy - this.y)*(newScale - 1)];
		this.x -= dis[0];
		this.y -= dis[1];
		this.update();
		this.editor.fire("zoom", { scale:newScale });

	};

    /**
     * 滚动到指定位置
     * @param {*} newScale 
     */
	zoomTo = (newScale) => {
		this.scale = newScale;
		this.update();
	};

    /**
     * 做指定转换
     * @param {*} newScale 
     * @param {*} x 
     * @param {*} y 
     */
	transform = (newScale, x = 0, y = 0) => {
		this.scale = newScale;
		this.x = x;
		this.y = y;
		this.update();
		this.editor.fire("panning")
		this.editor.fire("zoom")
	};

	panning = ev => {
		ev.preventDefault();
		const p1 = { x: ev.clientX, y: ev.clientY };
		const p2 = this.startPosition;

		const deltaP = [p1.x - p2.x, p1.y - p2.y];
		this.x += deltaP[0];
		this.y += deltaP[1]
		this.update();
		this.startPosition = p1;
		this.editor.fire("panning",{event:ev})
	};

    /**
     * 更新最新的matrix
     */
	update(){
		this.paper.transform(`matrix(${this.scale},0,0,${this.scale},${this.x},${this.y})`);
	}
}
export default Controller;