@umbraci/jsmind 0.10.10 → 0.10.12

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -11,11 +11,11 @@ jsMind is mind map library built by javascript, it base on html5 canvas and svg.
11
11
 
12
12
  jsMind now supports a new enhanced plugin system with:
13
13
 
14
- - **Synchronous Initialization**: Plugins initialize before rendering, allowing them to affect the initial render
15
- - **Preload Support**: Control plugin initialization order (before or after core modules)
16
- - **Lifecycle Management**: Proper cleanup with `beforePluginRemove()` and `beforePluginDestroy()` hooks
17
- - **Dynamic Plugin Management**: Add or remove plugins at runtime
18
- - **Backward Compatibility**: Works alongside the existing plugin system
14
+ - **Synchronous Initialization**: Plugins initialize before rendering, allowing them to affect the initial render
15
+ - **Preload Support**: Control plugin initialization order (before or after core modules)
16
+ - **Lifecycle Management**: Proper cleanup with `beforePluginRemove()` and `beforePluginDestroy()` hooks
17
+ - **Dynamic Plugin Management**: Add or remove plugins at runtime
18
+ - **Backward Compatibility**: Works alongside the existing plugin system
19
19
 
20
20
  See [Enhanced Plugin System Documentation](docs/enhanced-plugin-system.md) for more information.
21
21
 
@@ -28,13 +28,13 @@ import MultilineTextV2 from './plugins/jsmind.multiline-text-v2.js';
28
28
  // Register plugin (before creating instance)
29
29
  jsMind.usePlugin(MultilineTextV2, {
30
30
  text_width: 250,
31
- min_height: 40
31
+ min_height: 40,
32
32
  });
33
33
 
34
34
  // Create jsMind instance
35
35
  const jm = new jsMind({
36
36
  container: 'jsmind_container',
37
- editable: true
37
+ editable: true,
38
38
  });
39
39
 
40
40
  // Show mind map with multiline text
@@ -44,10 +44,8 @@ jm.show({
44
44
  data: {
45
45
  id: 'root',
46
46
  topic: 'Root Node',
47
- children: [
48
- { id: 'node1', topic: 'Line 1\nLine 2\nLine 3' }
49
- ]
50
- }
47
+ children: [{ id: 'node1', topic: 'Line 1\nLine 2\nLine 3' }],
48
+ },
51
49
  });
52
50
  ```
53
51
 
@@ -0,0 +1,9 @@
1
+ /**
2
+ * @license BSD-3-Clause
3
+ * @copyright 2014-2025 hizzgdev@163.com
4
+ *
5
+ * Project Home:
6
+ * https://github.com/hizzgdev/jsmind/
7
+ */
8
+ !function(e,t){"object"==typeof exports&&"undefined"!=typeof module?t(exports,require("@umbraci/jsmind")):"function"==typeof define&&define.amd?define(["exports","@umbraci/jsmind"],t):t((e="undefined"!=typeof globalThis?globalThis:e||self).jsMindCopyPaste={},e.jsMind)}(this,function(e,t){"use strict";function i(e){return e&&"object"==typeof e&&"default"in e?e:{default:e}}var n=i(t);const o=new class{constructor(e){this.w=e,this.d=e.document,this.g=function(e){return this.d.getElementById(e)},this.c=function(e){return this.d.createElement(e)},this.t=function(e,t){e.hasChildNodes()?e.firstChild.nodeValue=t:e.appendChild(this.d.createTextNode(t))},this.h=function(e,t){t instanceof HTMLElement?(e.innerHTML="",e.appendChild(t)):e.innerHTML=t},this.i=function(e){return!!e&&"object"==typeof e&&1===e.nodeType&&"object"==typeof e.style&&"object"==typeof e.ownerDocument},this.on=function(e,t,i){e.addEventListener?e.addEventListener(t,i,!1):e.attachEvent("on"+t,i)}}}(window),a={file:{read:function(e,t){var i=new FileReader;i.onload=function(){"function"==typeof t&&t(this.result,e.name)},i.readAsText(e)},save:function(e,t,i){var n;if("function"==typeof o.w.Blob)n=new Blob([e],{type:t});else{var a=new(o.w.BlobBuilder||o.w.MozBlobBuilder||o.w.WebKitBlobBuilder||o.w.MSBlobBuilder);a.append(e),n=a.getBlob(t)}if(navigator.msSaveBlob)navigator.msSaveBlob(n,i);else{var r=(o.w.URL||o.w.webkitURL).createObjectURL(n),s=o.c("a");if("download"in s){s.style.visibility="hidden",s.href=r,s.download=i,o.d.body.appendChild(s);var d=o.d.createEvent("MouseEvents");d.initEvent("click",!0,!0),s.dispatchEvent(d),o.d.body.removeChild(s)}else location.href=r}}},json:{json2string:function(e){return JSON.stringify(e)},string2json:function(e){return JSON.parse(e)},merge:function(e,t){for(var i in t)i in e?"object"!=typeof e[i]||"[object object]"!=Object.prototype.toString.call(e[i]).toLowerCase()||e[i].length?e[i]=t[i]:a.json.merge(e[i],t[i]):e[i]=t[i];return e}},uuid:{newid:function(){return((new Date).getTime().toString(16)+Math.random().toString(16).substring(2)).substring(2,18)}},text:{is_empty:function(e){return!e||0==e.replace(/\s*/,"").length}}};class r{constructor(e,t={}){this.jsMind=e,this.options={enabled:!0,shortcuts:{copy:"meta+c",paste:"meta+v",cut:"meta+x"},...t},this.clipboardData=null,this.logger={info:e=>console.info("[CopyPaste]",e),warn:e=>console.warn("[CopyPaste]",e),error:e=>console.error("[CopyPaste]",e),debug:e=>console.debug("[CopyPaste]",e)},this.init()}init(){return this.logger.info("CopyPaste plugin initialized"),!0}handleCopy(e){if(!this.options.enabled)return;const t=this.jsMind.get_selected_node();if(t)try{e.preventDefault(),this.clipboardData={id:t.id,topic:t.topic,data:t.data||{},direction:t.direction,expanded:t.expanded,children:this.copyChildren(t)},this.logger.info("Node copied:",t.topic),this.showMessage("节点已复制")}catch(e){this.logger.error("Copy failed:",e),this.showMessage("复制失败","error")}else this.logger.warn("No node selected for copying")}handlePaste(e){if(!this.options.enabled)return;if(!this.clipboardData)return this.logger.warn("No data in clipboard"),void this.showMessage("剪贴板为空,请先复制节点","warning");const t=this.jsMind.get_selected_node();if(!t)return this.logger.warn("No target node selected for pasting"),void this.showMessage("请选择目标节点","warning");try{e.preventDefault();const i=this.prepareBatchData(this.clipboardData),n=this.jsMind.add_nodes(t,[i]);n&&n.length>0&&(this.logger.info("Nodes pasted successfully:",this.clipboardData.topic),this.showMessage("粘贴成功"))}catch(e){this.logger.error("Paste failed:",e),this.showMessage("粘贴失败","error")}}handleCut(e){if(!this.options.enabled)return;const t=this.jsMind.get_selected_node();if(t){if(t.isroot)return this.logger.warn("Cannot cut root node"),void this.showMessage("不能剪切根节点","warning");try{e.preventDefault(),this.clipboardData={id:t.id,topic:t.topic,data:t.data||{},direction:t.direction,expanded:t.expanded,children:this.copyChildren(t)};this.jsMind.remove_node(t)&&(this.logger.info("Node cut:",t.topic),this.showMessage("节点已剪切"))}catch(e){this.logger.error("Cut failed:",e),this.showMessage("剪切失败","error")}}else this.logger.warn("No node selected for cutting")}copyChildren(e){return e.children&&0!==e.children.length?e.children.map(e=>({id:e.id,topic:e.topic,data:e.data||{},direction:e.direction,expanded:e.expanded,children:this.copyChildren(e)})):[]}prepareBatchData(e){return{id:a.uuid.newid(),topic:e.topic,data:e.data||{},children:e.children&&e.children.length>0?e.children.map(e=>this.prepareBatchData(e)):void 0}}pasteChildren(e,t){if(!t||0===t.length)return[];const i=t.map(e=>this.prepareBatchData(e));return this.jsMind.add_nodes(e,i)}showMessage(e,t="info"){console.log(`[CopyPaste ${t.toUpperCase()}]: ${e}`)}clear(){this.clipboardData=null,this.logger.info("Clipboard cleared")}getStatus(){return{hasData:!!this.clipboardData,data:this.clipboardData}}}const s=new n.default.plugin("copy-paste",function(e,t){var i=t||{};i.shortcuts=i.shortcuts||{copy:"meta+c",paste:"meta+v",cut:"meta+x"};var n=new r(e,i);e.view&&e.view.e_panel&&e.view.e_panel.addEventListener("keydown",function(e){!e.metaKey&&!e.ctrlKey||"c"!==e.key||e.shiftKey||e.altKey?!e.metaKey&&!e.ctrlKey||"v"!==e.key||e.shiftKey||e.altKey?!e.metaKey&&!e.ctrlKey||"x"!==e.key||e.shiftKey||e.altKey||n.handleCut(e):n.handlePaste(e):n.handleCopy(e)}),e.copy_paste_handler=n});n.default.register_plugin(s),e.copy_paste_plugin=s,e.default=r,Object.defineProperty(e,"__esModule",{value:!0})});
9
+ //# sourceMappingURL=jsmind.copy-paste.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"jsmind.copy-paste.js","sources":["../src/jsmind.dom.js","../src/jsmind.util.js","../src/plugins/copy-paste/copy-paste-plugin-simple.js"],"sourcesContent":["/**\n * @license BSD\n * @copyright 2014-2025 UmbraCi\n *\n * Project Home:\n * https://github.com/UmbraCi/jsmind/\n */\n\n/**\n * Lightweight DOM helpers bound to a window.\n */\nclass Dom {\n /**\n * @param {Window} w\n */\n constructor(w) {\n /** @type {Window} */\n this.w = w;\n /** @type {Document} */\n this.d = w.document;\n /**\n * Get element by id.\n * @param {string} id\n * @returns {HTMLElement|null}\n */\n this.g = function (id) {\n return this.d.getElementById(id);\n };\n /**\n * Create element with given tag.\n * @param {string} tag\n * @returns {HTMLElement}\n */\n this.c = function (tag) {\n return this.d.createElement(tag);\n };\n /**\n * Set text content for element.\n * @param {HTMLElement} n\n * @param {string} t\n */\n this.t = function (n, t) {\n if (n.hasChildNodes()) {\n n.firstChild.nodeValue = t;\n } else {\n n.appendChild(this.d.createTextNode(t));\n }\n };\n\n /**\n * Set inner HTML or append element.\n * @param {HTMLElement} n\n * @param {string|HTMLElement} t\n */\n this.h = function (n, t) {\n if (t instanceof HTMLElement) {\n n.innerHTML = '';\n n.appendChild(t);\n } else {\n n.innerHTML = t;\n }\n };\n // detect isElement\n /**\n * Runtime check for HTMLElement.\n * @param {unknown} el\n * @returns {el is HTMLElement}\n */\n this.i = function (el) {\n return (\n !!el &&\n typeof el === 'object' &&\n el.nodeType === 1 &&\n typeof el.style === 'object' &&\n typeof el.ownerDocument === 'object'\n );\n };\n\n //target,eventType,handler\n /**\n * Add event listener with legacy fallback.\n * @param {HTMLElement} t\n * @param {string} e\n * @param {(ev:Event)=>void} h\n */\n this.on = function (t, e, h) {\n if (!!t.addEventListener) {\n t.addEventListener(e, h, false);\n } else {\n t.attachEvent('on' + e, h);\n }\n };\n }\n}\n\nexport const $ = new Dom(window);\n","/**\n * @license BSD\n * @copyright 2014-2025 UmbraCi\n *\n * Project Home:\n * https://github.com/UmbraCi/jsmind/\n */\n\nimport { $ } from './jsmind.dom.js';\n\n/**\n * Misc utility collection.\n * @type {{\n * file: { read: (file: File, cb:(result:string,name:string)=>void)=>void, save:(data:string,type:string,name:string)=>void},\n * json: { json2string:(v:unknown)=>string, string2json:(s:string)=>unknown, merge:(b:object,a:object)=>object },\n * uuid: { newid:()=>string },\n * text: { is_empty:(s?:string)=>boolean }\n * }}\n */\nexport const util = {\n file: {\n read: function (file_data, fn_callback) {\n var reader = new FileReader();\n reader.onload = function () {\n if (typeof fn_callback === 'function') {\n fn_callback(this.result, file_data.name);\n }\n };\n reader.readAsText(file_data);\n },\n\n save: function (file_data, type, name) {\n var blob;\n if (typeof $.w.Blob === 'function') {\n blob = new Blob([file_data], { type: type });\n } else {\n var BlobBuilder =\n $.w.BlobBuilder ||\n $.w.MozBlobBuilder ||\n $.w.WebKitBlobBuilder ||\n $.w.MSBlobBuilder;\n var bb = new BlobBuilder();\n bb.append(file_data);\n blob = bb.getBlob(type);\n }\n if (navigator.msSaveBlob) {\n navigator.msSaveBlob(blob, name);\n } else {\n var URL = $.w.URL || $.w.webkitURL;\n var blob_url = URL.createObjectURL(blob);\n var anchor = $.c('a');\n if ('download' in anchor) {\n anchor.style.visibility = 'hidden';\n anchor.href = blob_url;\n anchor.download = name;\n $.d.body.appendChild(anchor);\n var evt = $.d.createEvent('MouseEvents');\n evt.initEvent('click', true, true);\n anchor.dispatchEvent(evt);\n $.d.body.removeChild(anchor);\n } else {\n location.href = blob_url;\n }\n }\n },\n },\n\n json: {\n json2string: function (json) {\n return JSON.stringify(json);\n },\n string2json: function (json_str) {\n return JSON.parse(json_str);\n },\n merge: function (b, a) {\n for (var o in a) {\n if (o in b) {\n if (\n typeof b[o] === 'object' &&\n Object.prototype.toString.call(b[o]).toLowerCase() == '[object object]' &&\n !b[o].length\n ) {\n util.json.merge(b[o], a[o]);\n } else {\n b[o] = a[o];\n }\n } else {\n b[o] = a[o];\n }\n }\n return b;\n },\n },\n\n uuid: {\n newid: function () {\n return (\n new Date().getTime().toString(16) + Math.random().toString(16).substring(2)\n ).substring(2, 18);\n },\n },\n\n text: {\n is_empty: function (s) {\n if (!s) {\n return true;\n }\n return s.replace(/\\s*/, '').length == 0;\n },\n },\n};\n","/**\n * @license BSD-3-Clause\n * @copyright 2014-2025 hizzgdev@163.com\n *\n * Project Home:\n * https://github.com/hizzgdev/jsmind/\n */\n\nimport { Plugin } from '../../jsmind.plugin.js';\nimport { util } from '../../jsmind.util.js';\nimport jsMind from '@umbraci/jsmind';\n\n/**\n * 简化版复制粘贴插件\n */\n\nclass CopyPasteHandler {\n constructor(jm, options = {}) {\n this.jsMind = jm;\n this.options = {\n enabled: true,\n shortcuts: {\n copy: 'meta+c',\n paste: 'meta+v',\n cut: 'meta+x'\n },\n ...options\n };\n this.clipboardData = null;\n\n this.logger = {\n info: (msg) => console.info('[CopyPaste]', msg),\n warn: (msg) => console.warn('[CopyPaste]', msg),\n error: (msg) => console.error('[CopyPaste]', msg),\n debug: (msg) => console.debug('[CopyPaste]', msg)\n };\n\n this.init();\n }\n\n init() {\n this.logger.info('CopyPaste plugin initialized');\n return true;\n }\n\n handleCopy(event) {\n if (!this.options.enabled) return;\n\n const selectedNode = this.jsMind.get_selected_node();\n if (!selectedNode) {\n this.logger.warn('No node selected for copying');\n return;\n }\n\n try {\n event.preventDefault();\n\n // 复制节点数据\n this.clipboardData = {\n id: selectedNode.id,\n topic: selectedNode.topic,\n data: selectedNode.data || {},\n direction: selectedNode.direction,\n expanded: selectedNode.expanded,\n children: this.copyChildren(selectedNode)\n };\n\n this.logger.info('Node copied:', selectedNode.topic);\n\n // 显示成功消息\n this.showMessage('节点已复制');\n\n } catch (error) {\n this.logger.error('Copy failed:', error);\n this.showMessage('复制失败', 'error');\n }\n }\n\n handlePaste(event) {\n if (!this.options.enabled) return;\n if (!this.clipboardData) {\n this.logger.warn('No data in clipboard');\n this.showMessage('剪贴板为空,请先复制节点', 'warning');\n return;\n }\n\n const targetNode = this.jsMind.get_selected_node();\n if (!targetNode) {\n this.logger.warn('No target node selected for pasting');\n this.showMessage('请选择目标节点', 'warning');\n return;\n }\n\n try {\n event.preventDefault();\n\n // 准备批量数据(包含所有子节点)\n const batchData = this.prepareBatchData(this.clipboardData);\n\n // 使用 add_nodes 批量创建节点\n const createdNodes = this.jsMind.add_nodes(targetNode, [batchData]);\n\n if (createdNodes && createdNodes.length > 0) {\n this.logger.info('Nodes pasted successfully:', this.clipboardData.topic);\n this.showMessage('粘贴成功');\n }\n\n } catch (error) {\n this.logger.error('Paste failed:', error);\n this.showMessage('粘贴失败', 'error');\n }\n }\n\n handleCut(event) {\n if (!this.options.enabled) return;\n\n const selectedNode = this.jsMind.get_selected_node();\n if (!selectedNode) {\n this.logger.warn('No node selected for cutting');\n return;\n }\n\n if (selectedNode.isroot) {\n this.logger.warn('Cannot cut root node');\n this.showMessage('不能剪切根节点', 'warning');\n return;\n }\n\n try {\n event.preventDefault();\n\n // 先复制到剪贴板\n this.clipboardData = {\n id: selectedNode.id,\n topic: selectedNode.topic,\n data: selectedNode.data || {},\n direction: selectedNode.direction,\n expanded: selectedNode.expanded,\n children: this.copyChildren(selectedNode)\n };\n\n // 删除原节点\n const removed = this.jsMind.remove_node(selectedNode);\n\n if (removed) {\n this.logger.info('Node cut:', selectedNode.topic);\n this.showMessage('节点已剪切');\n }\n\n } catch (error) {\n this.logger.error('Cut failed:', error);\n this.showMessage('剪切失败', 'error');\n }\n }\n\n copyChildren(node) {\n if (!node.children || node.children.length === 0) {\n return [];\n }\n\n return node.children.map(child => ({\n id: child.id,\n topic: child.topic,\n data: child.data || {},\n direction: child.direction,\n expanded: child.expanded,\n children: this.copyChildren(child)\n }));\n }\n\n /**\n * 准备批量数据,将节点数据转换为 add_nodes 所需的格式\n * @param {Object} nodeData - 节点数据\n * @returns {Object} - 格式化后的批量数据\n */\n prepareBatchData(nodeData) {\n return {\n id: util.uuid.newid(),\n topic: nodeData.topic,\n data: nodeData.data || {},\n children: nodeData.children && nodeData.children.length > 0\n ? nodeData.children.map(child => this.prepareBatchData(child))\n : undefined\n };\n }\n\n pasteChildren(parentNode, children) {\n // 保留此方法以备兼容性,但新的实现不再使用递归方式\n if (!children || children.length === 0) return [];\n\n const batchData = children.map(child => this.prepareBatchData(child));\n return this.jsMind.add_nodes(parentNode, batchData);\n }\n\n showMessage(message, type = 'info') {\n // 简单的消息显示\n console.log(`[CopyPaste ${type.toUpperCase()}]: ${message}`);\n }\n\n clear() {\n this.clipboardData = null;\n this.logger.info('Clipboard cleared');\n }\n\n getStatus() {\n return {\n hasData: !!this.clipboardData,\n data: this.clipboardData\n };\n }\n}\n\n/**\n * 复制粘贴插件注册\n */\nexport const copy_paste_plugin = new jsMind.plugin('copy-paste', function (jm, options) {\n // 确保options存在且包含shortcuts\n var pluginOptions = options || {};\n pluginOptions.shortcuts = pluginOptions.shortcuts || {\n copy: 'meta+c',\n paste: 'meta+v',\n cut: 'meta+x'\n };\n\n var handler = new CopyPasteHandler(jm, pluginOptions);\n\n // 直接注册键盘事件监听器,而不是使用不存在的add_shortcut方法\n if (jm.view && jm.view.e_panel) {\n jm.view.e_panel.addEventListener('keydown', function(event) {\n // 检查是否按下了复制快捷键\n if ((event.metaKey || event.ctrlKey) && event.key === 'c' && !event.shiftKey && !event.altKey) {\n handler.handleCopy(event);\n }\n // 检查是否按下了粘贴快捷键\n else if ((event.metaKey || event.ctrlKey) && event.key === 'v' && !event.shiftKey && !event.altKey) {\n handler.handlePaste(event);\n }\n // 检查是否按下了剪切快捷键\n else if ((event.metaKey || event.ctrlKey) && event.key === 'x' && !event.shiftKey && !event.altKey) {\n handler.handleCut(event);\n }\n });\n }\n\n // 将处理器实例附加到jm对象,方便外部调用\n jm.copy_paste_handler = handler;\n});\n\n// 注册插件\njsMind.register_plugin(copy_paste_plugin);\n\nexport default CopyPasteHandler;"],"names":["$","constructor","w","this","d","document","g","id","getElementById","c","tag","createElement","t","n","hasChildNodes","firstChild","nodeValue","appendChild","createTextNode","h","HTMLElement","innerHTML","i","el","nodeType","style","ownerDocument","on","e","addEventListener","attachEvent","window","util","file","read","file_data","fn_callback","reader","FileReader","onload","result","name","readAsText","save","type","blob","Blob","bb","BlobBuilder","MozBlobBuilder","WebKitBlobBuilder","MSBlobBuilder","append","getBlob","navigator","msSaveBlob","blob_url","URL","webkitURL","createObjectURL","anchor","visibility","href","download","body","evt","createEvent","initEvent","dispatchEvent","removeChild","location","json","json2string","JSON","stringify","string2json","json_str","parse","merge","b","a","o","Object","prototype","toString","call","toLowerCase","length","uuid","newid","Date","getTime","Math","random","substring","text","is_empty","s","replace","CopyPasteHandler","jm","options","jsMind","enabled","shortcuts","copy","paste","cut","clipboardData","logger","info","msg","console","warn","error","debug","init","handleCopy","event","selectedNode","get_selected_node","preventDefault","topic","data","direction","expanded","children","copyChildren","showMessage","handlePaste","targetNode","batchData","prepareBatchData","createdNodes","add_nodes","handleCut","isroot","remove_node","node","map","child","nodeData","undefined","pasteChildren","parentNode","message","log","toUpperCase","clear","getStatus","hasData","copy_paste_plugin","plugin","pluginOptions","handler","view","e_panel","metaKey","ctrlKey","key","shiftKey","altKey","copy_paste_handler","register_plugin"],"mappings":"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"}