我想合并成一个新的pdf多个pdf文件,并在每个页面上添加文具。
为此,我使用了PdfWriter,如下所示:
public class FusionPdf extends PdfPageEventHelper {
private Document document = null;
private PdfWriter writer = null;
private PdfReader markReader = null;
private PdfImportedPage fondDePage = null;
private String textFili = null;
private String textMark = null;
private static final Font FONT1 = FontFactory.getFont(FontFactory.HELVETICA, 50, Font.NORMAL, new GrayColor(0.7f));
private static final Font FONT2 = FontFactory.getFont(FontFactory.HELVETICA, 6, Font.NORMAL, BaseColor.BLACK);
/**
* liste des fichiers � concatener
*/
private List<File> pdfsOriginaux = new ArrayList<File>();
/**
*
* @param pdfDestination fichier de pdf de destination
* @param pdfsOriginaux fichiers pdf � concatener
* @param pdfTatoueur Fond de page (peut �tre null)
* @param textFiligrane filigran (peut �tre null)
* @throws IOException
* @throws DocumentException
* @throws Exception
*/
public FusionPdf(File pdfDestination, List<File> pdfsOriginaux, File pdfTatoueur, String textFiligrane, String textMarquage) throws DocumentException, IOException {
if (pdfsOriginaux.isEmpty()){
throw new IOException("Aucun document pdf a traiter !");
}
this.pdfsOriginaux = pdfsOriginaux;
this.init(pdfDestination, pdfTatoueur, textFiligrane, textMarquage);
}
/**
*
* @param pdfDestination fichier de pdf de destination
* @param pdfOriginal fichier pdf a traiter
* @param pdfTatoueur Fond de page (peut etre null)
* @param textFiligrane filigrane (peut etre null)
* @param textMarquage note technique en haut à droite en petit (peut etre null)
* @throws IOException
* @throws DocumentException
* @throws Exception
*/
public FusionPdf(File pdfDestination, File pdfOriginal, File pdfTatoueur, String textFiligrane, String textMarquage) throws DocumentException, IOException {
this.pdfsOriginaux.add(pdfOriginal);
this.init(pdfDestination, pdfTatoueur, textFiligrane, textMarquage);
}
/**
* initialisation des attributs
*
* @param pdfDestination
* @param pdfTatoueur
* @param textFiligrane
* @throws DocumentException
* @throws IOException
*/
private void init(File pdfDestination, File pdfTatoueur, String textFiligrane, String textMarquage) throws DocumentException, IOException{
document = new Document();
writer = PdfWriter.getInstance(document, new FileOutputStream(pdfDestination));
writer.setPageEvent(this);
if (pdfTatoueur != null) {
markReader = new PdfReader(pdfTatoueur.getAbsolutePath());
fondDePage = writer.getImportedPage(markReader, 1);
}
if (textFiligrane != null) {
textFili = textFiligrane;
}
if (textFiligrane != null) {
textMark = textMarquage;
}
}
/**
* applique la concatenation et la fusion
* @throws IOException
*/
public void fuse() throws IOException{
//---->initialisation d'un flux vers le pdf
PdfReader originalReader = new PdfReader(pdfsOriginaux.get(0).getAbsolutePath());
document.open();
for (File pdfOriginal : pdfsOriginaux) {
originalReader = new PdfReader(pdfOriginal.getAbsolutePath());
for (int i = 1; i <= originalReader.getNumberOfPages(); i++) {
document.setPageSize(originalReader.getPageSizeWithRotation(i));
document.newPage();
writer.getDirectContent().addTemplate(writer.getImportedPage(originalReader, i), 0, 0);
}
}
document.close();
originalReader.close();
if (markReader != null) {
markReader.close();
}
}
@Override
public void onEndPage(PdfWriter writer, Document document) {
PdfContentByte directContent;
Rectangle docPageSize = document.getPageSize();
//ajout du fond de page
if (markReader != null) {
directContent = writer.getDirectContentUnder();
Rectangle markPageSize = markReader.getPageSize(1);
float hScale = docPageSize.getWidth() / markPageSize.getWidth();
float vScale = docPageSize.getHeight() / markPageSize.getHeight();
float markScale = (hScale< vScale) ? hScale : vScale;
float hTrans = (float)((docPageSize.getWidth()-markPageSize.getWidth()* markScale) / 2.0);
float vTrans = (float)((docPageSize.getHeight()-markPageSize.getHeight()* markScale) / 2.0);
directContent.addTemplate(fondDePage, markScale, 0, 0, markScale, hTrans, vTrans );
}
//ajout du filigrane
if (textFili != null) {
directContent = writer.getDirectContent();
PdfGState gstate = new PdfGState();
gstate.setFillOpacity(0.3f);
gstate.setStrokeOpacity(0.3f);
directContent.saveState();
directContent.setGState(gstate);
ColumnText.showTextAligned(directContent,Element.ALIGN_CENTER, new Phrase(textFili, FONT1), docPageSize.getWidth()/2, docPageSize.getHeight()/2, 45);
directContent.restoreState();
}
//ajout de la marque en haut à droite en petit
if (textMark != null) {
directContent = writer.getDirectContent();
PdfGState gstate = new PdfGState();
directContent.saveState();
directContent.setGState(gstate);
ColumnText.showTextAligned(directContent,Element.ALIGN_RIGHT, new Phrase(textMark, FONT2), docPageSize.getWidth()-3, docPageSize.getHeight()-8, 0);
directContent.restoreState();
}
}
}
但这是错误的:根据原始pdf,结果有时是错误的。方向不正确。然后我在这里找到了该行为函数的答案,该函数可以使用iText将PDF连接/合并在一起,从而导致一些问题
=
public class FusionPdf2 extends PdfPageEventHelper {
private static final Font FONT1 = FontFactory.getFont(FontFactory.HELVETICA, 50, Font.NORMAL, new GrayColor(0.7f));
private static final Font FONT2 = FontFactory.getFont(FontFactory.HELVETICA, 6, Font.NORMAL, BaseColor.BLACK);
private List<File> pdfsOriginaux = new ArrayList<File>();
private File pdfDestination = null;
private File pdfTatoueur = null;
private String textFiligrane = null;
private String textMarquage = null;
/**
*
* @param pdfDestination fichier de pdf de destination
* @param pdfsOriginaux fichiers pdf à concatener
* @param pdfTatoueur Fond de page (peut être null)
* @param textFiligrane filigran (peut être null)
* @throws IOException
* @throws DocumentException
* @throws Exception
*/
public FusionPdf2(File pdfDestination, List<File> pdfsOriginaux, File pdfTatoueur, String textFiligrane, String textMarquage) throws DocumentException, IOException {
this.pdfDestination = pdfDestination;
if (pdfsOriginaux.isEmpty()){
throw new IOException("Aucun document pdf a traiter !");
}
this.pdfsOriginaux = pdfsOriginaux;
this.pdfTatoueur = pdfTatoueur;
this.textFiligrane = textFiligrane;
this.textMarquage = textMarquage;
}
/**
*
* @param pdfDestination fichier de pdf de destination
* @param pdfOriginal fichier pdf a traiter
* @param pdfTatoueur Fond de page (peut etre null)
* @param textFiligrane filigrane (peut etre null)
* @param textMarquage note technique en haut à droite en petit (peut etre null)
* @throws IOException
* @throws DocumentException
* @throws Exception
*/
public FusionPdf2(File pdfDestination, File pdfOriginal, File pdfTatoueur, String textFiligrane, String textMarquage) throws DocumentException, IOException {
this.pdfDestination = pdfDestination;
this.pdfsOriginaux.add(pdfOriginal);
this.pdfTatoueur = pdfTatoueur;
this.textFiligrane = textFiligrane;
this.textMarquage = textMarquage;
}
/**
* applique la concatenation et la fusion
*
* @param pdfDestination
* @param pdfTatoueur
* @param textFiligrane
* @throws DocumentException
* @throws IOException
* @throws IOException
* @throws DocumentException
*/
public void fuse() throws IOException, DocumentException{
Document document = new Document();
PdfCopy copy = new PdfCopy(document, new FileOutputStream(pdfDestination));
document.open();
PdfReader originalReader;
PdfReader fdpReader = null;
if (pdfTatoueur != null) {
fdpReader = new PdfReader(pdfTatoueur.getAbsolutePath());
}
for (File pdfOriginal : pdfsOriginaux) {
originalReader = new PdfReader(pdfOriginal.getAbsolutePath());
for (int i = 0 ; i < originalReader.getNumberOfPages(); ) {
PdfImportedPage page = copy.getImportedPage(originalReader, ++i);
PageStamp stamp = copy.createPageStamp(page);
Rectangle docPageSize = originalReader.getPageSizeWithRotation(i);
//ajout du fond de page
if (pdfTatoueur != null) {
PdfImportedPage fondDePage = copy.getImportedPage(fdpReader, 1);
PdfContentByte directContent = stamp.getUnderContent();
Rectangle markPageSize = fdpReader.getPageSize(1);
float hScale = docPageSize.getWidth() / markPageSize.getWidth();
float vScale = docPageSize.getHeight() / markPageSize.getHeight();
float markScale = (hScale< vScale) ? hScale : vScale;
float hTrans = (float)((docPageSize.getWidth()-markPageSize.getWidth()* markScale) / 2.0);
float vTrans = (float)((docPageSize.getHeight()-markPageSize.getHeight()* markScale) / 2.0);
directContent.addTemplate(fondDePage, markScale, 0, 0, markScale, hTrans, vTrans );
}
//ajout du filigrane
if (StringUtils.isNotBlank(textFiligrane)) {
PdfContentByte directContent = stamp.getOverContent();
PdfGState gstate = new PdfGState();
gstate.setFillOpacity(0.3f);
gstate.setStrokeOpacity(0.3f);
directContent.saveState();
directContent.setGState(gstate);
ColumnText.showTextAligned(directContent,Element.ALIGN_CENTER, new Phrase(textFiligrane, FONT1), docPageSize.getWidth()/2, docPageSize.getHeight()/2, 45);
directContent.restoreState();
}
//ajout de la marque en haut à droite en petit
if (StringUtils.isNotBlank(textMarquage)) {
PdfContentByte directContent = stamp.getOverContent();
ColumnText.showTextAligned(directContent,Element.ALIGN_RIGHT, new Phrase(textMarquage, FONT2), docPageSize.getWidth()-3, docPageSize.getHeight()-8, 0);
}
stamp.alterContents();
copy.addPage(page);
}
if (originalReader!=null) {
originalReader.close();
}
}
document.close();
}
}
这一次的结果在任何情况下都是好的:所有页面都处于良好的方向。
但是生成的pdf比以前的代码快10倍。经过分析,我发现了原因:在我添加文具之前,一切都很好。
这就像信纸的内容在每一页上都是重复的,而不是重复使用。
显然我又做错了。我读了这本书的第六章,但没有找到解决办法。
如果有人能帮助我!
这就像信纸的内容在每一页上都是重复的,而不是重复使用。
这是事实,它反映了FusionPdf
和FusionPdf2
类之间的根本区别:
在前一个类中,您在开始时导入固定页面一次
private void init(File pdfDestination, File pdfTatoueur, String textFiligrane, String textMarquage) throws DocumentException, IOException{
[...]
if (pdfTatoueur != null) {
markReader = new PdfReader(pdfTatoueur.getAbsolutePath());
fondDePage = writer.getImportedPage(markReader, 1);
}
[...]
}
并反复使用此导入的页面:
public void onEndPage(PdfWriter writer, Document document) {
[...]
//ajout du fond de page
if (markReader != null) {
[...]
directContent.addTemplate(fondDePage, markScale, 0, 0, markScale, hTrans, vTrans );
}
[...]
}
另一方面,在后一类中,您一次又一次地导入固定页面,每页应用一次:
for (int i = 0 ; i < originalReader.getNumberOfPages(); ) {
[...]
if (pdfTatoueur != null) {
PdfImportedPage fondDePage = copy.getImportedPage(fdpReader, 1);
[...]
directContent.addTemplate(fondDePage, markScale, 0, 0, markScale, hTrans, vTrans );
}
[...]
}
要使固定页面的内容仅在结果PDF中出现一次,只需导入一次。
问题内容: 如何使用iText将书签添加到现有PDF? 我将多个PDF合并为一个PDF,并且需要为最终PDF构建书签。例如,我有三个PDF:doc1.pdf,doc2.pdf和doc3.pdf,doc1和doc2属于Group1,doc3属于Group2。我需要合并它们,并且必须为生成的PDF构建嵌套书签,如下所示: 等等 问题答案: 我已经制作了一个MergeWithOutlines示例,该示例
//步骤1:创建文档-对象文档Document=new Document(); 提前道谢。
我正在尝试将多个pdf页面合并为一个pdf页面。有很多iText的例子展示了如何将pdf页面合并到一个文件中,但是我需要所有的页面都放在一个页面中(一路上缩小它们的宽度和高度) 编辑:尝试从这里这个代码,但它只是合并成一个文件的pdf页面,我需要他们收缩成一个单一的页面
问题内容: 如何使用iText在每个页面上添加总页数? 问题答案: 使用伪页面计数将输出从a 处理为第一个。 从中创建一个,调用以获取实际的页数。 重新创建PDF输出,知道页数是多少,并相应地更改页脚。 这很麻烦,但是如果没有两遍方法,就没有简单的方法来知道页数。有关处理PDF的详细信息,请参见示例代码。
问题内容: 我正在尝试使用具有以下代码的iText 7创建PDF文档,并且生成时,我的PDF文档内容在同一页面中重叠(即,在第1页中)。 我看到了 document.newPage(); iText 7中缺少该方法。如何在itext 7中不使用pdfDocumet.copyPages(…)或PDFmerger将页面添加到我的PDF文档中。 问题答案: 在iText 7中,该方法已成为区域中断的特殊
问题内容: 我的java项目分为两部分。 我需要填充pdf的字段 我需要在页面空白区域的填充部分下方添加一个表(并且该表需要能够移到下一页)。 我能够分别执行这些操作(填充pdf并创建表)。但是我无法有效地合并它们。我尝试做一个doc.add(table),这将导致该表在pdf的下一页上,这是我不想要的。 我基本上只需要能够指定表格在页面上的起始位置(这样它就不会与现有内容重叠),然后将表格标记到