plantuml/src/net/sourceforge/plantuml/svek/Cluster.java

946 lines
29 KiB
Java

/* ========================================================================
* 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.svek;
import java.awt.geom.Dimension2D;
import java.awt.geom.Point2D;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.EnumSet;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Set;
import net.sourceforge.plantuml.AlignmentParam;
import net.sourceforge.plantuml.ColorParam;
import net.sourceforge.plantuml.Dimension2DDouble;
import net.sourceforge.plantuml.FontParam;
import net.sourceforge.plantuml.ISkinParam;
import net.sourceforge.plantuml.LineParam;
import net.sourceforge.plantuml.SkinParam;
import net.sourceforge.plantuml.SkinParamUtils;
import net.sourceforge.plantuml.UmlDiagramType;
import net.sourceforge.plantuml.Url;
import net.sourceforge.plantuml.cucadiagram.EntityPosition;
import net.sourceforge.plantuml.cucadiagram.EntityUtils;
import net.sourceforge.plantuml.cucadiagram.IEntity;
import net.sourceforge.plantuml.cucadiagram.IGroup;
import net.sourceforge.plantuml.cucadiagram.Member;
import net.sourceforge.plantuml.cucadiagram.MethodsOrFieldsArea;
import net.sourceforge.plantuml.cucadiagram.Stereotype;
import net.sourceforge.plantuml.cucadiagram.dot.GraphvizVersion;
import net.sourceforge.plantuml.graphic.HorizontalAlignment;
import net.sourceforge.plantuml.graphic.StringBounder;
import net.sourceforge.plantuml.graphic.TextBlock;
import net.sourceforge.plantuml.graphic.TextBlockEmpty;
import net.sourceforge.plantuml.graphic.TextBlockWidth;
import net.sourceforge.plantuml.graphic.USymbol;
import net.sourceforge.plantuml.graphic.color.ColorType;
import net.sourceforge.plantuml.graphic.color.Colors;
import net.sourceforge.plantuml.posimo.Moveable;
import net.sourceforge.plantuml.skin.rose.Rose;
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.image.EntityImageState;
import net.sourceforge.plantuml.ugraphic.UComment;
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.HColorBackground;
import net.sourceforge.plantuml.ugraphic.color.HColorUtils;
import net.sourceforge.plantuml.utils.UniqueSequence;
public class Cluster implements Moveable {
private final Cluster parentCluster;
private final IGroup group;
private final List<Node> nodes = new ArrayList<Node>();
private final List<Cluster> children = new ArrayList<Cluster>();
private final int color;
private final int colorTitle;
private final ISkinParam skinParam;
private int titleAndAttributeWidth;
private int titleAndAttributeHeight;
private TextBlock ztitle;
private TextBlock zstereo;
private double xTitle;
private double yTitle;
private double minX;
private double minY;
private double maxX;
private double maxY;
public void moveSvek(double deltaX, double deltaY) {
this.xTitle += deltaX;
this.minX += deltaX;
this.maxX += deltaX;
this.yTitle += deltaY;
this.minY += deltaY;
this.maxY += deltaY;
}
private boolean hasEntryOrExitPoint() {
for (Node sh : nodes) {
if (sh.getEntityPosition() != EntityPosition.NORMAL) {
return true;
}
}
return false;
}
public Cluster(ColorSequence colorSequence, ISkinParam skinParam, IGroup root) {
this(null, colorSequence, skinParam, root);
}
private ColorParam border;
private Cluster(Cluster parentCluster, ColorSequence colorSequence, ISkinParam skinParam, IGroup group) {
if (group == null) {
throw new IllegalStateException();
}
this.parentCluster = parentCluster;
this.group = group;
if (group.getUSymbol() != null) {
border = group.getUSymbol().getColorParamBorder();
}
this.color = colorSequence.getValue();
this.colorTitle = colorSequence.getValue();
this.skinParam = group.getColors(skinParam).mute(skinParam);
}
@Override
public String toString() {
return super.toString() + " " + group;
}
public final Cluster getParentCluster() {
return parentCluster;
}
public void addNode(Node node) {
if (node == null) {
throw new IllegalArgumentException();
}
this.nodes.add(node);
node.setCluster(this);
}
public final List<Node> getNodes() {
return Collections.unmodifiableList(nodes);
}
private List<Node> getNodesOrderedTop(Collection<Line> lines) {
final List<Node> firsts = new ArrayList<Node>();
final Set<String> tops = new HashSet<String>();
final Map<String, Node> shs = new HashMap<String, Node>();
for (final Iterator<Node> it = nodes.iterator(); it.hasNext();) {
final Node node = it.next();
shs.put(node.getUid(), node);
if (node.isTop() && node.getEntityPosition() == EntityPosition.NORMAL) {
firsts.add(node);
tops.add(node.getUid());
}
}
for (Line l : lines) {
if (tops.contains(l.getStartUidPrefix())) {
final Node sh = shs.get(l.getEndUidPrefix());
if (sh != null && sh.getEntityPosition() == EntityPosition.NORMAL) {
firsts.add(0, sh);
}
}
if (l.isInverted()) {
final Node sh = shs.get(l.getStartUidPrefix());
if (sh != null && sh.getEntityPosition() == EntityPosition.NORMAL) {
firsts.add(0, sh);
}
}
}
return firsts;
}
private List<Node> getNodesEntryExit(EnumSet<EntityPosition> positions) {
final List<Node> result = new ArrayList<Node>();
for (final Iterator<Node> it = nodes.iterator(); it.hasNext();) {
final Node sh = it.next();
if (positions.contains(sh.getEntityPosition())) {
result.add(sh);
}
}
return result;
}
private List<Node> getNodesOrderedWithoutTop(Collection<Line> lines) {
final List<Node> all = new ArrayList<Node>(nodes);
final Set<String> tops = new HashSet<String>();
final Map<String, Node> shs = new HashMap<String, Node>();
for (final Iterator<Node> it = all.iterator(); it.hasNext();) {
final Node sh = it.next();
if (sh.getEntityPosition() != EntityPosition.NORMAL) {
it.remove();
continue;
}
shs.put(sh.getUid(), sh);
if (sh.isTop()) {
tops.add(sh.getUid());
it.remove();
}
}
for (Line l : lines) {
if (tops.contains(l.getStartUidPrefix())) {
final Node sh = shs.get(l.getEndUidPrefix());
if (sh != null) {
all.remove(sh);
}
}
if (l.isInverted()) {
final Node sh = shs.get(l.getStartUidPrefix());
if (sh != null) {
all.remove(sh);
}
}
}
return all;
}
public final List<Cluster> getChildren() {
return Collections.unmodifiableList(children);
}
public Cluster createChild(int titleAndAttributeWidth, int titleAndAttributeHeight, TextBlock title,
TextBlock stereo, ColorSequence colorSequence, ISkinParam skinParam, IGroup g) {
final Cluster child = new Cluster(this, colorSequence, skinParam, g);
child.titleAndAttributeWidth = titleAndAttributeWidth;
child.titleAndAttributeHeight = titleAndAttributeHeight;
child.ztitle = title;
child.zstereo = stereo;
this.children.add(child);
return child;
}
public final Set<IGroup> getGroups() {
return Collections.singleton(group);
}
public final int getTitleAndAttributeWidth() {
return titleAndAttributeWidth;
}
public final int getTitleAndAttributeHeight() {
return titleAndAttributeHeight;
}
public double getWidth() {
return maxX - minX;
}
public double getMinX() {
return minX;
}
public ClusterPosition getClusterPosition() {
return new ClusterPosition(minX, minY, maxX, maxY);
}
public void setTitlePosition(double x, double y) {
this.xTitle = x;
this.yTitle = y;
}
private static HColor getColor(ColorParam colorParam, ISkinParam skinParam, Stereotype stereotype) {
return SkinParamUtils.getColor(skinParam, stereotype, colorParam);
}
static public StyleSignature getDefaultStyleDefinition() {
return StyleSignature.of(SName.root, SName.element, SName.activityDiagram, SName.group);
}
public void drawU(UGraphic ug, UStroke stroke, final UmlDiagramType umlDiagramType, final ISkinParam skinParam2) {
if (group.isHidden()) {
return;
}
final String fullName = group.getCodeGetName();
if (fullName.startsWith("##") == false) {
ug.draw(new UComment("cluster " + fullName));
}
final Stereotype stereotype = group.getStereotype();
HColor borderColor;
if (SkinParam.USE_STYLES()) {
final Style style = getDefaultStyleDefinition().getMergedStyle(skinParam.getCurrentStyleBuilder());
borderColor = style.value(PName.LineColor).asColor(skinParam2.getIHtmlColorSet());
} else {
if (umlDiagramType == UmlDiagramType.STATE) {
borderColor = getColor(ColorParam.stateBorder, skinParam, stereotype);
} else if (umlDiagramType == UmlDiagramType.ACTIVITY) {
borderColor = getColor(ColorParam.packageBorder, skinParam, stereotype);
} else {
borderColor = getColor(ColorParam.packageBorder, skinParam, stereotype);
}
}
final Url url = group.getUrl99();
if (url != null) {
ug.startUrl(url);
}
try {
if (hasEntryOrExitPoint()) {
manageEntryExitPoint(ug.getStringBounder());
}
if (skinParam.useSwimlanes(umlDiagramType)) {
drawSwinLinesState(ug, borderColor);
return;
}
final boolean isState = umlDiagramType == UmlDiagramType.STATE;
if (isState) {
if (group.getColors(skinParam).getSpecificLineStroke() != null) {
stroke = group.getColors(skinParam).getSpecificLineStroke();
}
if (group.getColors(skinParam).getColor(ColorType.LINE) != null) {
borderColor = group.getColors(skinParam).getColor(ColorType.LINE);
}
drawUState(ug, borderColor, skinParam2, stroke, umlDiagramType);
return;
}
PackageStyle packageStyle = group.getPackageStyle();
if (packageStyle == null) {
packageStyle = skinParam2.getPackageStyle();
}
if (border != null) {
final HColor tmp = skinParam2.getHtmlColor(border, group.getStereotype(), false);
if (tmp != null) {
borderColor = tmp;
}
}
final double shadowing;
if (SkinParam.USE_STYLES()) {
final Style style = getDefaultStyleDefinition().getMergedStyle(skinParam.getCurrentStyleBuilder());
shadowing = style.value(PName.Shadowing).asDouble();
} else {
if (group.getUSymbol() == null) {
shadowing = skinParam2.shadowing2(group.getStereotype(), USymbol.PACKAGE.getSkinParameter()) ? 3
: 0;
} else {
shadowing = skinParam2.shadowing2(group.getStereotype(), group.getUSymbol().getSkinParameter()) ? 3
: 0;
}
}
HColor backColor = getBackColor(umlDiagramType);
backColor = getBackColor(backColor, skinParam2, group.getStereotype());
if (ztitle != null || zstereo != null) {
final double roundCorner = group.getUSymbol() == null ? 0
: group.getUSymbol().getSkinParameter().getRoundCorner(skinParam, stereotype);
final UStroke stroke2 = getStrokeInternal(skinParam2);
final ClusterDecoration decoration = new ClusterDecoration(packageStyle, group.getUSymbol(), ztitle,
zstereo, minX, minY, maxX, maxY, stroke2);
decoration.drawU(ug, backColor, borderColor, shadowing, roundCorner,
skinParam2.getHorizontalAlignment(AlignmentParam.packageTitleAlignment, null, false),
skinParam2.getStereotypeAlignment());
return;
}
final URectangle rect = new URectangle(maxX - minX, maxY - minY);
rect.setDeltaShadow(shadowing);
ug = ug.apply(backColor.bg()).apply(borderColor);
ug.apply(new UStroke(2)).apply(new UTranslate(minX, minY)).draw(rect);
} finally {
if (url != null) {
ug.closeAction();
}
}
}
private UStroke getStrokeInternal(ISkinParam skinParam) {
final Colors colors = group.getColors(skinParam);
if (colors.getSpecificLineStroke() != null) {
return colors.getSpecificLineStroke();
}
if (group.getUSymbol() != null) {
return group.getUSymbol().getSkinParameter().getStroke(skinParam, group.getStereotype());
}
UStroke stroke = skinParam.getThickness(LineParam.packageBorder, group.getStereotype());
if (stroke == null) {
stroke = new UStroke(1.5);
}
return stroke;
}
public void manageEntryExitPoint(StringBounder stringBounder) {
final Collection<ClusterPosition> insides = new ArrayList<ClusterPosition>();
final List<Point2D> points = new ArrayList<Point2D>();
for (Node sh : nodes) {
if (sh.getEntityPosition() == EntityPosition.NORMAL) {
insides.add(sh.getClusterPosition());
} else {
points.add(sh.getClusterPosition().getPointCenter());
}
}
for (Cluster in : children) {
insides.add(in.getClusterPosition());
}
final FrontierCalculator frontierCalculator = new FrontierCalculator(getClusterPosition(), insides, points);
if (titleAndAttributeHeight > 0 && titleAndAttributeWidth > 0) {
frontierCalculator.ensureMinWidth(titleAndAttributeWidth + 10);
}
final ClusterPosition forced = frontierCalculator.getSuggestedPosition();
xTitle += ((forced.getMinX() - minX) + (forced.getMaxX() - maxX)) / 2;
minX = forced.getMinX();
minY = forced.getMinY();
maxX = forced.getMaxX();
maxY = forced.getMaxY();
yTitle = minY + IEntityImage.MARGIN;
final double widthTitle = ztitle.calculateDimension(stringBounder).getWidth();
xTitle = minX + ((maxX - minX - widthTitle) / 2);
}
private void drawSwinLinesState(UGraphic ug, HColor borderColor) {
if (ztitle != null) {
ztitle.drawU(ug.apply(UTranslate.dx(xTitle)));
}
final ULine line = ULine.vline(maxY - minY);
ug = ug.apply(borderColor);
ug.apply(UTranslate.dx(minX)).draw(line);
ug.apply(UTranslate.dx(maxX)).draw(line);
}
private HColor getColor(ISkinParam skinParam, ColorParam colorParam, Stereotype stereo) {
return new Rose().getHtmlColor(skinParam, stereo, colorParam);
}
private void drawUState(UGraphic ug, HColor borderColor, ISkinParam skinParam2, UStroke stroke,
UmlDiagramType umlDiagramType) {
final Dimension2D total = new Dimension2DDouble(maxX - minX, maxY - minY);
final double suppY;
if (ztitle == null) {
suppY = 0;
} else {
suppY = ztitle.calculateDimension(ug.getStringBounder()).getHeight() + IEntityImage.MARGIN
+ IEntityImage.MARGIN_LINE;
}
HColor stateBack = getBackColor(umlDiagramType);
if (stateBack == null) {
stateBack = getColor(skinParam2, ColorParam.stateBackground, group.getStereotype());
}
final HColor background = getColor(skinParam2, ColorParam.background, null);
final TextBlockWidth attribute = getTextBlockAttribute(skinParam2);
final double attributeHeight = attribute.calculateDimension(ug.getStringBounder()).getHeight();
final RoundedContainer r = new RoundedContainer(total, suppY,
attributeHeight + (attributeHeight > 0 ? IEntityImage.MARGIN : 0), borderColor, stateBack, background,
stroke);
r.drawU(ug.apply(new UTranslate(minX, minY)), skinParam2.shadowing(group.getStereotype()));
if (ztitle != null) {
ztitle.drawU(ug.apply(new UTranslate(xTitle, yTitle)));
}
if (attributeHeight > 0) {
attribute.asTextBlock(total.getWidth()).drawU(
ug.apply(new UTranslate(minX + IEntityImage.MARGIN, minY + suppY + IEntityImage.MARGIN / 2.0)));
}
final Stereotype stereotype = group.getStereotype();
final boolean withSymbol = stereotype != null && stereotype.isWithOOSymbol();
if (withSymbol) {
EntityImageState.drawSymbol(ug.apply(borderColor), maxX, maxY);
}
}
private TextBlockWidth getTextBlockAttribute(ISkinParam skinParam) {
final TextBlockWidth attribute;
final List<Member> members = group.getBodier().getFieldsToDisplay();
if (members.size() == 0) {
attribute = new TextBlockEmpty();
} else {
attribute = new MethodsOrFieldsArea(members, FontParam.STATE_ATTRIBUTE, skinParam, group.getStereotype(),
null);
}
return attribute;
}
public void setPosition(double minX, double minY, double maxX, double maxY) {
this.minX = minX;
this.maxX = maxX;
this.minY = minY;
this.maxY = maxY;
}
private boolean isThereALinkFromOrToGroup(Collection<Line> lines) {
for (Line line : lines) {
if (line.isLinkFromOrTo(group)) {
return true;
}
}
return false;
}
public void printCluster1(StringBuilder sb, Collection<Line> lines, StringBounder stringBounder) {
for (Node node : getNodesOrderedTop(lines)) {
node.appendShape(sb, stringBounder);
}
}
private List<IShapePseudo> addProtection(List<Node> entries, double width) {
final List<IShapePseudo> result = new ArrayList<IShapePseudo>();
result.add(entries.get(0));
for (int i = 1; i < entries.size(); i++) {
result.add(new ShapePseudoImpl("psd" + UniqueSequence.getValue(), width, 5));
result.add(entries.get(i));
}
return result;
}
private double getMaxWidthFromLabelForEntryExit(List<Node> entries, StringBounder stringBounder) {
double result = -Double.MAX_VALUE;
for (Node node : entries) {
final double w = getMaxWidthFromLabelForEntryExit(node, stringBounder);
if (w > result) {
result = w;
}
}
return result;
}
private double getMaxWidthFromLabelForEntryExit(Node node, StringBounder stringBounder) {
return node.getMaxWidthFromLabelForEntryExit(stringBounder);
}
public void printClusterEntryExit(StringBuilder sb, StringBounder stringBounder) {
final List<Node> nodesEntryExitList = getNodesEntryExit(EntityPosition.getInputs());
final double maxWith = getMaxWidthFromLabelForEntryExit(nodesEntryExitList, stringBounder);
final double naturalSpace = 70;
final List<? extends IShapePseudo> entries;
if (maxWith > naturalSpace) {
entries = addProtection(nodesEntryExitList, maxWith - naturalSpace);
} else {
entries = nodesEntryExitList;
}
if (entries.size() > 0) {
sb.append("{rank=source;");
for (IShapePseudo sh : entries) {
sb.append(sh.getUid() + ";");
}
sb.append("}");
for (IShapePseudo sh : entries) {
sh.appendShape(sb, stringBounder);
}
}
final List<Node> exits = getNodesEntryExit(EntityPosition.getOutputs());
if (exits.size() > 0) {
sb.append("{rank=sink;");
for (Node sh : exits) {
sb.append(sh.getUid() + ";");
}
sb.append("}");
for (Node sh : exits) {
sh.appendShape(sb, stringBounder);
}
}
}
public boolean printCluster2(StringBuilder sb, Collection<Line> lines, StringBounder stringBounder, DotMode dotMode,
GraphvizVersion graphvizVersion, UmlDiagramType type) {
// Log.println("Cluster::printCluster " + this);
boolean added = false;
for (Node node : getNodesOrderedWithoutTop(lines)) {
node.appendShape(sb, stringBounder);
added = true;
}
if (skinParam.useRankSame() && dotMode != DotMode.NO_LEFT_RIGHT_AND_XLABEL
&& graphvizVersion.ignoreHorizontalLinks() == false) {
appendRankSame(sb, lines);
}
for (Cluster child : getChildren()) {
child.printInternal(sb, lines, stringBounder, dotMode, graphvizVersion, type);
}
return added;
}
private void appendRankSame(StringBuilder sb, Collection<Line> lines) {
for (String same : getRankSame(lines)) {
sb.append(same);
SvekUtils.println(sb);
}
}
private Set<String> getRankSame(Collection<Line> lines) {
final Set<String> rankSame = new HashSet<String>();
for (Line l : lines) {
if (l.hasEntryPoint()) {
continue;
}
final String startUid = l.getStartUidPrefix();
final String endUid = l.getEndUidPrefix();
if (isInCluster(startUid) && isInCluster(endUid)) {
final String same = l.rankSame();
if (same != null) {
rankSame.add(same);
}
}
}
return rankSame;
}
public void fillRankMin(Set<String> rankMin) {
for (Node sh : getNodes()) {
if (sh.isTop()) {
rankMin.add(sh.getUid());
}
}
for (Cluster child : getChildren()) {
child.fillRankMin(rankMin);
}
}
private boolean isInCluster(String uid) {
for (Node node : nodes) {
if (node.getUid().equals(uid)) {
return true;
}
}
return false;
}
public String getClusterId() {
return "cluster" + color;
}
public static String getSpecialPointId(IEntity group) {
return CENTER_ID + group.getUid();
}
public final static String CENTER_ID = "za";
private boolean protection0(UmlDiagramType type) {
if (skinParam.useSwimlanes(type)) {
return false;
}
return true;
}
private boolean protection1(UmlDiagramType type) {
if (group.getUSymbol() == USymbol.NODE) {
return true;
}
if (skinParam.useSwimlanes(type)) {
return false;
}
return true;
}
public String getMinPoint(UmlDiagramType type) {
if (skinParam.useSwimlanes(type)) {
return "minPoint" + color;
}
return null;
}
public String getMaxPoint(UmlDiagramType type) {
if (skinParam.useSwimlanes(type)) {
return "maxPoint" + color;
}
return null;
}
private String getSourceInPoint(UmlDiagramType type) {
if (skinParam.useSwimlanes(type)) {
return "sourceIn" + color;
}
return null;
}
private String getSinkInPoint(UmlDiagramType type) {
if (skinParam.useSwimlanes(type)) {
return "sinkIn" + color;
}
return null;
}
private void printInternal(StringBuilder sb, Collection<Line> lines, StringBounder stringBounder, DotMode dotMode,
GraphvizVersion graphvizVersion, UmlDiagramType type) {
final boolean thereALinkFromOrToGroup2 = isThereALinkFromOrToGroup(lines);
boolean thereALinkFromOrToGroup1 = thereALinkFromOrToGroup2;
final boolean useProtectionWhenThereALinkFromOrToGroup = graphvizVersion
.useProtectionWhenThereALinkFromOrToGroup();
if (useProtectionWhenThereALinkFromOrToGroup == false) {
thereALinkFromOrToGroup1 = false;
}
// final boolean thereALinkFromOrToGroup1 = false;
if (thereALinkFromOrToGroup1) {
subgraphCluster(sb, "a");
}
final boolean hasEntryOrExitPoint = hasEntryOrExitPoint();
if (hasEntryOrExitPoint) {
for (Line line : lines) {
if (line.isLinkFromOrTo(group)) {
line.setProjectionCluster(this);
}
}
}
boolean protection0 = protection0(type);
boolean protection1 = protection1(type);
if (hasEntryOrExitPoint || useProtectionWhenThereALinkFromOrToGroup == false) {
protection0 = false;
protection1 = false;
}
// if (graphvizVersion.modeSafe()) {
// protection0 = false;
// protection1 = false;
// }
if (protection0) {
subgraphCluster(sb, "p0");
}
sb.append("subgraph " + getClusterId() + " {");
sb.append("style=solid;");
sb.append("color=\"" + DotStringFactory.sharp000000(color) + "\";");
final String label;
if (isLabel()) {
final StringBuilder sblabel = new StringBuilder("<");
Line.appendTable(sblabel, getTitleAndAttributeWidth(), getTitleAndAttributeHeight() - 5, colorTitle);
sblabel.append(">");
label = sblabel.toString();
final HorizontalAlignment align = skinParam.getHorizontalAlignment(AlignmentParam.packageTitleAlignment,
null, false);
sb.append("labeljust=\"" + align.getGraphVizValue() + "\";");
} else {
label = "\"\"";
}
if (hasEntryOrExitPoint) {
printClusterEntryExit(sb, stringBounder);
subgraphCluster(sb, "ee", label);
} else {
sb.append("label=" + label + ";");
SvekUtils.println(sb);
}
// if (hasEntryOrExitPoint) {
// printClusterEntryExit(sb);
// subgraphCluster(sb, "ee");
// }
if (thereALinkFromOrToGroup2) {
sb.append(getSpecialPointId(group) + " [shape=point,width=.01,label=\"\"];");
}
if (thereALinkFromOrToGroup1) {
subgraphCluster(sb, "i");
}
if (protection1) {
subgraphCluster(sb, "p1");
}
if (skinParam.useSwimlanes(type)) {
sb.append("{rank = source; ");
sb.append(getSourceInPoint(type));
sb.append(" [shape=point,width=.01,label=\"\"];");
sb.append(getMinPoint(type) + "->" + getSourceInPoint(type) + " [weight=999];");
sb.append("}");
SvekUtils.println(sb);
sb.append("{rank = sink; ");
sb.append(getSinkInPoint(type));
sb.append(" [shape=point,width=.01,label=\"\"];");
sb.append("}");
sb.append(getSinkInPoint(type) + "->" + getMaxPoint(type) + " [weight=999];");
SvekUtils.println(sb);
}
SvekUtils.println(sb);
printCluster1(sb, lines, stringBounder);
final boolean added = printCluster2(sb, lines, stringBounder, dotMode, graphvizVersion, type);
if (hasEntryOrExitPoint && added == false) {
final String empty = "empty" + color;
sb.append(empty + " [shape=point,width=.01,label=\"\"];");
}
sb.append("}");
if (protection1) {
sb.append("}");
}
if (thereALinkFromOrToGroup1) {
sb.append("}");
sb.append("}");
}
if (hasEntryOrExitPoint) {
sb.append("}");
}
if (protection0) {
sb.append("}");
}
SvekUtils.println(sb);
}
public boolean isLabel() {
return getTitleAndAttributeHeight() > 0 && getTitleAndAttributeWidth() > 0;
}
private void subgraphCluster(StringBuilder sb, String id) {
subgraphCluster(sb, id, "\"\"");
}
private void subgraphCluster(StringBuilder sb, String id, String label) {
final String uid = getClusterId() + id;
sb.append("subgraph " + uid + " {");
sb.append("label=" + label + ";");
}
public int getColor() {
return color;
}
public int getTitleColor() {
return colorTitle;
}
private final HColor getBackColor(final UmlDiagramType umlDiagramType) {
if (EntityUtils.groupRoot(group)) {
return null;
}
final HColor result = group.getColors(skinParam).getColor(ColorType.BACK);
if (result != null) {
return result;
}
final Stereotype stereo = group.getStereotype();
final USymbol sym = group.getUSymbol() == null ? USymbol.PACKAGE : group.getUSymbol();
final ColorParam backparam = umlDiagramType == UmlDiagramType.ACTIVITY ? ColorParam.partitionBackground
: sym.getColorParamBack();
final HColor c1 = skinParam.getHtmlColor(backparam, stereo, false);
if (c1 != null) {
return c1;
}
if (parentCluster == null) {
return null;
}
return parentCluster.getBackColor(umlDiagramType);
}
public boolean isClusterOf(IEntity ent) {
if (ent.isGroup() == false) {
return false;
}
return group == ent;
}
public static HColor getBackColor(HColor backColor, ISkinParam skinParam, Stereotype stereotype) {
if (SkinParam.USE_STYLES()) {
final Style style = getDefaultStyleDefinition().getMergedStyle(skinParam.getCurrentStyleBuilder());
if (backColor == null) {
backColor = style.value(PName.BackGroundColor).asColor(skinParam.getIHtmlColorSet());
}
if (backColor == null || backColor.equals(HColorUtils.transparent()) /* || stateBack instanceof HtmlColorTransparent */) {
backColor = new HColorBackground(skinParam.getBackgroundColor(true));
}
return backColor;
}
if (backColor == null) {
backColor = skinParam.getHtmlColor(ColorParam.packageBackground, stereotype, false);
}
if (backColor == null) {
backColor = skinParam.getHtmlColor(ColorParam.background, stereotype, false);
}
if (backColor == null || backColor.equals(HColorUtils.transparent()) /* || stateBack instanceof HtmlColorTransparent */) {
backColor = new HColorBackground(skinParam.getBackgroundColor(true));
}
return backColor;
}
public double checkFolderPosition(Point2D pt, StringBounder stringBounder) {
if (getClusterPosition().isPointJustUpper(pt)) {
if (ztitle == null) {
return 0;
}
final Dimension2D dimTitle = ztitle.calculateDimension(stringBounder);
if (pt.getX() < getClusterPosition().getMinX() + dimTitle.getWidth()) {
return 0;
}
return getClusterPosition().getMinY() - pt.getY() + dimTitle.getHeight();
}
return 0;
}
// public Point2D projection(double x, double y) {
// final double v1 = Math.abs(minX - x);
// final double v2 = Math.abs(maxX - x);
// final double v3 = Math.abs(minY - y);
// final double v4 = Math.abs(maxY - y);
// if (v1 <= v2 && v1 <= v3 && v1 <= v4) {
// return new Point2D.Double(minX, y);
// }
// if (v2 <= v1 && v2 <= v3 && v2 <= v4) {
// return new Point2D.Double(maxX, y);
// }
// if (v3 <= v1 && v3 <= v2 && v3 <= v4) {
// return new Point2D.Double(x, minY);
// }
// if (v4 <= v1 && v4 <= v1 && v4 <= v3) {
// return new Point2D.Double(x, maxY);
// }
// throw new IllegalStateException();
// }
}