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;