mirror of
https://github.com/octoleo/plantuml.git
synced 2024-09-26 22:19:02 +00:00
320 lines
11 KiB
Java
320 lines
11 KiB
Java
/* ========================================================================
|
|
* PlantUML : a free UML diagram generator
|
|
* ========================================================================
|
|
*
|
|
* (C) Copyright 2009-2023, 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.style;
|
|
|
|
import java.util.EnumMap;
|
|
import java.util.Map;
|
|
import java.util.Map.Entry;
|
|
import java.util.StringTokenizer;
|
|
|
|
import net.sourceforge.plantuml.ISkinSimple;
|
|
import net.sourceforge.plantuml.LineBreakStrategy;
|
|
import net.sourceforge.plantuml.cucadiagram.Display;
|
|
import net.sourceforge.plantuml.graphic.FontConfiguration;
|
|
import net.sourceforge.plantuml.graphic.HorizontalAlignment;
|
|
import net.sourceforge.plantuml.graphic.SymbolContext;
|
|
import net.sourceforge.plantuml.graphic.TextBlock;
|
|
import net.sourceforge.plantuml.graphic.TextBlockUtils;
|
|
import net.sourceforge.plantuml.graphic.color.ColorType;
|
|
import net.sourceforge.plantuml.graphic.color.Colors;
|
|
import net.sourceforge.plantuml.ugraphic.UFont;
|
|
import net.sourceforge.plantuml.ugraphic.UGraphic;
|
|
import net.sourceforge.plantuml.ugraphic.UStroke;
|
|
import net.sourceforge.plantuml.ugraphic.color.HColor;
|
|
import net.sourceforge.plantuml.ugraphic.color.HColorSet;
|
|
import net.sourceforge.plantuml.ugraphic.color.HColors;
|
|
|
|
public class Style {
|
|
|
|
private final Map<PName, Value> map;
|
|
private final StyleSignatureBasic signature;
|
|
|
|
public Style(StyleSignatureBasic signature, Map<PName, Value> map) {
|
|
this.map = map;
|
|
this.signature = signature;
|
|
}
|
|
|
|
public Style deltaPriority(int delta) {
|
|
if (signature.isStarred() == false)
|
|
throw new UnsupportedOperationException();
|
|
|
|
final EnumMap<PName, Value> copy = new EnumMap<PName, Value>(PName.class);
|
|
for (Entry<PName, Value> ent : this.map.entrySet())
|
|
copy.put(ent.getKey(), ((ValueImpl) ent.getValue()).addPriority(delta));
|
|
|
|
return new Style(this.signature, copy);
|
|
|
|
}
|
|
|
|
public void printMe() {
|
|
if (map.size() == 0)
|
|
return;
|
|
|
|
System.err.println(signature + " {");
|
|
for (Entry<PName, Value> ent : map.entrySet())
|
|
System.err.println(" " + ent.getKey() + ": " + ent.getValue().asString());
|
|
|
|
System.err.println("}");
|
|
|
|
}
|
|
|
|
@Override
|
|
public String toString() {
|
|
return signature + " " + map;
|
|
}
|
|
|
|
public Value value(PName name) {
|
|
final Value result = map.get(name);
|
|
if (result == null)
|
|
return ValueNull.NULL;
|
|
|
|
return result;
|
|
}
|
|
|
|
public boolean hasValue(PName name) {
|
|
return map.containsKey(name);
|
|
}
|
|
|
|
public Style mergeWith(Style other, MergeStrategy strategy) {
|
|
if (other == null)
|
|
return this;
|
|
|
|
final EnumMap<PName, Value> both = new EnumMap<PName, Value>(this.map);
|
|
for (Entry<PName, Value> ent : other.map.entrySet()) {
|
|
final Value previous = this.map.get(ent.getKey());
|
|
if (previous != null && previous.getPriority() > StyleLoader.DELTA_PRIORITY_FOR_STEREOTYPE
|
|
&& strategy == MergeStrategy.KEEP_EXISTING_VALUE_OF_STEREOTYPE)
|
|
continue;
|
|
final PName key = ent.getKey();
|
|
both.put(key, ((ValueImpl) ent.getValue()).mergeWith(previous));
|
|
}
|
|
return new Style(this.signature.mergeWith(other.getSignature()), both);
|
|
}
|
|
|
|
public Style eventuallyOverride(PName param, HColor color) {
|
|
if (color == null)
|
|
return this;
|
|
|
|
final EnumMap<PName, Value> result = new EnumMap<PName, Value>(this.map);
|
|
final Value old = result.get(param);
|
|
result.put(param, new ValueColor(color, old.getPriority()));
|
|
return new Style(this.signature, result);
|
|
}
|
|
|
|
public Style eventuallyOverride(PName param, double value) {
|
|
return eventuallyOverride(param, "" + value);
|
|
}
|
|
|
|
public Style eventuallyOverride(PName param, String value) {
|
|
final EnumMap<PName, Value> result = new EnumMap<PName, Value>(this.map);
|
|
result.put(param, ValueImpl.regular(value, Integer.MAX_VALUE));
|
|
return new Style(this.signature, result);
|
|
}
|
|
|
|
public Style eventuallyOverride(Colors colors) {
|
|
Style result = this;
|
|
if (colors != null) {
|
|
final HColor back = colors.getColor(ColorType.BACK);
|
|
if (back != null)
|
|
result = result.eventuallyOverride(PName.BackGroundColor, back);
|
|
|
|
final HColor line = colors.getColor(ColorType.LINE);
|
|
if (line != null)
|
|
result = result.eventuallyOverride(PName.LineColor, line);
|
|
|
|
final HColor text = colors.getColor(ColorType.TEXT);
|
|
if (text != null)
|
|
result = result.eventuallyOverride(PName.FontColor, text);
|
|
|
|
}
|
|
return result;
|
|
}
|
|
|
|
public Style eventuallyOverride(SymbolContext symbolContext) {
|
|
Style result = this;
|
|
if (symbolContext != null) {
|
|
final HColor back = symbolContext.getBackColor();
|
|
if (back != null)
|
|
result = result.eventuallyOverride(PName.BackGroundColor, back);
|
|
|
|
}
|
|
return result;
|
|
}
|
|
|
|
public StyleSignatureBasic getSignature() {
|
|
return signature;
|
|
}
|
|
|
|
public UFont getUFont() {
|
|
final String family = value(PName.FontName).asString();
|
|
final int fontStyle = value(PName.FontStyle).asFontStyle();
|
|
int size = value(PName.FontSize).asInt();
|
|
if (size == 0)
|
|
size = 14;
|
|
return new UFont(family, fontStyle, size);
|
|
}
|
|
|
|
public FontConfiguration getFontConfiguration(HColorSet set) {
|
|
return getFontConfiguration(set, null);
|
|
}
|
|
|
|
public FontConfiguration getFontConfiguration(HColorSet set, Colors colors) {
|
|
final UFont font = getUFont();
|
|
HColor color = colors == null ? null : colors.getColor(ColorType.TEXT);
|
|
if (color == null)
|
|
color = value(PName.FontColor).asColor(set);
|
|
|
|
final HColor hyperlinkColor = value(PName.HyperLinkColor).asColor(set);
|
|
final UStroke stroke = getStroke(PName.HyperlinkUnderlineThickness, PName.HyperlinkUnderlineStyle);
|
|
return FontConfiguration.create(font, color, hyperlinkColor, stroke);
|
|
}
|
|
|
|
public SymbolContext getSymbolContext(HColorSet set, Colors colors) {
|
|
HColor backColor = colors == null ? null : colors.getColor(ColorType.BACK);
|
|
if (backColor == null)
|
|
backColor = value(PName.BackGroundColor).asColor(set);
|
|
HColor foreColor = colors == null ? null : colors.getColor(ColorType.LINE);
|
|
if (foreColor == null)
|
|
foreColor = value(PName.LineColor).asColor(set);
|
|
final double deltaShadowing = value(PName.Shadowing).asDouble();
|
|
final double roundCorner = value(PName.RoundCorner).asDouble();
|
|
final double diagonalCorner = value(PName.DiagonalCorner).asDouble();
|
|
return new SymbolContext(backColor, foreColor).withStroke(getStroke()).withDeltaShadow(deltaShadowing)
|
|
.withCorner(roundCorner, diagonalCorner);
|
|
}
|
|
|
|
public SymbolContext getSymbolContext(HColorSet set) {
|
|
return getSymbolContext(set, null);
|
|
}
|
|
|
|
public Style eventuallyOverride(UStroke stroke) {
|
|
if (stroke == null)
|
|
return this;
|
|
|
|
Style result = this.eventuallyOverride(PName.LineThickness, stroke.getThickness());
|
|
final double space = stroke.getDashSpace();
|
|
final double visible = stroke.getDashVisible();
|
|
result = result.eventuallyOverride(PName.LineStyle, "" + visible + ";" + space);
|
|
return result;
|
|
}
|
|
|
|
public UStroke getStroke() {
|
|
return getStroke(PName.LineThickness, PName.LineStyle);
|
|
}
|
|
|
|
private UStroke getStroke(final PName thicknessParam, final PName styleParam) {
|
|
final double thickness = value(thicknessParam).asDouble();
|
|
final String dash = value(styleParam).asString();
|
|
if (dash.length() == 0)
|
|
return new UStroke(thickness);
|
|
|
|
try {
|
|
final StringTokenizer st = new StringTokenizer(dash, "-;,");
|
|
final double dashVisible = Double.parseDouble(st.nextToken().trim());
|
|
double dashSpace = dashVisible;
|
|
if (st.hasMoreTokens())
|
|
dashSpace = Double.parseDouble(st.nextToken().trim());
|
|
|
|
return new UStroke(dashVisible, dashSpace, thickness);
|
|
} catch (Exception e) {
|
|
return new UStroke(thickness);
|
|
}
|
|
}
|
|
|
|
public UStroke getStroke(Colors colors) {
|
|
final UStroke stroke = colors.getSpecificLineStroke();
|
|
if (stroke == null)
|
|
return getStroke();
|
|
|
|
return stroke;
|
|
}
|
|
|
|
public LineBreakStrategy wrapWidth() {
|
|
final String value = value(PName.MaximumWidth).asString();
|
|
return new LineBreakStrategy(value);
|
|
}
|
|
|
|
public ClockwiseTopRightBottomLeft getPadding() {
|
|
final String padding = value(PName.Padding).asString();
|
|
return ClockwiseTopRightBottomLeft.read(padding);
|
|
}
|
|
|
|
public ClockwiseTopRightBottomLeft getMargin() {
|
|
final String margin = value(PName.Margin).asString();
|
|
return ClockwiseTopRightBottomLeft.read(margin);
|
|
}
|
|
|
|
public HorizontalAlignment getHorizontalAlignment() {
|
|
return value(PName.HorizontalAlignment).asHorizontalAlignment();
|
|
}
|
|
|
|
private TextBlock createTextBlockInternal(Display display, HColorSet set, ISkinSimple spriteContainer,
|
|
HorizontalAlignment alignment) {
|
|
final FontConfiguration fc = getFontConfiguration(set);
|
|
return display.create(fc, alignment, spriteContainer);
|
|
}
|
|
|
|
public static final String ID_TITLE = "_title";
|
|
public static final String ID_CAPTION = "_caption";
|
|
public static final String ID_LEGEND = "_legend";
|
|
|
|
public TextBlock createTextBlockBordered(Display note, HColorSet set, ISkinSimple spriteContainer, String id) {
|
|
final HorizontalAlignment alignment = this.getHorizontalAlignment();
|
|
final TextBlock textBlock = this.createTextBlockInternal(note, set, spriteContainer, alignment);
|
|
|
|
final HColor backgroundColor = this.value(PName.BackGroundColor).asColor(set);
|
|
final HColor lineColor = this.value(PName.LineColor).asColor(set);
|
|
final UStroke stroke = this.getStroke();
|
|
final int cornersize = this.value(PName.RoundCorner).asInt();
|
|
final ClockwiseTopRightBottomLeft margin = this.getMargin();
|
|
final ClockwiseTopRightBottomLeft padding = this.getPadding();
|
|
final TextBlock result = TextBlockUtils.bordered(textBlock, stroke, lineColor, backgroundColor, cornersize,
|
|
padding, id);
|
|
return TextBlockUtils.withMargin(result, margin);
|
|
}
|
|
|
|
public UGraphic applyStrokeAndLineColor(UGraphic ug, HColorSet colorSet) {
|
|
final HColor color = value(PName.LineColor).asColor(colorSet);
|
|
if (color == null)
|
|
ug = ug.apply(HColors.none());
|
|
else
|
|
ug = ug.apply(color);
|
|
|
|
ug = ug.apply(getStroke());
|
|
return ug;
|
|
}
|
|
|
|
} |