公司研發的管理系統有工做流圖形化設計和查看功能,這個功能的開發歷史比較久遠。在那個暗無天日的年月裏,IE幾乎一統江湖,因此瓜熟蒂落地採用了當時紅極一時的VML技術。javascript
後來的事情你們都知道了,IE開始走下坡路,VML這個技術如今早已滅絕,致使原來的工做流圖形化功能徹底不能使用,因此須要採用新技術來重寫工做流圖形化功能。java
多方對比以後,決定採用zrender庫來實現(關於zrender庫的介紹,請看http://ecomfe.github.io/zrender/),花了一天的時間,終於作出了一個大體的效果模型,以下圖所示:git
流程圖由兩類部件組成:活動部件和鏈接弧部件,每一類部件包含多個性狀不一樣的部件。github
以活動部件爲例,圓形的是開始活動,平行四邊形是自動活動,長方形是人工活動,等等。canvas
在代碼實現上,定義了Unit(部件基類),全部的部件都繼承自這個基類。經過Graph類來管理整個流程圖,包括全部部件、上下文菜單等等都由Graph來統一管理和調度,代碼以下:app
var Libra = {}; Libra.Workflow = {}; Libra.Workflow.Graph = function(type, options){ var graph = this, activities = {}, transitions = {}; var zrenderInstance, contextMenuContainer; this.type = type; this.addActivity = function(activity){ activity.graph = graph; activities[activity.id] = {object:activity}; }; this.getActivity = function(id){ return activities[id].object; }; this.addTransition = function(transition){ transition.graph = graph; transitions[transition.id] = {object:transition}; }; function modElements(shapes){ shapes.each(function(shape){ zrenderInstance.modElement(shape); }); return shapes; } // 當前正在拖放的節點 var dragingActivity = null; // 活動節點拖放開始 this.onActivityDragStart = function(activity){ dragingActivity = activity; }; // 活動節點拖放結束 this.onActivityDragEnd = function(){ if(dragingActivity) refreshActivityTransitions(dragingActivity); dragingActivity = null; }; // 拖動過程處理 function zrenderInstanceOnMouseMove(){ if(dragingActivity != null) refreshActivityTransitions(dragingActivity); } // 刷新活動相關的全部鏈接弧 function refreshActivityTransitions(activity){ var activityId = activity.id; for(var key in transitions){ var transition = transitions[key].object; if(transition.from === activityId || transition.to == activityId){ zrenderInstance.refreshShapes(modElements(transition.refresh(graph))); } } } // 當前選中的部件 var selectedUnit = null; this.onUnitSelect = function(unit){ if(selectedUnit) zrenderInstance.refreshShapes(modElements(selectedUnit.unselect(graph))); zrenderInstance.refreshShapes(modElements(unit.select(graph))); selectedUnit = unit; }; // 記錄當前鼠標在哪一個部件上,能夠用來生成上下文相關菜單 var currentUnit = null; this.onUnitMouseOver = function(unit){ currentUnit = unit; }; this.onUnitMouseOut = function(unit){ if(currentUnit === unit) currentUnit = null; }; // 上下文菜單事件響應 function onContextMenu(event){ Event.stop(event); if(currentUnit) currentUnit.showContextMenu(event, contextMenuContainer, graph); } this.addShape = function(shape){ zrenderInstance.addShape(shape); }; // 初始化 this.init = function(){ var canvasElement = options.canvas.element; canvasElement.empty(); canvasElement.setStyle({height: document.viewport.getHeight() + 'px'}); zrenderInstance = graph.type.zrender.init(document.getElementById(canvasElement.identify())); for(var key in activities){ activities[key].object.addTo(graph); } for(var key in transitions){ transitions[key].object.addTo(graph); } // 建立上下文菜單容器 contextMenuContainer = new Element('div', {'class': 'context-menu'}); contextMenuContainer.hide(); document.body.appendChild(contextMenuContainer); Event.observe(contextMenuContainer, 'mouseout', function(event){ // 關閉時,應判斷鼠標是否已經移出菜單容器 if(!Position.within(contextMenuContainer, event.clientX, event.clientY)){ contextMenuContainer.hide(); } }); // 偵聽拖動過程 zrenderInstance.on('mousemove', zrenderInstanceOnMouseMove); // 上下文菜單 Event.observe(document, 'contextmenu', onContextMenu); }; // 呈現或刷新呈現 this.render = function(){ var canvasElement = options.canvas.element; canvasElement.setStyle({height: document.viewport.getHeight() + 'px'}); zrenderInstance.render(); }; }; /* * 部件(包括活動和鏈接弧) */ Libra.Workflow.Unit = Class.create({ id: null, title: null, graph: null, // 當前是否被選中 selected: false, // 上下文菜單項集合 contextMenuItems: [], initialize: function(options){ var _this = this; _this.id = options.id; _this.title = options.title; }, createShapeOptions: function(){ var _this = this; return { hoverable : true, clickable : true, onclick: function(params){ // 選中並高亮 _this.graph.onUnitSelect(_this); }, onmouseover: function(params){ _this.graph.onUnitMouseOver(_this); }, onmouseout: function(params){ _this.graph.onUnitMouseOut(_this); } }; }, addTo: function(graph){}, // 刷新顯示 refresh: function(graph){ return []; }, // 選中 select: function(graph){ this.selected = true; return this.refresh(graph); }, // 取消選中 unselect: function(graph){ this.selected = false; return this.refresh(graph); }, // 顯示上下文菜單 showContextMenu: function(event, container, graph){ container.hide(); container.innerHTML = ''; var ul = new Element('ul'); container.appendChild(ul); this.buildContextMenuItems(ul, graph); // 加偏移,讓鼠標位於菜單內 var offset = -5; var rightEdge = document.body.clientWidth - event.clientX; var bottomEdge = document.body.clientHeight - event.clientY; if (rightEdge < container.offsetWidth) container.style.left = document.body.scrollLeft + event.clientX - container.offsetWidth + offset; else container.style.left = document.body.scrollLeft + event.clientX + offset; if (bottomEdge < container.offsetHeight) container.style.top = document.body.scrollTop + event.clientY - container.offsetHeight + offset; else container.style.top = document.body.scrollTop + event.clientY + offset; container.show(); }, // 建立上下文菜單項 buildContextMenuItems: function(container, graph){ var unit = this; unit.contextMenuItems.each(function(item){ item.addTo(container); }); } });
zrender默認已經支持了對圖形的拖動,因此活動部件的拖動只須要設置dragable屬性爲真便可。不過雖然活動部件能夠拖動,但活動部件上的鏈接線不會跟着一塊兒動,這須要偵聽拖動開始事件、拖動結束事件以及拖動過程當中的鼠標移動事件,來實現鏈接線的實時重繪。在Graph中偵聽鼠標移動事件,就是爲了實現鏈接線等相關圖形的實時重繪。ide
每一個部件都規劃了八個鏈接點,默認狀況下,鏈接弧不固定與某個鏈接點,而是根據活動部件的位置關係,自動找出最近的鏈接點,因此在拖動活動部件的時候,能夠看到鏈接弧在活動部件上的鏈接點在不斷變化。測試
上面只是以最簡化的方式實現了工做流圖形化設計的基本功能,完善的圖形化設計應包含曲線、鏈接點的拖放等等,以下圖所示:ui
上面是公司產品中的工做流圖形化設計功能,功能相對於上面的範例要完善許多,但基本原理不變,無非就是細節處理更多一些。this
特別是在畫曲線的地方花了不少時間,中學的平面幾何知識幾乎都忘記了,因此作起來花了很多功夫,這部分準備之後專門寫篇文章來詳談。
本文的結尾會給出前期建模測試階段的完整代碼下載,是前期代碼,不是最終代碼,緣由你懂的,見諒。