annotations - iText 中的 Box Cloud 注释外观
问题描述
这是我绘制框云注释的示例代码。我在 PDFBox 的实现中使用了代码来绘制一个框云,但是在 iText 中使用时我遇到了一个小问题。我修改了边框类和一些部分以在 iText 中使用。
你可以在这里找到边界类。
我的问题是,没有绘制顶部和右侧的边界云。似乎它们的位置超出了矩形差异。我认为问题在于在 cloudyPolygonImpl() 中绘制曲线。也许itext有不同的方式来绘制PdfAppearance?我不知道。
public class Test {
public static void main(String[] args) throws Exception {
PdfReader reader = new PdfReader("src.pdf");
PdfStamper stamper = new PdfStamper(reader, new FileOutputStream("result.pdf"));
PdfDictionary be = new PdfDictionary();
be.put(PdfName.S, PdfName.C);
be.put(PdfName.I, new PdfNumber(1));
Rectangle location = new Rectangle(123.6f, 584.4f, 252.6f, 653.4f);
PdfAnnotation stamp = PdfAnnotation.createSquareCircle(stamper.getWriter(), location, "", true);
stamp.setBorderStyle(new PdfBorderDictionary(1, PdfBorderDictionary.STYLE_SOLID));
stamp.put(new PdfName("BE"), be);
stamp.setColor(BaseColor.RED);
PdfContentByte cb = stamper.getOverContent(1);
PdfAppearance app = cb.createAppearance(location.getWidth(), location.getHeight());
stamp.setAppearance(PdfName.N, app);
PdfArray stickyRect = stamp.getAsArray(PdfName.RECT);
Rectangle annotRect = new Rectangle(stickyRect.getAsNumber(0).floatValue(),
stickyRect.getAsNumber(1).floatValue(),
stickyRect.getAsNumber(2).floatValue(),
stickyRect.getAsNumber(3).floatValue());
PdfArray arrDiff = annotation.getAsArray(PdfName.RD);
Rectangle annotRectDiff = null;
if (arrDiff != null) {
annotRectDiff = new Rectangle(arrDiff.getAsNumber(0).floatValue(), arrDiff.getAsNumber(1).floatValue(),
arrDiff.getAsNumber(2).floatValue(), arrDiff.getAsNumber(3).floatValue()
}
// Create cloud appearance
CBorder cborder = new CBorder(app, 1, 1, annotRect);
cborder.createCloudyRectangle(annotRectDiff);
stamp.put(PdfName.RECT, new PdfRectangle(cborder.getRectangle()));
stamp.put(PdfName.RD, new PdfArray(new float[] {
cborder.getRectDifference().getLeft(),
cborder.getRectDifference().getBottom(),
cborder.getRectDifference().getRight(),
cborder.getRectDifference().getTop() }));
app.rectangle(cborder.getBBox());
app.transform(cborder.getMatrix());
app.setColorStroke(BaseColor.RED);
app.setLineWidth(1);
app.stroke();
stamper.addAnnotation(stamp, 1);
stamper.close();
reader.close();
}
}
正确的输出应该是所有边框都用云绘制,但目前只绘制了左侧和底部。
解决方案
(此答案基于您问题的修订版 3 中的代码,因为修订版 4中的更改引入了多个错误。)
您的代码在这里创建了一个无效的注释外观流:
CBorder cborder = new CBorder(app, 1, 1, annotRect);
cborder.createCloudyRectangle(null);
stamp.put(PdfName.RECT, new PdfRectangle(cborder.getRectangle()));
stamp.put(PdfName.RD, new PdfArray(new float[] {
cborder.getRectDifference().getLeft(),
cborder.getRectDifference().getBottom(),
cborder.getRectDifference().getRight(),
cborder.getRectDifference().getTop() }));
app.rectangle(cborder.getBBox());
app.transform(cborder.getMatrix());
app.setColorStroke(BaseColor.RED);
app.setLineWidth(1);
app.stroke();
它的上半部分创建了一条路径:
2 j
121.58 588.63 m
122.06 588.95 122.6 589.18 123.16 589.3 c
120.73 588.78 119.18 586.4 119.7 583.96 c
120.19 581.67 122.35 580.14 124.68 580.44 c
...
122.06 596.42 122.6 596.64 123.16 596.76 c
121.09 596.32 119.6 594.49 119.6 592.36 c
119.6 590.87 120.34 589.47 121.58 588.63 c
h
然后app.rectangle(cborder.getBBox())
不会创建任何东西(请注意,此rectangle
重载不会像您期望的那样!)。
然后app.transform(cborder.getMatrix())
对当前的变换矩阵app.setColorStroke(BaseColor.RED)
添加一个变化,添加一个描边颜色的变化,并app.setLineWidth(1)
添加一个线宽的变化:
1 0 0 1 -118.68 -579.48 cm
1 0 0 RG
1 w
最后app.stroke()
添加命令来描边路径:
S
但是在路径的定义和对应的路径绘制命令之间,只允许剪切路径指令!参照。图 9——图形对象——在 PDF 规范 ISO 32000-1 中。
您可以像这样修复此代码,拉起颜色和线宽更改,并直接使用云边界框:
// Create cloud appearance
app.setColorStroke(BaseColor.RED);
app.setLineWidth(1);
CBorder cborder = new CBorder(app, 1, 1, annotRect);
cborder.createCloudyRectangle(null);
stamp.put(PdfName.RECT, new PdfRectangle(cborder.getRectangle()));
stamp.put(PdfName.RD, new PdfArray(new float[] {
cborder.getRectDifference().getLeft(),
cborder.getRectDifference().getBottom(),
cborder.getRectDifference().getRight(),
cborder.getRectDifference().getTop() }));
app.stroke();
app.setBoundingBox(cborder.getBBox());
(CloudBoxAnnotation测试testDrawLikeChitgoksImproved
)
这特别改变了结果(如在 Adobe Acrobat 中看到的)
至
推荐阅读
- javascript - 如何检查元素是否包含至少一个直接文本节点
- random - Neo4j:匹配随机节点时,with子句中rand()和rand()之间的区别
- r - 如何在不输入所有列名的情况下添加列?
- android - AdMob 横幅的默认尺寸是多少?
- teradata - 关于唯一主键与唯一主索引的问题
- ruby-on-rails - 在 html.erb 中完成了几个月的渲染
- multithreading - 带有 CompletableFuture 的 Spring-Boot App,如何管理大量请求?
- google-sheets - Countif 区分大小写的谷歌表格
- javascript - 尝试添加属性时,Javascript Map Array Object会更改原始数组
- traefik - Traefik - 测试Referer的中间件