Spring Boot集成IText实现电子签章
Spring Boot集成IText实现电子签章
一、生成数字证书
1、引入依赖
<dependency>
<groupId>org.bouncycastle</groupId>
<artifactId>bcpkix-jdk15on</artifactId>
<version>1.70</version>
</dependency>
<dependency>
<groupId>org.bouncycastle</groupId>
<artifactId>bcprov-ext-jdk15on</artifactId>
<version>1.70</version>
</dependency>
2、生成数字证书的工具类
import org.bouncycastle.asn1.ASN1ObjectIdentifier;
import org.bouncycastle.asn1.x500.X500Name;
import org.bouncycastle.asn1.x509.*;
import org.bouncycastle.cert.X509CertificateHolder;
import org.bouncycastle.cert.X509v3CertificateBuilder;
import org.bouncycastle.cert.jcajce.JcaX509v3CertificateBuilder;
import org.bouncycastle.jce.provider.BouncyCastleProvider;
import org.bouncycastle.operator.ContentSigner;
import org.bouncycastle.operator.jcajce.JcaContentSignerBuilder;
import java.io.*;
import java.math.BigInteger;
import java.security.*;
import java.security.cert.CertificateFactory;
import java.security.cert.X509Certificate;
import java.text.SimpleDateFormat;
import java.util.*;
public class PkcsUtils {
/**
* 生成证书
*
* @return
* @throws NoSuchAlgorithmException
*/
private static KeyPair getKey() throws NoSuchAlgorithmException {
KeyPairGenerator generator = KeyPairGenerator.getInstance("RSA",
new BouncyCastleProvider());
generator.initialize(1024);
// 证书中的密钥 公钥和私钥
KeyPair keyPair = generator.generateKeyPair();
return keyPair;
}
/**
* 生成证书
*
* @param password
* @param issuerStr
* @param subjectStr
* @param certificateCRL
* @return
*/
public static Map<String, byte[]> createCert(String password, String issuerStr, String subjectStr, String certificateCRL) {
Map<String, byte[]> result = new HashMap<String, byte[]>();
try(ByteArrayOutputStream out= new ByteArrayOutputStream()) {
// 标志生成PKCS12证书
KeyStore keyStore = KeyStore.getInstance("PKCS12", new BouncyCastleProvider());
keyStore.load(null, null);
KeyPair keyPair = getKey();
// issuer与 subject相同的证书就是CA证书
X509Certificate cert = generateCertificateV3(issuerStr, subjectStr,
keyPair, result, certificateCRL);
// 证书序列号
keyStore.setKeyEntry("cretkey", keyPair.getPrivate(),
password.toCharArray(), new X509Certificate[]{cert});
cert.verify(keyPair.getPublic());
keyStore.store(out, password.toCharArray());
byte[] keyStoreData = out.toByteArray();
result.put("keyStoreData", keyStoreData);
return result;
} catch (Exception e) {
e.printStackTrace();
}
return result;
}
/**
* 生成证书
* @param issuerStr
* @param subjectStr
* @param keyPair
* @param result
* @param certificateCRL
* @return
*/
public static X509Certificate generateCertificateV3(String issuerStr,
String subjectStr, KeyPair keyPair, Map<String, byte[]> result,
String certificateCRL) {
ByteArrayInputStream bint = null;
X509Certificate cert = null;
try {
PublicKey publicKey = keyPair.getPublic();
PrivateKey privateKey = keyPair.getPrivate();
Date notBefore = new Date();
Calendar rightNow = Calendar.getInstance();
rightNow.setTime(notBefore);
// 日期加1年
rightNow.add(Calendar.YEAR, 1);
Date notAfter = rightNow.getTime();
// 证书序列号
BigInteger serial = BigInteger.probablePrime(256, new Random());
X509v3CertificateBuilder builder = new JcaX509v3CertificateBuilder(
new X500Name(issuerStr), serial, notBefore, notAfter,
new X500Name(subjectStr), publicKey);
JcaContentSignerBuilder jBuilder = new JcaContentSignerBuilder(
"SHA1withRSA");
SecureRandom secureRandom = new SecureRandom();
jBuilder.setSecureRandom(secureRandom);
ContentSigner singer = jBuilder.setProvider(
new BouncyCastleProvider()).build(privateKey);
// 分发点
ASN1ObjectIdentifier cRLDistributionPoints = new ASN1ObjectIdentifier(
"2.5.29.31");
GeneralName generalName = new GeneralName(
GeneralName.uniformResourceIdentifier, certificateCRL);
GeneralNames seneralNames = new GeneralNames(generalName);
DistributionPointName distributionPoint = new DistributionPointName(
seneralNames);
DistributionPoint[] points = new DistributionPoint[1];
points[0] = new DistributionPoint(distributionPoint, null, null);
CRLDistPoint cRLDistPoint = new CRLDistPoint(points);
builder.addExtension(cRLDistributionPoints, true, cRLDistPoint);
// 用途
ASN1ObjectIdentifier keyUsage = new ASN1ObjectIdentifier(
"2.5.29.15");
// | KeyUsage.nonRepudiation | KeyUsage.keyCertSign
builder.addExtension(keyUsage, true, new KeyUsage(
KeyUsage.digitalSignature | KeyUsage.keyEncipherment));
// 基本限制 X509Extension.java
ASN1ObjectIdentifier basicConstraints = new ASN1ObjectIdentifier(
"2.5.29.19");
builder.addExtension(basicConstraints, true, new BasicConstraints(
true));
X509CertificateHolder holder = builder.build(singer);
CertificateFactory cf = CertificateFactory.getInstance("X.509");
bint = new ByteArrayInputStream(holder.toASN1Structure()
.getEncoded());
cert = (X509Certificate) cf.generateCertificate(bint);
byte[] certBuf = holder.getEncoded();
SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd");
// 证书数据
result.put("certificateData", certBuf);
//公钥
result.put("publicKey", publicKey.getEncoded());
//私钥
result.put("privateKey", privateKey.getEncoded());
//证书有效开始时间
result.put("notBefore", format.format(notBefore).getBytes("utf-8"));
//证书有效结束时间
result.put("notAfter", format.format(notAfter).getBytes("utf-8"));
} catch (Exception e) {
e.printStackTrace();
} finally {
if (bint != null) {
try {
bint.close();
} catch (IOException e) {
}
}
}
return cert;
}
public static void main(String[] args) throws Exception {
// CN: 名字与姓氏 OU : 组织单位名称
// O :组织名称 L : 城市或区域名称 E : 电子邮件
// ST: 州或省份名称 C: 单位的两字母国家代码
String issuerStr = "CN=xx,OU=yy,O=zz,C=CN,E=xx@gmail.com,L=hz,ST=hz";
String subjectStr = "CN=xx,OU=yy,O=zz,C=CN,E=xx@gmail.com,L=hz,ST=hz";
String certificateCRL = "http://www.bbb.xxx";
Map<String, byte[]> result = createCert("123456", issuerStr, subjectStr, certificateCRL);
FileOutputStream outPutStream = new FileOutputStream("keystore.p12");
outPutStream.write(result.get("keyStoreData"));
outPutStream.close();
FileOutputStream fos = new FileOutputStream(new File("keystore.cer"));
fos.write(result.get("certificateData"));
fos.flush();
fos.close();
}
}
运行该方法,在我们当前工程目录下生成 keystore.p12
和 keystore.cer
两个文件。
其中 keystore.cer
文件通常是一个以 DER
或 PEM
格式存储的 X.509
公钥证书,它包含了公钥以及证书所有者的信息,如姓名、组织、地理位置等。
keystore.p12
文件是一个 PKCS#12
格式的文件,它是一个个人信息交换标准,用于存储一个或多个证书以及它们对应的私钥。.p12
文件是加密的,通常需要密码才能打开。这种文件格式便于将证书和私钥一起分发或存储,常用于需要在不同系统或设备间传输证书和私钥的场景。
二、生成印章图片
1、最终方法
public class SealSample {
public static void main(String[] args) throws Exception {
Seal seal = new Seal();
seal.setSize(200);
SealCircle sealCircle = new SealCircle();
sealCircle.setLine(4);
sealCircle.setWidth(95);
sealCircle.setHeight(95);
seal.setBorderCircle(sealCircle);
SealFont mainFont = new SealFont();
mainFont.setText("xxx有限公司");
mainFont.setSize(22);
mainFont.setFamily("隶书");
mainFont.setSpace(22.0);
mainFont.setMargin(4);
seal.setMainFont(mainFont);
SealFont centerFont = new SealFont();
centerFont.setText("★");
centerFont.setSize(60);
seal.setCenterFont(centerFont);
SealFont titleFont = new SealFont();
titleFont.setText("xx专用章");
titleFont.setSize(16);
titleFont.setSpace(8.0);
titleFont.setMargin(54);
seal.setTitleFont(titleFont);
seal.draw("公章1.png");
}
}
2、工具类-Seal
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.File;
public class Seal {
// 起始位置
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 boolean draw(String pngPath) throws Exception {
if (borderSquare != null) {
return draw2(pngPath); // 画私章
}
//1.画布
BufferedImage bi = new BufferedImage(size, size, BufferedImage.TYPE_4BYTE_ABGR);
//2.画笔
Graphics2D g2d = bi.createGraphics();
//2.1抗锯齿设置
//文本不抗锯齿,否则圆中心的文字会被拉长
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);
//2.2设置背景透明度
g2d.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC, 0));
//2.3填充矩形
g2d.fillRect(0, 0, size, size);
//2.4重设透明度,开始画图
g2d.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC, 1));
//2.5设置画笔颜色
g2d.setPaint(color == null ? Color.RED : color);
// 背景色,要搭配clearRect才可以设置背景生效
g2d.setBackground(Color.WHITE);
g2d.clearRect(0, 0, size, size);//通过使用当前绘图表面的背景色进行填充来清除指定的矩形,
//3.画边线圆
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();
//4.画内边线圆
if (borderInnerCircle != null) {
int x = INIT_BEGIN + borderCircleWidth - borderInnerCircle.getWidth();
int y = INIT_BEGIN + borderCircleHeight - borderInnerCircle.getHeight();
drawCircle(g2d, borderInnerCircle, x, y);
}
//5.画内环线圆
if (innerCircle != null) {
int x = INIT_BEGIN + borderCircleWidth - innerCircle.getWidth();
int y = INIT_BEGIN + borderCircleHeight - innerCircle.getHeight();
drawCircle(g2d, innerCircle, x, y);
}
//6.画弧形主文字
if (borderCircleHeight != borderCircleWidth) {
drawArcFont4Oval(g2d, borderCircle, mainFont, true);
} else {
drawArcFont4Circle(g2d, borderCircleHeight, mainFont, true);
}
//7.画弧形副文字
if (borderCircleHeight != borderCircleWidth) {
drawArcFont4Oval(g2d, borderCircle, viceFont, false);
} else {
drawArcFont4Circle(g2d, borderCircleHeight, viceFont, false);
}
//8.画中心字
drawFont(g2d, (borderCircleWidth + INIT_BEGIN) * 2, (borderCircleHeight + INIT_BEGIN) * 2, centerFont);
//9.画抬头文字
drawFont(g2d, (borderCircleWidth + INIT_BEGIN) * 2, (borderCircleHeight + INIT_BEGIN) * 2, titleFont);
g2d.dispose();
return ImageIO.write(bi, "PNG", new File(pngPath));
}
/**
* 绘制圆弧形文字
*/
private static void drawArcFont4Circle(Graphics2D g2d, int circleRadius, SealFont font, boolean isTop) {
if (font == null) {
return;
}
//1.字体长度
int textLen = font.getText().length();
//2.字体大小,默认根据字体长度动态设定
int size = font.getSize() == null ? (55 - textLen * 2) : font.getSize();
//3.字体样式
int style = font.getBold() ? Font.BOLD : Font.PLAIN;
//4.构造字体
Font f = new Font(font.getFamily(), style, size);
FontRenderContext context = g2d.getFontRenderContext();
Rectangle2D rectangle = f.getStringBounds(font.getText(), context);
//5.文字之间间距,默认动态调整
double space;
if (font.getSpace() != null) {
space = font.getSpace();
} else {
if (textLen == 1) {
space = 0;
} else {
space = rectangle.getWidth() / (textLen - 1) * 0.9;
}
}
//6.距离外圈距离
int margin = font.getMargin() == null ? INIT_BEGIN : font.getMargin();
//7.写字
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();
//1.字体长度
int textLen = font.getText().length();
//2.字体大小,默认根据字体长度动态设定
int size = font.getSize() == null ? 25 + (10 - textLen) / 2 : font.getSize();
//3.字体样式
int style = font.getBold() ? Font.BOLD : Font.PLAIN;
//4.构造字体
Font f = new Font(font.getFamily(), style, size);
//5.总的角跨度
double totalArcAng = font.getSpace() * textLen;
//6.从边线向中心的移动因子
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);
FontMetrics fm = g2d.getFontMetrics();
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;
}
//1.字体长度
int textLen = font.getText().length();
//2.字体大小,默认根据字体长度动态设定
int size = font.getSize() == null ? (55 - textLen * 2) : font.getSize();
//3.字体样式
int style = font.getBold() ? Font.BOLD : Font.PLAIN;
//4.构造字体
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());
}
//5.设置上边距
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);
//5.设置上边距,默认在中心
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;
}
//1.圆线条粗细默认是圆直径的1/35
int lineSize = circle.getLine() == null ? circle.getHeight() * 2 / (35) : circle.getLine();
//2.画圆
g2d.setStroke(new BasicStroke(lineSize));
g2d.drawOval(x, y, circle.getWidth() * 2, circle.getHeight() * 2);
}
/**
* 画私章
*/
public boolean draw2(String pngPath) throws Exception {
if (mainFont == null || mainFont.getText().length() < 2 || mainFont.getText().length() > 4) {
throw new IllegalArgumentException("请输入2-4个字");
}
int fixH = 18;
int fixW = 2;
//1.画布
BufferedImage bi = new BufferedImage(size, size / 2, BufferedImage.TYPE_4BYTE_ABGR);// 这种是背景色马赛克的
//2.画笔
Graphics2D g2d = bi.createGraphics();
//2.1设置画笔颜色
g2d.setPaint(Color.RED);
// 背景色,要搭配clearRect才可以设置背景生效
g2d.setBackground(Color.WHITE);
g2d.clearRect(0, 0, size, size / 2);//通过使用当前绘图表面的背景色进行填充来清除指定的矩形,
//2.2抗锯齿设置
g2d.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
//3.写签名
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.setBackground(Color.WHITE);
//画正方形
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();
return ImageIO.write(bi, "PNG", new File(pngPath));
}
/**
* 画三字
*/
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 Integer getSize() {
return size;
}
public void setSize(Integer size) {
this.size = size;
}
public Color getColor() {
return color;
}
public void setColor(Color color) {
this.color = color;
}
public SealFont getMainFont() {
return mainFont;
}
public void setMainFont(SealFont mainFont) {
this.mainFont = mainFont;
}
public SealFont getViceFont() {
return viceFont;
}
public void setViceFont(SealFont viceFont) {
this.viceFont = viceFont;
}
public SealFont getTitleFont() {
return titleFont;
}
public void setTitleFont(SealFont titleFont) {
this.titleFont = titleFont;
}
public SealFont getCenterFont() {
return centerFont;
}
public void setCenterFont(SealFont centerFont) {
this.centerFont = centerFont;
}
public SealCircle getBorderCircle() {
return borderCircle;
}
public void setBorderCircle(SealCircle borderCircle) {
this.borderCircle = borderCircle;
}
public SealCircle getBorderInnerCircle() {
return borderInnerCircle;
}
public void setBorderInnerCircle(SealCircle borderInnerCircle) {
this.borderInnerCircle = borderInnerCircle;
}
public SealCircle getInnerCircle() {
return innerCircle;
}
public void setInnerCircle(SealCircle innerCircle) {
this.innerCircle = innerCircle;
}
public Integer getBorderSquare() {
return borderSquare;
}
public void setBorderSquare(Integer borderSquare) {
this.borderSquare = borderSquare;
}
public String getStamp() {
return stamp;
}
public void setStamp(String stamp) {
this.stamp = stamp;
}
}
3、工具类-SealCircle
public class SealCircle {
private Integer line;
private Integer width;
private Integer height;
public Integer getLine() {
return line;
}
public void setLine(Integer line) {
this.line = line;
}
public Integer getWidth() {
return width;
}
public void setWidth(Integer width) {
this.width = width;
}
public Integer getHeight() {
return height;
}
public void setHeight(Integer height) {
this.height = height;
}
}
4、工具类-SealFont
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;
}
public String getText() {
return text;
}
public void setText(String text) {
this.text = text;
}
public void setFamily(String family) {
this.family = family;
}
public Integer getSize() {
return size;
}
public void setSize(Integer size) {
this.size = size;
}
public void setBold(Boolean bold) {
this.bold = bold;
}
public Double getSpace() {
return space;
}
public void setSpace(Double space) {
this.space = space;
}
public Integer getMargin() {
return margin;
}
public void setMargin(Integer margin) {
this.margin = margin;
}
}
PDF签名
1、引入依赖
<dependency>
<groupId>com.itextpdf</groupId>
<artifactId>itextpdf</artifactId>
<version>5.5.13.4</version>
</dependency>
<dependency>
<groupId>com.itextpdf</groupId>
<artifactId>html2pdf</artifactId>
<version>5.0.5</version>
</dependency>
2、对 PDF 文件进行签名
public class SignPdf {
/**
* @param password pkcs12证书密码
* @param keyStorePath pkcs12证书路径
* @param signPdfSrc 签名pdf路径
* @param signImage 签名图片
* @param x
* @param y
* @return
*/
public static byte[] sign(String password, String keyStorePath, String signPdfSrc, String signImage,
float x, float y) {
File signPdfSrcFile = new File(signPdfSrc);
PdfReader reader = null;
ByteArrayOutputStream signPDFData = null;
PdfStamper stp = null;
FileInputStream fos = null;
try {
BouncyCastleProvider provider = new BouncyCastleProvider();
Security.addProvider(provider);
KeyStore ks = KeyStore.getInstance("PKCS12", new BouncyCastleProvider());
fos = new FileInputStream(keyStorePath);
// 私钥密码 为Pkcs生成证书是的私钥密码 123456
ks.load(fos, password.toCharArray());
String alias = (String) ks.aliases().nextElement();
PrivateKey key = (PrivateKey) ks.getKey(alias, password.toCharArray());
Certificate[] chain = ks.getCertificateChain(alias);
reader = new PdfReader(signPdfSrc);
signPDFData = new ByteArrayOutputStream();
// 临时pdf文件
File temp = new File(signPdfSrcFile.getParent(), System.currentTimeMillis() + ".pdf");
stp = PdfStamper.createSignature(reader, signPDFData, '\0', temp, true);
stp.setFullCompression();
PdfSignatureAppearance sap = stp.getSignatureAppearance();
sap.setReason("数字签名,不可改变");
// 使用png格式透明图片
Image image = Image.getInstance(signImage);
sap.setImageScale(0);
sap.setSignatureGraphic(image);
sap.setRenderingMode(PdfSignatureAppearance.RenderingMode.GRAPHIC);
// 是对应x轴和y轴坐标
sap.setVisibleSignature(new Rectangle(x, y, x + 185, y + 68), 1,
UUID.randomUUID().toString().replaceAll("-", ""));
stp.getWriter().setCompressionLevel(5);
ExternalDigest digest = new BouncyCastleDigest();
ExternalSignature signature = new PrivateKeySignature(key, DigestAlgorithms.SHA512, provider.getName());
MakeSignature.signDetached(sap, digest, signature, chain, null, null, null, 0, MakeSignature.CryptoStandard.CADES);
stp.close();
reader.close();
return signPDFData.toByteArray();
} catch (Exception e) {
e.printStackTrace();
} finally {
if (signPDFData != null) {
try {
signPDFData.close();
} catch (IOException e) {
}
}
if (fos != null) {
try {
fos.close();
} catch (IOException e) {
}
}
}
return null;
}
public static void main(String[] args) throws Exception {
byte[] fileData = sign("123456", "keystore.p12",
"待签名.pdf",//
"公章1.png", 100, 290);
FileOutputStream f = new FileOutputStream(new File("已签名.pdf"));
f.write(fileData);
f.close();
}
}