我只想更改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中提取当前页面,以及如何仅更改那些页面的页面方向?
答案 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。
但是有一些限制:
1)一般而言,pdf文件不仅仅是页面上呈现的一组元素。在HtmlConverter.convertToPdf下,iText还可以处理其他一些功能(轮廓等)。但是,对于您来说,该方法似乎可以正常工作。
2)如您所见,我没有检查是否有一些表不是body标签的直接子代。实际上,为了在一般情况下正确执行,您需要修改循环:检查所有元素的子元素(可能使用AbstractElement#getChildren()),并在其中放置一些AreaBreaks。但这听起来像是另一个问题。