16

我可以像这样插入简单的文本:

document = new PDDocument();
page = new PDPage(PDPage.PAGE_SIZE_A4);
document.addPage(page);
PDPageContentStream content = new PDPageContentStream(document, page);
content.beginText();
content.moveTextPositionByAmount (10 , 10);
content.drawString ("test text");
content.endText();
content.close();

但是如何使用 width 属性创建类似于 HTML 的段落?

<p style="width:200px;">test text</p>
4

2 回答 2

25

警告:此答案适用于旧版本的 PDFBox,并且依赖于已弃用的功能。有关更多详细信息,请参阅下面的评论。

根据这个答案,不可能在某些文本中插入换行符并让 PDF 正确显示(无论是使用 PDFBox 还是其他东西),所以我相信自动换行一些文本以适应某个宽度也可能是它不能自动做。(此外,有很多方法可以包装文本 - 仅整个单词,将它们分成较小的部分等)

This answer to another question (about centering a string) 提供了一些关于如何自己执行此操作的指示。假设您编写了一个函数possibleWrapPoints(String):int[]来列出文本中可能发生自动换行的所有点(不包括“零”,包括“文本长度”),一种可能的解决方案可能是:

PDFont font = PDType1Font.HELVETICA_BOLD; // Or whatever font you want.
int fontSize = 16; // Or whatever font size you want.
int paragraphWidth = 200;
String text = "test text";

int start = 0;
int end = 0;
int height = 10;
for ( int i : possibleWrapPoints(text) ) {
    float width = font.getStringWidth(text.substring(start,i)) / 1000 * fontSize;
    if ( start < end && width > paragraphWidth ) {
        // Draw partial text and increase height
        content.moveTextPositionByAmount(10 , height);
        content.drawString(text.substring(start,end));
        height += font.getFontDescriptor().getFontBoundingBox().getHeight() / 1000 * fontSize;
        start = end;
    }
    end = i;
}
// Last piece of text
content.moveTextPositionByAmount(10 , height);
content.drawString(text.substring(start));

的一个示例possibleWrapPoints,它允许在不属于单词(reference)的任何点进行换行,可能是:

int[] possibleWrapPoints(String text) {
    String[] split = text.split("(?<=\\W)");
    int[] ret = new int[split.length];
    ret[0] = split[0].length();
    for ( int i = 1 ; i < split.length ; i++ )
        ret[i] = ret[i-1] + split[i].length();
    return ret;
}

更新:一些附加信息:

  • PDF 文件格式旨在在不同情况下看起来相同,您要求的功能在 PDF 编辑器/创建器中是有意义的,但在 PDF 文件本身中则不然。出于这个原因,大多数“低级”工具倾向于专注于处理文件格式本身,而放弃类似的东西。

  • 更高级别的工具OTOH通常有办法进行这种转换。一个例子是Platypus(不过对于 Python),它确实有创建段落的简单方法,并且依赖于较低级别的ReportLab函数来进行实际的 PDF 渲染。我不知道用于 PDFBox 的类似工具,但这篇文章提供了一些关于如何在 Java 环境中使用免费提供的工具将 HTML 内容转换为 PDF 的提示。我自己没有尝试过,但我在这里发帖,因为它可能有用(以防我上面的手工尝试还不够)。

于 2013-01-01T01:21:49.117 回答
7

我一直在努力将 Lukas 的答案与 mgibsonbr 结合起来,并达到了这一点。我认为对寻找开箱即用解决方案的人更有帮助。

private void write(Paragraph paragraph) throws IOException {
    out.beginText();
    out.appendRawCommands(paragraph.getFontHeight() + " TL\n");
    out.setFont(paragraph.getFont(), paragraph.getFontSize());
    out.moveTextPositionByAmount(paragraph.getX(), paragraph.getY());
    out.setStrokingColor(paragraph.getColor());

    List<String> lines = paragraph.getLines();
    for (Iterator<String> i = lines.iterator(); i.hasNext(); ) {
        out.drawString(i.next().trim());
        if (i.hasNext()) {
            out.appendRawCommands("T*\n");
        }
    }
    out.endText();

}

public class Paragraph {

    /** position X */
    private float x;

    /** position Y */
    private float y;

    /** width of this paragraph */
    private int width = 500;

    /** text to write */
    private String text;

    /** font to use */
    private PDType1Font font = PDType1Font.HELVETICA;

    /** font size to use */
    private int fontSize = 10;

    private int color = 0;

    public Paragraph(float x, float y, String text) {
        this.x = x;
        this.y = y;
        this.text = text;
    }

    /**
     * Break the text in lines
     * @return
     */
    public List<String> getLines() throws IOException {
        List<String> result = Lists.newArrayList();

        String[] split = text.split("(?<=\\W)");
        int[] possibleWrapPoints = new int[split.length];
        possibleWrapPoints[0] = split[0].length();
        for ( int i = 1 ; i < split.length ; i++ ) {
            possibleWrapPoints[i] = possibleWrapPoints[i-1] + split[i].length();
        }

        int start = 0;
        int end = 0;
        for ( int i : possibleWrapPoints ) {
            float width = font.getStringWidth(text.substring(start,i)) / 1000 * fontSize;
            if ( start < end && width > this.width ) {
                result.add(text.substring(start,end));
                start = end;
            }
            end = i;
        }
        // Last piece of text
        result.add(text.substring(start));
        return result;
    }

    public float getFontHeight() {
        return font.getFontDescriptor().getFontBoundingBox().getHeight() / 1000 * fontSize;
    }

    public Paragraph withWidth(int width) {
        this.width = width;
        return this;
    }

    public Paragraph withFont(PDType1Font font, int fontSize) {
        this.font = font;
        this.fontSize = fontSize;
        return this;
    }

    public Paragraph withColor(int color) {
        this.color = color;
        return this;
    }

    public int getColor() {
        return color;
    }

    public float getX() {
        return x;
    }

    public float getY() {
        return y;
    }

    public int getWidth() {
        return width;
    }

    public String getText() {
        return text;
    }

    public PDType1Font getFont() {
        return font;
    }

    public int getFontSize() {
        return fontSize;
    }

}
于 2013-12-21T10:11:34.000 回答