diff --git a/pom.xml b/pom.xml index 5cb3fa7ad..298d521d5 100644 --- a/pom.xml +++ b/pom.xml @@ -94,6 +94,21 @@ 1.0.7 test + + org.eclipse.elk + org.eclipse.elk.core + 0.7.1 + + + org.eclipse.elk + org.eclipse.elk.alg.layered + 0.7.1 + + + org.eclipse.elk + org.eclipse.elk.alg.mrtree + 0.7.1 + diff --git a/skin/plantuml.skin b/skin/plantuml.skin index a9cd7ac2b..ae4a21a2b 100644 --- a/skin/plantuml.skin +++ b/skin/plantuml.skin @@ -290,6 +290,15 @@ ganttDiagram { timeline { BackgroundColor transparent } + task { + RoundCorner 0 + Margin 2 2 2 2 + Padding 0 + } + milestone { + Margin 2 + Padding 3 + } } diff --git a/src/net/sourceforge/plantuml/TitledDiagram.java b/src/net/sourceforge/plantuml/TitledDiagram.java index 6815d0562..cdd5eb22a 100644 --- a/src/net/sourceforge/plantuml/TitledDiagram.java +++ b/src/net/sourceforge/plantuml/TitledDiagram.java @@ -202,11 +202,20 @@ public abstract class TitledDiagram extends AbstractPSystem implements Diagram, } private boolean useSmetana; + private boolean useElk; public void setUseSmetana(boolean useSmetana) { this.useSmetana = useSmetana; } + public void setUseElk(boolean useElk) { + this.useElk = useElk; + } + + public boolean isUseElk() { + return this.useElk; + } + public boolean isUseSmetana() { if (FORCE_SMETANA) return true; @@ -234,7 +243,6 @@ public abstract class TitledDiagram extends AbstractPSystem implements Diagram, @Override public ImageBuilder createImageBuilder(FileFormatOption fileFormatOption) throws IOException { - return super.createImageBuilder(fileFormatOption) - .styled(this); + return super.createImageBuilder(fileFormatOption).styled(this); } } diff --git a/src/net/sourceforge/plantuml/command/CommandPragma.java b/src/net/sourceforge/plantuml/command/CommandPragma.java index 068ff9768..020ddaab6 100644 --- a/src/net/sourceforge/plantuml/command/CommandPragma.java +++ b/src/net/sourceforge/plantuml/command/CommandPragma.java @@ -77,11 +77,15 @@ public class CommandPragma extends SingleLineCommand2 { } else { system.getPragma().define(name, value); if (name.equalsIgnoreCase("graphviz_dot") && value.equalsIgnoreCase("jdot")) { - return CommandExecutionResult.error("This directive has been renamed to '!pragma graphviz_dot smetana'. Please update your diagram."); + return CommandExecutionResult.error( + "This directive has been renamed to '!pragma graphviz_dot smetana'. Please update your diagram."); } if (name.equalsIgnoreCase("graphviz_dot") && value.equalsIgnoreCase("smetana")) { system.setUseSmetana(true); } + if (name.equalsIgnoreCase("graphviz_dot") && value.equalsIgnoreCase("elk")) { + system.setUseElk(true); + } if (name.equalsIgnoreCase("graphviz_dot") && value.equalsIgnoreCase(GraphvizUtils.VIZJS)) { system.getSkinParam().setUseVizJs(true); } diff --git a/src/net/sourceforge/plantuml/cucadiagram/CucaDiagram.java b/src/net/sourceforge/plantuml/cucadiagram/CucaDiagram.java index ef71ecf5a..f060cc7e8 100644 --- a/src/net/sourceforge/plantuml/cucadiagram/CucaDiagram.java +++ b/src/net/sourceforge/plantuml/cucadiagram/CucaDiagram.java @@ -57,6 +57,7 @@ import net.sourceforge.plantuml.core.ImageData; import net.sourceforge.plantuml.creole.CreoleMode; import net.sourceforge.plantuml.cucadiagram.dot.CucaDiagramTxtMaker; import net.sourceforge.plantuml.cucadiagram.entity.EntityFactory; +import net.sourceforge.plantuml.elk.CucaDiagramFileMakerElk; import net.sourceforge.plantuml.graphic.USymbol; import net.sourceforge.plantuml.sdot.CucaDiagramFileMakerSmetana; import net.sourceforge.plantuml.security.SecurityUtils; @@ -650,9 +651,14 @@ public abstract class CucaDiagram extends UmlDiagram implements GroupHierarchy, entityFactory.buildSuperGroups(); - final CucaDiagramFileMaker maker = this.isUseSmetana() - ? new CucaDiagramFileMakerSmetana(this, fileFormatOption.getDefaultStringBounder(getSkinParam())) - : new CucaDiagramFileMakerSvek(this); + final CucaDiagramFileMaker maker; + if (this.isUseElk()) { + maker = new CucaDiagramFileMakerElk(this, fileFormatOption.getDefaultStringBounder(getSkinParam())); + } else if (this.isUseSmetana()) { + maker = new CucaDiagramFileMakerSmetana(this, fileFormatOption.getDefaultStringBounder(getSkinParam())); + } else { + maker = new CucaDiagramFileMakerSvek(this); + } final ImageData result = maker.createFile(os, getDotStrings(), fileFormatOption); if (result == null) { diff --git a/src/net/sourceforge/plantuml/elk/CucaDiagramFileMakerElk.java b/src/net/sourceforge/plantuml/elk/CucaDiagramFileMakerElk.java new file mode 100644 index 000000000..7dce40f64 --- /dev/null +++ b/src/net/sourceforge/plantuml/elk/CucaDiagramFileMakerElk.java @@ -0,0 +1,247 @@ +/* ======================================================================== + * PlantUML : a free UML diagram generator + * ======================================================================== + * + * (C) Copyright 2009-2020, Arnaud Roques + * + * Project Info: http://plantuml.com + * + * If you like this project or if you find it useful, you can support us at: + * + * http://plantuml.com/patreon (only 1$ per month!) + * http://plantuml.com/paypal + * + * This file is part of PlantUML. + * + * PlantUML is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * PlantUML distributed in the hope that it will be useful, but + * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY + * or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public + * License for more details. + * + * You should have received a copy of the GNU General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, + * USA. + * + * + * Original Author: Arnaud Roques + * + * + */ +package net.sourceforge.plantuml.elk; + +import java.awt.geom.Dimension2D; +import java.awt.geom.Point2D; +import java.io.IOException; +import java.io.OutputStream; +import java.util.ArrayList; +import java.util.EnumSet; +import java.util.LinkedHashMap; +import java.util.List; +import java.util.Map; +import java.util.Map.Entry; + +import org.eclipse.elk.core.RecursiveGraphLayoutEngine; +import org.eclipse.elk.core.math.ElkPadding; +import org.eclipse.elk.core.options.CoreOptions; +import org.eclipse.elk.core.options.NodeLabelPlacement; +import org.eclipse.elk.core.options.SizeConstraint; +import org.eclipse.elk.core.options.SizeOptions; +import org.eclipse.elk.core.util.NullElkProgressMonitor; +import org.eclipse.elk.graph.ElkEdge; +import org.eclipse.elk.graph.ElkLabel; +import org.eclipse.elk.graph.ElkNode; +import org.eclipse.elk.graph.util.ElkGraphUtil; + +import net.sourceforge.plantuml.FileFormatOption; +import net.sourceforge.plantuml.ISkinParam; +import net.sourceforge.plantuml.StringUtils; +import net.sourceforge.plantuml.UmlDiagram; +import net.sourceforge.plantuml.api.ImageDataSimple; +import net.sourceforge.plantuml.core.ImageData; +import net.sourceforge.plantuml.cucadiagram.CucaDiagram; +import net.sourceforge.plantuml.cucadiagram.ILeaf; +import net.sourceforge.plantuml.cucadiagram.Link; +import net.sourceforge.plantuml.graphic.AbstractTextBlock; +import net.sourceforge.plantuml.graphic.QuoteUtils; +import net.sourceforge.plantuml.graphic.StringBounder; +import net.sourceforge.plantuml.graphic.TextBlock; +import net.sourceforge.plantuml.graphic.TextBlockUtils; +import net.sourceforge.plantuml.svek.Bibliotekon; +import net.sourceforge.plantuml.svek.CucaDiagramFileMaker; +import net.sourceforge.plantuml.svek.DotStringFactory; +import net.sourceforge.plantuml.svek.GeneralImageBuilder; +import net.sourceforge.plantuml.svek.GraphvizCrash; +import net.sourceforge.plantuml.svek.IEntityImage; +import net.sourceforge.plantuml.svek.TextBlockBackcolored; +import net.sourceforge.plantuml.ugraphic.MinMax; +import net.sourceforge.plantuml.ugraphic.UGraphic; +import net.sourceforge.plantuml.ugraphic.UTranslate; +import net.sourceforge.plantuml.ugraphic.color.HColor; + +public class CucaDiagramFileMakerElk implements CucaDiagramFileMaker { + + private final CucaDiagram diagram; + private final StringBounder stringBounder; + private final DotStringFactory dotStringFactory; + + private final Map nodes = new LinkedHashMap(); + + public CucaDiagramFileMakerElk(CucaDiagram diagram, StringBounder stringBounder) { + this.diagram = diagram; + this.stringBounder = stringBounder; + this.dotStringFactory = new DotStringFactory(stringBounder, diagram); + + } + + // The Drawing class does the real drawing + class Drawing extends AbstractTextBlock implements TextBlockBackcolored { + + // min and max of all coord + private final MinMax minMax; + + public Drawing(MinMax minMax) { + this.minMax = minMax; + } + + public void drawU(UGraphic ug) { + + // Draw all nodes + for (Entry ent : nodes.entrySet()) { + final ILeaf leaf = ent.getKey(); + final ElkNode agnode = ent.getValue(); + + final IEntityImage image = printEntityInternal(leaf); + + // Retrieve coord from ELK + final Point2D corner = new Point2D.Double(agnode.getX(), agnode.getY()); + + // Print the node image at right coord + image.drawU(ug.apply(new UTranslate(corner))); + } + + } + + public Dimension2D calculateDimension(StringBounder stringBounder) { + if (minMax == null) { + throw new UnsupportedOperationException(); + } + return minMax.getDimension(); + } + + public HColor getBackcolor() { + return null; + } + + } + + @Override + public ImageData createFile(OutputStream os, List dotStrings, FileFormatOption fileFormatOption) + throws IOException { + + try { + final ElkNode root = ElkGraphUtil.createGraph(); + final ElkPadding labelPadding = new ElkPadding(2.0); + + // Convert all "leaf" to ELK node + for (ILeaf leaf : diagram.getLeafsvalues()) { + final IEntityImage image = printEntityInternal(leaf); + + // Expected dimension of the node + final Dimension2D dimension = image.calculateDimension(stringBounder); + + // Here, we try to tell ELK to use this dimension as node dimension + final ElkNode node = ElkGraphUtil.createNode(root); + node.setDimensions(dimension.getWidth(), dimension.getHeight()); + + // There is no real "label" here + // We just would like to force node dimension + final ElkLabel label = ElkGraphUtil.createLabel(node); + label.setDimensions(dimension.getWidth(), dimension.getHeight()); + + // No idea of what we are doing here :-) + label.setProperty(CoreOptions.NODE_LABELS_PLACEMENT, EnumSet.of(NodeLabelPlacement.INSIDE, + NodeLabelPlacement.H_CENTER, NodeLabelPlacement.V_CENTER)); + label.setProperty(CoreOptions.NODE_LABELS_PADDING, labelPadding); + node.setProperty(CoreOptions.NODE_SIZE_CONSTRAINTS, EnumSet.of(SizeConstraint.NODE_LABELS)); + node.setProperty(CoreOptions.NODE_SIZE_OPTIONS, EnumSet.noneOf(SizeOptions.class)); + + // Let's store this + nodes.put(leaf, node); + } + + for (final Link link : diagram.getLinks()) { + final ElkEdge edge = ElkGraphUtil.createEdge(root); + System.err.println("edge=" + edge); + edge.getSources().add(nodes.get(link.getEntity1())); + edge.getTargets().add(nodes.get(link.getEntity2())); + } + + final RecursiveGraphLayoutEngine engine = new RecursiveGraphLayoutEngine(); + engine.layout(root, new NullElkProgressMonitor()); + + // Debug + for (final ElkNode node : nodes.values()) { + final String name = node.getLabels().get(0).getText(); + System.out.println("node " + name + " : " + node.getX() + ", " + node.getY() + " (" + node.getWidth() + + ", " + node.getHeight() + ")"); + } + + final MinMax minMax = TextBlockUtils.getMinMax(new Drawing(null), stringBounder, false); + + final TextBlock drawable = new Drawing(minMax); + return diagram.createImageBuilder(fileFormatOption) // + .drawable(drawable) // + .write(os); // + + } catch (Throwable e) { + UmlDiagram.exportDiagramError(os, e, fileFormatOption, diagram.seed(), diagram.getMetadata(), + diagram.getFlashData(), getFailureText3(e)); + return ImageDataSimple.error(); + } + + } + + static private List getFailureText3(Throwable exception) { + exception.printStackTrace(); + final List strings = new ArrayList(); + strings.add("An error has occured : " + exception); + final String quote = StringUtils.rot(QuoteUtils.getSomeQuote()); + strings.add("" + quote); + strings.add(" "); + GraphvizCrash.addProperties(strings); + strings.add(" "); + strings.add("Sorry, ELK intregration is really alpha feature..."); + strings.add(" "); + strings.add("You should send this diagram and this image to plantuml@gmail.com or"); + strings.add("post to http://plantuml.com/qa to solve this issue."); + strings.add(" "); + return strings; + } + + private Bibliotekon getBibliotekon() { + return dotStringFactory.getBibliotekon(); + } + + private IEntityImage printEntityInternal(ILeaf ent) { + if (ent.isRemoved()) { + throw new IllegalStateException(); + } + if (ent.getSvekImage() == null) { + final ISkinParam skinParam = diagram.getSkinParam(); + if (skinParam.sameClassWidth()) { + System.err.println("NOT YET IMPLEMENED"); + } + + return GeneralImageBuilder.createEntityImageBlock(ent, skinParam, diagram.isHideEmptyDescriptionForState(), + diagram, getBibliotekon(), null, diagram.getUmlDiagramType(), diagram.getLinks()); + } + return ent.getSvekImage(); + } + +} diff --git a/src/net/sourceforge/plantuml/math/ScientificEquationSafe.java b/src/net/sourceforge/plantuml/math/ScientificEquationSafe.java index 90f2603f4..2d624c200 100644 --- a/src/net/sourceforge/plantuml/math/ScientificEquationSafe.java +++ b/src/net/sourceforge/plantuml/math/ScientificEquationSafe.java @@ -93,35 +93,35 @@ public class ScientificEquationSafe { private ImageData dimSvg; public UImageSvg getSvg(double scale, Color foregroundColor, Color backgroundColor) { - try { - final UImageSvg svg = equation.getSvg(scale, foregroundColor, backgroundColor); - dimSvg = new ImageDataSimple(equation.getDimension()); - return svg; - } catch (Exception e) { - printTrace(e); - final ByteArrayOutputStream baos = new ByteArrayOutputStream(); + if (equation != null) try { - dimSvg = plainImageBuilder(getRollback(), new FileFormatOption(FileFormat.SVG)) - .write(baos); - } catch (IOException e1) { - return null; + final UImageSvg svg = equation.getSvg(scale, foregroundColor, backgroundColor); + dimSvg = new ImageDataSimple(equation.getDimension()); + return svg; + } catch (Exception e) { + printTrace(e); } - return new UImageSvg(new String(baos.toByteArray()), scale); + final ByteArrayOutputStream baos = new ByteArrayOutputStream(); + try { + dimSvg = plainImageBuilder(getRollback(), new FileFormatOption(FileFormat.SVG)).write(baos); + } catch (IOException e1) { + return null; } + return new UImageSvg(new String(baos.toByteArray()), scale); } public MutableImage getImage(Color foregroundColor, Color backgroundColor) { - try { - return equation.getImage(foregroundColor, backgroundColor); - } catch (Exception e) { - printTrace(e); + if (equation != null) try { - final byte[] bytes = plainPngBuilder(getRollback()).writeByteArray(); - return new PixelImage(ImageIO.read(new ByteArrayInputStream(bytes)), - AffineTransformType.TYPE_BILINEAR); - } catch (IOException e1) { - return null; + return equation.getImage(foregroundColor, backgroundColor); + } catch (Exception e) { + printTrace(e); } + try { + final byte[] bytes = plainPngBuilder(getRollback()).writeByteArray(); + return new PixelImage(ImageIO.read(new ByteArrayInputStream(bytes)), AffineTransformType.TYPE_BILINEAR); + } catch (IOException e1) { + return null; } } diff --git a/src/net/sourceforge/plantuml/nwdiag/GridTextBlockDecorated.java b/src/net/sourceforge/plantuml/nwdiag/GridTextBlockDecorated.java index c9a15cedb..8349442cd 100644 --- a/src/net/sourceforge/plantuml/nwdiag/GridTextBlockDecorated.java +++ b/src/net/sourceforge/plantuml/nwdiag/GridTextBlockDecorated.java @@ -169,6 +169,9 @@ public class GridTextBlockDecorated extends GridTextBlockSimple { for (int j = i + 1; j < groups.size(); j++) { final NwGroup group1 = groups.get(i); final NwGroup group2 = groups.get(j); + if (group1.size() == 0 || group2.size() == 0) { + continue; + } if (group1.getNetwork() != group2.getNetwork()) { continue; } diff --git a/src/net/sourceforge/plantuml/nwdiag/NwGroup.java b/src/net/sourceforge/plantuml/nwdiag/NwGroup.java index 4e79c577e..fd406f527 100644 --- a/src/net/sourceforge/plantuml/nwdiag/NwGroup.java +++ b/src/net/sourceforge/plantuml/nwdiag/NwGroup.java @@ -71,6 +71,10 @@ public class NwGroup { this.network = network; } + public int size() { + return elements.size(); + } + public final String getName() { return name; } diff --git a/src/net/sourceforge/plantuml/project/GanttArrow.java b/src/net/sourceforge/plantuml/project/GanttArrow.java index 5b9b70bfe..85319a03b 100644 --- a/src/net/sourceforge/plantuml/project/GanttArrow.java +++ b/src/net/sourceforge/plantuml/project/GanttArrow.java @@ -43,8 +43,12 @@ import net.sourceforge.plantuml.project.core.TaskAttribute; import net.sourceforge.plantuml.project.core.TaskInstant; import net.sourceforge.plantuml.project.draw.TaskDraw; import net.sourceforge.plantuml.project.timescale.TimeScale; +import net.sourceforge.plantuml.style.ClockwiseTopRightBottomLeft; import net.sourceforge.plantuml.style.PName; +import net.sourceforge.plantuml.style.SName; import net.sourceforge.plantuml.style.Style; +import net.sourceforge.plantuml.style.StyleBuilder; +import net.sourceforge.plantuml.style.StyleSignature; import net.sourceforge.plantuml.ugraphic.UGraphic; import net.sourceforge.plantuml.ugraphic.ULine; import net.sourceforge.plantuml.ugraphic.UStroke; @@ -62,9 +66,11 @@ public class GanttArrow implements UDrawable { private final HColorSet colorSet; private final Style style; private final ToTaskDraw toTaskDraw; + private final StyleBuilder styleBuilder; public GanttArrow(HColorSet colorSet, Style style, TimeScale timeScale, TaskInstant source, TaskInstant dest, - ToTaskDraw toTaskDraw) { + ToTaskDraw toTaskDraw, StyleBuilder styleBuilder) { + this.styleBuilder = styleBuilder; this.toTaskDraw = toTaskDraw; this.style = style; this.colorSet = colorSet; @@ -98,30 +104,34 @@ public class GanttArrow implements UDrawable { public void drawU(UGraphic ug) { ug = style.applyStrokeAndLineColor(ug, colorSet); - // ug = ug.apply(color.bg()).apply(color).apply(style.getStroke3(new - // UStroke(1.5))); - double x1 = getX(source.withDelta(0), atStart); + double x1 = getX(source.getAttribute(), getSource(), atStart); final StringBounder stringBounder = ug.getStringBounder(); double y1 = getSource().getY(stringBounder, atStart); - final double x2 = getX(dest, atEnd.getInv()); + final double x2 = getX(dest.getAttribute(), getDestination(), atEnd.getInv()); final double y2 = getDestination().getY(stringBounder, atEnd); if (atStart == Direction.DOWN && y2 < y1) { y1 = getSource().getY(stringBounder, atStart.getInv()); } + final double minimalWidth = 8; +// final Style style = getStyleSignatureTask().getMergedStyle(styleBuilder); +// final ClockwiseTopRightBottomLeft margin = style.getMargin(); +// final ClockwiseTopRightBottomLeft padding = style.getPadding(); + if (this.atStart == Direction.DOWN && this.atEnd == Direction.RIGHT) { if (x2 > x1) { - if (x2 - x1 < 8) { - x1 = x2 - 8; + if (x2 - x1 < minimalWidth) { + x1 = x2 - minimalWidth; } drawLine(ug, x1, y1, x1, y2, x2, y2); } else { - x1 = getX(source.withDelta(0), Direction.RIGHT); + x1 = getX(source.getAttribute(), getSource(), Direction.RIGHT); y1 = getSource().getY(stringBounder, Direction.RIGHT); - drawLine(ug, x1, y1, x1 + 6, y1, x1 + 6, y1 + 8, x2 - 8, y1 + 8, x2 - 8, y2, x2, y2); + final double y1b = getDestination().getY(stringBounder); + drawLine(ug, x1, y1, x1 + 6, y1, x1 + 6, y1b, x2 - 8, y1b, x2 - 8, y2, x2, y2); } } else if (this.atStart == Direction.RIGHT && this.atEnd == Direction.LEFT) { final double xmax = Math.max(x1, x2) + 8; @@ -151,15 +161,17 @@ public class GanttArrow implements UDrawable { } - private double getX(TaskInstant when, Direction direction) { - final double x1 = timeScale.getStartingPosition(when.getInstantTheorical()); - final double x2 = timeScale.getEndingPosition(when.getInstantTheorical()); + private StyleSignature getStyleSignatureTask() { + return StyleSignature.of(SName.root, SName.element, SName.ganttDiagram, SName.task); + } + + private double getX(TaskAttribute taskAttribute, TaskDraw task, Direction direction) { if (direction == Direction.LEFT) { - return x1; + return task.getX1(taskAttribute) - 1; } if (direction == Direction.RIGHT) { - return x2; + return task.getX2(taskAttribute) + 1; } - return (x1 + x2) / 2; + return (task.getX1(taskAttribute) + (task.getX2(taskAttribute))) / 2; } } diff --git a/src/net/sourceforge/plantuml/project/GanttConstraint.java b/src/net/sourceforge/plantuml/project/GanttConstraint.java index 3e517fc65..6df95741c 100644 --- a/src/net/sourceforge/plantuml/project/GanttConstraint.java +++ b/src/net/sourceforge/plantuml/project/GanttConstraint.java @@ -101,7 +101,7 @@ public class GanttConstraint extends WithLinkType { Style style = styleBuilder.getMergedStyle(getStyleSignature()).eventuallyOverride(PName.LineColor, getSpecificColor()); style = style.eventuallyOverride(getType().getStroke3(style.getStroke())); - return new GanttArrow(colorSet, style, timeScale, source, dest, toTaskDraw); + return new GanttArrow(colorSet, style, timeScale, source, dest, toTaskDraw, styleBuilder); } public boolean isHidden(Day min, Day max) { diff --git a/src/net/sourceforge/plantuml/project/GanttDiagram.java b/src/net/sourceforge/plantuml/project/GanttDiagram.java index 939c0feb6..ad9d011b9 100644 --- a/src/net/sourceforge/plantuml/project/GanttDiagram.java +++ b/src/net/sourceforge/plantuml/project/GanttDiagram.java @@ -60,6 +60,7 @@ import net.sourceforge.plantuml.core.ImageData; import net.sourceforge.plantuml.cucadiagram.Display; import net.sourceforge.plantuml.graphic.InnerStrategy; import net.sourceforge.plantuml.graphic.StringBounder; +import net.sourceforge.plantuml.graphic.UDrawable; import net.sourceforge.plantuml.project.core.Moment; import net.sourceforge.plantuml.project.core.MomentImpl; import net.sourceforge.plantuml.project.core.PrintScale; @@ -93,6 +94,7 @@ import net.sourceforge.plantuml.style.PName; import net.sourceforge.plantuml.style.SName; import net.sourceforge.plantuml.style.Style; import net.sourceforge.plantuml.style.StyleSignature; +import net.sourceforge.plantuml.svek.GraphvizCrash; import net.sourceforge.plantuml.svek.TextBlockBackcolored; import net.sourceforge.plantuml.ugraphic.MinMax; import net.sourceforge.plantuml.ugraphic.UGraphic; @@ -215,28 +217,35 @@ public class GanttDiagram extends TitledDiagram implements ToTaskDraw, WithSprit return new TextBlockBackcolored() { public void drawU(UGraphic ug) { - final Style timelineStyle = StyleSignature - .of(SName.root, SName.element, SName.ganttDiagram, SName.timeline) - .getMergedStyle(getCurrentStyleBuilder()); + try { + final Style timelineStyle = StyleSignature + .of(SName.root, SName.element, SName.ganttDiagram, SName.timeline) + .getMergedStyle(getCurrentStyleBuilder()); - final HColor back = timelineStyle.value(PName.BackGroundColor).asColor(getIHtmlColorSet()); - if (HColorUtils.isTransparent(back) == false) { - final URectangle rect1 = new URectangle(calculateDimension(ug.getStringBounder()).getWidth(), - timeHeader.getTimeHeaderHeight()); - final URectangle rect2 = new URectangle(calculateDimension(ug.getStringBounder()).getWidth(), - timeHeader.getTimeFooterHeight()); - ug.apply(back.bg()).draw(rect1); - ug.apply(back.bg()).apply(UTranslate.dy(totalHeightWithoutFooter)).draw(rect2); - } + final HColor back = timelineStyle.value(PName.BackGroundColor).asColor(getIHtmlColorSet()); + if (HColorUtils.isTransparent(back) == false) { + final URectangle rect1 = new URectangle(calculateDimension(ug.getStringBounder()).getWidth(), + timeHeader.getTimeHeaderHeight()); + final URectangle rect2 = new URectangle(calculateDimension(ug.getStringBounder()).getWidth(), + timeHeader.getTimeFooterHeight()); + ug.apply(back.bg()).draw(rect1); + ug.apply(back.bg()).apply(UTranslate.dy(totalHeightWithoutFooter)).draw(rect2); + } - timeHeader.drawTimeHeader(ug, totalHeightWithoutFooter); + timeHeader.drawTimeHeader(ug, totalHeightWithoutFooter); + + drawConstraints(ug, timeHeader.getTimeScale()); + drawTasksRect(ug); + drawTasksTitle(ug); + drawResources(ug); + if (showFootbox) { + timeHeader.drawTimeFooter(ug.apply(UTranslate.dy(totalHeightWithoutFooter))); + } + } catch (Throwable t) { + t.printStackTrace(); + final UDrawable crash = new GraphvizCrash(getSource().getPlainString(), false, t); + crash.drawU(ug); - drawConstraints(ug, timeHeader.getTimeScale()); - drawTasksRect(ug); - drawTasksTitle(ug); - drawResources(ug); - if (showFootbox) { - timeHeader.drawTimeFooter(ug.apply(UTranslate.dy(totalHeightWithoutFooter))); } } @@ -366,7 +375,7 @@ public class GanttDiagram extends TitledDiagram implements ToTaskDraw, WithSprit draw.setColorsAndCompletion(tmp.getColors(), tmp.getCompletion(), tmp.getUrl(), tmp.getNote()); } if (task.getRow() == null) { - y += draw.getHeightTask(stringBounder); + y += draw.getFullHeightTask(stringBounder); } draws.put(task, draw); } diff --git a/src/net/sourceforge/plantuml/project/draw/AbstractTaskDraw.java b/src/net/sourceforge/plantuml/project/draw/AbstractTaskDraw.java index 545e533a0..daa37bf76 100644 --- a/src/net/sourceforge/plantuml/project/draw/AbstractTaskDraw.java +++ b/src/net/sourceforge/plantuml/project/draw/AbstractTaskDraw.java @@ -46,6 +46,7 @@ import net.sourceforge.plantuml.project.core.Task; import net.sourceforge.plantuml.project.lang.CenterBorderColor; import net.sourceforge.plantuml.project.time.Day; import net.sourceforge.plantuml.project.timescale.TimeScale; +import net.sourceforge.plantuml.style.ClockwiseTopRightBottomLeft; import net.sourceforge.plantuml.style.PName; import net.sourceforge.plantuml.style.Style; import net.sourceforge.plantuml.style.StyleBuilder; @@ -68,8 +69,6 @@ public abstract class AbstractTaskDraw implements TaskDraw { private final Task task; private final ToTaskDraw toTaskDraw; - protected final double margin = 2; - @Override final public String toString() { return super.toString() + " " + task; @@ -112,12 +111,12 @@ public abstract class AbstractTaskDraw implements TaskDraw { return getStyleSignature().getMergedStyle(styleBuilder); } - final protected double getShapeHeight(StringBounder stringBounder) { - return getHeightTask(stringBounder) - 2 * margin; - } + abstract protected double getShapeHeight(StringBounder stringBounder); - final public double getHeightTask(StringBounder stringBounder) { - return getFontConfiguration().getFont().getSize2D() + 5; + final public double getFullHeightTask(StringBounder stringBounder) { + final Style style = getStyle(); + final ClockwiseTopRightBottomLeft margin = style.getMargin(); + return margin.getTop() + getShapeHeight(stringBounder) + margin.getBottom(); } public TaskDraw getTrueRow() { @@ -142,13 +141,21 @@ public abstract class AbstractTaskDraw implements TaskDraw { } public final double getY(StringBounder stringBounder, Direction direction) { + final Style style = getStyle(); + final ClockwiseTopRightBottomLeft margin = style.getMargin(); + final ClockwiseTopRightBottomLeft padding = style.getPadding(); + + final double y1 = margin.getTop() + getY(stringBounder); + final double y2 = y1 + getShapeHeight(stringBounder); + if (direction == Direction.UP) { - return getY(stringBounder); + return y1; } if (direction == Direction.DOWN) { - return getY(stringBounder) + getHeightTask(stringBounder); + return y2; } - return getY(stringBounder) + getHeightTask(stringBounder) / 2; + return (y1 + y2) / 2; + } protected final StyleBuilder getStyleBuilder() { diff --git a/src/net/sourceforge/plantuml/project/draw/PathUtils.java b/src/net/sourceforge/plantuml/project/draw/PathUtils.java index a3a7b20d9..4123dd007 100644 --- a/src/net/sourceforge/plantuml/project/draw/PathUtils.java +++ b/src/net/sourceforge/plantuml/project/draw/PathUtils.java @@ -41,26 +41,26 @@ import net.sourceforge.plantuml.ugraphic.UPath; public class PathUtils { - private final static double round = 4; - - public static UPath UtoRight(double width, double height) { + public static UPath UtoRight(double width, double height, double round) { + final double halfRound = round / 2; final UPath result = new UPath(); result.moveTo(0, 0); - result.lineTo(width - round, 0); - result.arcTo(new Point2D.Double(width, round), round, 0, 1); - result.lineTo(width, height - round); - result.arcTo(new Point2D.Double(width - round, height), round, 0, 1); + result.lineTo(width - halfRound, 0); + result.arcTo(new Point2D.Double(width, halfRound), halfRound, 0, 1); + result.lineTo(width, height - halfRound); + result.arcTo(new Point2D.Double(width - halfRound, height), halfRound, 0, 1); result.lineTo(0, height); return result; } - public static UPath UtoLeft(double width, double height) { + public static UPath UtoLeft(double width, double height, double round) { + final double halfRound = round / 2; final UPath result = new UPath(); result.moveTo(width, height); - result.lineTo(round, height); - result.arcTo(new Point2D.Double(0, height - round), round, 0, 1); - result.lineTo(0, round); - result.arcTo(new Point2D.Double(round, 0), round, 0, 1); + result.lineTo(halfRound, height); + result.arcTo(new Point2D.Double(0, height - halfRound), halfRound, 0, 1); + result.lineTo(0, halfRound); + result.arcTo(new Point2D.Double(halfRound, 0), halfRound, 0, 1); result.lineTo(width, 0); return result; } diff --git a/src/net/sourceforge/plantuml/project/draw/RectangleTask.java b/src/net/sourceforge/plantuml/project/draw/RectangleTask.java new file mode 100644 index 000000000..876199150 --- /dev/null +++ b/src/net/sourceforge/plantuml/project/draw/RectangleTask.java @@ -0,0 +1,195 @@ +/* ======================================================================== + * PlantUML : a free UML diagram generator + * ======================================================================== + * + * (C) Copyright 2009-2020, Arnaud Roques + * + * Project Info: http://plantuml.com + * + * If you like this project or if you find it useful, you can support us at: + * + * http://plantuml.com/patreon (only 1$ per month!) + * http://plantuml.com/paypal + * + * This file is part of PlantUML. + * + * PlantUML is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * PlantUML distributed in the hope that it will be useful, but + * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY + * or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public + * License for more details. + * + * You should have received a copy of the GNU General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, + * USA. + * + * + * Original Author: Arnaud Roques + * + * + */ +package net.sourceforge.plantuml.project.draw; + +import java.util.ArrayList; +import java.util.Collection; +import java.util.Collections; +import java.util.List; + +import net.sourceforge.plantuml.sequencediagram.graphic.Segment; +import net.sourceforge.plantuml.ugraphic.UGraphic; +import net.sourceforge.plantuml.ugraphic.ULine; +import net.sourceforge.plantuml.ugraphic.URectangle; +import net.sourceforge.plantuml.ugraphic.UStroke; +import net.sourceforge.plantuml.ugraphic.UTranslate; +import net.sourceforge.plantuml.ugraphic.color.HColor; +import net.sourceforge.plantuml.ugraphic.color.HColorNone; + +public class RectangleTask { + + private final List segments; + private final double round; + private final int completion; + + public RectangleTask(double startPos, double endPos, double round, int completion, Collection paused) { + this.round = round; + this.completion = completion; + if (startPos < endPos) { + this.segments = new ArrayList(new Segment(startPos, endPos).cutSegmentIfNeed(paused)); + } else { + this.segments = Collections.singletonList(new Segment(startPos, startPos + 1)); + } + } + + private void draw2hlines(UGraphic ug, double height, ULine hline) { + ug.draw(hline); + ug.apply(UTranslate.dy(height)).draw(hline); + } + + private void drawRect(UGraphic ug, int completion, HColor documentBackground, double width, double height) { + if (completion == 100 || completion == 0) { + if (completion == 0) + ug = ug.apply(documentBackground.bg()); + final URectangle rect = new URectangle(width, height); + ug.draw(rect); + } else { + final URectangle rect1 = new URectangle(width * completion / 100, height); + ug.draw(rect1); + final URectangle rect2 = new URectangle(width * (100 - completion) / 100, height); + ug.apply(documentBackground.bg()).apply(UTranslate.dx(width * completion / 100)).draw(rect2); + } + } + + public void draw(UGraphic ug, double height, HColor documentBackground, boolean oddStart, boolean oddEnd) { + + if (round == 0) { + drawWithoutRound(ug, height, documentBackground, oddStart, oddEnd); + return; + } + + if (segments.size() != 1) { + drawWithRound(ug, height, documentBackground); + return; + } + + assert segments.size() == 1; + assert round > 0; + final Segment segment = segments.get(0); + + final double width = segment.getLength(); + final URectangle partial = new URectangle(width, height).rounded(round); + if (completion == 100 || completion == 0) { + if (completion == 0) + ug = ug.apply(documentBackground.bg()); + if (oddStart && !oddEnd) + ug.apply(UTranslate.dx(segment.getPos1())).draw(PathUtils.UtoRight(width, height, round)); + else if (!oddStart && oddEnd) + ug.apply(UTranslate.dx(segment.getPos1())).draw(PathUtils.UtoLeft(width, height, round)); + else + ug.apply(UTranslate.dx(segment.getPos1())).draw(partial); + } else { + final double x1 = width * completion / 100; + ug.apply(new HColorNone()).apply(UTranslate.dx(segment.getPos1())) + .draw(PathUtils.UtoLeft(x1, height, round)); + ug.apply(documentBackground.bg()).apply(new HColorNone()).apply(UTranslate.dx(segment.getPos1() + x1)) + .draw(PathUtils.UtoRight(width * (100 - completion) / 100, height, round)); + ug.apply(new HColorNone().bg()).apply(UTranslate.dx(segment.getPos1())).draw(partial); + } + + } + + private void drawWithRound(UGraphic ug, double height, HColor documentBackground) { + + final Segment first = segments.get(0); + ug.apply(UTranslate.dx(first.getPos1())).draw(PathUtils.UtoLeft(first.getLength(), height, round)); + + for (int i = 1; i < segments.size() - 1; i++) { + final Segment segment = segments.get(i); + drawPartly(ug, segment, height, documentBackground, i); + } + + final Segment last = segments.get(segments.size() - 1); + ug.apply(UTranslate.dx(last.getPos1())).draw(PathUtils.UtoRight(last.getLength(), height, round)); + + drawIntermediateDotted(ug, height); + } + + private void drawWithoutRound(UGraphic ug, double height, HColor documentBackground, boolean oddStart, + boolean oddEnd) { + final ULine vline = ULine.vline(height); + + for (int i = 0; i < segments.size(); i++) { + final Segment segment = segments.get(i); + drawPartly(ug, segment, height, documentBackground, i); + + if (!oddStart && i == 0) { + ug.apply(UTranslate.dx(segment.getPos1())).draw(vline); + } + if (!oddEnd && i == segments.size() - 1) { + ug.apply(UTranslate.dx(segment.getPos2())).draw(vline); + } + } + drawIntermediateDotted(ug, height); + } + + private void drawIntermediateDotted(UGraphic ug, double height) { + ug = ug.apply(new UStroke(2, 3, 1)); + for (int i = 0; i < segments.size() - 1; i++) { + final double v1 = segments.get(i).getPos2() + 3; + final double v2 = segments.get(i + 1).getPos1() - 3; + if (v2 > v1) { + draw2hlines(ug.apply(UTranslate.dx(v1)), height, ULine.hline(v2 - v1)); + } + } + } + + private void drawPartly(UGraphic ug, final Segment segment, double height, HColor documentBackground, int i) { + double width = segment.getLength(); + if (i != segments.size() - 1) { + width++; + } + if (width > 0) { + drawRect(ug.apply(new HColorNone()).apply(UTranslate.dx(segment.getPos1())), completion, documentBackground, + width, height); + } + + double pos1 = segment.getPos1(); + double len = segment.getLength(); + if (i == 0) { + if (segments.size() > 1) { + len--; + } + } else { + pos1++; + len--; + } + if (len > 0) { + draw2hlines(ug.apply(UTranslate.dx(pos1)), height, ULine.hline(len)); + } + } + +} diff --git a/src/net/sourceforge/plantuml/project/draw/TaskDraw.java b/src/net/sourceforge/plantuml/project/draw/TaskDraw.java index a18398f97..4d521a9b4 100644 --- a/src/net/sourceforge/plantuml/project/draw/TaskDraw.java +++ b/src/net/sourceforge/plantuml/project/draw/TaskDraw.java @@ -41,6 +41,7 @@ import net.sourceforge.plantuml.cucadiagram.Display; import net.sourceforge.plantuml.graphic.StringBounder; import net.sourceforge.plantuml.graphic.UDrawable; import net.sourceforge.plantuml.project.core.Task; +import net.sourceforge.plantuml.project.core.TaskAttribute; import net.sourceforge.plantuml.project.lang.CenterBorderColor; import net.sourceforge.plantuml.ugraphic.UGraphic; @@ -58,7 +59,7 @@ public interface TaskDraw extends UDrawable { public void drawTitle(UGraphic ug); - public double getHeightTask(StringBounder stringBounder); + public double getFullHeightTask(StringBounder stringBounder); public double getHeightMax(StringBounder stringBounder); @@ -68,5 +69,9 @@ public interface TaskDraw extends UDrawable { public FingerPrint getFingerPrintNote(StringBounder stringBounder); + public double getX1(TaskAttribute taskAttribute); + + public double getX2(TaskAttribute taskAttribute); + } diff --git a/src/net/sourceforge/plantuml/project/draw/TaskDrawDiamond.java b/src/net/sourceforge/plantuml/project/draw/TaskDrawDiamond.java index 8274d3aef..191b5960b 100644 --- a/src/net/sourceforge/plantuml/project/draw/TaskDrawDiamond.java +++ b/src/net/sourceforge/plantuml/project/draw/TaskDrawDiamond.java @@ -43,9 +43,12 @@ import net.sourceforge.plantuml.graphic.StringBounder; import net.sourceforge.plantuml.graphic.TextBlock; import net.sourceforge.plantuml.project.ToTaskDraw; import net.sourceforge.plantuml.project.core.Task; +import net.sourceforge.plantuml.project.core.TaskAttribute; import net.sourceforge.plantuml.project.time.Day; import net.sourceforge.plantuml.project.timescale.TimeScale; +import net.sourceforge.plantuml.style.ClockwiseTopRightBottomLeft; import net.sourceforge.plantuml.style.SName; +import net.sourceforge.plantuml.style.Style; import net.sourceforge.plantuml.style.StyleBuilder; import net.sourceforge.plantuml.style.StyleSignature; import net.sourceforge.plantuml.ugraphic.UGraphic; @@ -67,27 +70,53 @@ public class TaskDrawDiamond extends AbstractTaskDraw { } public double getHeightMax(StringBounder stringBounder) { - return getHeightTask(stringBounder); + return getFullHeightTask(stringBounder); } -// final UFont font = UFont.serif(11); -// return new FontConfiguration(font, HColorUtils.BLACK, HColorUtils.BLACK, false); + @Override + protected double getShapeHeight(StringBounder stringBounder) { +// final Style style = getStyle(); +// final ClockwiseTopRightBottomLeft padding = style.getPadding(); + int result = (int) getFontConfiguration().getFont().getSize2D(); + if (result % 2 == 1) + result--; + return result; + + } final public void drawTitle(UGraphic ug) { + + final Style style = getStyle(); + final ClockwiseTopRightBottomLeft margin = style.getMargin(); + final ClockwiseTopRightBottomLeft padding = style.getPadding(); + ug = ug.apply(UTranslate.dy(margin.getTop())); + final TextBlock title = Display.getWithNewlines(prettyDisplay).create(getFontConfiguration(), HorizontalAlignment.LEFT, new SpriteContainerEmpty()); final StringBounder stringBounder = ug.getStringBounder(); final double titleHeight = title.calculateDimension(stringBounder).getHeight(); - final double h = (margin + getShapeHeight(stringBounder) - titleHeight) / 2; - final double endingPosition = timeScale.getStartingPosition(start) + getHeightTask(stringBounder); - title.drawU(ug.apply(new UTranslate(endingPosition, h))); + final double h = (getShapeHeight(stringBounder) - titleHeight) / 2; + + final double x1 = timeScale.getStartingPosition(start); + final double x2 = timeScale.getEndingPosition(start); + final double width = getShapeHeight(ug.getStringBounder()); + final double delta = x2 - x1 - width; + + final double x = x2 - delta / 2 + padding.getLeft(); + title.drawU(ug.apply(new UTranslate(x, h))); } - public void drawU(UGraphic ug1) { - final double startPos = timeScale.getStartingPosition(start); - ug1 = applyColors(ug1); - UGraphic ug2 = ug1.apply(new UTranslate(startPos + margin, margin)); - drawShape(ug2); + public void drawU(UGraphic ug) { + + final Style style = getStyle(); + final ClockwiseTopRightBottomLeft margin = style.getMargin(); + ug = ug.apply(UTranslate.dy(margin.getTop())); + + final double x1 = timeScale.getStartingPosition(start); + final double x2 = timeScale.getEndingPosition(start); + final double width = getShapeHeight(ug.getStringBounder()); + final double delta = x2 - x1 - width; + drawShape(applyColors(ug).apply(UTranslate.dx(x1 + delta / 2))); } private UGraphic applyColors(UGraphic ug) { @@ -106,13 +135,13 @@ public class TaskDrawDiamond extends AbstractTaskDraw { } public FingerPrint getFingerPrint(StringBounder stringBounder) { - final double h = getHeightTask(stringBounder); + final double h = getFullHeightTask(stringBounder); final double startPos = timeScale.getStartingPosition(start); return new FingerPrint(startPos, getY(stringBounder), startPos + h, getY(stringBounder) + h); } private UShape getDiamond(StringBounder stringBounder) { - final double h = getHeightTask(stringBounder) - 2 * margin; + final double h = getShapeHeight(stringBounder); final UPolygon result = new UPolygon(); result.addPoint(h / 2, 0); result.addPoint(h, h / 2); @@ -121,4 +150,20 @@ public class TaskDrawDiamond extends AbstractTaskDraw { return result; } + public double getX1(TaskAttribute taskAttribute) { + final double x1 = timeScale.getStartingPosition(start); + final double x2 = timeScale.getEndingPosition(start); + final double width = getShapeHeight(null); + final double delta = x2 - x1 - width; + return x1 + delta; + } + + public double getX2(TaskAttribute taskAttribute) { + final double x1 = timeScale.getStartingPosition(start); + final double x2 = timeScale.getEndingPosition(start); + final double width = getShapeHeight(null); + final double delta = x2 - x1 - width; + return x2 - delta; + } + } diff --git a/src/net/sourceforge/plantuml/project/draw/TaskDrawRegular.java b/src/net/sourceforge/plantuml/project/draw/TaskDrawRegular.java index debadf74f..d45e66f94 100644 --- a/src/net/sourceforge/plantuml/project/draw/TaskDrawRegular.java +++ b/src/net/sourceforge/plantuml/project/draw/TaskDrawRegular.java @@ -36,6 +36,7 @@ package net.sourceforge.plantuml.project.draw; import java.awt.geom.Dimension2D; +import java.util.ArrayList; import java.util.Collection; import java.util.TreeSet; @@ -54,9 +55,12 @@ import net.sourceforge.plantuml.graphic.TextBlock; import net.sourceforge.plantuml.project.GanttConstraint; import net.sourceforge.plantuml.project.ToTaskDraw; import net.sourceforge.plantuml.project.core.Task; +import net.sourceforge.plantuml.project.core.TaskAttribute; import net.sourceforge.plantuml.project.core.TaskImpl; import net.sourceforge.plantuml.project.time.Day; import net.sourceforge.plantuml.project.timescale.TimeScale; +import net.sourceforge.plantuml.sequencediagram.graphic.Segment; +import net.sourceforge.plantuml.style.ClockwiseTopRightBottomLeft; import net.sourceforge.plantuml.style.PName; import net.sourceforge.plantuml.style.SName; import net.sourceforge.plantuml.style.Style; @@ -82,7 +86,7 @@ public class TaskDrawRegular extends AbstractTaskDraw { private final Collection constraints; private final ISkinParam skinParam; - private final double margin = 2; + // private final double margin = 2; public TaskDrawRegular(TimeScale timeScale, double y, String prettyDisplay, Day start, Day end, boolean oddStart, boolean oddEnd, ISkinParam skinParam, Task task, ToTaskDraw toTaskDraw, @@ -103,12 +107,23 @@ public class TaskDrawRegular extends AbstractTaskDraw { } } + @Override + protected double getShapeHeight(StringBounder stringBounder) { + final Style style = getStyle(); + final ClockwiseTopRightBottomLeft padding = style.getPadding(); + return padding.getTop() + getTextBlock().calculateDimension(stringBounder).getHeight() + padding.getBottom(); + } + public void drawTitle(UGraphic ug) { - final TextBlock title = Display.getWithNewlines(prettyDisplay).create(getFontConfiguration(), - HorizontalAlignment.LEFT, new SpriteContainerEmpty()); + final TextBlock title = getTextBlock(); final StringBounder stringBounder = ug.getStringBounder(); final Dimension2D dim = title.calculateDimension(stringBounder); - final double h = (margin + getShapeHeight(stringBounder) - dim.getHeight()) / 2; + + final Style style = getStyleSignature().getMergedStyle(getStyleBuilder()); + final ClockwiseTopRightBottomLeft margin = style.getMargin(); + final ClockwiseTopRightBottomLeft padding = style.getPadding(); + ug = ug.apply(UTranslate.dy(margin.getTop() + padding.getTop())); + final double pos1 = timeScale.getStartingPosition(start) + 6; final double pos2 = timeScale.getEndingPosition(end) - 6; final double pos; @@ -116,7 +131,12 @@ public class TaskDrawRegular extends AbstractTaskDraw { pos = pos1; else pos = getOutPosition(pos2); - title.drawU(ug.apply(new UTranslate(pos, h))); + title.drawU(ug.apply(UTranslate.dx(pos))); + } + + private TextBlock getTextBlock() { + return Display.getWithNewlines(prettyDisplay).create(getFontConfiguration(), HorizontalAlignment.LEFT, + new SpriteContainerEmpty()); } private double getOutPosition(double pos2) { @@ -142,14 +162,16 @@ public class TaskDrawRegular extends AbstractTaskDraw { public void drawU(UGraphic ug) { final double startPos = timeScale.getStartingPosition(start); - drawNote(ug.apply((new UTranslate(startPos + margin, getYNotePosition(ug.getStringBounder()))))); + drawNote(ug.apply((new UTranslate(startPos, getYNotePosition(ug.getStringBounder()))))); - ug = applyColors(ug).apply(new UTranslate(margin, margin)); + ug = applyColors(ug); drawShape(ug); } private double getYNotePosition(StringBounder stringBounder) { - return getShapeHeight(stringBounder) + margin * 3; + final Style style = getStyle(); + final ClockwiseTopRightBottomLeft margin = style.getMargin(); + return margin.getTop() + getShapeHeight(stringBounder) + margin.getBottom(); } private void drawNote(UGraphic ug) { @@ -162,7 +184,7 @@ public class TaskDrawRegular extends AbstractTaskDraw { public double getHeightMax(StringBounder stringBounder) { if (note == null) { - return getHeightTask(stringBounder); + return getFullHeightTask(stringBounder); } return getYNotePosition(stringBounder) + getOpaleNote().calculateDimension(stringBounder).getHeight(); } @@ -186,7 +208,7 @@ public class TaskDrawRegular extends AbstractTaskDraw { } public FingerPrint getFingerPrint(StringBounder stringBounder) { - final double h = getHeightTask(stringBounder); + final double h = getFullHeightTask(stringBounder); final double startPos = timeScale.getStartingPosition(start); final double endPos = timeScale.getEndingPosition(end); return new FingerPrint(startPos, getY(stringBounder), endPos - startPos, h); @@ -210,37 +232,101 @@ public class TaskDrawRegular extends AbstractTaskDraw { return ug.apply(getLineColor()).apply(getBackgroundColor().bg()); } - private void drawShape(UGraphic ug) { - final double startPos = timeScale.getStartingPosition(start); - final double endPos = timeScale.getEndingPosition(end); + public double getX1(TaskAttribute taskAttribute) { + final Style style = getStyleSignature().getMergedStyle(getStyleBuilder()); + final ClockwiseTopRightBottomLeft margin = style.getMargin(); + final double startPos = taskAttribute == TaskAttribute.START ? timeScale.getStartingPosition(start) + : timeScale.getStartingPosition(end) + margin.getLeft(); + return startPos; + } - double fullLength = endPos - startPos - 2 * margin; + public double getX2(TaskAttribute taskAttribute) { + final Style style = getStyleSignature().getMergedStyle(getStyleBuilder()); + final ClockwiseTopRightBottomLeft margin = style.getMargin(); + final double endPos = taskAttribute == TaskAttribute.START ? timeScale.getEndingPosition(start) + : timeScale.getEndingPosition(end) - margin.getLeft(); + return endPos; + } + + private void drawShape(UGraphic ug) { + final Style style = getStyleSignature().getMergedStyle(getStyleBuilder()); + final ClockwiseTopRightBottomLeft margin = style.getMargin(); + + final double startPos = timeScale.getStartingPosition(start) + margin.getLeft(); + final double endPos = timeScale.getEndingPosition(end) - margin.getRight(); + + if (url != null) { + ug.startUrl(url); + } + + ug = ug.apply(UTranslate.dy(margin.getTop())); + + final StringBounder stringBounder = ug.getStringBounder(); + + final double round = style.value(PName.RoundCorner).asDouble(); + + final Collection off = new ArrayList(); + for (Day pause : paused) { + final double x1 = timeScale.getStartingPosition(pause); + final double x2 = timeScale.getEndingPosition(pause); + off.add(new Segment(x1, x2)); + } + + final HColor back2 = StyleSignature.of(SName.root, SName.document, SName.ganttDiagram) + .getMergedStyle(getStyleBuilder()).value(PName.BackGroundColor).asColor(getColorSet()); + + final RectangleTask rectangleTask = new RectangleTask(startPos, endPos, round, completion, off); + + rectangleTask.draw(ug, getShapeHeight(stringBounder), back2, oddStart, oddEnd); + + if (url != null) { + ug.closeUrl(); + } + + } + + private void drawShapeOld(UGraphic ug) { + final Style style = getStyleSignature().getMergedStyle(getStyleBuilder()); + final ClockwiseTopRightBottomLeft margin = style.getMargin(); + + final double startPos = timeScale.getStartingPosition(start) + margin.getLeft(); + final double endPos = timeScale.getEndingPosition(end) - margin.getRight(); + + double fullLength = endPos - startPos; if (fullLength < 3) { fullLength = 3; } if (url != null) { ug.startUrl(url); } + + ug = ug.apply(UTranslate.dy(margin.getTop())); + final StringBounder stringBounder = ug.getStringBounder(); + + final double round = style.value(PName.RoundCorner).asDouble(); + if (oddStart && !oddEnd) { - ug.apply(UTranslate.dx(startPos)).draw(PathUtils.UtoRight(fullLength, getShapeHeight(stringBounder))); + ug.apply(UTranslate.dx(startPos)) + .draw(PathUtils.UtoRight(fullLength, getShapeHeight(stringBounder), round)); } else if (!oddStart && oddEnd) { - ug.apply(UTranslate.dx(startPos)).draw(PathUtils.UtoLeft(fullLength, getShapeHeight(stringBounder))); + ug.apply(UTranslate.dx(startPos)).draw(PathUtils.UtoLeft(fullLength, getShapeHeight(stringBounder), round)); } else { - final URectangle full = new URectangle(fullLength, getShapeHeight(stringBounder)).rounded(8); + final URectangle full = new URectangle(fullLength, getShapeHeight(stringBounder)).rounded(round); if (completion == 100) { ug.apply(UTranslate.dx(startPos)).draw(full); } else { final double partialLength = fullLength * completion / 100.; ug.apply(UTranslate.dx(startPos)).apply(HColorUtils.WHITE).apply(HColorUtils.WHITE.bg()).draw(full); if (partialLength > 2) { - final URectangle partial = new URectangle(partialLength, getShapeHeight(stringBounder)).rounded(8); + final URectangle partial = new URectangle(partialLength, getShapeHeight(stringBounder)) + .rounded(round); ug.apply(UTranslate.dx(startPos)).apply(new HColorNone()).draw(partial); } if (partialLength > 10 && partialLength < fullLength - 10) { - final URectangle patch = new URectangle(8, getShapeHeight(stringBounder)); - ug.apply(UTranslate.dx(startPos)).apply(new HColorNone()).apply(UTranslate.dx(partialLength - 8)) - .draw(patch); + final URectangle patch = new URectangle(round, getShapeHeight(stringBounder)); + ug.apply(UTranslate.dx(startPos)).apply(new HColorNone()) + .apply(UTranslate.dx(partialLength - round)).draw(patch); } ug.apply(UTranslate.dx(startPos)).apply(new HColorNone().bg()).draw(full); } diff --git a/src/net/sourceforge/plantuml/project/draw/TaskDrawSeparator.java b/src/net/sourceforge/plantuml/project/draw/TaskDrawSeparator.java index 522ae96cb..a92bf79a8 100644 --- a/src/net/sourceforge/plantuml/project/draw/TaskDrawSeparator.java +++ b/src/net/sourceforge/plantuml/project/draw/TaskDrawSeparator.java @@ -45,6 +45,7 @@ import net.sourceforge.plantuml.graphic.StringBounder; import net.sourceforge.plantuml.graphic.TextBlock; import net.sourceforge.plantuml.graphic.TextBlockUtils; import net.sourceforge.plantuml.project.core.Task; +import net.sourceforge.plantuml.project.core.TaskAttribute; import net.sourceforge.plantuml.project.lang.CenterBorderColor; import net.sourceforge.plantuml.project.time.Day; import net.sourceforge.plantuml.project.timescale.TimeScale; @@ -152,12 +153,12 @@ public class TaskDrawSeparator implements TaskDraw { } public FingerPrint getFingerPrint(StringBounder stringBounder) { - final double h = getHeightTask(stringBounder); + final double h = getFullHeightTask(stringBounder); final double end = timeScale.getEndingPosition(max); return new FingerPrint(0, y, end, y + h); } - public double getHeightTask(StringBounder stringBounder) { + public double getFullHeightTask(StringBounder stringBounder) { final ClockwiseTopRightBottomLeft padding = getStyle().getPadding(); final ClockwiseTopRightBottomLeft margin = getStyle().getMargin(); return margin.getTop() + padding.getTop() + getTextHeight(stringBounder) + padding.getBottom() @@ -196,7 +197,15 @@ public class TaskDrawSeparator implements TaskDraw { } public double getHeightMax(StringBounder stringBounder) { - return getHeightTask(stringBounder); + return getFullHeightTask(stringBounder); + } + + public double getX1(TaskAttribute taskAttribute) { + throw new UnsupportedOperationException(); + } + + public double getX2(TaskAttribute taskAttribute) { + throw new UnsupportedOperationException(); } } diff --git a/src/net/sourceforge/plantuml/sequencediagram/graphic/Segment.java b/src/net/sourceforge/plantuml/sequencediagram/graphic/Segment.java index 62233cb1c..96d4b4249 100644 --- a/src/net/sourceforge/plantuml/sequencediagram/graphic/Segment.java +++ b/src/net/sourceforge/plantuml/sequencediagram/graphic/Segment.java @@ -99,23 +99,27 @@ public class Segment { Collections.sort(sortedDelay, new SortPos1()); final List result2 = new ArrayList(); double pendingStart = pos1; - for (Segment d : sortedDelay) { - if (d.pos1 <= pendingStart) { + for (Segment pause : sortedDelay) { + if (pause.pos1 == pendingStart) { + pendingStart = pause.pos2; continue; } - if (d.pos1 > this.pos2) { - result2.add(new Segment(pendingStart, this.pos2)); + if (pause.pos1 < pendingStart) { + continue; + } + if (pause.pos1 > this.pos2) { + if (pendingStart < this.pos2) + result2.add(new Segment(pendingStart, this.pos2)); return Collections.unmodifiableCollection(result2); } - // if (this.contains(d) == false) { - // throw new IllegalStateException(); - // } - if (this.contains(d)) { - result2.add(new Segment(pendingStart, d.pos1)); - pendingStart = d.pos2; + if (this.contains(pause)) { + if (pendingStart < pause.pos1) + result2.add(new Segment(pendingStart, pause.pos1)); + pendingStart = pause.pos2; } } - result2.add(new Segment(pendingStart, this.pos2)); + if (pendingStart < this.pos2) + result2.add(new Segment(pendingStart, this.pos2)); return Collections.unmodifiableCollection(result2); } diff --git a/src/net/sourceforge/plantuml/ugraphic/color/HColorSimple.java b/src/net/sourceforge/plantuml/ugraphic/color/HColorSimple.java index a3e0726df..81454156b 100644 --- a/src/net/sourceforge/plantuml/ugraphic/color/HColorSimple.java +++ b/src/net/sourceforge/plantuml/ugraphic/color/HColorSimple.java @@ -49,12 +49,16 @@ public class HColorSimple extends HColorAbstract implements HColor { @Override public String toString() { - if (color.getAlpha() == 0) { + if (isTransparent()) { return "transparent"; } return color.toString() + " alpha=" + color.getAlpha() + " monochrome=" + monochrome; } + public boolean isTransparent() { + return color.getAlpha() == 0; + } + @Override public boolean equals(Object other) { if (other instanceof HColorSimple == false) { diff --git a/src/net/sourceforge/plantuml/ugraphic/color/HColorUtils.java b/src/net/sourceforge/plantuml/ugraphic/color/HColorUtils.java index 6456c0fe0..ead06dbec 100644 --- a/src/net/sourceforge/plantuml/ugraphic/color/HColorUtils.java +++ b/src/net/sourceforge/plantuml/ugraphic/color/HColorUtils.java @@ -146,6 +146,9 @@ public class HColorUtils { if (back instanceof HColorBackground && ((HColorBackground) back).getBack() == TRANSPARENT) { return true; } + if (back instanceof HColorSimple && ((HColorSimple) back).isTransparent()) { + return true; + } return false; } diff --git a/src/net/sourceforge/plantuml/ugraphic/eps/DriverTextEps.java b/src/net/sourceforge/plantuml/ugraphic/eps/DriverTextEps.java index 7b7c2200f..b025b359e 100644 --- a/src/net/sourceforge/plantuml/ugraphic/eps/DriverTextEps.java +++ b/src/net/sourceforge/plantuml/ugraphic/eps/DriverTextEps.java @@ -60,6 +60,7 @@ import net.sourceforge.plantuml.ugraphic.UShape; import net.sourceforge.plantuml.ugraphic.UText; import net.sourceforge.plantuml.ugraphic.color.ColorMapper; import net.sourceforge.plantuml.ugraphic.color.HColor; +import net.sourceforge.plantuml.ugraphic.color.HColorUtils; public class DriverTextEps implements UDriver { @@ -84,12 +85,16 @@ public class DriverTextEps implements UDriver { final UText shape = (UText) ushape; + final FontConfiguration fontConfiguration = shape.getFontConfiguration(); + if (HColorUtils.isTransparent(fontConfiguration.getColor())) { + return; + } + if (strategy == EpsStrategy.WITH_MACRO_AND_TEXT) { drawAsText(shape, x, y, param, eps, mapper); return; } - final FontConfiguration fontConfiguration = shape.getFontConfiguration(); final UFont font = fontConfiguration.getFont(); final TextLayout textLayout = new TextLayout(shape.getText(), font.getUnderlayingFont(), fontRenderContext); diff --git a/src/net/sourceforge/plantuml/ugraphic/g2d/DriverTextG2d.java b/src/net/sourceforge/plantuml/ugraphic/g2d/DriverTextG2d.java index 8abfebf62..11f49133d 100644 --- a/src/net/sourceforge/plantuml/ugraphic/g2d/DriverTextG2d.java +++ b/src/net/sourceforge/plantuml/ugraphic/g2d/DriverTextG2d.java @@ -61,6 +61,7 @@ import net.sourceforge.plantuml.ugraphic.UText; import net.sourceforge.plantuml.ugraphic.color.ColorMapper; import net.sourceforge.plantuml.ugraphic.color.HColor; import net.sourceforge.plantuml.ugraphic.color.HColorGradient; +import net.sourceforge.plantuml.ugraphic.color.HColorUtils; public class DriverTextG2d implements UDriver { @@ -73,6 +74,10 @@ public class DriverTextG2d implements UDriver { public void draw(UShape ushape, double x, double y, ColorMapper mapper, UParam param, Graphics2D g2d) { final UText shape = (UText) ushape; final FontConfiguration fontConfiguration = shape.getFontConfiguration(); + + if (HColorUtils.isTransparent(fontConfiguration.getColor())) { + return; + } final String text = shape.getText(); final List strings = StyledString.build(text); @@ -82,8 +87,7 @@ public class DriverTextG2d implements UDriver { for (StyledString styledString : strings) { final FontConfiguration fc = styledString.getStyle() == FontStyle.BOLD ? fontConfiguration.bold() : fontConfiguration; - final Dimension2D dim = calculateDimension( - FileFormat.PNG.getDefaultStringBounder(), fc.getFont(), + final Dimension2D dim = calculateDimension(FileFormat.PNG.getDefaultStringBounder(), fc.getFont(), styledString.getText()); printSingleText(g2d, fc, styledString.getText(), x, y, mapper, param); x += dim.getWidth(); @@ -109,8 +113,7 @@ public class DriverTextG2d implements UDriver { } else if (orientation == 0) { - final Dimension2D dimBack = calculateDimension( - FileFormat.PNG.getDefaultStringBounder(), font, text); + final Dimension2D dimBack = calculateDimension(FileFormat.PNG.getDefaultStringBounder(), font, text); if (fontConfiguration.containsStyle(FontStyle.BACKCOLOR)) { final Rectangle2D.Double area = new Rectangle2D.Double(x, y - dimBack.getHeight() + 1.5, dimBack.getWidth(), dimBack.getHeight()); @@ -140,16 +143,14 @@ public class DriverTextG2d implements UDriver { if (extended != null) { g2d.setColor(mapper.toColor(extended)); } - final Dimension2D dim = calculateDimension( - FileFormat.PNG.getDefaultStringBounder(), font, text); + final Dimension2D dim = calculateDimension(FileFormat.PNG.getDefaultStringBounder(), font, text); final int ypos = (int) (y + 2.5); g2d.setStroke(new BasicStroke((float) 1)); g2d.drawLine((int) x, ypos, (int) (x + dim.getWidth()), ypos); g2d.setStroke(new BasicStroke()); } if (fontConfiguration.containsStyle(FontStyle.WAVE)) { - final Dimension2D dim = calculateDimension( - FileFormat.PNG.getDefaultStringBounder(), font, text); + final Dimension2D dim = calculateDimension(FileFormat.PNG.getDefaultStringBounder(), font, text); final int ypos = (int) (y + 2.5) - 1; if (extended != null) { g2d.setColor(mapper.toColor(extended)); @@ -160,8 +161,7 @@ public class DriverTextG2d implements UDriver { } } if (fontConfiguration.containsStyle(FontStyle.STRIKE)) { - final Dimension2D dim = calculateDimension( - FileFormat.PNG.getDefaultStringBounder(), font, text); + final Dimension2D dim = calculateDimension(FileFormat.PNG.getDefaultStringBounder(), font, text); final FontMetrics fm = g2d.getFontMetrics(font.getUnderlayingFont()); final int ypos = (int) (y - fm.getDescent() - 0.5); if (extended != null) { diff --git a/src/net/sourceforge/plantuml/ugraphic/svg/DriverTextSvg.java b/src/net/sourceforge/plantuml/ugraphic/svg/DriverTextSvg.java index 1d970e3af..46303cf0c 100644 --- a/src/net/sourceforge/plantuml/ugraphic/svg/DriverTextSvg.java +++ b/src/net/sourceforge/plantuml/ugraphic/svg/DriverTextSvg.java @@ -52,6 +52,7 @@ import net.sourceforge.plantuml.ugraphic.UText; import net.sourceforge.plantuml.ugraphic.color.ColorMapper; import net.sourceforge.plantuml.ugraphic.color.HColor; import net.sourceforge.plantuml.ugraphic.color.HColorGradient; +import net.sourceforge.plantuml.ugraphic.color.HColorUtils; public class DriverTextSvg implements UDriver { @@ -72,6 +73,9 @@ public class DriverTextSvg implements UDriver { final UText shape = (UText) ushape; final FontConfiguration fontConfiguration = shape.getFontConfiguration(); + if (HColorUtils.isTransparent(fontConfiguration.getColor())) { + return; + } final UFont font = fontConfiguration.getFont(); String fontWeight = null; if (fontConfiguration.containsStyle(FontStyle.BOLD) || font.isBold()) { diff --git a/src/net/sourceforge/plantuml/version/Version.java b/src/net/sourceforge/plantuml/version/Version.java index d3f1425d2..8765e3ed1 100644 --- a/src/net/sourceforge/plantuml/version/Version.java +++ b/src/net/sourceforge/plantuml/version/Version.java @@ -80,7 +80,7 @@ public class Version { } public static int beta() { - final int beta = 3; + final int beta = 4; return beta; } diff --git a/test/nonreg/simple/A0003_TestResult.java b/test/nonreg/simple/A0003_TestResult.java index 3599af751..381325cc8 100644 --- a/test/nonreg/simple/A0003_TestResult.java +++ b/test/nonreg/simple/A0003_TestResult.java @@ -5,7 +5,7 @@ public class A0003_TestResult { /* """ DPI: 96 -dimension: [ 367.7447 ; 78.0000 ] +dimension: [ 367.7447 ; 76.0000 ] scaleFactor: 2.0000 seed: -6040919743496430850 svgLinkTarget: _top @@ -14,7 +14,7 @@ preserveAspectRatio: none RECTANGLE: pt1: [ 8.0000 ; 29.0000 ] - pt2: [ 16.0000 ; 61.0000 ] + pt2: [ 16.0000 ; 59.0000 ] xCorner: 0 yCorner: 0 stroke: 0.0-0.0-1.0 @@ -24,7 +24,7 @@ RECTANGLE: RECTANGLE: pt1: [ 36.0000 ; 29.0000 ] - pt2: [ 44.0000 ; 61.0000 ] + pt2: [ 44.0000 ; 59.0000 ] xCorner: 0 yCorner: 0 stroke: 0.0-0.0-1.0 @@ -34,7 +34,7 @@ RECTANGLE: RECTANGLE: pt1: [ 44.0000 ; 29.0000 ] - pt2: [ 72.0000 ; 61.0000 ] + pt2: [ 72.0000 ; 59.0000 ] xCorner: 0 yCorner: 0 stroke: 0.0-0.0-1.0 @@ -44,7 +44,7 @@ RECTANGLE: RECTANGLE: pt1: [ 92.0000 ; 29.0000 ] - pt2: [ 100.0000 ; 61.0000 ] + pt2: [ 100.0000 ; 59.0000 ] xCorner: 0 yCorner: 0 stroke: 0.0-0.0-1.0 @@ -54,7 +54,7 @@ RECTANGLE: RECTANGLE: pt1: [ 120.0000 ; 29.0000 ] - pt2: [ 128.0000 ; 61.0000 ] + pt2: [ 128.0000 ; 59.0000 ] xCorner: 0 yCorner: 0 stroke: 0.0-0.0-1.0 @@ -64,7 +64,7 @@ RECTANGLE: RECTANGLE: pt1: [ 148.0000 ; 29.0000 ] - pt2: [ 156.0000 ; 61.0000 ] + pt2: [ 156.0000 ; 59.0000 ] xCorner: 0 yCorner: 0 stroke: 0.0-0.0-1.0 @@ -122,49 +122,49 @@ TEXT: LINE: pt1: [ 16.0000 ; 16.0000 ] - pt2: [ 16.0000 ; 61.0000 ] + pt2: [ 16.0000 ; 59.0000 ] stroke: 0.0-0.0-1.0 shadow: 0 color: ffc0c0c0 LINE: pt1: [ 44.0000 ; 16.0000 ] - pt2: [ 44.0000 ; 61.0000 ] + pt2: [ 44.0000 ; 59.0000 ] stroke: 0.0-0.0-1.0 shadow: 0 color: ffc0c0c0 LINE: pt1: [ 72.0000 ; 16.0000 ] - pt2: [ 72.0000 ; 61.0000 ] + pt2: [ 72.0000 ; 59.0000 ] stroke: 0.0-0.0-1.0 shadow: 0 color: ffc0c0c0 LINE: pt1: [ 100.0000 ; 16.0000 ] - pt2: [ 100.0000 ; 61.0000 ] + pt2: [ 100.0000 ; 59.0000 ] stroke: 0.0-0.0-1.0 shadow: 0 color: ffc0c0c0 LINE: pt1: [ 128.0000 ; 16.0000 ] - pt2: [ 128.0000 ; 61.0000 ] + pt2: [ 128.0000 ; 59.0000 ] stroke: 0.0-0.0-1.0 shadow: 0 color: ffc0c0c0 LINE: pt1: [ 156.0000 ; 16.0000 ] - pt2: [ 156.0000 ; 61.0000 ] + pt2: [ 156.0000 ; 59.0000 ] stroke: 0.0-0.0-1.0 shadow: 0 color: ffc0c0c0 LINE: pt1: [ 168.0000 ; 16.0000 ] - pt2: [ 168.0000 ; 61.0000 ] + pt2: [ 168.0000 ; 59.0000 ] stroke: 0.0-0.0-1.0 shadow: 0 color: ffc0c0c0 @@ -228,25 +228,25 @@ LINE: color: ffc0c0c0 LINE: - pt1: [ 132.0000 ; 45.0000 ] - pt2: [ 132.0000 ; 53.0000 ] + pt1: [ 131.0000 ; 42.0000 ] + pt2: [ 131.0000 ; 51.5000 ] stroke: 0.0-0.0-1.5 shadow: 0 color: ffa80036 LINE: - pt1: [ 132.0000 ; 53.0000 ] - pt2: [ 140.0000 ; 53.0000 ] + pt1: [ 131.0000 ; 51.5000 ] + pt2: [ 139.0000 ; 51.5000 ] stroke: 0.0-0.0-1.5 shadow: 0 color: ffa80036 POLYGON: points: - - [ 136.0000 ; 49.0000 ] - - [ 136.0000 ; 53.0000 ] - - [ 136.0000 ; 57.0000 ] - - [ 140.0000 ; 53.0000 ] + - [ 135.0000 ; 47.5000 ] + - [ 135.0000 ; 51.5000 ] + - [ 135.0000 ; 55.5000 ] + - [ 139.0000 ; 51.5000 ] stroke: 0.0-0.0-1.5 shadow: 0 color: ffa80036 @@ -254,171 +254,311 @@ POLYGON: RECTANGLE: pt1: [ 2.0000 ; 31.0000 ] - pt2: [ 138.0000 ; 43.0000 ] - xCorner: 8 - yCorner: 8 + pt2: [ 9.0000 ; 42.0000 ] + xCorner: 0 + yCorner: 0 stroke: 0.0-0.0-1.0 shadow: 0 - color: ffadd8e6 + color: NULL_COLOR backcolor: ffe6e6fa +LINE: + pt1: [ 2.0000 ; 31.0000 ] + pt2: [ 7.0000 ; 31.0000 ] + stroke: 0.0-0.0-1.0 + shadow: 0 + color: ffadd8e6 + +LINE: + pt1: [ 2.0000 ; 42.0000 ] + pt2: [ 7.0000 ; 42.0000 ] + stroke: 0.0-0.0-1.0 + shadow: 0 + color: ffadd8e6 + +LINE: + pt1: [ 2.0000 ; 31.0000 ] + pt2: [ 2.0000 ; 42.0000 ] + stroke: 0.0-0.0-1.0 + shadow: 0 + color: ffadd8e6 + RECTANGLE: - pt1: [ 9.0000 ; 31.0000 ] - pt2: [ 16.0000 ; 44.0000 ] + pt1: [ 16.0000 ; 31.0000 ] + pt2: [ 37.0000 ; 42.0000 ] xCorner: 0 yCorner: 0 stroke: 0.0-0.0-1.0 shadow: 0 - color: ffffffff - backcolor: ffffffff + color: NULL_COLOR + backcolor: ffe6e6fa LINE: - pt1: [ 9.0000 ; 31.0000 ] - pt2: [ 16.0000 ; 31.0000 ] - stroke: 2.0-3.0-1.0 + pt1: [ 17.0000 ; 31.0000 ] + pt2: [ 36.0000 ; 31.0000 ] + stroke: 0.0-0.0-1.0 shadow: 0 color: ffadd8e6 LINE: - pt1: [ 9.0000 ; 43.0000 ] - pt2: [ 16.0000 ; 43.0000 ] - stroke: 2.0-3.0-1.0 + pt1: [ 17.0000 ; 42.0000 ] + pt2: [ 36.0000 ; 42.0000 ] + stroke: 0.0-0.0-1.0 shadow: 0 color: ffadd8e6 RECTANGLE: - pt1: [ 37.0000 ; 31.0000 ] - pt2: [ 44.0000 ; 44.0000 ] + pt1: [ 44.0000 ; 31.0000 ] + pt2: [ 65.0000 ; 42.0000 ] xCorner: 0 yCorner: 0 stroke: 0.0-0.0-1.0 shadow: 0 - color: ffffffff - backcolor: ffffffff + color: NULL_COLOR + backcolor: ffe6e6fa LINE: - pt1: [ 37.0000 ; 31.0000 ] - pt2: [ 44.0000 ; 31.0000 ] - stroke: 2.0-3.0-1.0 + pt1: [ 45.0000 ; 31.0000 ] + pt2: [ 64.0000 ; 31.0000 ] + stroke: 0.0-0.0-1.0 shadow: 0 color: ffadd8e6 LINE: - pt1: [ 37.0000 ; 43.0000 ] - pt2: [ 44.0000 ; 43.0000 ] - stroke: 2.0-3.0-1.0 + pt1: [ 45.0000 ; 42.0000 ] + pt2: [ 64.0000 ; 42.0000 ] + stroke: 0.0-0.0-1.0 shadow: 0 color: ffadd8e6 RECTANGLE: - pt1: [ 65.0000 ; 31.0000 ] - pt2: [ 72.0000 ; 44.0000 ] + pt1: [ 72.0000 ; 31.0000 ] + pt2: [ 93.0000 ; 42.0000 ] xCorner: 0 yCorner: 0 stroke: 0.0-0.0-1.0 shadow: 0 - color: ffffffff - backcolor: ffffffff + color: NULL_COLOR + backcolor: ffe6e6fa LINE: - pt1: [ 65.0000 ; 31.0000 ] - pt2: [ 72.0000 ; 31.0000 ] - stroke: 2.0-3.0-1.0 + pt1: [ 73.0000 ; 31.0000 ] + pt2: [ 92.0000 ; 31.0000 ] + stroke: 0.0-0.0-1.0 shadow: 0 color: ffadd8e6 LINE: - pt1: [ 65.0000 ; 43.0000 ] - pt2: [ 72.0000 ; 43.0000 ] - stroke: 2.0-3.0-1.0 + pt1: [ 73.0000 ; 42.0000 ] + pt2: [ 92.0000 ; 42.0000 ] + stroke: 0.0-0.0-1.0 shadow: 0 color: ffadd8e6 RECTANGLE: - pt1: [ 93.0000 ; 31.0000 ] - pt2: [ 100.0000 ; 44.0000 ] + pt1: [ 100.0000 ; 31.0000 ] + pt2: [ 121.0000 ; 42.0000 ] xCorner: 0 yCorner: 0 stroke: 0.0-0.0-1.0 shadow: 0 - color: ffffffff - backcolor: ffffffff + color: NULL_COLOR + backcolor: ffe6e6fa LINE: - pt1: [ 93.0000 ; 31.0000 ] - pt2: [ 100.0000 ; 31.0000 ] - stroke: 2.0-3.0-1.0 + pt1: [ 101.0000 ; 31.0000 ] + pt2: [ 120.0000 ; 31.0000 ] + stroke: 0.0-0.0-1.0 shadow: 0 color: ffadd8e6 LINE: - pt1: [ 93.0000 ; 43.0000 ] - pt2: [ 100.0000 ; 43.0000 ] - stroke: 2.0-3.0-1.0 + pt1: [ 101.0000 ; 42.0000 ] + pt2: [ 120.0000 ; 42.0000 ] + stroke: 0.0-0.0-1.0 shadow: 0 color: ffadd8e6 RECTANGLE: - pt1: [ 121.0000 ; 31.0000 ] - pt2: [ 128.0000 ; 44.0000 ] + pt1: [ 128.0000 ; 31.0000 ] + pt2: [ 138.0000 ; 42.0000 ] xCorner: 0 yCorner: 0 stroke: 0.0-0.0-1.0 shadow: 0 - color: ffffffff - backcolor: ffffffff + color: NULL_COLOR + backcolor: ffe6e6fa LINE: - pt1: [ 121.0000 ; 31.0000 ] - pt2: [ 128.0000 ; 31.0000 ] + pt1: [ 129.0000 ; 31.0000 ] + pt2: [ 138.0000 ; 31.0000 ] + stroke: 0.0-0.0-1.0 + shadow: 0 + color: ffadd8e6 + +LINE: + pt1: [ 129.0000 ; 42.0000 ] + pt2: [ 138.0000 ; 42.0000 ] + stroke: 0.0-0.0-1.0 + shadow: 0 + color: ffadd8e6 + +LINE: + pt1: [ 138.0000 ; 31.0000 ] + pt2: [ 138.0000 ; 42.0000 ] + stroke: 0.0-0.0-1.0 + shadow: 0 + color: ffadd8e6 + +LINE: + pt1: [ 11.0000 ; 31.0000 ] + pt2: [ 13.0000 ; 31.0000 ] stroke: 2.0-3.0-1.0 shadow: 0 color: ffadd8e6 LINE: - pt1: [ 121.0000 ; 43.0000 ] - pt2: [ 128.0000 ; 43.0000 ] + pt1: [ 11.0000 ; 42.0000 ] + pt2: [ 13.0000 ; 42.0000 ] + stroke: 2.0-3.0-1.0 + shadow: 0 + color: ffadd8e6 + +LINE: + pt1: [ 39.0000 ; 31.0000 ] + pt2: [ 41.0000 ; 31.0000 ] + stroke: 2.0-3.0-1.0 + shadow: 0 + color: ffadd8e6 + +LINE: + pt1: [ 39.0000 ; 42.0000 ] + pt2: [ 41.0000 ; 42.0000 ] + stroke: 2.0-3.0-1.0 + shadow: 0 + color: ffadd8e6 + +LINE: + pt1: [ 67.0000 ; 31.0000 ] + pt2: [ 69.0000 ; 31.0000 ] + stroke: 2.0-3.0-1.0 + shadow: 0 + color: ffadd8e6 + +LINE: + pt1: [ 67.0000 ; 42.0000 ] + pt2: [ 69.0000 ; 42.0000 ] + stroke: 2.0-3.0-1.0 + shadow: 0 + color: ffadd8e6 + +LINE: + pt1: [ 95.0000 ; 31.0000 ] + pt2: [ 97.0000 ; 31.0000 ] + stroke: 2.0-3.0-1.0 + shadow: 0 + color: ffadd8e6 + +LINE: + pt1: [ 95.0000 ; 42.0000 ] + pt2: [ 97.0000 ; 42.0000 ] + stroke: 2.0-3.0-1.0 + shadow: 0 + color: ffadd8e6 + +LINE: + pt1: [ 123.0000 ; 31.0000 ] + pt2: [ 125.0000 ; 31.0000 ] + stroke: 2.0-3.0-1.0 + shadow: 0 + color: ffadd8e6 + +LINE: + pt1: [ 123.0000 ; 42.0000 ] + pt2: [ 125.0000 ; 42.0000 ] stroke: 2.0-3.0-1.0 shadow: 0 color: ffadd8e6 RECTANGLE: - pt1: [ 142.0000 ; 47.0000 ] - pt2: [ 166.0000 ; 59.0000 ] - xCorner: 8 - yCorner: 8 + pt1: [ 142.0000 ; 46.0000 ] + pt2: [ 149.0000 ; 57.0000 ] + xCorner: 0 + yCorner: 0 stroke: 0.0-0.0-1.0 shadow: 0 - color: ffa80036 + color: NULL_COLOR backcolor: fffefece +LINE: + pt1: [ 142.0000 ; 46.0000 ] + pt2: [ 147.0000 ; 46.0000 ] + stroke: 0.0-0.0-1.0 + shadow: 0 + color: ffa80036 + +LINE: + pt1: [ 142.0000 ; 57.0000 ] + pt2: [ 147.0000 ; 57.0000 ] + stroke: 0.0-0.0-1.0 + shadow: 0 + color: ffa80036 + +LINE: + pt1: [ 142.0000 ; 46.0000 ] + pt2: [ 142.0000 ; 57.0000 ] + stroke: 0.0-0.0-1.0 + shadow: 0 + color: ffa80036 + RECTANGLE: - pt1: [ 149.0000 ; 47.0000 ] - pt2: [ 156.0000 ; 60.0000 ] + pt1: [ 156.0000 ; 46.0000 ] + pt2: [ 166.0000 ; 57.0000 ] xCorner: 0 yCorner: 0 stroke: 0.0-0.0-1.0 shadow: 0 - color: ffffffff - backcolor: ffffffff + color: NULL_COLOR + backcolor: fffefece LINE: - pt1: [ 149.0000 ; 47.0000 ] - pt2: [ 156.0000 ; 47.0000 ] + pt1: [ 157.0000 ; 46.0000 ] + pt2: [ 166.0000 ; 46.0000 ] + stroke: 0.0-0.0-1.0 + shadow: 0 + color: ffa80036 + +LINE: + pt1: [ 157.0000 ; 57.0000 ] + pt2: [ 166.0000 ; 57.0000 ] + stroke: 0.0-0.0-1.0 + shadow: 0 + color: ffa80036 + +LINE: + pt1: [ 166.0000 ; 46.0000 ] + pt2: [ 166.0000 ; 57.0000 ] + stroke: 0.0-0.0-1.0 + shadow: 0 + color: ffa80036 + +LINE: + pt1: [ 151.0000 ; 46.0000 ] + pt2: [ 153.0000 ; 46.0000 ] stroke: 2.0-3.0-1.0 shadow: 0 color: ffa80036 LINE: - pt1: [ 149.0000 ; 59.0000 ] - pt2: [ 156.0000 ; 59.0000 ] + pt1: [ 151.0000 ; 57.0000 ] + pt2: [ 153.0000 ; 57.0000 ] stroke: 2.0-3.0-1.0 shadow: 0 color: ffa80036 TEXT: text: Prototype design - position: [ 142.0000 ; 39.0556 ] + position: [ 142.0000 ; 39.5556 ] orientation: 0 font: SansSerif.plain/11 [] color: ff000000 @@ -426,59 +566,59 @@ TEXT: TEXT: text: Testing - position: [ 170.0000 ; 55.0556 ] + position: [ 170.0000 ; 54.5556 ] orientation: 0 font: SansSerif.plain/11 [] color: ff000000 extendedColor: NULL_COLOR LINE: - pt1: [ 0.0000 ; 61.0000 ] - pt2: [ 168.0000 ; 61.0000 ] + pt1: [ 0.0000 ; 59.0000 ] + pt2: [ 168.0000 ; 59.0000 ] stroke: 0.0-0.0-1.0 shadow: 0 color: ffc0c0c0 LINE: - pt1: [ 0.0000 ; 61.0000 ] - pt2: [ 0.0000 ; 77.0000 ] + pt1: [ 0.0000 ; 59.0000 ] + pt2: [ 0.0000 ; 75.0000 ] stroke: 0.0-0.0-1.0 shadow: 0 color: ffc0c0c0 LINE: - pt1: [ 68.0000 ; 61.0000 ] - pt2: [ 68.0000 ; 77.0000 ] + pt1: [ 68.0000 ; 59.0000 ] + pt2: [ 68.0000 ; 75.0000 ] stroke: 0.0-0.0-1.0 shadow: 0 color: ffc0c0c0 TEXT: text: Oct - position: [ 17.4989 ; 70.3333 ] + position: [ 17.4989 ; 68.3333 ] orientation: 0 font: Serif.bold/12 [BOLD] color: ff000000 extendedColor: NULL_COLOR LINE: - pt1: [ 168.0000 ; 61.0000 ] - pt2: [ 168.0000 ; 77.0000 ] + pt1: [ 168.0000 ; 59.0000 ] + pt2: [ 168.0000 ; 75.0000 ] stroke: 0.0-0.0-1.0 shadow: 0 color: ffc0c0c0 TEXT: text: Nov 2020 - position: [ 72.7816 ; 70.3333 ] + position: [ 72.7816 ; 68.3333 ] orientation: 0 font: Serif.bold/12 [BOLD] color: ff000000 extendedColor: NULL_COLOR LINE: - pt1: [ 0.0000 ; 77.0000 ] - pt2: [ 168.0000 ; 77.0000 ] + pt1: [ 0.0000 ; 75.0000 ] + pt2: [ 168.0000 ; 75.0000 ] stroke: 0.0-0.0-1.0 shadow: 0 color: ffc0c0c0