工具类
import com.Jake_955.common.business.BusinessException;
import lombok.Builder;
import lombok.Getter;
import javax.imageio.ImageIO;
import java.awt.*;
import java.awt.font.FontRenderContext;
import java.awt.geom.AffineTransform;
import java.awt.geom.Rectangle2D;
import java.awt.image.BufferedImage;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.Arrays;
import java.util.HashMap;
import java.util.Map;
@Builder
@Getter
public class SealUtil {
private static final int INIT_BEGIN = 5;
private Integer size;
private Color color;
private SealFont mainFont;
private SealFont viceFont;
private SealFont titleFont;
private SealFont centerFont;
private SealCircle borderCircle;
private SealCircle borderInnerCircle;
private SealCircle innerCircle;
private Integer borderSquare;
private String stamp;
public InputStream draw() throws Exception {
if (borderSquare != null) {
return draw2();
}
BufferedImage bi = new BufferedImage(size, size, BufferedImage.TYPE_4BYTE_ABGR);
Graphics2D g2d = bi.createGraphics();
RenderingHints hints = new RenderingHints(RenderingHints.KEY_TEXT_ANTIALIASING, RenderingHints.VALUE_TEXT_ANTIALIAS_OFF);
hints.put(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
g2d.setRenderingHints(hints);
g2d.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC, 0));
g2d.fillRect(0, 0, size, size);
g2d.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC, 1));
g2d.setPaint(color == null ? Color.RED : color);
if (borderCircle != null) {
drawCircle(g2d, borderCircle, INIT_BEGIN, INIT_BEGIN);
} else {
throw new Exception("BorderCircle can not null!");
}
int borderCircleWidth = borderCircle.getWidth();
int borderCircleHeight = borderCircle.getHeight();
if (borderInnerCircle != null) {
int x = INIT_BEGIN + borderCircleWidth - borderInnerCircle.getWidth();
int y = INIT_BEGIN + borderCircleHeight - borderInnerCircle.getHeight();
drawCircle(g2d, borderInnerCircle, x, y);
}
if (innerCircle != null) {
int x = INIT_BEGIN + borderCircleWidth - innerCircle.getWidth();
int y = INIT_BEGIN + borderCircleHeight - innerCircle.getHeight();
drawCircle(g2d, innerCircle, x, y);
}
if (borderCircleHeight != borderCircleWidth) {
drawArcFont4Oval(g2d, borderCircle, mainFont, true);
} else {
drawArcFont4Circle(g2d, borderCircleHeight, mainFont, true);
}
if (borderCircleHeight != borderCircleWidth) {
drawArcFont4Oval(g2d, borderCircle, viceFont, false);
} else {
drawArcFont4Circle(g2d, borderCircleHeight, viceFont, false);
}
drawFont(g2d, (borderCircleWidth + INIT_BEGIN) * 2, (borderCircleHeight + INIT_BEGIN) * 2, centerFont);
drawFont(g2d, (borderCircleWidth + INIT_BEGIN) * 2, (borderCircleHeight + INIT_BEGIN) * 2, titleFont);
g2d.dispose();
ByteArrayOutputStream os = new ByteArrayOutputStream();
try {
ImageIO.write(bi, "png", os);
InputStream input = new ByteArrayInputStream(os.toByteArray());
return input;
} catch (IOException e) {
throw new BusinessException("自动生成印章错误");
}
}
private static void drawArcFont4Circle(Graphics2D g2d, int circleRadius, SealFont font, boolean isTop) {
if (font == null) {
return;
}
int textLen = font.getText().length();
int size = font.getSize() == null ? (55 - textLen * 2) : font.getSize();
int style = font.getBold() ? Font.BOLD : Font.PLAIN;
Font f = new Font(font.getFamily(), style, size);
FontRenderContext context = g2d.getFontRenderContext();
Rectangle2D rectangle = f.getStringBounds(font.getText(), context);
double space;
if (font.getSpace() != null) {
space = font.getSpace();
} else {
if (textLen == 1) {
space = 0;
} else {
space = rectangle.getWidth() / (textLen - 1) * 0.9;
}
}
int margin = font.getMargin() == null ? INIT_BEGIN : font.getMargin();
double newRadius = circleRadius + rectangle.getY() - margin;
double radianPerInterval = 2 * Math.asin(space / (2 * newRadius));
double fix = 0.04;
if (isTop) {
fix = 0.18;
}
double firstAngle;
if (!isTop) {
if (textLen % 2 == 1) {
firstAngle = Math.PI + Math.PI / 2 - (textLen - 1) * radianPerInterval / 2.0 - fix;
} else {
firstAngle = Math.PI + Math.PI / 2 - ((textLen / 2.0 - 0.5) * radianPerInterval) - fix;
}
} else {
if (textLen % 2 == 1) {
firstAngle = (textLen - 1) * radianPerInterval / 2.0 + Math.PI / 2 + fix;
} else {
firstAngle = (textLen / 2.0 - 0.5) * radianPerInterval + Math.PI / 2 + fix;
}
}
for (int i = 0; i < textLen; i++) {
double theta;
double thetaX;
double thetaY;
if (!isTop) {
theta = firstAngle + i * radianPerInterval;
thetaX = newRadius * Math.sin(Math.PI / 2 - theta);
thetaY = newRadius * Math.cos(theta - Math.PI / 2);
} else {
theta = firstAngle - i * radianPerInterval;
thetaX = newRadius * Math.sin(Math.PI / 2 - theta);
thetaY = newRadius * Math.cos(theta - Math.PI / 2);
}
AffineTransform transform;
if (!isTop) {
transform = AffineTransform.getRotateInstance(Math.PI + Math.PI / 2 - theta);
} else {
transform = AffineTransform.getRotateInstance(Math.PI / 2 - theta + Math.toRadians(8));
}
Font f2 = f.deriveFont(transform);
g2d.setFont(f2);
g2d.drawString(font.getText().substring(i, i + 1), (float) (circleRadius + thetaX + INIT_BEGIN), (float) (circleRadius - thetaY + INIT_BEGIN));
}
}
private static void drawArcFont4Oval(Graphics2D g2d, SealCircle sealCircle, SealFont font, boolean isTop) {
if (font == null) {
return;
}
float radiusX = sealCircle.getWidth();
float radiusY = sealCircle.getHeight();
float radiusWidth = radiusX + sealCircle.getLine();
float radiusHeight = radiusY + sealCircle.getLine();
int textLen = font.getText().length();
int size = font.getSize() == null ? 25 + (10 - textLen) / 2 : font.getSize();
int style = font.getBold() ? Font.BOLD : Font.PLAIN;
Font f = new Font(font.getFamily(), style, size);
double totalArcAng = font.getSpace() * textLen;
float minRat = 0.90f;
double startAngle = isTop ? -90f - totalArcAng / 2f : 90f - totalArcAng / 2f;
double step = 0.5;
int alCount = (int) Math.ceil(totalArcAng / step) + 1;
double[] angleArr = new double[alCount];
double[] arcLenArr = new double[alCount];
int num = 0;
double accArcLen = 0.0;
angleArr[num] = startAngle;
arcLenArr[num] = accArcLen;
num++;
double angR = startAngle * Math.PI / 180.0;
double lastX = radiusX * Math.cos(angR) + radiusWidth;
double lastY = radiusY * Math.sin(angR) + radiusHeight;
for (double i = startAngle + step; num < alCount; i += step) {
angR = i * Math.PI / 180.0;
double x = radiusX * Math.cos(angR) + radiusWidth, y = radiusY * Math.sin(angR) + radiusHeight;
accArcLen += Math.sqrt((lastX - x) * (lastX - x) + (lastY - y) * (lastY - y));
angleArr[num] = i;
arcLenArr[num] = accArcLen;
lastX = x;
lastY = y;
num++;
}
double arcPer = accArcLen / textLen;
for (int i = 0; i < textLen; i++) {
double arcL = i * arcPer + arcPer / 2.0;
double ang = 0.0;
for (int p = 0; p < arcLenArr.length - 1; p++) {
if (arcLenArr[p] <= arcL && arcL <= arcLenArr[p + 1]) {
ang = (arcL >= ((arcLenArr[p] + arcLenArr[p + 1]) / 2.0)) ? angleArr[p + 1] : angleArr[p];
break;
}
}
angR = (ang * Math.PI / 180f);
Float x = radiusX * (float) Math.cos(angR) + radiusWidth;
Float y = radiusY * (float) Math.sin(angR) + radiusHeight;
double qxang = Math.atan2(radiusY * Math.cos(angR), -radiusX * Math.sin(angR));
double fxang = qxang + Math.PI / 2.0;
int subIndex = isTop ? i : textLen - 1 - i;
String c = font.getText().substring(subIndex, subIndex + 1);
FontMetrics fm = sun.font.FontDesignMetrics.getMetrics(f);
int w = fm.stringWidth(c), h = fm.getHeight();
if (isTop) {
x += h * minRat * (float) Math.cos(fxang);
y += h * minRat * (float) Math.sin(fxang);
x += -w / 2f * (float) Math.cos(qxang);
y += -w / 2f * (float) Math.sin(qxang);
} else {
x += (h * minRat) * (float) Math.cos(fxang);
y += (h * minRat) * (float) Math.sin(fxang);
x += w / 2f * (float) Math.cos(qxang);
y += w / 2f * (float) Math.sin(qxang);
}
AffineTransform affineTransform = new AffineTransform();
affineTransform.scale(0.8, 1);
if (isTop)
affineTransform.rotate(Math.toRadians((fxang * 180.0 / Math.PI - 90)), 0, 0);
else
affineTransform.rotate(Math.toRadians((fxang * 180.0 / Math.PI + 180 - 90)), 0, 0);
Font f2 = f.deriveFont(affineTransform);
g2d.setFont(f2);
g2d.drawString(c, x.intValue() + INIT_BEGIN, y.intValue() + INIT_BEGIN);
}
}
private static void drawFont(Graphics2D g2d, int circleWidth, int circleHeight, SealFont font) {
if (font == null) {
return;
}
int textLen = font.getText().length();
int size = font.getSize() == null ? (55 - textLen * 2) : font.getSize();
int style = font.getBold() ? Font.BOLD : Font.PLAIN;
Font f = new Font(font.getFamily(), style, size);
g2d.setFont(f);
FontRenderContext context = g2d.getFontRenderContext();
String[] fontTexts = font.getText().split("\n");
if (fontTexts.length > 1) {
int y = 0;
for (String fontText : fontTexts) {
y += Math.abs(f.getStringBounds(fontText, context).getHeight());
}
float margin = INIT_BEGIN + (float) (circleHeight / 2 - y / 2);
for (String fontText : fontTexts) {
Rectangle2D rectangle2D = f.getStringBounds(fontText, context);
g2d.drawString(fontText, (float) (circleWidth / 2 - rectangle2D.getCenterX()), margin);
margin += Math.abs(rectangle2D.getHeight());
}
} else {
Rectangle2D rectangle2D = f.getStringBounds(font.getText(), context);
float margin = font.getMargin() == null ?
(float) (circleHeight / 2 - rectangle2D.getCenterY()) :
(float) (circleHeight / 2 - rectangle2D.getCenterY()) + (float) font.getMargin();
g2d.drawString(font.getText(), (float) (circleWidth / 2 - rectangle2D.getCenterX()), margin);
}
}
private static void drawCircle(Graphics2D g2d, SealCircle circle, int x, int y) {
if (circle == null) {
return;
}
int lineSize = circle.getLine() == null ? circle.getHeight() * 2 / (35) : circle.getLine();
g2d.setStroke(new BasicStroke(lineSize));
g2d.drawOval(x, y, circle.getWidth() * 2, circle.getHeight() * 2);
}
public InputStream draw2() throws Exception {
if (mainFont == null || mainFont.getText().length() < 2 || mainFont.getText().length() > 4) {
throw new IllegalArgumentException("请输入2-4个字");
}
int fixH = 18;
int fixW = 2;
BufferedImage bi = new BufferedImage(size, size / 2, BufferedImage.TYPE_4BYTE_ABGR);
Graphics2D g2d = bi.createGraphics();
g2d.setPaint(Color.RED);
g2d.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
int marginW = fixW + borderSquare;
float marginH;
FontRenderContext context = g2d.getFontRenderContext();
Rectangle2D rectangle;
Font f;
if (mainFont.getText().length() == 2) {
if (stamp != null && stamp.trim().length() > 0) {
bi = drawThreeFont(bi, g2d, mainFont.append(stamp), borderSquare, size, fixH, fixW, true);
} else {
f = new Font(mainFont.getFamily(), Font.BOLD, mainFont.getSize());
g2d.setFont(f);
rectangle = f.getStringBounds(mainFont.getText().substring(0, 1), context);
marginH = (float) (Math.abs(rectangle.getCenterY()) * 2 + marginW) + fixH - 4;
g2d.drawString(mainFont.getText().substring(0, 1), marginW, marginH);
marginW += Math.abs(rectangle.getCenterX()) * 2 + (mainFont.getSpace() == null ? INIT_BEGIN : mainFont.getSpace());
g2d.drawString(mainFont.getText().substring(1), marginW, marginH);
BufferedImage nbi = new BufferedImage(size, size, bi.getType());
Graphics2D ng2d = nbi.createGraphics();
ng2d.setPaint(Color.RED);
ng2d.drawImage(bi, 0, 0, size, size, null);
ng2d.setStroke(new BasicStroke(borderSquare));
ng2d.drawRect(0, 0, size, size);
ng2d.dispose();
bi = nbi;
}
} else if (mainFont.getText().length() == 3) {
if (stamp != null && stamp.trim().length() > 0) {
bi = drawFourFont(bi, mainFont.append(stamp), borderSquare, size, fixH, fixW);
} else {
bi = drawThreeFont(bi, g2d, mainFont, borderSquare, size, fixH, fixW, false);
}
} else {
bi = drawFourFont(bi, mainFont, borderSquare, size, fixH, fixW);
}
g2d.dispose();
ByteArrayOutputStream os = new ByteArrayOutputStream();
try {
ImageIO.write(bi, "png", os);
InputStream input = new ByteArrayInputStream(os.toByteArray());
return input;
} catch (IOException e) {
throw new BusinessException("自动生成印章错误");
}
}
private static BufferedImage drawThreeFont(BufferedImage bi, Graphics2D g2d, SealFont font, int lineSize, int imageSize, int fixH, int fixW, boolean isWithYin) {
fixH -= 9;
int marginW = fixW + lineSize;
Font f = new Font(font.getFamily(), Font.BOLD, font.getSize());
g2d.setFont(f);
FontRenderContext context = g2d.getFontRenderContext();
Rectangle2D rectangle = f.getStringBounds(font.getText().substring(0, 1), context);
float marginH = (float) (Math.abs(rectangle.getCenterY()) * 2 + marginW) + fixH;
int oldW = marginW;
if (isWithYin) {
g2d.drawString(font.getText().substring(2, 3), marginW, marginH);
marginW += rectangle.getCenterX() * 2 + (font.getSpace() == null ? INIT_BEGIN : font.getSpace());
} else {
marginW += rectangle.getCenterX() * 2 + (font.getSpace() == null ? INIT_BEGIN : font.getSpace());
g2d.drawString(font.getText().substring(0, 1), marginW, marginH);
}
BufferedImage nbi = new BufferedImage(imageSize, imageSize, bi.getType());
Graphics2D ng2d = nbi.createGraphics();
ng2d.setPaint(Color.RED);
ng2d.drawImage(bi, 0, 0, imageSize, imageSize, null);
ng2d.setStroke(new BasicStroke(lineSize));
ng2d.drawRect(0, 0, imageSize, imageSize);
ng2d.dispose();
bi = nbi;
g2d = bi.createGraphics();
g2d.setPaint(Color.RED);
g2d.setFont(f);
g2d.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
if (isWithYin) {
g2d.drawString(font.getText().substring(0, 1), marginW, marginH += fixH);
rectangle = f.getStringBounds(font.getText(), context);
marginH += Math.abs(rectangle.getHeight());
g2d.drawString(font.getText().substring(1), marginW, marginH);
} else {
g2d.drawString(font.getText().substring(1, 2), oldW, marginH += fixH);
rectangle = f.getStringBounds(font.getText(), context);
marginH += Math.abs(rectangle.getHeight());
g2d.drawString(font.getText().substring(2, 3), oldW, marginH);
}
return bi;
}
private static BufferedImage drawFourFont(BufferedImage bi, SealFont font, int lineSize, int imageSize, int fixH, int fixW) {
int marginW = fixW + lineSize;
BufferedImage nbi = new BufferedImage(imageSize, imageSize, bi.getType());
Graphics2D ng2d = nbi.createGraphics();
ng2d.setPaint(Color.RED);
ng2d.drawImage(bi, 0, 0, imageSize, imageSize, null);
ng2d.setStroke(new BasicStroke(lineSize));
ng2d.drawRect(0, 0, imageSize, imageSize);
ng2d.dispose();
bi = nbi;
Graphics2D g2d = bi.createGraphics();
g2d.setPaint(Color.RED);
g2d.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
FontRenderContext context = g2d.getFontRenderContext();
Font f = new Font(font.getFamily(), Font.BOLD, font.getSize());
g2d.setFont(f);
Rectangle2D rectangle = f.getStringBounds(font.getText().substring(0, 1), context);
float marginH = (float) (Math.abs(rectangle.getCenterY()) * 2 + marginW) + fixH;
g2d.drawString(font.getText().substring(2, 3), marginW, marginH);
int oldW = marginW;
marginW += Math.abs(rectangle.getCenterX()) * 2 + (font.getSpace() == null ? INIT_BEGIN : font.getSpace());
g2d.drawString(font.getText().substring(0, 1), marginW, marginH);
marginH += Math.abs(rectangle.getHeight());
g2d.drawString(font.getText().substring(3, 4), oldW, marginH);
g2d.drawString(font.getText().substring(1, 2), marginW, marginH);
return bi;
}
public static Map automaticGeneration(String text,String fontDesign, Integer width,Integer height,
Integer align,Integer size) throws Exception {
size = size == null ? 40 : size;
align = align == null ? 0 : size;
width = 30;
height = 30;
Font font = new Font("楷体", Font.PLAIN, size);
return createImage(text, font, width, height, align);
}
public static Map newDate(String text) throws Exception {
Font font = new Font("楷体", Font.PLAIN, 100);
return createImage(text, font, 40, 20, 1);
}
private static int[] getWidthAndHeight(String text, Font font) {
Rectangle2D r = font.getStringBounds(text, new FontRenderContext(
AffineTransform.getScaleInstance(1, 1), false, false));
int unitHeight = (int) Math.floor(r.getHeight());
int width = (int) Math.round(r.getWidth()) + 1;
int height = unitHeight ;
return new int[]{width, height};
}
public static Map createImage(String text, Font font, Integer width, Integer height,
Integer align)
throws Exception {
if(text.length() == 2){
char[] chars = text.toCharArray();
text = chars[0] + " " + chars[1];
}
int[] arr = getWidthAndHeight(text, font);
int fontWidth = arr[0];
int fontHeight = arr[1];
height = height < fontHeight ? fontHeight : height;
width = width < fontWidth ? fontWidth : width;
BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_4BYTE_ABGR);
Graphics2D g = image.createGraphics();
RenderingHints hints = new RenderingHints(RenderingHints.KEY_TEXT_ANTIALIASING, RenderingHints.VALUE_TEXT_ANTIALIAS_LCD_HRGB);
g.setRenderingHints(hints);
g.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC, 0));
g.fillRect(0, 0, width, height);
g.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC, 1));
g.setPaint(Color.black);
g.setFont(font);
Integer fontX = 0;
g.drawString(text, fontX, font.getSize());
g.dispose();
ByteArrayOutputStream os = new ByteArrayOutputStream();
ImageIO.write(image, "png", os);
InputStream input = new ByteArrayInputStream(os.toByteArray());
String base64FromInputStream = ImagePointCutUtils.getBase64FromInputStream(input);
os.close();
input.close();
Map responseMap = new HashMap();
responseMap.put("bs64",base64FromInputStream);
responseMap.put("width", width);
responseMap.put("height", height);
return responseMap;
}
}
字体类
import lombok.Builder;
import lombok.Getter;
@Getter
@Builder
public class SealFont {
private String text;
private String family;
private Integer size;
private Boolean bold;
private Double space;
private Integer margin;
public String getFamily() {
return family == null ? "宋体" : family;
}
public boolean getBold() {
return bold == null ? true : bold;
}
public SealFont append(String text) {
this.text += text;
return this;
}
}
变量
@Getter
@Builder
public class SealCircle {
private Integer line;
private Integer width;
private Integer height;
}
测试 生成印章
int size = 300;
int scale = size/300;
int fontScale = scale * 1;
int width = size/2 - 5 * scale;
int height = size/2 - 5 * scale;
int fontSize = 35;
Enterprise enterprise = enterpriseMapper.selectById(user.getLastEnterpriseId());
try {
InputStream inputStream = SealUtil.builder()
.size(size)
.borderCircle(SealCircle.builder().line(5 * scale).width(width).height(height).build())
.mainFont(SealFont.builder().text(enterprise.getName()).size(fontSize * fontScale).space(35.0 * scale).margin(10 * scale).build())
.centerFont(SealFont.builder().text("★").size(100 * fontScale).build())
.titleFont(SealFont.builder().text(" ").size(22 * fontScale).space(10.0 * scale).margin(68 * scale).build())
.build()
.draw();
测试 生成签名
Map sealInfoMap = SealUtil.automaticGeneration(user.getName(), "正楷", 0, 0, 0, 100);
String b64 = (String) sealInfoMap.get("bs64");