为什么添加SVG支持?
可伸缩矢量图形 (Scalable Vector Graphics)简称SVG,相对于普通的PNG图形文件,SVG格式有更好的展示效果,可以明显增加用户的体验。FoxBPM流程引擎同时支持PNG和SVG两种格式的输出。开发者可以根据用户需求动态的抉择。针对BPM产品SVG本身有着更多的优势:首先SVG本质就是一个XML,其对应的字符串可以直接在支持SVG的浏览器中展示,所以相对普通的图形文件SVG可以更加容易的输出到前端浏览器,在网络带宽不足的情况下这个优势就更加明显; 其次既然SVG本身是一个XML,那么我们就可以在前端像操作其他的DOM模型一样操作SVG,从而我们在浏览器前端也可以动态的改变流程图内容,从而添加更多功能(比如流程实例的运行轨迹,运行状态等)。
主要知识点和难点:
主要知识点包括:采用JAXB实现XML 和POJO之间的映射,VO的克隆,三次贝塞尔曲线的控制点计算,线条拐点的中心点计算等等。相关知识点核心代码如下所示:
VO对象克隆方法:
/** * DefsVO克隆 * @param DefsVO 原对象 * @return clone之后的对象 */ public final static GVO cloneGVO(GVO gVo) { return (GVO) clone(gVo); } public final static DefsVO cloneDefsVO(DefsVO defsVo) { return (DefsVO) clone(defsVo); } /** * SvgVO模板对象需要多次引用,所以要克隆,避免产生问题 * @param SvgVO 原对象 * @return clone之后的对象 */ public final static SvgVO cloneSVGVo(SvgVO svgVo) { return (SvgVO) clone(svgVo); } /** * 克隆对象 * @param object 原对象 * @return 目标对象 */ public final static Object clone(Object object) { ByteArrayOutputStream bos = null; ObjectOutputStream oos = null; ObjectInputStream ois = null; Object cloneObject = null; try { bos = new ByteArrayOutputStream(); oos = new ObjectOutputStream(bos); oos.writeObject(object); ois = new ObjectInputStream(new ByteArrayInputStream(bos.toByteArray())); cloneObject = ois.readObject(); } catch (Exception e) { throw new FoxBPMException("SVG对象G节点克隆出现问题", e); } finally { try { if (bos != null) { bos.close(); } if (oos != null) { oos.close(); } if (ois != null) { ois.close(); } } catch (Exception e) { throw new FoxBPMException("克隆之后关闭对象流时出现问题", e); } } return cloneObject; }
JAXB映射方法(templateName是bpmn2.0官网提供的SVG模板名称):
/** * 第一次需要从svg文档加载 * * @param templateName */ private void init(String templateName) { try { JAXBContext context = JAXBContext.newInstance(SvgVO.class); Unmarshaller unMarshaller = context.createUnmarshaller(); SAXParserFactory factory = SAXParserFactory.newInstance(); // 解析的时候忽略SVG命名空间,否则会出错 factory.setNamespaceAware(true); XMLReader reader = factory.newSAXParser().getXMLReader(); String sourcePath = new StringBuffer(BPMN_PATH).append(FILE_SPERATOR) .append(templateName).toString(); Source source = new SAXSource(reader, new InputSource( ReflectUtil.getResourceAsStream(sourcePath))); VONode object = (VONode) unMarshaller.unmarshal(source); //将模板VO对象加入系统缓存 this.svgTemplets.put(templateName, object); } catch (Exception e) { throw new FoxBPMException("template svg file load exception", e); } finally { } } /** * 操作之后的SVG转化成String字符串 * @param svgVo 容器对象 * @return SVG字符串 */ public final static String createSVGString(VONode svgVo) { try { JAXBContext context = JAXBContext.newInstance(SvgVO.class); Marshaller marshal = context.createMarshaller(); marshal.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true); StringWriter writer = new StringWriter(); marshal.marshal(svgVo, writer); return writer.toString(); } catch (Exception e) { throw new FoxBPMException("svg object convert to String exception", e); } }
三次贝塞尔曲线的控制点以及线条中心点计算方法:
/** * Copyright 1996-2014 FoxBPM ORG. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * * @author MAENLIANG */ package org.foxbpm.engine.impl.diagramview.svg; import java.util.ArrayList; import java.util.Arrays; import java.util.Iterator; import java.util.List; /** * 坐标工具类 * @author MAENLIANG * @date 2014-06-19 * */ public final class PointUtils { /** * X、Y允许的最大偏移量,超过最大偏移量需要设置文本相对线条的中心位置,如果小于这个 * 值则不需要设置 */ private static final float X_Y_LOCATION_MAXSHIFT = 100F; // TODO后续改善成动态规划算法 /** * 默认圆角的大小,暂时分三个梯度 */ private final static float SEQUENCE_ROUNDCONTROL_FLAG = 15.0f; private final static float SEQUENCE_ROUNDCONTROL_MIN_FLAG = 3.0f; private final static float NONE_SCALE = 1.0f; /** * 计算三次贝塞尔曲线的起始点,控制点以及终点 * @param start * @param center * @param end * @return 起始点 控制点 终点坐标数组 */ public final static Point[] caclBeralPoints(Point start, Point center, Point end) { float lengthStartHerizon = 0.0f; float lengthStartVertical = 0.0f; float lengthStartCenter = 0.0f; float lengthEndHerizon = 0.0f; float lengthEndVertical = 0.0f; float lengthEndCenter = 0.0f; lengthStartHerizon = Math.abs(start.getX() - center.getX()); lengthStartVertical = Math.abs(start.getY() - center.getY()); lengthStartCenter = (float) Math.sqrt(lengthStartHerizon * lengthStartHerizon + lengthStartVertical * lengthStartVertical); lengthEndHerizon = Math.abs(end.getX() - center.getX()); lengthEndVertical = Math.abs(end.getY() - center.getY()); lengthEndCenter = (float) Math.sqrt(lengthEndHerizon * lengthEndHerizon + lengthEndVertical * lengthEndVertical); float scale = SEQUENCE_ROUNDCONTROL_FLAG; if (lengthEndCenter <= SEQUENCE_ROUNDCONTROL_FLAG || lengthStartCenter <= SEQUENCE_ROUNDCONTROL_FLAG) { if (lengthEndCenter <= SEQUENCE_ROUNDCONTROL_MIN_FLAG || lengthStartCenter <= SEQUENCE_ROUNDCONTROL_MIN_FLAG) { scale = NONE_SCALE; } else { scale = SEQUENCE_ROUNDCONTROL_MIN_FLAG; } } float controlScale = scale / 3; Point[] points = new Point[4]; points[0] = caclBeralControlPoint(start, center, scale, lengthStartHerizon, lengthStartVertical, lengthStartCenter); points[1] = caclBeralControlPoint(start, center, controlScale, lengthStartHerizon, lengthStartVertical, lengthStartCenter); points[2] = caclBeralControlPoint(end, center, controlScale, lengthEndHerizon, lengthEndVertical, lengthEndCenter); points[3] = caclBeralControlPoint(end, center, scale, lengthEndHerizon, lengthEndVertical, lengthEndCenter); return points; } /** * 计算三次贝塞尔曲线控制点 * * @param startEndPoint * @param center * @return */ private final static Point caclBeralControlPoint(Point startEndPoint, Point center, float scale, float lengthHerizon, float lengthVertical, float lengthCenter) { float startX = 0.0f; float startY = 0.0f; if (startEndPoint.getX() > center.getX()) { startX = ((scale * lengthHerizon) / lengthCenter) + center.getX(); } else { startX = (((lengthCenter - scale) * lengthHerizon) / lengthCenter) + startEndPoint.getX(); } if (startEndPoint.getY() > center.getY()) { startY = ((scale * lengthVertical) / lengthCenter) + center.getY(); } else { startY = (((lengthCenter - scale) * lengthVertical) / lengthCenter) + startEndPoint.getY(); } Point resultPoint = new Point(Math.round(startX), Math.round(startY)); return resultPoint; } /** * 计算中心点位置,包括复杂情况,和简单情况 * * @param pointList * @return */ public final static Point caclDetailCenterPoint(List<Point> pointList) { Float[][] xyArrays = getXYLocationArray(pointList); Float[] xArrays = xyArrays[0]; Float[] yArrays = xyArrays[1]; float xShift = calCariance(xArrays); float yShift = calCariance(yArrays); // 如果偏差过大就计算中心点 if (xShift > X_Y_LOCATION_MAXSHIFT && yShift < X_Y_LOCATION_MAXSHIFT) { // X坐标拐点幅度大,Y不大,计算线条X中心位置,Y取平均值 return caclXCenter(xArrays, yArrays); } else if (yShift > X_Y_LOCATION_MAXSHIFT && xShift < X_Y_LOCATION_MAXSHIFT) { // Y坐标拐点幅度大,X不大,计算线条Y中心位置,X取平均值 return caclYCenter(xArrays, yArrays); } else if (yShift > X_Y_LOCATION_MAXSHIFT && xShift > X_Y_LOCATION_MAXSHIFT) { // XY拐点幅度都比较大 return caclCenterPoint(pointList); } // 如果没有计算中心点 return null; } /** * 勾股定理, 取线段长度 * * @param pointA * @param pointB * @return */ public final static float segmentLength(Point pointA, Point pointB) { double length = 0; float width = pointA.getX() - pointA.getX(); float height = pointA.getY() - pointB.getY(); double dWidth = (double) width; double dHeight = (double) height; length = Math.sqrt(Math.pow(dWidth, 2) + Math.pow(dHeight, 2)); return (float) length; } /** * 统计所有线段的长度 * * @param pointList * @return */ private final static List<Float> getSegmentsLength(List<Point> pointList) { List<Float> segList = new ArrayList<Float>(); float segLen = 0f; int size = pointList.size(); size = size - 1; Point pointA = null; Point pointB = null; for (int i = 0; i < size; i++) { pointA = pointList.get(i); pointB = pointList.get(i + 1); segLen = segmentLength(pointA, pointB); segList.add(segLen); } return segList; } /** * 计算所有x坐标值或者y坐标值的方差,判断是否需要重新设置所有拐点的中心点 * * @param array * @return */ public final static Float calCariance(Float[] array) { int arrayLength = array.length; Float ave = 0.0F; for (int i = 0; i < arrayLength; i++) { ave += array[i]; } ave /= arrayLength; Float sum = 0.0F; for (int i = 0; i < arrayLength; i++) { sum += (array[i] - ave) * (array[i] - ave); } sum = sum / arrayLength; return sum; } /** * 获取所有点的XY坐标 * * @param pointList * @return */ private final static Float[][] getXYLocationArray(List<Point> pointList) { int pointListsize = pointList.size(); Float[][] xyLocationArray = new Float[2][pointListsize]; Iterator<Point> iterator = pointList.iterator(); for (int i = 0; i < pointListsize; i++) { Point next = iterator.next(); xyLocationArray[0][i] = next.getX(); xyLocationArray[1][i] = next.getY(); } return xyLocationArray; } /** * 计算线条在X方向上的中心位置 * * @param xArrays * @param yArrays * @return */ public final static Point caclXCenter(Float[] xArrays, Float[] yArrays) { Arrays.sort(xArrays); int length = yArrays.length; Float xCenter = xArrays[0] + (xArrays[xArrays.length - 1] - xArrays[0]) / 2; Float tempV = 0.0f; for (int i = 0; i < length; i++) { tempV += yArrays[i]; } Float y = tempV / length; return new Point(xCenter, y); } /** * 计算线条在y方向上的中心位置 * * @param xArrays * @param yArrays * @return */ public final static Point caclYCenter(Float[] xArrays, Float[] yArrays) { Arrays.sort(yArrays); Float yCenter = yArrays[0] + (yArrays[yArrays.length - 1] - yArrays[0]) / 2; Float tempV = 0.0f; int length = xArrays.length; for (int i = 0; i < length; i++) { tempV += xArrays[i]; } Float x = tempV / length; return new Point(yCenter, x); } /** * 计算中心点 * * @param svgLine * @return */ public final static Point caclCenterPoint(List<Point> pointList) { List<Float> segList = PointUtils.getSegmentsLength(pointList); float totalLength = 0f; for (float seg : segList) { totalLength += seg; } float centerLen = totalLength / 2; int keySegIndex = 0; float keySeg = 0f; float tempLen = 0f; for (float seg : segList) { keySeg = seg; keySegIndex++; if ((tempLen + keySeg) > centerLen) { break; } else { tempLen += keySeg; } } float lastLen = centerLen - tempLen; Point startPoint = pointList.get(keySegIndex - 1); Point endPoint = pointList.get(keySegIndex); float scale = lastLen / keySeg; float x = startPoint.getX() + (endPoint.getX() - startPoint.getX()) * scale; float y = startPoint.getY() + (endPoint.getY() - startPoint.getY()) * scale; Point point = new Point(x, y); return point; } }
模块结构详细说明:
结构如下图所示:
1、模块构建基于工厂和创建者设计模式,独立于SVG,对其他矢量标记语言扩展友好。
2、FoxBpmnViewBuilder为VO属性构建的上层接口,当创建新的元素时,可以针对元素的特性进行实现。流程引擎后期的维护和功能扩展也主要是针对该接口的实现。
3、AbstractFlowElementVOFactory也是VO属性构建的上层抽象类,负责VO对象的获取、过滤、以及根据特定元素选择特定Builder构建VO属性。
4、svg包模块是针对FoxBpmnViewBuilder和AbstractFlowElementVOFactory的svg实现,包括svg工厂、svg 构建者、svg VO模型、svg工具类。
SVG构建详细说明:
1、流程引擎启动初始化的时候,将BPMN2.0官网提供的所有SVG模版(其模版以单个组件的形式提供)初始化,采用映射工具JAXB 将模版转化成VO对象并且加入缓存。
VO定义如下代码所示:
/** * Copyright 1996-2014 FoxBPM ORG. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * * @author MAENLIANG */ package org.foxbpm.engine.impl.diagramview.vo; import java.io.Serializable; import javax.xml.bind.annotation.XmlAccessType; import javax.xml.bind.annotation.XmlAccessorType; import javax.xml.bind.annotation.XmlAttribute; import javax.xml.bind.annotation.XmlType; import javax.xml.bind.annotation.XmlValue; /** * SVG对象的超类,包括 节点SVG对象,连接线SVG对象 * * @author MAENLIANG * @date 2014-06-10 * */ @XmlType @XmlAccessorType(XmlAccessType.FIELD) public abstract class VONode implements Serializable { private static final long serialVersionUID = -817550474604039751L; /** * ID Name 用于标记流程元素ID用于 流程图的前端操作 */ @XmlAttribute(name = "id") protected String id; @XmlAttribute(name = "name") protected String name; @XmlAttribute(name = "style") protected String style; @XmlAttribute(name = "fill") protected String fill; @XmlAttribute(name = "stroke") protected String stroke; @XmlAttribute(name = "stroke-width") protected Float strokeWidth; @XmlAttribute(name = "stroke-linecap") protected String strokeLinecap; @XmlAttribute(name = "stroke-linejoin") protected String strokeLinejoin; @XmlAttribute(name = "width") protected Float width; @XmlAttribute(name = "height") protected Float height; @XmlAttribute(name = "x") protected Float x; @XmlAttribute(name = "y") protected Float y; @XmlValue protected String elementValue; public String getElementValue() { return elementValue; } public void setElementValue(String elementValue) { this.elementValue = elementValue; } public Float getX() { return x; } public void setX(Float x) { this.x = x; } public Float getY() { return y; } public void setY(Float y) { this.y = y; } public String getStyle() { return style; } public void setStyle(String style) { this.style = style; } public String getFill() { return fill; } public void setFill(String fill) { this.fill = fill; } public String getStroke() { return stroke; } public void setStroke(String stroke) { this.stroke = stroke; } public float getStrokeWidth() { return strokeWidth; } public void setStrokeWidth(float strokeWidth) { this.strokeWidth = strokeWidth; } public float getWidth() { return width; } public void setWidth(float width) { this.width = width; } public float getHeight() { return height; } public void setHeight(float height) { this.height = height; } public String getId() { return id; } public void setId(String id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } public String getStrokeLinecap() { return strokeLinecap; } public void setStrokeLinecap(String strokeLinecap) { this.strokeLinecap = strokeLinecap; } public String getStrokeLinejoin() { return strokeLinejoin; } public void setStrokeLinejoin(String strokeLinejoin) { this.strokeLinejoin = strokeLinejoin; } }
2、SVG模版相对独立,当创建时流程引擎提供包含所有流程元素定义的流程定义对象,流程元素包括 节点、线条、泳道、部件,其中节点包括活动任务、事件、网关。SVG创建者首先创建一个VO集合,然后根据流程元素从缓存中获取每个元素对应的自己的VO对象,获取VO 对象之后根据每个元素的式样属性来设置其VO属性,需要实现的接口为FoxBpmnViewBuilder,VO对象属性设置好之后添加到VO集合中。
FoxBpmnViewBuilder代码如下:
/** * Copyright 1996-2014 FoxBPM ORG. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * * @author MAENLIANG */ package org.foxbpm.engine.impl.diagramview.builder; import java.util.List; import org.foxbpm.engine.impl.diagramview.svg.Point; /** * 流程图形信息构造接口 * * @author MAENLIANG * @date 2014-06-10 */ public interface FoxBpmnViewBuilder { public static final String SEQUENCE_STROKEWIDTH_DEFAULT = "2"; public static final String COLOR_FLAG = "#"; public static final String STROKE_DEFAULT = "black"; public static final String STROKEWIDTH_DEFAULT = "1"; public static final int LINEARGRADIENT_INDEX = 0; public static final String COMMA = ","; public static final String BACK_GROUND_PREFIX = "url(#"; public static final String BACK_GROUND_SUFFIX = ") #"; public static final String BRACKET_SUFFIX = ")"; public static final String TRANSLANT_PREFIX = "translate("; public static final String ARIAL = "arial"; /** * 设置节点信息 */ public void setWayPoints(List<Point> pointList); public void setWidth(float width); public void setHeight(float height); public void setXAndY(float x, float y); public void setStroke(String stroke); public void setStrokeWidth(float strokeWidth); public void setFill(String fill); public void setID(String id); public void setName(String name); public void setStyle(String style); /** * 设置文本信息 */ public void setText(String text); public void setTextFont(String font); public void setTextStrokeWidth(float textStrokeWidth); public void setTextX(float textX); public void setTextY(float textY); public void setTextFontSize(String textFontSize); public void setTextStroke(String textStroke); public void setTextFill(String textFill); /** * 设置子类型 */ public void setTypeStroke(String stroke); public void setTypeStrokeWidth(float strokeWidth); public void setTypeFill(String fill); public void setTypeStyle(String style); /** * 泳道负责重写该方法 setTextLocationByHerizonFlag */ public void setTextLocationByHerizonFlag(boolean herizonFlag); }
3、所有元素的VO对象设置好之后,再根据流程定义提供的基本信息创建SVG模版对象,然后将所有的VO对象克隆并且添加到模版对象,然后将模版对象用JAXB 工具转化成最终的SVG XML文档
流程定义对象所包含的元素如下所以:
List<KernelFlowNodeImpl> flowNodes = new ArrayList<KernelFlowNodeImpl>(); Map<String, KernelSequenceFlowImpl> sequenceFlows = new HashMap<String, KernelSeque nceFlowImpl>(); List<KernelLaneSet> laneSets = new ArrayList<KernelLaneSet>(); List<KernelArtifact> artifacts = new ArrayList<KernelArtifact>();
foxbpm流程引擎构建SVG示意图如下: