import Canvg from 'canvg';
import { Snap } from "../MMEditor";
// 使用html
/**
* @class
*/
class MiniMap {
constructor(editor) {
this.editor = editor;
const {minimap=[]} = editor.config
this.width = minimap.width||160;
this.height = minimap.height||160;
this.padding = minimap.padding||20;
this.scale = minimap.scale||10;
}
init() {
const dom = `<div class="mm-minimap" >
<canvas width="100%" height="100%"></canvas>
<div class="drag-rect" style="left:${this.padding}px;top:${this.padding}px">
<div class="drag-point"></div>
</div>
</div>`;
const can = document.createElement("div");
can.innerHTML = dom;
this.container = can.querySelector(".mm-minimap");
this.editor.container.append(this.container);
this.canvas = this.container.querySelector("canvas");
this.ctx = this.canvas.getContext("2d");
this.drag = Snap(this.container.querySelector(".drag-rect"));
this.dragPoint = Snap(this.container.querySelector(".drag-point"));
const bbox = this.container.getBoundingClientRect();
this.width = bbox.width;
this.height = bbox.height;
this.canvas.width = this.width;
this.canvas.height = this.height;
this.initEvent();
}
initEvent() {
const {controller} = this.editor;
this.canvas.addEventListener('click',(e)=>{
const left = e.offsetX - this.dragBBox.width/2;
const top = e.offsetY - this.dragBBox.height/2;
this.drag.node.style.left = left + "px";
this.drag.node.style.top = top + "px";
controller.moveTo(
-(left - this.padding)*this.scale*controller.scale,
-(top - this.padding)*this.scale*controller.scale
);
this.resetDrag();
})
this.drag.drag((dx, dy) => {
const dleft = dx + this.dragStart.x;
const dtop = dy + this.dragStart.y;
const left = Math.min(Math.max(dleft, 0),this.width - this.dragBBox.width + this.padding*2);
const top = Math.min(Math.max(dtop, 0),this.height - this.dragBBox.height + this.padding*2);
this.drag.node.style.left = left + "px";
this.drag.node.style.top = top + "px";
controller.moveTo(
-(left - this.padding)*this.scale*controller.scale,
-(top - this.padding)*this.scale*controller.scale
)
}, () => {
const { style } = this.drag.node;
this.dragStart = {
x: style.left ? parseInt(style.left.split("px")[0]) : 0,
y: style.top ? parseInt(style.top.split("px")[0]) : 0
};
});
this.dragPoint.drag((dx, dy) => {
const ratio = this.svgBBox.width/this.svgBBox.height;
let height = Math.max(dy + this.dragStartBBox.height, 10);
let width = height*ratio;
this.drag.node.style.width = width+'px';
this.drag.node.style.height = height+'px';
controller.scale = this.svgBBox.width/(width* this.scale);
controller.update();
}, (x, y, e) => {
e.preventDefault();
e.stopPropagation()
this.dragStartBBox = this.drag.node.getBoundingClientRect();
return false;
});
this.editor.on("change", this.render);
this.editor.on("format", this.render);
this.editor.on("autofit", this.resetDrag);
this.editor.on("panning",this.resetDrag);
this.editor.on("zoom",this.resetDrag);
}
/**
* 重新计算拖拽框位置
*/
resetDrag=()=>{
const {x,y,scale} = this.editor.controller;
const {padding} = this;
if(!this.svgBBox)return;
/**
* 这里虽然坐标整体都缩小了10倍,但是用户画布放大的scale倍,在这个坐标系下永远都是1倍,不会随着用户放大而放大,
* 所以这里求得的左上角便宜坐标实际上还是标准倍率吸下的,需要再放大用户的倍率才能得到最终的效果,
* 用户画布=》缩小10倍画布到用户scale*this.sclae=》还原回基准this.scale
*/
this.drag.node.style.left = -x/scale/this.scale + padding +'px';
this.drag.node.style.top = -y/scale/this.scale + padding +'px';
this.dragBBox = {
width:this.svgBBox.width / this.scale /scale,
height:this.svgBBox.height/ this.scale /scale
}
this.drag.node.style.width = this.dragBBox.width+'px';
this.drag.node.style.height = this.dragBBox.height+'px';
}
/**
* 重新渲染小地图
*/
render = () => {
clearTimeout(this.timeout);
this.timeout = setTimeout(async () => {
const node = this.editor.svg.node;
const svgBBox = node.getBoundingClientRect();
const images = node.querySelectorAll("image")||[];
images.forEach(img=>{
img.setAttribute("xmlns:xlink", "http://www.w3.org/1999/xlink");
})
const svg = node.innerHTML.replace(/\"mm-editor-paper\" transform=\"matrix\([-\d\,]+\)\"/i,`"mm-editor-paper" `).replace(
/transform=\"matrix\([-\d\,]+\)\" class=\"mm-editor-paper\"/i,`class="mm-editor-paper" `
);
const paperBBox = this.editor.paper.node.getBBox();
this.scale = Math.max((paperBBox.width)/(this.width-this.padding*2),paperBBox.height/(this.height-this.padding*2),10);
const x = this.padding;
const y = this.padding;
const m = new Snap.Matrix();
m.translate(x,y);
m.scale(1 / this.scale);
this.svgBBox = svgBBox;
this.resetDrag();
this.converting = await Canvg.fromString(this.ctx, `<g transform="${m.toString()}" class="minimap-graph">${svg}</g>`,{
ignoreMouse:true,
ignoreDimensions:true,
ignoreAnimation:true
});
this.converting.render();
}, 200)
}
destroy() {
clearTimeout(this.timeout);
this.editor.off("change", this.render);
this.editor.off("format", this.render);
this.editor.off("autofit", this.resetDrag);
this.editor.off("panning",this.resetDrag);
this.editor.off("zoom",this.resetDrag);
this.drag.undrag();
this.dragPoint.undrag();
this.drag.remove();
this.dragPoint.remove();
}
}
export default MiniMap;