1
0
mirror of https://github.com/octoleo/plantuml.git synced 2025-01-03 07:12:29 +00:00
Arnaud Roques 2023-03-13 21:46:30 +01:00
parent 66918ad185
commit da21293630
13 changed files with 134 additions and 50 deletions

View File

@ -0,0 +1,57 @@
/* ========================================================================
* 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.decoration.symbol;
import net.sourceforge.plantuml.klimt.Fashion;
import net.sourceforge.plantuml.klimt.shape.TextBlock;
import net.sourceforge.plantuml.skin.ActorStyle;
import net.sourceforge.plantuml.style.SName;
public class USymbolActorBusiness extends USymbolSimpleAbstract {
private final ActorStyle actorStyle = ActorStyle.STICKMAN_BUSINESS;
@Override
public SName getSName() {
return SName.business;
}
@Override
protected TextBlock getDrawing(Fashion symbolContext) {
return actorStyle.getTextBlock(symbolContext);
}
}

View File

@ -77,8 +77,7 @@ public abstract class USymbols {
public final static USymbol COLLECTIONS = record("COLLECTIONS", new USymbolCollections()); public final static USymbol COLLECTIONS = record("COLLECTIONS", new USymbolCollections());
public final static USymbol AGENT = record("AGENT", new USymbolRectangle(SName.agent)); public final static USymbol AGENT = record("AGENT", new USymbolRectangle(SName.agent));
public final static USymbol ACTOR_STICKMAN = record("ACTOR_STICKMAN", new USymbolActor(ActorStyle.STICKMAN)); public final static USymbol ACTOR_STICKMAN = record("ACTOR_STICKMAN", new USymbolActor(ActorStyle.STICKMAN));
public final static USymbol ACTOR_STICKMAN_BUSINESS = record("ACTOR_STICKMAN_BUSINESS", public final static USymbol ACTOR_STICKMAN_BUSINESS = record("ACTOR_STICKMAN_BUSINESS", new USymbolActorBusiness());
new USymbolActor(ActorStyle.STICKMAN_BUSINESS));
public final static USymbol ACTOR_AWESOME = record("ACTOR_AWESOME", new USymbolActor(ActorStyle.AWESOME)); public final static USymbol ACTOR_AWESOME = record("ACTOR_AWESOME", new USymbolActor(ActorStyle.AWESOME));
public final static USymbol ACTOR_HOLLOW = record("ACTOR_HOLLOW", new USymbolActor(ActorStyle.HOLLOW)); public final static USymbol ACTOR_HOLLOW = record("ACTOR_HOLLOW", new USymbolActor(ActorStyle.HOLLOW));
public final static USymbol USECASE = null; public final static USymbol USECASE = null;

View File

@ -58,6 +58,10 @@ public class DescriptionDiagram extends AbstractEntityDiagram {
return null; return null;
if (id.startsWith("()")) if (id.startsWith("()"))
id = StringUtils.trin(id.substring(2)); id = StringUtils.trin(id.substring(2));
if (id.startsWith(":") && id.endsWith(":/"))
return id.substring(1, id.length() - 2);
if (id.startsWith("(") && id.endsWith(")/"))
return id.substring(1, id.length() - 2);
return super.cleanId(id); return super.cleanId(id);
} }

View File

@ -57,6 +57,7 @@ import net.sourceforge.plantuml.regex.RegexConcat;
import net.sourceforge.plantuml.regex.RegexLeaf; import net.sourceforge.plantuml.regex.RegexLeaf;
import net.sourceforge.plantuml.regex.RegexOptional; import net.sourceforge.plantuml.regex.RegexOptional;
import net.sourceforge.plantuml.regex.RegexResult; import net.sourceforge.plantuml.regex.RegexResult;
import net.sourceforge.plantuml.skin.ActorStyle;
import net.sourceforge.plantuml.stereo.Stereotype; import net.sourceforge.plantuml.stereo.Stereotype;
import net.sourceforge.plantuml.utils.Direction; import net.sourceforge.plantuml.utils.Direction;
import net.sourceforge.plantuml.utils.LineLocation; import net.sourceforge.plantuml.utils.LineLocation;
@ -77,9 +78,13 @@ public class CommandLinkElement extends SingleLineCommand2<DescriptionDiagram> {
static IRegex getRegexConcat() { static IRegex getRegexConcat() {
return RegexConcat.build(CommandLinkElement.class.getName(), RegexLeaf.start(), // return RegexConcat.build(CommandLinkElement.class.getName(), RegexLeaf.start(), //
getGroup("ENT1"), // getGroup("ENT1"), //
RegexLeaf.spaceZeroOrMore(), // RegexLeaf.spaceZeroOrMore(), //
new RegexOptional(new RegexLeaf("FIRST_LABEL", "[%g]([^%g]+)[%g]")), // new RegexOptional(new RegexLeaf("FIRST_LABEL", "[%g]([^%g]+)[%g]")), //
RegexLeaf.spaceZeroOrMore(), // RegexLeaf.spaceZeroOrMore(), //
// new RegexOptional(new RegexLeaf("STEREO1", "(\\<\\<.*\\>\\>)")), //
// RegexLeaf.spaceZeroOrMore(), //
new RegexLeaf("HEAD2", "(0\\)|<<|<_|[<^*+#0@)]|<\\|[\\|\\:]?|[%s]+o)?"), // new RegexLeaf("HEAD2", "(0\\)|<<|<_|[<^*+#0@)]|<\\|[\\|\\:]?|[%s]+o)?"), //
new RegexLeaf("BODY1", "([-=.~]+)"), // new RegexLeaf("BODY1", "([-=.~]+)"), //
new RegexLeaf("ARROW_STYLE1", "(?:\\[(" + LINE_STYLE_MUTILPLES + ")\\])?"), // new RegexLeaf("ARROW_STYLE1", "(?:\\[(" + LINE_STYLE_MUTILPLES + ")\\])?"), //
@ -88,11 +93,16 @@ public class CommandLinkElement extends SingleLineCommand2<DescriptionDiagram> {
new RegexLeaf("ARROW_STYLE2", "(?:\\[(" + LINE_STYLE + ")\\])?"), // new RegexLeaf("ARROW_STYLE2", "(?:\\[(" + LINE_STYLE + ")\\])?"), //
new RegexLeaf("BODY2", "([-=.~]*)"), // new RegexLeaf("BODY2", "([-=.~]*)"), //
new RegexLeaf("HEAD1", "(\\(0|>>|_>|[>^*+#0@(]|[\\:\\|]?\\|>|\\\\\\\\|o[%s]+)?"), // new RegexLeaf("HEAD1", "(\\(0|>>|_>|[>^*+#0@(]|[\\:\\|]?\\|>|\\\\\\\\|o[%s]+)?"), //
RegexLeaf.spaceZeroOrMore(), // RegexLeaf.spaceZeroOrMore(), //
new RegexOptional(new RegexLeaf("SECOND_LABEL", "[%g]([^%g]+)[%g]")), // new RegexOptional(new RegexLeaf("SECOND_LABEL", "[%g]([^%g]+)[%g]")), //
RegexLeaf.spaceZeroOrMore(), // RegexLeaf.spaceZeroOrMore(), //
getGroup("ENT2"), // getGroup("ENT2"), //
RegexLeaf.spaceZeroOrMore(), // RegexLeaf.spaceZeroOrMore(), //
// new RegexOptional(new RegexLeaf("STEREO2", "(\\<\\<.*\\>\\>)")), //
// RegexLeaf.spaceZeroOrMore(), //
color().getRegex(), // color().getRegex(), //
RegexLeaf.spaceZeroOrMore(), // RegexLeaf.spaceZeroOrMore(), //
new RegexLeaf("STEREOTYPE", "(\\<\\<.*\\>\\>)?"), // new RegexLeaf("STEREOTYPE", "(\\<\\<.*\\>\\>)?"), //
@ -229,11 +239,11 @@ public class CommandLinkElement extends SingleLineCommand2<DescriptionDiagram> {
"|" + // "|" + //
"\\(\\)[%s]*[%g][^%g]+[%g]" + // "\\(\\)[%s]*[%g][^%g]+[%g]" + //
"|" + // "|" + //
":[^:]+:" + // ":[^:]+:/?" + //
"|" + // "|" + //
"(?!\\[\\*\\])\\[[^\\[\\]]+\\]" + // "(?!\\[\\*\\])\\[[^\\[\\]]+\\]" + //
"|" + // "|" + //
"\\((?!\\*\\))[^)]+\\)" + // "\\((?!\\*\\))[^)]+\\)/?" + //
")"); ")");
} }
@ -244,14 +254,6 @@ public class CommandLinkElement extends SingleLineCommand2<DescriptionDiagram> {
final String ent2 = arg.get("ENT2", 0); final String ent2 = arg.get("ENT2", 0);
final String ent1clean = diagram.cleanId(ent1); final String ent1clean = diagram.cleanId(ent1);
final String ent2clean = diagram.cleanId(ent2); final String ent2clean = diagram.cleanId(ent2);
// final String ent1 = StringUtils.eventuallyRemoveStartingAndEndingDoubleQuote(ent1String);
// final String ent2 = StringUtils.eventuallyRemoveStartingAndEndingDoubleQuote(ent2String);
// final Quark ident1 = diagram.buildFullyQualified(ent1);
// final Quark ident2 = diagram.buildFullyQualified(ent2);
// Quark ident1pure = diagram.getPlasma().root().child(ent1);
// Quark ident2pure = diagram.getPlasma().root().child(ent2);
// final Quark code1 = diagram.buildFromFullPath(ent1String);
// final Quark code2 = diagram.buildFromFullPath(ent2String);
final LinkType linkType = getLinkType(arg); final LinkType linkType = getLinkType(arg);
final Direction dir = getDirection(arg); final Direction dir = getDirection(arg);
@ -308,6 +310,7 @@ public class CommandLinkElement extends SingleLineCommand2<DescriptionDiagram> {
} }
final char codeChar = ident.length() > 2 ? ident.charAt(0) : 0; final char codeChar = ident.length() > 2 ? ident.charAt(0) : 0;
final boolean endWithSlash = ident.endsWith("/");
ident = diagram.cleanId(ident); ident = diagram.cleanId(ident);
final Quark<Entity> quark = diagram.quarkInContext(ident, false); final Quark<Entity> quark = diagram.quarkInContext(ident, false);
@ -318,10 +321,18 @@ public class CommandLinkElement extends SingleLineCommand2<DescriptionDiagram> {
final Display display = Display.getWithNewlines(quark.getName()); final Display display = Display.getWithNewlines(quark.getName());
if (codeChar == '(') { if (codeChar == '(') {
return diagram.reallyCreateLeaf(quark, display, LeafType.USECASE, USymbols.USECASE); if (endWithSlash)
return diagram.reallyCreateLeaf(quark, display, LeafType.USECASE_BUSINESS, USymbols.USECASE);
else
return diagram.reallyCreateLeaf(quark, display, LeafType.USECASE, USymbols.USECASE);
} else if (codeChar == ':') { } else if (codeChar == ':') {
return diagram.reallyCreateLeaf(quark, display, LeafType.DESCRIPTION, if (endWithSlash)
diagram.getSkinParam().actorStyle().toUSymbol()); return diagram.reallyCreateLeaf(quark, display, LeafType.DESCRIPTION,
ActorStyle.STICKMAN_BUSINESS.toUSymbol());
else
return diagram.reallyCreateLeaf(quark, display, LeafType.DESCRIPTION,
diagram.getSkinParam().actorStyle().toUSymbol());
} else if (codeChar == '[') { } else if (codeChar == '[') {
final USymbol sym = diagram.getSkinParam().componentStyle().toUSymbol(); final USymbol sym = diagram.getSkinParam().componentStyle().toUSymbol();
return diagram.reallyCreateLeaf(quark, display, LeafType.DESCRIPTION, sym); return diagram.reallyCreateLeaf(quark, display, LeafType.DESCRIPTION, sym);

View File

@ -53,12 +53,6 @@ import net.sourceforge.plantuml.style.Style;
public class FontConfiguration { public class FontConfiguration {
// ::uncomment when __HAXE__
// public UFont getFont() {
// return null;
// }
// ::done__
public static FontConfiguration create(UFont font, HColor color, HColor hyperlinkColor, public static FontConfiguration create(UFont font, HColor color, HColor hyperlinkColor,
UStroke hyperlinkUnderlineStroke, int tabSize) { UStroke hyperlinkUnderlineStroke, int tabSize) {
return new FontConfiguration(getStyles(font), font, color, font, color, null, FontPosition.NORMAL, return new FontConfiguration(getStyles(font), font, color, font, color, null, FontPosition.NORMAL,
@ -87,7 +81,6 @@ public class FontConfiguration {
private FontConfiguration(EnumSet<FontStyle> styles, UFont motherFont, HColor motherColor, UFont currentFont, private FontConfiguration(EnumSet<FontStyle> styles, UFont motherFont, HColor motherColor, UFont currentFont,
HColor currentColor, HColor extendedColor, FontPosition fontPosition, SvgAttributes svgAttributes, HColor currentColor, HColor extendedColor, FontPosition fontPosition, SvgAttributes svgAttributes,
HColor hyperlinkColor, UStroke hyperlinkUnderlineStroke, int tabSize) { HColor hyperlinkColor, UStroke hyperlinkUnderlineStroke, int tabSize) {
// ::comment when __HAXE__
this.styles = styles; this.styles = styles;
this.currentFont = currentFont; this.currentFont = currentFont;
this.motherFont = motherFont; this.motherFont = motherFont;
@ -99,10 +92,16 @@ public class FontConfiguration {
this.hyperlinkColor = hyperlinkColor; this.hyperlinkColor = hyperlinkColor;
this.hyperlinkUnderlineStroke = hyperlinkUnderlineStroke; this.hyperlinkUnderlineStroke = hyperlinkUnderlineStroke;
this.tabSize = tabSize; this.tabSize = tabSize;
// ::done
} }
// ::comment when __HAXE__ public UFont getFont() {
UFont result = currentFont;
for (FontStyle style : styles)
result = style.mutateFont(result);
return fontPosition.mute(result);
}
private final EnumSet<FontStyle> styles; private final EnumSet<FontStyle> styles;
private final UFont currentFont; private final UFont currentFont;
private final UFont motherFont; private final UFont motherFont;
@ -117,6 +116,7 @@ public class FontConfiguration {
private final int tabSize; private final int tabSize;
// ::comment when __HAXE__
public String toStringDebug() { public String toStringDebug() {
return getFont().toStringDebug() + " " + styles.toString(); return getFont().toStringDebug() + " " + styles.toString();
// return "" + currentFont + " " + styles.toString() + currentColor + extendedColor + hyperlinkColor // return "" + currentFont + " " + styles.toString() + currentColor + extendedColor + hyperlinkColor
@ -303,14 +303,6 @@ public class FontConfiguration {
svgAttributes, hyperlinkColor, hyperlinkUnderlineStroke, tabSize); svgAttributes, hyperlinkColor, hyperlinkUnderlineStroke, tabSize);
} }
public UFont getFont() {
UFont result = currentFont;
for (FontStyle style : styles)
result = style.mutateFont(result);
return fontPosition.mute(result);
}
public HColor getColor() { public HColor getColor() {
return currentColor; return currentColor;
} }

View File

@ -48,8 +48,6 @@ public enum FontPosition {
return 0; return 0;
} }
// ::comment when __HAXE__
public UFont mute(UFont font) { public UFont mute(UFont font) {
if (this == NORMAL) if (this == NORMAL)
return font; return font;
@ -61,6 +59,7 @@ public enum FontPosition {
return font.withSize((float) size); return font.withSize((float) size);
} }
// ::comment when __HAXE__
public String getHtmlTag() { public String getHtmlTag() {
if (this == EXPOSANT) if (this == EXPOSANT)
return "sup"; return "sup";

View File

@ -51,6 +51,8 @@ public enum ActorStyle {
return USymbols.ACTOR_AWESOME; return USymbols.ACTOR_AWESOME;
else if (this == HOLLOW) else if (this == HOLLOW)
return USymbols.ACTOR_HOLLOW; return USymbols.ACTOR_HOLLOW;
else if (this == STICKMAN_BUSINESS)
return USymbols.ACTOR_STICKMAN_BUSINESS;
throw new IllegalStateException(); throw new IllegalStateException();
} }

View File

@ -49,6 +49,7 @@ public enum SName {
boundary, // boundary, //
box, // box, //
boxless, // boxless, //
business, //
caption, // caption, //
card, // card, //
circle, // circle, //

View File

@ -49,6 +49,7 @@ import net.sourceforge.plantuml.abel.Link;
import net.sourceforge.plantuml.cucadiagram.BodyFactory; import net.sourceforge.plantuml.cucadiagram.BodyFactory;
import net.sourceforge.plantuml.cucadiagram.PortionShower; import net.sourceforge.plantuml.cucadiagram.PortionShower;
import net.sourceforge.plantuml.decoration.symbol.USymbol; import net.sourceforge.plantuml.decoration.symbol.USymbol;
import net.sourceforge.plantuml.decoration.symbol.USymbolActorBusiness;
import net.sourceforge.plantuml.decoration.symbol.USymbols; import net.sourceforge.plantuml.decoration.symbol.USymbols;
import net.sourceforge.plantuml.klimt.Fashion; import net.sourceforge.plantuml.klimt.Fashion;
import net.sourceforge.plantuml.klimt.Shadowable; import net.sourceforge.plantuml.klimt.Shadowable;
@ -67,7 +68,6 @@ import net.sourceforge.plantuml.klimt.geom.HorizontalAlignment;
import net.sourceforge.plantuml.klimt.geom.MagneticBorder; import net.sourceforge.plantuml.klimt.geom.MagneticBorder;
import net.sourceforge.plantuml.klimt.geom.MagneticBorderNone; import net.sourceforge.plantuml.klimt.geom.MagneticBorderNone;
import net.sourceforge.plantuml.klimt.geom.XDimension2D; import net.sourceforge.plantuml.klimt.geom.XDimension2D;
import net.sourceforge.plantuml.klimt.geom.XPoint2D;
import net.sourceforge.plantuml.klimt.shape.TextBlock; import net.sourceforge.plantuml.klimt.shape.TextBlock;
import net.sourceforge.plantuml.klimt.shape.TextBlockUtils; import net.sourceforge.plantuml.klimt.shape.TextBlockUtils;
import net.sourceforge.plantuml.klimt.shape.UComment; import net.sourceforge.plantuml.klimt.shape.UComment;
@ -131,8 +131,13 @@ public class EntityImageDescription extends AbstractEntityImage {
final Colors colors = entity.getColors(); final Colors colors = entity.getColors();
final StyleSignatureBasic tmp = StyleSignatureBasic.of(SName.root, SName.element, styleName, symbol.getSName(), final StyleSignatureBasic tmp;
SName.title); if (symbol instanceof USymbolActorBusiness)
tmp = StyleSignatureBasic.of(SName.root, SName.element, styleName, SName.actor, SName.business,
SName.title);
else
tmp = StyleSignatureBasic.of(SName.root, SName.element, styleName, symbol.getSName(), SName.title);
final Stereotype stereotype = entity.getStereotype(); final Stereotype stereotype = entity.getStereotype();
final Style styleTitle = tmp.withTOBECHANGED(stereotype).getMergedStyle(getSkinParam().getCurrentStyleBuilder()) final Style styleTitle = tmp.withTOBECHANGED(stereotype).getMergedStyle(getSkinParam().getCurrentStyleBuilder())
.eventuallyOverride(colors); .eventuallyOverride(colors);

View File

@ -194,6 +194,11 @@ public class EntityImageUseCase extends AbstractEntityImage {
} }
private StyleSignature getDefaultStyleDefinition() { private StyleSignature getDefaultStyleDefinition() {
final LeafType type = getEntity().getLeafType();
if (type == LeafType.USECASE_BUSINESS)
return StyleSignatureBasic
.of(SName.root, SName.element, SName.componentDiagram, SName.usecase, SName.business)
.withTOBECHANGED(getStereo());
return StyleSignatureBasic.of(SName.root, SName.element, SName.componentDiagram, SName.usecase) return StyleSignatureBasic.of(SName.root, SName.element, SName.componentDiagram, SName.usecase)
.withTOBECHANGED(getStereo()); .withTOBECHANGED(getStereo());
} }

View File

@ -49,25 +49,26 @@ public class EaterAffectation extends Eater {
checkAndEatChar("!"); checkAndEatChar("!");
skipSpaces(); skipSpaces();
String varname = eatAndGetVarname(); String varname = eatAndGetVarname();
TVariableScope scope = null; TVariableScope scope = TVariableScope.lazzyParse(varname);
if (scope != null) {
skipSpaces();
if (peekChar() == '?' || peekChar() == '=') {
// The variable itself is "local" or "glocal", which is not a good idea by the way
scope = null;
} else
varname = eatAndGetVarname();
}
skipSpaces(); skipSpaces();
boolean conditional = false; boolean conditional = false;
if (peekChar() == '?') { if (peekChar() == '?') {
checkAndEatChar('?'); checkAndEatChar('?');
conditional = true; conditional = true;
} }
if (peekChar() != '=') {
scope = TVariableScope.valueOf(varname.toUpperCase());
varname = eatAndGetVarname();
skipSpaces();
}
checkAndEatChar('='); checkAndEatChar('=');
if (conditional) { if (conditional)
final TValue already = memory.getVariable(varname); if (memory.getVariable(varname) != null)
if (already != null) {
return; return;
}
}
skipSpaces(); skipSpaces();
final TValue value = eatExpression(context, memory); final TValue value = eatExpression(context, memory);
memory.putVariable(varname, value, scope); memory.putVariable(varname, value, scope);

View File

@ -35,5 +35,13 @@
package net.sourceforge.plantuml.tim; package net.sourceforge.plantuml.tim;
public enum TVariableScope { public enum TVariableScope {
LOCAL, GLOBAL LOCAL, GLOBAL;
public static TVariableScope lazzyParse(String value) {
if ("local".equalsIgnoreCase(value))
return LOCAL;
if ("global".equalsIgnoreCase(value))
return GLOBAL;
return null;
}
} }

View File

@ -82,7 +82,7 @@ public class Version {
} }
public static int beta() { public static int beta() {
final int beta = 0; final int beta = 1;
return beta; return beta;
} }