mirror of
https://github.com/wanglin2/mind-map.git
synced 2026-02-17 14:04:47 +08:00
1 line
9.2 KiB
JavaScript
1 line
9.2 KiB
JavaScript
(window["webpackJsonp"]=window["webpackJsonp"]||[]).push([["chunk-2d2254a4"],{e487:function(s,t,n){"use strict";n.r(t);var i=function(){var s=this;s._self._c;return s._m(0)},a=[function(){var s=this,t=s._self._c;return t("div",[t("h1",[s._v("如何渲染一个小地图")]),t("blockquote",[t("p",[s._v("要使用小地图需要注册小地图插件")])]),t("p",[s._v("小地图可以方便看到当前画布可视区域在思维导图树的哪个部分。")]),t("p",[s._v("虽然小地图需要你自行开发,不过"),t("code",[s._v("simple-mind-map")]),s._v("提供了一些方法来帮你快速的完成这一工作。")]),t("p",[s._v("小地图由两部分组成,一个是当前的画布内容,一个是视口框,当缩放、移动、元素过多时画布上可能只显示了思维导图的部分内容,可以通过视口框来查看当前视口所在位置,以及可以通过在小地图上拖动来快速定位。")]),t("p",[s._v("当注册了小地图插件后可以通过"),t("code",[s._v("mindMap.miniMap")]),s._v("获取到插件实例,然后通过"),t("code",[s._v("mindMap.miniMap.calculationMiniMap")]),s._v("方法即可获取小地图渲染需要的数据,返回的数据结构如下:")]),t("pre",{staticClass:"hljs"},[t("code",[s._v("{\n getImgUrl,"),t("span",{staticClass:"hljs-comment"},[s._v("// v0.8.0+,一个异步函数,你可以调用该函数,传递一个回调函数,回调函数可以接收一个参数,代表图片类型的小地图,你可以通过img标签进行渲染")]),s._v("\n svgHTML, "),t("span",{staticClass:"hljs-comment"},[s._v("// 小地图html")]),s._v("\n viewBoxStyle, "),t("span",{staticClass:"hljs-comment"},[s._v("// 视图框的位置信息")]),s._v("\n miniMapBoxScale, "),t("span",{staticClass:"hljs-comment"},[s._v("// 视图框的缩放值")]),s._v("\n miniMapBoxLeft, "),t("span",{staticClass:"hljs-comment"},[s._v("// 视图框的left值")]),s._v("\n miniMapBoxTop, "),t("span",{staticClass:"hljs-comment"},[s._v("// 视图框的top值")]),s._v("\n}\n")])]),t("p",[s._v("完整实现思路如下:")]),t("p",[s._v("1.准备一个宽高不为0的容器元素container,定位不为static")]),t("p",[s._v("2.在container内创建一个小地图容器元素miniMapContainer,绝对定位,设置变换中心点为"),t("code",[s._v("left top")]),s._v(":")]),t("pre",{staticClass:"hljs"},[t("code",[t("span",{staticClass:"hljs-attribute"},[s._v("transform-origin")]),s._v(": left top;\n")])]),t("p",[s._v("3.在container内创建一个视口框元素viewBoxContainer,绝对定位,设置边框样式,过渡属性(可选)")]),t("p",[s._v("4.监听data_change和view_data_change事件,最好也监听一下node_tree_render_end事件,防止初次渲染完毕后小地图没有刷新,在该事件内调用calculationMiniMap方法获取计算数据,然后将返回数据中的svgHTML渲染到miniMapContainer元素内:")]),t("pre",{staticClass:"hljs"},[t("code",[s._v("miniMapContainer.innerHTML = svgHTML\n")])]),t("p",[s._v("并且给miniMapContainer元素设置或更新如下样式:")]),t("pre",{staticClass:"hljs"},[t("code",[s._v("{\n "),t("span",{staticClass:"hljs-attr"},[s._v("transform")]),s._v(": "),t("span",{staticClass:"hljs-string"},[s._v("`scale("),t("span",{staticClass:"hljs-subst"},[s._v("${miniMapBoxScale}")]),s._v(")`")]),s._v(",\n "),t("span",{staticClass:"hljs-attr"},[s._v("left")]),s._v(": miniMapBoxLeft + "),t("span",{staticClass:"hljs-string"},[s._v("'px'")]),s._v(",\n "),t("span",{staticClass:"hljs-attr"},[s._v("top")]),s._v(": miniMapBoxTop + "),t("span",{staticClass:"hljs-string"},[s._v("'px'")]),s._v(",\n}\n")])]),t("p",[s._v("5.将viewBoxStyle对象设置为viewBoxContainer元素的样式")]),t("p",[s._v("到这一步,当画布上的思维导图变化了,小地图也会实时更新,并且视口框元素会实时反映视口在思维导图图形上的位置")]),t("p",[s._v("6.监听container元素的mousedown、mousemove事件,并且监听window的mouseup事件(如果将mouseup绑定到container元素上,那么鼠标移出container元素整个拖拽行为无法停止),分别调用小地图插件实例的三个方法即可实现鼠标拖动时画布上的思维导图也随之拖动的效果")]),t("p",[s._v("插件的完整信息可以参考"),t("a",{attrs:{href:"https://wanglin2.github.io/mind-map/#/doc/zh/miniMap"}},[s._v("miniMap")]),s._v("。")]),t("p",[s._v("在"),t("code",[s._v("v0.8.0+")]),s._v("版本之后,"),t("code",[s._v("calculationMiniMap")]),s._v("方法会返回"),t("code",[s._v("getImgUrl")]),s._v("属性,这是一个异步函数,你可以调用它并传递一个回调函数,回调函数可以接收一个参数,代表小地图图片数据,然后可以通过"),t("code",[s._v("img")]),s._v("标签进行渲染,替代前面的"),t("code",[s._v("svgHTML")]),s._v(",这样可以减少页面上的节点数量,能优化一定的性能:")]),t("pre",{staticClass:"hljs"},[t("code",[s._v("getImgUrl("),t("span",{staticClass:"hljs-function"},[t("span",{staticClass:"hljs-params"},[s._v("img")]),s._v(" =>")]),s._v(" {\n img.src = img\n})\n")])]),t("p",[s._v("在"),t("code",[s._v("v0.10.2+")]),s._v("版本之后,支持拖拽小地图内的视口框元素来同步移动画布位置,也就是可以拖拽"),t("code",[s._v("viewBoxContainer")]),s._v("元素,要实现这个特性,需要监听"),t("code",[s._v("viewBoxContainer")]),s._v("元素的"),t("code",[s._v("mousedown")]),s._v("(需要阻止冒泡,否则会触发container元素的mousedown事件)、"),t("code",[s._v("mousemove")]),s._v("事件,分别调用小地图插件实例的方法:")]),t("pre",{staticClass:"hljs"},[t("code",[t("span",{staticClass:"hljs-comment"},[s._v("// mousedown事件调用")]),s._v("\nmindMap.miniMap.onViewBoxMousedown(e)\n"),t("span",{staticClass:"hljs-comment"},[s._v("// mousemove事件调用")]),s._v("\nmindMap.miniMap.onViewBoxMousemove(e)\n")])]),t("p",[s._v("同时需要监听"),t("code",[s._v("mini_map_view_box_position_change")]),s._v("事件来实时更新"),t("code",[s._v("viewBoxContainer")]),s._v("元素:")]),t("pre",{staticClass:"hljs"},[t("code",[s._v("mindMap.on("),t("span",{staticClass:"hljs-string"},[s._v("'mini_map_view_box_position_change'")]),s._v(", "),t("span",{staticClass:"hljs-function"},[s._v("("),t("span",{staticClass:"hljs-params"},[s._v("{ left, right, top, bottom }")]),s._v(") =>")]),s._v(" {\n viewBoxStyle.left = left\n viewBoxStyle.right = right\n viewBoxStyle.top = top\n viewBoxStyle.bottom = bottom\n})\n")])]),t("p",[s._v("需要注意,如果你给"),t("code",[s._v("viewBoxContainer")]),s._v("元素设置了"),t("code",[s._v("css")]),s._v("的"),t("code",[s._v("transition")]),s._v("属性来增加过渡效果,那么在"),t("code",[s._v("mini_map_view_box_position_change")]),s._v("事件里需要临时去除,否则拖动会不流畅,可以在前面的"),t("code",[s._v("mouseup")]),s._v("事件里恢复。")]),t("h2",[s._v("完整示例")]),t("p",[s._v("在线Demo小地图代码:"),t("a",{attrs:{href:"https://github.com/wanglin2/mind-map/blob/main/web/src/pages/Edit/components/Navigator.vue"}},[s._v("Navigator.vue")]),s._v("。")]),t("iframe",{staticStyle:{width:"100%",height:"455px",border:"none"},attrs:{src:"https://wanglin2.github.io/playground/#eNrFV1tvG0UU/iujBWQHnLXbUgQmqUpLJZAShEoED2zlbrxje6v1zmp3bKeyLEGFArSp8sCt0EqkApFKSFGEEGrShl/jS/LEX+Cc2Znd2bXdhCcs2d6Z853LnDm37RvvBIHZ7VCjaixF9dANOIko7wSXLN9tByzkpE9C2igR5q+yjs+pUyJRy/Y81rtOG2RAGiFrkwJIKCQcq67vrNpBTLKMCLY9utiG3cW2HViG5RNi+R7lBPcQuUz8judZvuWXy2S0vz16tD96+Pdo72D045PJT1+M9o5Ofn9g+XXmR5zAL7ddn4afuA5vAe+5SmWK9h51my0OxItAE2LHD/8cf7+fCFccYIILJlxVjMAC5y2iPQsK03Vp7wrb+Ijf9qik9wcJtefy1lpo+5HLXeZLOg87NEFIHSiibicizs2gr9AGGo3kygzyGkNfZamdwLE5BZ+7q0gsLpDlS6SPPoZDH+89nuz9kBx6/N3++P4e0tD7wGcDh7wEU6oxwcR6x7PxMEKqHRSzLi/l3Qy2CG3jp3+Mf/5m9JXuZTLlYbNrex1quj7yr62ugAloiRl1m7hEFt3hMVyBdIomXLk2C84Rs3h09Tw40rJo8Pw8MJAsf4BRFl+IyJSIOqyHwVCk6YXkPa0hATZLSJt1UeMZhCByjpCOiAtNRDZik3NhzL5AQydI5WNo7W6Otn8ZP96EBD15vjPe+XK89fXw6b3h4b3hs79SEz6Ob+zsPskzKK0zdU52n43uPnmRzjO5MM8w96TDo/uTo73Rr7sQ4uNvD0bbD4aHm/883zp59Nnxb5/HNWbKjg9Z7OmrLdtvCmP6kIANXiIhpk+JcBaUyDrjHArm4NR7atheJC5qOk1MFAsQ/JuDECqxiuD/HAzYg/GAgT2TLk1dljZLVyUtoqgVIa3E055qDUVBIoR6VeKweqdNfW42Kb/mUXy8cvt9p1iQnEndKCyUYi7Mv2osHT+WgRuWoW3F25xucNy2jPHOwfHdO5M7B3Hzwc9ACkNgveV6Tkh9BH+aysiJm6klr2l4uDU53M0ryyqcofRGStNx/5MF6lHuKT7IFn6dMf4Bc6gKaeAsYLAVSqRQh6uDa7oh4IOFt+M+n6Yb84sFPEKtLvIAWPTeJZqIDsW4q/0HvA9m1XhIaQ3O5NCwBr+zeHrAw3qm7TjXumDxihtxCuEFARdXOeBJKt6UEiwZNZhiasK6dbZRC6QnUivn5D1WFPguleM5CyYsWHAKwxHYBytClhy3S+qeHUXLliG1vgv1yDIEWQJcJ6UmyQGQpTJQdaC6RiUx6dppWFwWZ8YqC3StHSmExWME1sQUgSuFkAotnlGpac23f8vQUTDLzMRokGqElQdQubDnWBkbLGxXyc0I+3vx5X6u4w8WbmZiHueeBq/mB67XSCHYKOSQUP10II5eMS6pG9zig9RS6QhBkU/6jeTuRPeQLLBzPJSePzP75DBKVj/XNKr58VSzOBsAZgQnFnecb8ApBx45FxD5zqmJz0Zk8qyelspa8MNSHFMgLssXCcswy/Hbg+wbJo3aZj2KLCMpLtjEVZ6owtjDKbWK7wWvCBwhKkmrEHA42napIIhKh9+X8vmkRKWM9nrEvA6PGVUgVeRKBItaTKtviTG5Sl6vVIINpXm23leV5rYdNl3Qq6QGULBcv6k2EtPNJK3PaPM5ZYM0W1unlidbyvSLmuWp57OvTqfpTzJ2kcH0gYcT4wpYkZOcz4hTJa+zECp+lZwPNggQXIeEzfXi+Qtvlsgbb+E37kbSBCXD80jFvBCdotzMZVBShubKEtKgzouANkpGHM748mveipgP79pChiUJEM5JT4cshyFcNHKzDI9mCBOV26YY+YvrIetFYNAt4JDVasbrdcw7nTfIJU86MAb/Ak6Okok="}})])}],e={},o=e,v=n("2877"),p=Object(v["a"])(o,i,a,!1,null,null,null);t["default"]=p.exports}}]); |