1
0
mirror of https://github.com/octoleo/plantuml.git synced 2025-01-05 08:02:11 +00:00

Merge pull request #933 from kolrami/chru-ditaa-font

Add Support for Font Selection and Transparent Background in Ditaa
This commit is contained in:
PlantUML 2022-02-28 14:19:34 +01:00 committed by GitHub
commit d8cbe985ef
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
7 changed files with 183 additions and 125 deletions

View File

@ -34,6 +34,8 @@
*/ */
package net.sourceforge.plantuml.ditaa; package net.sourceforge.plantuml.ditaa;
import java.awt.Color;
import java.awt.Font;
import java.awt.image.BufferedImage; import java.awt.image.BufferedImage;
import java.io.IOException; import java.io.IOException;
import java.io.OutputStream; import java.io.OutputStream;
@ -61,11 +63,13 @@ public class PSystemDitaa extends AbstractPSystem {
private final boolean dropShadows; private final boolean dropShadows;
private final String data; private final String data;
private final float scale; private final float scale;
private final boolean transparentBackground;
private final Font font;
private final boolean performSeparationOfCommonEdges; private final boolean performSeparationOfCommonEdges;
private final boolean allCornersAreRound; private final boolean allCornersAreRound;
public PSystemDitaa(UmlSource source, String data, boolean performSeparationOfCommonEdges, boolean dropShadows, public PSystemDitaa(UmlSource source, String data, boolean performSeparationOfCommonEdges, boolean dropShadows,
boolean allCornersAreRound, float scale) { boolean allCornersAreRound, boolean transparentBackground, float scale, Font font) {
super(source); super(source);
this.data = data; this.data = data;
this.dropShadows = dropShadows; this.dropShadows = dropShadows;
@ -83,12 +87,14 @@ public class PSystemDitaa extends AbstractPSystem {
e.printStackTrace(); e.printStackTrace();
this.processingOptions = null; this.processingOptions = null;
} }
this.transparentBackground = transparentBackground;
this.scale = scale; this.scale = scale;
this.font = font;
} }
PSystemDitaa add(String line) { PSystemDitaa add(String line) {
return new PSystemDitaa(getSource(), data + line + BackSlash.NEWLINE, performSeparationOfCommonEdges, return new PSystemDitaa(getSource(), data + line + BackSlash.NEWLINE, performSeparationOfCommonEdges,
dropShadows, allCornersAreRound, scale); dropShadows, allCornersAreRound, transparentBackground, scale, font);
} }
public DiagramDescription getDescription() { public DiagramDescription getDescription() {
@ -113,6 +119,14 @@ public class PSystemDitaa extends AbstractPSystem {
final Field f_renderingOptions = options.getClass().getField("renderingOptions"); final Field f_renderingOptions = options.getClass().getField("renderingOptions");
final Object renderingOptions = f_renderingOptions.get(options); final Object renderingOptions = f_renderingOptions.get(options);
// renderingOptions.setBackgroundColor(font);
final Method setBackgroundColor = renderingOptions.getClass().getMethod("setBackgroundColor", Color.class);
setBackgroundColor.invoke(renderingOptions, transparentBackground ? new Color(0, 0, 0, 0) : Color.WHITE);
// renderingOptions.setFont(font);
final Method setFont = renderingOptions.getClass().getMethod("setFont", Font.class);
setFont.invoke(renderingOptions, font);
// renderingOptions.setScale(scale); // renderingOptions.setScale(scale);
final Method setScale = renderingOptions.getClass().getMethod("setScale", float.class); final Method setScale = renderingOptions.getClass().getMethod("setScale", float.class);
setScale.invoke(renderingOptions, scale); setScale.invoke(renderingOptions, scale);

View File

@ -34,6 +34,7 @@
*/ */
package net.sourceforge.plantuml.ditaa; package net.sourceforge.plantuml.ditaa;
import java.awt.Font;
import java.util.regex.Matcher; import java.util.regex.Matcher;
import java.util.regex.Pattern; import java.util.regex.Pattern;
@ -69,11 +70,16 @@ public class PSystemDitaaFactory extends PSystemBasicFactory<PSystemDitaa> {
if (startLine != null && (startLine.contains("-r") || startLine.contains("--round-corners"))) if (startLine != null && (startLine.contains("-r") || startLine.contains("--round-corners")))
allCornersAreRound = true; allCornersAreRound = true;
boolean transparentBackground = false;
if (startLine != null && (startLine.contains("-T") || startLine.contains("--transparent")))
transparentBackground = true;
final float scale = extractScale(startLine); final float scale = extractScale(startLine);
final Font font = extractFont(startLine);
if (getDiagramType() == DiagramType.UML) if (getDiagramType() == DiagramType.UML)
return null; return null;
else if (getDiagramType() == DiagramType.DITAA) else if (getDiagramType() == DiagramType.DITAA)
return new PSystemDitaa(source, "", performSeparationOfCommonEdges, dropShadows, allCornersAreRound, scale); return new PSystemDitaa(source, "", performSeparationOfCommonEdges, dropShadows, allCornersAreRound, transparentBackground, scale, font);
else else
throw new IllegalStateException(getDiagramType().name()); throw new IllegalStateException(getDiagramType().name());
@ -94,8 +100,13 @@ public class PSystemDitaaFactory extends PSystemBasicFactory<PSystemDitaa> {
if (line.contains("-r") || line.contains("--round-corners")) if (line.contains("-r") || line.contains("--round-corners"))
allCornersAreRound = true; allCornersAreRound = true;
boolean transparentBackground = false;
if (line.contains("-T") || line.contains("--transparent"))
transparentBackground = true;
final float scale = extractScale(line); final float scale = extractScale(line);
return new PSystemDitaa(source, "", performSeparationOfCommonEdges, dropShadows, allCornersAreRound, scale); final Font font = extractFont(line);
return new PSystemDitaa(source, "", performSeparationOfCommonEdges, dropShadows, allCornersAreRound, transparentBackground, scale, font);
} }
if (system == null) if (system == null)
return null; return null;
@ -115,4 +126,46 @@ public class PSystemDitaaFactory extends PSystemBasicFactory<PSystemDitaa> {
} }
return 1; return 1;
} }
private Font extractFont(String line) {
if (line == null)
return new Font("Dialog", Font.BOLD, 12);
final Pattern pName = Pattern.compile("font-family=([a-zA-Z0-0 ]+)");
final Matcher mName = pName.matcher(line);
String fontName = "Dialog";
if (mName.find())
{
fontName = mName.group(1);
}
final Pattern pVariant = Pattern.compile("font-variant=(BOLD|ITALIC|PLAIN)");
final Matcher mVariant = pVariant.matcher(line);
int fontVariant = Font.BOLD;
if (mVariant.find())
{
switch (mVariant.group(1))
{
case "BOLD":
fontVariant = Font.BOLD;
break;
case "ITALIC":
fontVariant = Font.ITALIC;
break;
case "PLAIN":
fontVariant = Font.PLAIN;
break;
}
}
final Pattern pSize = Pattern.compile("font-size=([\\d]+)");
final Matcher mSize = pSize.matcher(line);
int fontSize = 12;
if (mSize.find())
{
fontSize = Integer.parseInt(mSize.group(1));
}
return new Font(fontName, fontVariant, fontSize);
}
} }

View File

@ -20,6 +20,7 @@
*/ */
package org.stathissideris.ascii2image.core; package org.stathissideris.ascii2image.core;
import java.awt.*;
import java.util.HashMap; import java.util.HashMap;
import org.stathissideris.ascii2image.graphics.CustomShapeDefinition; import org.stathissideris.ascii2image.graphics.CustomShapeDefinition;
@ -41,6 +42,10 @@ public class RenderingOptions {
private float scale = 1; private float scale = 1;
private Color backgroundColor = Color.white;
private Font font = new Font("Dialog", Font.BOLD, 12);
/** /**
* @return * @return
*/ */
@ -113,4 +118,26 @@ public class RenderingOptions {
antialias = b; antialias = b;
} }
public Color getBackgroundColor() {
return backgroundColor;
}
public void setBackgroundColor(Color backgroundColor) {
this.backgroundColor = backgroundColor;
}
public boolean needsTransparency() {
return backgroundColor.getAlpha() < 255;
}
public Font getFont()
{
return font;
}
public void setFont(Font font)
{
this.font = font;
}
} }

View File

@ -55,13 +55,19 @@ public class BitmapRenderer {
Stroke normalStroke; Stroke normalStroke;
Stroke dashStroke; Stroke dashStroke;
public RenderedImage renderToImage(Diagram diagram, RenderingOptions options){ public RenderedImage renderToImage(Diagram diagram, RenderingOptions options){
BufferedImage image = new BufferedImage( BufferedImage image;
if(options.needsTransparency()) {
image = new BufferedImage(
diagram.getWidth(),
diagram.getHeight(),
BufferedImage.TYPE_INT_ARGB);
} else {
image = new BufferedImage(
diagram.getWidth(), diagram.getWidth(),
diagram.getHeight(), diagram.getHeight(),
BufferedImage.TYPE_INT_RGB); BufferedImage.TYPE_INT_RGB);
}
return render(diagram, image, options); return render(diagram, image, options);
} }
@ -75,7 +81,7 @@ public class BitmapRenderer {
g2.setRenderingHint(RenderingHints.KEY_ANTIALIASING, antialiasSetting); g2.setRenderingHint(RenderingHints.KEY_ANTIALIASING, antialiasSetting);
g2.setColor(Color.white); g2.setColor(options.getBackgroundColor());
//TODO: find out why the next line does not work //TODO: find out why the next line does not work
g2.fillRect(0, 0, image.getWidth()+10, image.getHeight()+10); g2.fillRect(0, 0, image.getWidth()+10, image.getHeight()+10);
/*for(int y = 0; y < diagram.getHeight(); y ++) /*for(int y = 0; y < diagram.getHeight(); y ++)

View File

@ -111,6 +111,8 @@ public class Diagram {
this.cellWidth = options.renderingOptions.getCellWidth(); this.cellWidth = options.renderingOptions.getCellWidth();
this.cellHeight = options.renderingOptions.getCellHeight(); this.cellHeight = options.renderingOptions.getCellHeight();
FontMeasurer fontMeasurer = new FontMeasurer(options.renderingOptions.getFont());
width = grid.getWidth() * cellWidth; width = grid.getWidth() * cellWidth;
height = grid.getHeight() * cellHeight; height = grid.getHeight() * cellHeight;
@ -526,7 +528,7 @@ public class Diagram {
ArrayList textGroups = nonBlank.breakIntoDistinctBoundaries(); ArrayList textGroups = nonBlank.breakIntoDistinctBoundaries();
if(DEBUG) System.out.println(textGroups.size()+" text groups found"); if(DEBUG) System.out.println(textGroups.size()+" text groups found");
Font font = FontMeasurer.instance().getFontFor(cellHeight); Font font = fontMeasurer.getFontFor(cellHeight);
Iterator textGroupIt = textGroups.iterator(); Iterator textGroupIt = textGroups.iterator();
while(textGroupIt.hasNext()){ while(textGroupIt.hasNext()){
@ -550,10 +552,10 @@ public class Diagram {
int maxX = getCellMaxX(lastCell); int maxX = getCellMaxX(lastCell);
DiagramText textObject; DiagramText textObject;
if(FontMeasurer.instance().getWidthFor(string, font) > maxX - minX){ //does not fit horizontally if(fontMeasurer.getWidthFor(string, font) > maxX - minX){ //does not fit horizontally
Font lessWideFont = FontMeasurer.instance().getFontFor(maxX - minX, string); Font lessWideFont = fontMeasurer.getFontFor(maxX - minX, string);
textObject = new DiagramText(minX, y, string, lessWideFont); textObject = new DiagramText(minX, y, string, lessWideFont, fontMeasurer);
} else textObject = new DiagramText(minX, y, string, font); } else textObject = new DiagramText(minX, y, string, font, fontMeasurer);
textObject.centerVerticallyBetween(getCellMinY(cell), getCellMaxY(cell)); textObject.centerVerticallyBetween(getCellMinY(cell), getCellMaxY(cell));

View File

@ -39,8 +39,9 @@ public class DiagramText extends DiagramComponent {
private boolean isTextOnLine = false; private boolean isTextOnLine = false;
private boolean hasOutline = false; private boolean hasOutline = false;
private Color outlineColor = Color.white; private Color outlineColor = Color.white;
private final FontMeasurer fontMeasurer;
public DiagramText(int x, int y, String text, Font font){ public DiagramText(int x, int y, String text, Font font, FontMeasurer fontMeasurer){
if(text == null) throw new IllegalArgumentException("DiagramText cannot be initialised with a null string"); if(text == null) throw new IllegalArgumentException("DiagramText cannot be initialised with a null string");
if(font == null) throw new IllegalArgumentException("DiagramText cannot be initialised with a null font"); if(font == null) throw new IllegalArgumentException("DiagramText cannot be initialised with a null font");
@ -48,6 +49,7 @@ public class DiagramText extends DiagramComponent {
this.yPos = y; this.yPos = y;
this.text = text; this.text = text;
this.font = font; this.font = font;
this.fontMeasurer = fontMeasurer;
} }
public void centerInBounds(Rectangle2D bounds){ public void centerInBounds(Rectangle2D bounds){
@ -56,20 +58,20 @@ public class DiagramText extends DiagramComponent {
} }
public void centerHorizontallyBetween(int minX, int maxX){ public void centerHorizontallyBetween(int minX, int maxX){
int width = FontMeasurer.instance().getWidthFor(text, font); int width = fontMeasurer.getWidthFor(text, font);
int center = Math.abs(maxX - minX) / 2; int center = Math.abs(maxX - minX) / 2;
xPos += Math.abs(center - width / 2); xPos += Math.abs(center - width / 2);
} }
public void centerVerticallyBetween(int minY, int maxY){ public void centerVerticallyBetween(int minY, int maxY){
int zHeight = FontMeasurer.instance().getZHeight(font); int zHeight = fontMeasurer.getZHeight(font);
int center = Math.abs(maxY - minY) / 2; int center = Math.abs(maxY - minY) / 2;
yPos -= Math.abs(center - zHeight / 2); yPos -= Math.abs(center - zHeight / 2);
} }
public void alignRightEdgeTo(int x){ public void alignRightEdgeTo(int x){
int width = FontMeasurer.instance().getWidthFor(text, font); int width = fontMeasurer.getWidthFor(text, font);
xPos = x - width; xPos = x - width;
} }
@ -145,7 +147,7 @@ public class DiagramText extends DiagramComponent {
} }
public Rectangle2D getBounds(){ public Rectangle2D getBounds(){
Rectangle2D bounds = FontMeasurer.instance().getBoundsFor(text, font); Rectangle2D bounds = fontMeasurer.getBoundsFor(text, font);
bounds.setRect( bounds.setRect(
bounds.getMinX() + xPos, bounds.getMinX() + xPos,
bounds.getMinY() + yPos, bounds.getMinY() + yPos,

View File

@ -26,7 +26,6 @@ import java.awt.Graphics2D;
import java.awt.font.FontRenderContext; import java.awt.font.FontRenderContext;
import java.awt.geom.Rectangle2D; import java.awt.geom.Rectangle2D;
import java.awt.image.BufferedImage; import java.awt.image.BufferedImage;
import java.util.Locale;
/** /**
* *
@ -34,25 +33,19 @@ import java.util.Locale;
*/ */
public class FontMeasurer { public class FontMeasurer {
private static final String fontFamilyName = "Dialog"; private final Font baseFont;
//private static final String fontFamilyName = "Helvetica"; private FontRenderContext fakeRenderContext;
private Graphics2D fakeGraphics;
private static final boolean DEBUG = false; public FontMeasurer(Font font){
baseFont = font;
private static final FontMeasurer instance = new FontMeasurer();
FontRenderContext fakeRenderContext;
Graphics2D fakeGraphics;
{
BufferedImage image = new BufferedImage(1,1, BufferedImage.TYPE_INT_RGB); BufferedImage image = new BufferedImage(1,1, BufferedImage.TYPE_INT_RGB);
fakeGraphics = image.createGraphics(); fakeGraphics = image.createGraphics();
if (DEBUG) System.out.println("Locale: "+Locale.getDefault());
fakeRenderContext = fakeGraphics.getFontRenderContext(); fakeRenderContext = fakeGraphics.getFontRenderContext();
} }
public int getWidthFor(String str, int pixelHeight){ public int getWidthFor(String str, int pixelHeight){
Font font = getFontFor(pixelHeight); Font font = getFontFor(pixelHeight);
Rectangle2D rectangle = font.getStringBounds(str, fakeRenderContext); Rectangle2D rectangle = font.getStringBounds(str, fakeRenderContext);
@ -79,120 +72,81 @@ public class FontMeasurer {
return font.getStringBounds(str, fakeRenderContext); return font.getStringBounds(str, fakeRenderContext);
} }
public Font getFontFor(int pixelHeight){
BufferedImage image = new BufferedImage(1,1, BufferedImage.TYPE_INT_RGB);
Graphics2D g2 = image.createGraphics();
return getFontFor(pixelHeight, fakeRenderContext);
}
public int getAscent(Font font){ public int getAscent(Font font){
fakeGraphics.setFont(font); fakeGraphics.setFont(font);
FontMetrics metrics = fakeGraphics.getFontMetrics(); FontMetrics metrics = fakeGraphics.getFontMetrics();
if(DEBUG) System.out.println("Ascent: "+metrics.getAscent());
return metrics.getAscent(); return metrics.getAscent();
} }
public int getZHeight(Font font){ public int getZHeight(Font font){
int height = (int) font.createGlyphVector(fakeRenderContext, "Z").getOutline().getBounds().getHeight(); return (int) font.createGlyphVector(fakeRenderContext, "Z").getOutline().getBounds().getHeight();
if(DEBUG) System.out.println("Z height: "+height);
return height;
} }
public Font getFontFor(int maxWidth, String string){ public Font getFontFor(final int maxWidth, final String string){
float size = 12; FontPredicate predicate = new FontPredicate() {
Font currentFont = new Font(fontFamilyName, Font.BOLD, (int) size); @Override
public boolean test(Font font)
{
int width = getWidthFor(string, font);
return width > maxWidth;
}
};
return deriveFont(predicate, 1.0f);
}
public Font getFontFor(final int pixelHeight){
FontPredicate predicate = new FontPredicate() {
@Override
public boolean test(Font font)
{
//ascent is the distance between the baseline and the tallest character //ascent is the distance between the baseline and the tallest character
int width = getWidthFor(string, currentFont); int ascent = getAscent(font);
return ascent > pixelHeight;
}
};
return deriveFont(predicate, 0.5f);
}
private Font deriveFont(FontPredicate predicate, float sizeDelta)
{
Font currentFont = baseFont;
float size = baseFont.getSize2D();
int direction; //direction of size change (towards smaller or bigger) int direction; //direction of size change (towards smaller or bigger)
if(width > maxWidth){ if(predicate.test(currentFont)){
currentFont = currentFont.deriveFont(size - 1); currentFont = currentFont.deriveFont(size - 1f);
size--; size--;
direction = -1; direction = -1;
} else { } else {
currentFont = currentFont.deriveFont(size + 1); currentFont = currentFont.deriveFont(size + 1f);
size++; size++;
direction = 1; direction = 1;
} }
while(size > 0){ while(size > 0){
currentFont = currentFont.deriveFont(size); currentFont = currentFont.deriveFont(size);
//rectangle = currentFont.getStringBounds(testString, frc); //rectangle = currentFont.getStringBounds(testString, frc);
width = getWidthFor(string, currentFont);
if (direction == 1) { if (direction == 1) {
if(width > maxWidth){ if (predicate.test(currentFont)) {
size = size - 1; size = size - sizeDelta;
return currentFont.deriveFont(size); return currentFont.deriveFont(size);
}
else size = size + 1;
} else { } else {
if(width < maxWidth) size = size + sizeDelta;
}
} else {
if (!predicate.test(currentFont)) {
return currentFont; return currentFont;
else size = size - 1; } else {
size = size - sizeDelta;
}
} }
} }
return null; return null;
} }
private interface FontPredicate {
boolean test(Font font);
public Font getFontFor(int pixelHeight, FontRenderContext frc){
float size = 12;
Font currentFont = new Font(fontFamilyName, Font.BOLD, (int) size);
// Font currentFont = new Font("Times", Font.BOLD, (int) size);
if (DEBUG) System.out.println(currentFont.getFontName());
//ascent is the distance between the baseline and the tallest character
int ascent = getAscent(currentFont);
int direction; //direction of size change (towards smaller or bigger)
if(ascent > pixelHeight){
currentFont = currentFont.deriveFont(size - 1);
size--;
direction = -1;
} else {
currentFont = currentFont.deriveFont(size + 1);
size++;
direction = 1;
}
while(size > 0){
currentFont = currentFont.deriveFont(size);
//rectangle = currentFont.getStringBounds(testString, frc);
ascent = getAscent(currentFont);
if(direction == 1){
if(ascent > pixelHeight){
size = size - 0.5f;
return currentFont.deriveFont(size);
}
else size = size + 0.5f;
} else {
if(ascent < pixelHeight)
return currentFont;
else size = size - 0.5f;
}
}
return null;
}
public static FontMeasurer instance(){
return instance;
}
public FontMeasurer(){
}
public static void main(String[] args) {
//FontMeasurer.instance().getFontFor(7);
float size = 12;
Font currentFont = new Font("Sans", Font.BOLD, (int) size);
System.out.println(currentFont.getSize());
currentFont = currentFont.deriveFont(--size);
System.out.println(currentFont.getSize());
currentFont = currentFont.deriveFont(--size);
System.out.println(currentFont.getSize());
currentFont = currentFont.deriveFont(--size);
System.out.println(currentFont.getSize());
currentFont = currentFont.deriveFont(--size);
System.out.println(currentFont.getSize());
currentFont = currentFont.deriveFont(--size);
System.out.println(currentFont.getSize());
} }
} }