更改结果PDF中仅某些页面的页面方向(由html创建)

时间:2019-01-16 11:01:19

标签: java itext7 html2pdf

我只想更改PDF文档中特定页面的页面方向。使用html2pdf从html模板创建PDF文档。它是这样的:如果页面(通常是表格)的内容太宽而无法以纵向显示,则以横向显示页面。

遵循[如何将页面旋转为横向且页面内容应以纵向iTextpdf格式] [1]

中的提示

[1]:how to rotate pages into landscape and page content should be in portrait iTextpdf我已经创建了自定义标签和TagWorker。

public class LandscapeTagWorker extends BodyTagWorker {

public LandscapeTagWorker(IElementNode element, ProcessorContext context) {
    super(element, context);
}

/**
 * @param element
 * @param context
 * @see com.itextpdf.html2pdf.attach.ITagWorker#processEnd(com.itextpdf.html2pdf.html.node.IElementNode, com.itextpdf.html2pdf.attach.ProcessorContext)
 */
@Override
public void processEnd(IElementNode element, ProcessorContext context) {
    super.processEnd(element, context);
    String value = element.getAttribute("value");
     if ( "true".equalsIgnoreCase(value) ) {
        PdfDocument doc = context.getPdfDocument();

        doc.setDefaultPageSize(doc.getDefaultPageSize().rotate());
     }
}

}

问题是:首先,这什么都不做。即使可行,我也不想更改整个文档的方向,而只是更改找到<landscape value="true">内容的页面的方向。

如何从ProcessorContext / PdfDocument中提取当前页面,以及如何仅更改那些页面的页面方向?

1 个答案:

答案 0 :(得分:0)

在下面的答案中,我将显示如何将类属性设置为横向的表按照您想要的方式进行处理。

首先,让我们为此类表创建一个自定义CSS应用程序。其目的是在表格标签上设置自定义属性(例如10001)。然后在布局期间,我们将检查该属性并决定是否更改页面大小。

    class CustomCssApplierFactory extends DefaultCssApplierFactory {
    @Override
    public ICssApplier getCustomCssApplier(IElementNode tag) {
        if ("table".equals(tag.name()) && "landscape".equals(tag.getAttribute(AttributeConstants.CLASS))) {
            return new CustomLandscapeCssApplier();
        }
        return null;
    }
}

class CustomLandscapeCssApplier extends TableTagCssApplier {
    @Override
    public void apply(ProcessorContext context, IStylesContainer stylesContainer, ITagWorker tagWorker) {
        super.apply(context, stylesContainer, tagWorker);
        IPropertyContainer container = tagWorker.getElementResult();
        if (null != container) {
            container.setProperty(10001, true);
        }
    }
}

现在,让我们将您的html 不是转换为pdf文件,而是转换为形成该html的 IElements列表 (您特别对最后一行感兴趣):

    PdfDocument pdfDocument = new PdfDocument(new PdfWriter(pdfDest));
    Document document = new Document(pdfDocument);

    ConverterProperties converterProperties = new ConverterProperties();
    converterProperties.setCssApplierFactory(new CustomCssApplierFactory());
    List<IElement> list = HtmlConverter.convertToElements(new FileInputStream(htmlSource), converterProperties);

现在让我们将这些元素一一添加到文档中,并检查当前元素(要添加的元素)是否具有魔术10001属性。如果没有,则只需添加元素。如果存在,让我们分页并在分页时设置下一页的大小:

        for (IElement element : list) {
        if (element instanceof IBlockElement) {
            if (!element.hasProperty(10001)) {
                document.add((IBlockElement) element);
            } else {
                document.add(new AreaBreak(new PageSize(PageSize.A4).rotate()));
                document.add((IBlockElement) element);
                document.getPdfDocument().setDefaultPageSize(PageSize.A4);
            }
        }
    }

如您在上面的代码段中所见,我将页面弄碎了(设置下一页的大小),添加了元素,然后再次设置下一个布局区域的大小,这样,下一页之后的下一页包含该元素将具有默认的pagesize。

结果就是这样: enter image description here

但是有一些限制:

1)一般而言,pdf文件不仅仅是页面上呈现的一组元素。在HtmlConverter.convertToPdf下,iText还可以处理其他一些功能(轮廓等)。但是,对于您来说,该方法似乎可以正常工作。

2)如您所见,我没有检查是否有一些表不是body标签的直接子代。实际上,为了在一般情况下正确执行,您需要修改循环:检查所有元素的子元素(可能使用AbstractElement#getChildren()),并在其中放置一些AreaBreaks。但这听起来像是另一个问题。