Source: Plugins/Minimap.js

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;