如何使用PDFbox创建固定宽度的段落?

Qpi*_*Qpi 15 html paragraph pdfbox

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

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();
Run Code Online (Sandbox Code Playgroud)

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

<p style="width:200px;">test text</p>
Run Code Online (Sandbox Code Playgroud)

mgi*_*nbr 24

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

对另一个问题(关于字符串居中)的回答给出了一些关于如何自己做这个的指示.假设您编写了一个函数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));
Run Code Online (Sandbox Code Playgroud)

possibleWrapPoints允许包装在不属于单词(引用)的任何点的一个示例可以是:

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;
}
Run Code Online (Sandbox Code Playgroud)

更新:一些额外的信息:

  • PDF文件格式设计为在不同情况下看起来相同,您所请求的功能在PDF编辑器/创建者中是有意义的,但在PDF文件本身中则不然.出于这个原因,大多数"低级"工具倾向于专注于处理文件格式本身并留下类似的东西.

  • 更高级别的工具OTOH通常有进行此转换的手段.一个例子是鸭嘴兽(对于Python,虽然),该做的都创造段落的简单的方法,并依赖于较低水平ReportLab的功能做实际的PDF渲染.我不知道PDFBox的类似工具,但这篇文章提供了一些关于如何使用免费工具在Java环境中将HTML内容转换为PDF的提示.没有自己试过,但我在这里发帖,因为它可能有用(如果上面的手工制作尝试还不够).


Nac*_*oma 6

我一直在努力将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;
    }

}
Run Code Online (Sandbox Code Playgroud)