在页面上移动文本,iText7保留字体,颜色,样式......但更改文本大小

时间:2016-11-11 15:51:02

标签: java pdf itext7

我想用iText7移动文字。我有一个源边界框,可以在页面的某个地方,我有一个目标边界框,它有一个固定的位置(包括宽度和高度)。我会留在同一页上。源框和目标框可以重叠。源边界框也可以大于目标框。在这种情况下,我必须减少字体大小。文本应保留字体,颜色等。

iText网站上有一个cut and paste example。但是在结果pdf文件中,您可以选择新旧位置的文本(仅使用普通的pdf阅读器进行尝试)。我不希望文本在旧位置可选。

我想,也许我可以选择文字并将其放在新位置并将其从旧位置移除。对于后者我需要pdfSweep,但这没关系。在新位置添加文本应该没问题。即使文本有不同的字体,大小等。 iText网站上有很多例子。我知道选择文本的唯一方法就像这个example中所示。这只给我文字。但是要将它放在具有相同字体,颜色等的目标位置,我也需要所有这些信息。 我知道,pdf不适合编辑。在StackOverflow的答案中经常提到这一点。 有没有办法用iText7做到这一点?

1 个答案:

答案 0 :(得分:0)

iText中没有高级API允许您移动页面内容,特别是来自某个矩形的所有内容。一个原因可能是,一般来说这不仅仅是移动。 PDF通常包含影响较大区域的结构,这些结构不仅需要移动而是复制,并且每个副本都限制在其区域内。

但确实可以将找到的cut and paste example OP与OP已经考虑过的pdfSweep模块组合到一个解决方案中,以防止文本在旧位置被选择,例如像这样:

public void moveCleanSection(PdfReader pdfReader, String targetFile, int page, Rectangle from, Rectangle to) throws IOException
{
    LicenseKey.loadLicenseFile("itextkey-multiple-products.xml");

    ByteArrayOutputStream interimMain = new ByteArrayOutputStream();
    ByteArrayOutputStream interimPage = new ByteArrayOutputStream();
    ByteArrayOutputStream interimSection = new ByteArrayOutputStream();

    try (   PdfDocument pdfMainDocument = new PdfDocument(pdfReader);
            PdfDocument pdfPageDocument = new PdfDocument(new PdfWriter(interimPage)) )
    {
        pdfMainDocument.setCloseReader(false);
        pdfMainDocument.copyPagesTo(page, page, pdfPageDocument);
    }

    try (   PdfDocument pdfMainDocument = new PdfDocument(pdfReader, new PdfWriter(interimMain));
            PdfDocument pdfSectionDocument = new PdfDocument(new PdfReader(new ByteArrayInputStream(interimPage.toByteArray())), 
            new PdfWriter(interimSection))  )
    {

        List<PdfCleanUpLocation> cleanUpLocations = new ArrayList<PdfCleanUpLocation>();
        cleanUpLocations.add(new PdfCleanUpLocation(page, from, null));
        cleanUpLocations.add(new PdfCleanUpLocation(page, to, null));

        PdfCleanUpTool cleaner = new PdfCleanUpTool(pdfMainDocument, cleanUpLocations);
        cleaner.cleanUp();

        cleanUpLocations = new ArrayList<PdfCleanUpLocation>();
        Rectangle mediaBox = pdfSectionDocument.getPage(1).getMediaBox();

        if (from.getTop() < mediaBox.getTop())
            cleanUpLocations.add(new PdfCleanUpLocation(1, new Rectangle(mediaBox.getLeft(), from.getTop(), mediaBox.getWidth(), mediaBox.getTop() - from.getTop()), null));
        if (from.getBottom() > mediaBox.getBottom())
            cleanUpLocations.add(new PdfCleanUpLocation(1, new Rectangle(mediaBox.getLeft(), mediaBox.getBottom(), mediaBox.getWidth(), from.getBottom() -  mediaBox.getBottom()), null));
        if (from.getLeft() > mediaBox.getLeft())
            cleanUpLocations.add(new PdfCleanUpLocation(1, new Rectangle(mediaBox.getLeft(), mediaBox.getBottom(), from.getLeft() - mediaBox.getLeft(), mediaBox.getHeight()), null));
        if (from.getRight() < mediaBox.getRight())
            cleanUpLocations.add(new PdfCleanUpLocation(1, new Rectangle(from.getRight(), mediaBox.getBottom(), mediaBox.getRight() - from.getRight(), mediaBox.getHeight()), null));

        cleaner = new PdfCleanUpTool(pdfSectionDocument, cleanUpLocations);
        cleaner.cleanUp();
    }

    try (   PdfDocument pdfSectionDocument = new PdfDocument(new PdfReader(new ByteArrayInputStream(interimSection.toByteArray())));
            PdfDocument pdfMainDocument = new PdfDocument(new PdfReader(new ByteArrayInputStream(interimMain.toByteArray())), new PdfWriter(targetFile)) )
    {
        float scale = Math.min(to.getHeight() / from.getHeight(), to.getWidth() / from.getWidth());
        pdfSectionDocument.getPage(1).setMediaBox(from);
        PdfFormXObject pageXObject = pdfSectionDocument.getFirstPage().copyAsFormXObject(pdfMainDocument);
        PdfPage pdfPage = pdfMainDocument.getPage(page);
        PdfCanvas pdfCanvas = new PdfCanvas(pdfPage);
        pdfCanvas.addXObject(pageXObject, scale, 0, 0, scale, (to.getLeft() - from.getLeft() * scale), (to.getBottom() - from.getBottom() * scale));
    }
}

(来自MoveSectionCleanly.java

注意:由于pdfSweep的性质,源区域边界上的文本将从源文件的副本和副本中删除。