我想使用itext7生成类似此pdf文件的内容:
但我找不到任何方法来实现这一目标。
我在教程iText - clickable image should open ms word attachment中看到的内容无法将附件和文本放在一起。您只能将附件放在单独的页面上。
如果是itext5,我会这样:
PdfAnnotation anno = PdfAnnotation.createFileAttachment(writer, null, fileDescribe, pdfFileSpecification);
chunk.setAnnotation(anno);
paragrah.add(chunk);
所以我可以在一个段落中添加文本,注释,但是itext7教程就是这样:
PdfDocument pdfDoc = new PdfDocument(new PdfWriter(DEST));
Rectangle rect = new Rectangle(36, 700, 100, 100);
PdfFileSpec fs = PdfFileSpec.createEmbeddedFileSpec(pdfDoc, PATH, null, "test.docx", null, null, false);
PdfAnnotation attachment = new PdfFileAttachmentAnnotation(rect, fs)
.setContents("Click me");
PdfFormXObject xObject = new PdfFormXObject(rect);
ImageData imageData = ImageDataFactory.create(IMG);
PdfCanvas canvas = new PdfCanvas(xObject, pdfDoc);
canvas.addImage(imageData, rect, true);
attachment.setNormalAppearance(xObject.getPdfObject());
pdfDoc.addNewPage().addAnnotation(attachment);
pdfDoc.close();
有人能帮我吗?
最佳答案
如果我理解正确,则希望在文档布局的其他布局元素中添加注释。
当前在iText7
中没有实现它的快速方法(例如setAnnotation
中的iText5
方法)。但是,iText7
具有足够的灵活性,可以允许您创建自定义元素,而不必深入了解代码。
初始部分将与当前示例相同。此处已设置注释本身:
PdfDocument pdfDoc = new PdfDocument(new PdfWriter(DEST));
Rectangle rect = new Rectangle(36, 700, 50, 50);
PdfFileSpec fs = PdfFileSpec.createEmbeddedFileSpec(pdfDoc, PATH, null, "test.docx", null, null, false);
PdfAnnotation attachment = new PdfFileAttachmentAnnotation(rect, fs)
.setContents("Click me");
PdfFormXObject xObject = new PdfFormXObject(rect);
ImageData imageData = ImageDataFactory.create(IMG);
PdfCanvas canvas = new PdfCanvas(xObject, pdfDoc);
canvas.addImage(imageData, rect, true);
attachment.setNormalAppearance(xObject.getPdfObject());
然后,我们要实现的是能够将自定义注释元素添加到布局
Document
流中。最好的情况是,代码如下所示:
Document document = new Document(pdfDoc);
Paragraph p = new Paragraph("There are two").add(new AnnotationElement(attachment)).add(new Text("elements"));
document.add(p);
document.close();
现在剩下定义
AnnotationElement
和相应的渲染器了。除了创建自定义渲染器并将注释传递给元素之外,元素本身没有任何特定的逻辑:private static class AnnotationElement extends AbstractElement<AnnotationElement> implements ILeafElement {
private PdfAnnotation annotation;
public AnnotationElement(PdfAnnotation annotation) {
this.annotation = annotation;
}
@Override
protected IRenderer makeNewRenderer() {
return new AnnotationRenderer(annotation);
}
}
渲染器实现具有更多代码,但它只是定义了
layout
上的已占用区域并将注释添加到draw
上的页面上。请注意,它不能涵盖所有情况(例如,没有足够的空间容纳注释),但是对于简单的情况和演示目的而言,这已经足够了。private static class AnnotationRenderer extends AbstractRenderer implements ILeafElementRenderer {
private PdfAnnotation annotation;
public AnnotationRenderer(PdfAnnotation annotat) {
this.annotation = annotat;
}
@Override
public float getAscent() {
return annotation.getRectangle().toRectangle().getHeight();
}
@Override
public float getDescent() {
return 0;
}
@Override
public LayoutResult layout(LayoutContext layoutContext) {
occupiedArea = layoutContext.getArea().clone();
float myHeight = annotation.getRectangle().toRectangle().getHeight();
float myWidth = annotation.getRectangle().toRectangle().getWidth();
occupiedArea.getBBox().moveUp(occupiedArea.getBBox().getHeight() - myHeight).setHeight(myHeight);
occupiedArea.getBBox().setWidth(myWidth);
return new LayoutResult(LayoutResult.FULL, occupiedArea, null, null);
}
@Override
public void draw(DrawContext drawContext) {
super.draw(drawContext);
annotation.setRectangle(new PdfArray(occupiedArea.getBBox()));
drawContext.getDocument().getPage(occupiedArea.getPageNumber()).addAnnotation(annotation);
}
@Override
public IRenderer getNextRenderer() {
return new AnnotationRenderer(annotation);
}
}
请注意,该示例适用于当前
7.0.3-SNAPSHOT
版本。对于更高版本的7.0.2
可能仍然无法工作,因为后来添加了ILeafElementRenderer
接口,但是如果确实需要,仍然可以将代码修改为7.0.2
。