plantuml/src/net/sourceforge/plantuml/abel/Link.java

565 lines
14 KiB
Java

/* ========================================================================
* PlantUML : a free UML diagram generator
* ========================================================================
*
* (C) Copyright 2009-2024, Arnaud Roques
*
* Project Info: https://plantuml.com
*
* If you like this project or if you find it useful, you can support us at:
*
* https://plantuml.com/patreon (only 1$ per month!)
* https://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.abel;
import java.util.Objects;
import net.sourceforge.plantuml.OptionFlags;
import net.sourceforge.plantuml.cucadiagram.EntityPort;
import net.sourceforge.plantuml.cucadiagram.ICucaDiagram;
import net.sourceforge.plantuml.cucadiagram.LinkConstraint;
import net.sourceforge.plantuml.decoration.LinkDecor;
import net.sourceforge.plantuml.decoration.LinkType;
import net.sourceforge.plantuml.decoration.WithLinkType;
import net.sourceforge.plantuml.decoration.symbol.USymbolInterface;
import net.sourceforge.plantuml.klimt.creole.Display;
import net.sourceforge.plantuml.klimt.font.FontConfiguration;
import net.sourceforge.plantuml.klimt.font.StringBounder;
import net.sourceforge.plantuml.klimt.font.UFont;
import net.sourceforge.plantuml.klimt.geom.HorizontalAlignment;
import net.sourceforge.plantuml.klimt.geom.XDimension2D;
import net.sourceforge.plantuml.klimt.shape.TextBlock;
import net.sourceforge.plantuml.klimt.shape.UComment;
import net.sourceforge.plantuml.skin.VisibilityModifier;
import net.sourceforge.plantuml.stereo.Stereotype;
import net.sourceforge.plantuml.style.ISkinSimple;
import net.sourceforge.plantuml.style.StyleBuilder;
import net.sourceforge.plantuml.svek.Bibliotekon;
import net.sourceforge.plantuml.url.Url;
import net.sourceforge.plantuml.utils.LineLocation;
public class Link extends WithLinkType implements Hideable, Removeable {
public final StyleBuilder getStyleBuilder() {
return styleBuilder;
}
final private Entity cl1;
final private Entity cl2;
private String port1;
private String port2;
private final LinkArg linkArg;
final private String uid;
private CucaNote note;
private boolean invis = false;
private double weight = 1.0;
private boolean constraint = true;
private boolean inverted = false;
private LinkArrow linkArrow = LinkArrow.NONE_OR_SEVERAL;
private boolean opale;
private boolean horizontalSolitary;
private String sametail;
private final StyleBuilder styleBuilder;
private Stereotype stereotype;
private final IEntityFactory entityFactory;
private Url url;
public LinkStrategy getLinkStrategy() {
// return LinkStrategy.LEGACY;
return LinkStrategy.SIMPLIER;
}
public String idCommentForSvg() {
if (type.looksLikeRevertedForSvg())
return getEntity1().getName() + "-backto-" + getEntity2().getName();
if (type.looksLikeNoDecorAtAllSvg())
return getEntity1().getName() + "-" + getEntity2().getName();
return getEntity1().getName() + "-to-" + getEntity2().getName();
}
public UComment commentForSvg() {
if (type.looksLikeRevertedForSvg())
return new UComment("reverse link " + getEntity1().getName() + " to " + getEntity2().getName());
return new UComment("link " + getEntity1().getName() + " to " + getEntity2().getName());
}
public Link(IEntityFactory entityFactory, StyleBuilder styleBuilder, Entity cl1, Entity cl2, LinkType type,
LinkArg linkArg) {
if (linkArg.getLength() < 1)
throw new IllegalArgumentException();
this.entityFactory = entityFactory;
this.styleBuilder = styleBuilder;
this.cl1 = Objects.requireNonNull(cl1);
this.cl2 = Objects.requireNonNull(cl2);
this.type = type;
final ICucaDiagram diagram = ((Entity) cl1).getDiagram();
this.uid = "LNK" + diagram.getUniqueSequence();
this.linkArg = linkArg;
if (diagram.getPragma().useKermor()) {
if (cl1.getEntityPosition().isNormal() == false ^ cl2.getEntityPosition().isNormal() == false)
setConstraint(false);
}
}
public Link getInv() {
final Link result = new Link(entityFactory, styleBuilder, cl2, cl1, getType().getInversed(), linkArg.getInv());
result.inverted = !this.inverted;
result.port1 = this.port2;
result.port2 = this.port1;
result.url = this.url;
result.linkConstraint = this.linkConstraint;
result.stereotype = stereotype;
result.linkArg.setVisibilityModifier(this.linkArg.getVisibilityModifier());
return result;
}
@Override
public void goNorank() {
setConstraint(false);
}
public String getLabeldistance() {
// Default in dot 1.0
return getLinkArg().getLabeldistance();
}
public String getLabelangle() {
// Default in dot -25
return getLinkArg().getLabelangle();
}
public String getUid() {
return uid;
}
public final boolean isInvis() {
if (type.isInvisible())
return true;
return invis;
}
public final void setInvis(boolean invis) {
this.invis = invis;
}
public boolean isBetween(Entity cl1, Entity cl2) {
if (cl1 == this.cl1 && cl2 == this.cl2)
return true;
if (cl1 == this.cl2 && cl2 == this.cl1)
return true;
return false;
}
@Override
public String toString() {
return super.toString() + " {" + linkArg.getLength() + "} " + cl1 + "-->" + cl2;
}
public Entity getEntity1() {
return cl1;
}
public Entity getEntity2() {
return cl2;
}
public String getPortName1() {
return port1;
}
public String getPortName2() {
return port2;
}
public EntityPort getEntityPort1(Bibliotekon bibliotekon) {
return getEntityPort(cl1, port1, bibliotekon);
}
public EntityPort getEntityPort2(Bibliotekon bibliotekon) {
return getEntityPort(cl2, port2, bibliotekon);
}
private EntityPort getEntityPort(Entity leaf, String port, Bibliotekon bibliotekon) {
if (leaf.getEntityPosition().usePortP())
return EntityPort.forPort(bibliotekon.getNodeUid(leaf));
return EntityPort.create(bibliotekon.getNodeUid(leaf), port);
}
@Override
public LinkType getType() {
if (opale)
return new LinkType(LinkDecor.NONE, LinkDecor.NONE);
if (getSametail() != null)
return new LinkType(LinkDecor.NONE, LinkDecor.NONE);
LinkType result = type;
if (OptionFlags.USE_INTERFACE_EYE1) {
if (isLollipopInterfaceEye(cl1))
type = type.withLollipopInterfaceEye1();
if (isLollipopInterfaceEye(cl2))
type = type.withLollipopInterfaceEye2();
}
return result;
}
private boolean isReallyGroup(Entity ent) {
if (ent.isGroup() == false)
return false;
final Entity group = (Entity) ent;
return group.groups().size() + group.leafs().size() > 0;
}
public LinkType getTypePatchCluster() {
LinkType result = getType();
if (isReallyGroup(getEntity1()))
result = result.withoutDecors2();
if (isReallyGroup(getEntity2()))
result = result.withoutDecors1();
return result;
}
private LinkType getTypeSpecialForPrinting() {
if (opale)
return new LinkType(LinkDecor.NONE, LinkDecor.NONE);
LinkType result = type;
if (OptionFlags.USE_INTERFACE_EYE1) {
if (isLollipopInterfaceEye(cl1))
type = type.withLollipopInterfaceEye1();
if (isLollipopInterfaceEye(cl2))
type = type.withLollipopInterfaceEye2();
}
return result;
}
private boolean isLollipopInterfaceEye(Entity ent) {
return ent.getUSymbol() instanceof USymbolInterface;
}
public Display getLabel() {
return getLinkArg().getLabel();
}
public int getLength() {
return getLinkArg().getLength();
}
public final void setLength(int length) {
this.getLinkArg().setLength(length);
}
public String getQuantifier1() {
return getLinkArg().getQuantifier1();
}
public String getQuantifier2() {
return getLinkArg().getQuantifier2();
}
public final double getWeight() {
return weight;
}
public final void setWeight(double weight) {
this.weight = weight;
}
public final CucaNote getNote() {
return note;
}
public final void addNote(CucaNote note) {
this.note = note;
}
public final void addNoteFrom(Link other, NoteLinkStrategy strategy) {
if (other.note != null)
this.note = other.note.withStrategy(strategy);
}
public boolean isAutoLinkOfAGroup() {
if (getEntity1().isGroup() == false)
return false;
if (getEntity2().isGroup() == false)
return false;
if (getEntity1() == getEntity2())
return true;
return false;
}
public boolean containsType(LeafType type) {
if (getEntity1().getLeafType() == type || getEntity2().getLeafType() == type)
return true;
return false;
}
public boolean contains(Entity entity) {
if (getEntity1() == entity)
return true;
if (getEntity2() == entity)
return true;
return false;
}
public Entity getOther(Entity entity) {
if (getEntity1() == entity)
return getEntity2();
if (getEntity2() == entity)
return getEntity1();
throw new IllegalArgumentException();
}
// public double getMarginDecors1(StringBounder stringBounder, UFont fontQualif, ISkinSimple spriteContainer) {
// final double q = getQualifierMargin(stringBounder, fontQualif, linkArg.getQualifier1(), spriteContainer);
// final LinkDecor decor = getType().getDecor1();
// return decor.getMargin() + q;
// }
//
// public double getMarginDecors2(StringBounder stringBounder, UFont fontQualif, ISkinSimple spriteContainer) {
// final double q = getQualifierMargin(stringBounder, fontQualif, linkArg.getQualifier2(), spriteContainer);
// final LinkDecor decor = getType().getDecor2();
// return decor.getMargin() + q;
// }
private double getQuantifierMargin(StringBounder stringBounder, UFont fontQualif, String qualif,
ISkinSimple spriteContainer) {
if (qualif != null) {
final TextBlock b = Display.create(qualif).create(FontConfiguration.blackBlueTrue(fontQualif),
HorizontalAlignment.LEFT, spriteContainer);
final XDimension2D dim = b.calculateDimension(stringBounder);
return Math.max(dim.getWidth(), dim.getHeight());
}
return 0;
}
public final boolean isConstraint() {
return constraint;
}
public final void setConstraint(boolean constraint) {
this.constraint = constraint;
}
public void setOpale(boolean opale) {
this.opale = opale;
}
public final void setHorizontalSolitary(boolean horizontalSolitary) {
this.horizontalSolitary = horizontalSolitary;
}
public final boolean isHorizontalSolitary() {
return horizontalSolitary;
}
public final LinkArrow getLinkArrow() {
if (inverted)
return linkArrow.reverse();
return linkArrow;
}
public final void setLinkArrow(LinkArrow linkArrow) {
this.linkArrow = linkArrow;
}
public final boolean isInverted() {
return inverted;
}
public boolean hasEntryPoint() {
return (getEntity1().isGroup() == false && getEntity1().getEntityPosition() != EntityPosition.NORMAL)
|| (getEntity2().isGroup() == false && getEntity2().getEntityPosition() != EntityPosition.NORMAL);
}
public boolean hasTwoEntryPointsSameContainer() {
return getEntity1().isGroup() == false && getEntity2().isGroup() == false
&& getEntity1().getEntityPosition() != EntityPosition.NORMAL
&& getEntity2().getEntityPosition() != EntityPosition.NORMAL
&& getEntity1().getParentContainer() == getEntity2().getParentContainer();
}
public Url getUrl() {
return url;
}
public void setUrl(Url url) {
this.url = url;
}
public boolean isHidden() {
return hidden || cl1.isHidden() || cl2.isHidden();
}
public boolean sameConnections(Link other) {
if (this.cl1 == other.cl1 && this.cl2 == other.cl2)
return true;
if (this.cl1 == other.cl2 && this.cl2 == other.cl1)
return true;
return false;
}
public boolean doesTouch(Link other) {
if (this.cl1 == other.cl1)
return true;
if (this.cl1 == other.cl2)
return true;
if (this.cl2 == other.cl1)
return true;
if (this.cl2 == other.cl2)
return true;
return false;
}
public boolean isAutolink() {
return cl1 == cl2;
}
public boolean isRemoved() {
final Stereotype stereotype = getStereotype();
if (stereotype != null && entityFactory.isRemoved(stereotype))
return true;
return cl1.isRemoved() || cl2.isRemoved();
}
public boolean hasUrl() {
if (Display.isNull(linkArg.getLabel()) == false && linkArg.getLabel().hasUrl())
return true;
return getUrl() != null;
}
public String getSametail() {
return sametail;
}
public void setSametail(String sametail) {
this.sametail = sametail;
}
public void setPortMembers(String port1, String port2) {
this.port1 = port1;
this.port2 = port2;
if (port1 != null)
cl1.addPortShortName(port1);
if (port2 != null)
cl2.addPortShortName(port2);
}
private LinkConstraint linkConstraint;
public void setLinkConstraint(LinkConstraint linkConstraint) {
this.linkConstraint = linkConstraint;
}
public final LinkConstraint getLinkConstraint() {
return linkConstraint;
}
private LineLocation codeLine;
public String getCodeLine() {
if (codeLine == null)
return null;
return "" + codeLine.getPosition();
}
public void setCodeLine(LineLocation location) {
this.codeLine = location;
}
public void setStereotype(Stereotype stereotype) {
this.stereotype = stereotype;
}
public final Stereotype getStereotype() {
return stereotype;
}
public final LinkArg getLinkArg() {
return linkArg;
}
public final VisibilityModifier getVisibilityModifier() {
return getLinkArg().getVisibilityModifier();
}
public final boolean isOpale() {
return opale;
}
public final boolean hasKal1() {
return this.linkArg.getKal1() != null && !this.linkArg.getKal1().isEmpty();
}
public final boolean hasKal2() {
return this.linkArg.getKal2() != null && !this.linkArg.getKal2().isEmpty();
}
}