打印一个大型Swing组件

7
我有一个带有自定义表格的Swing表单在JScrollPane中(它只是一个JPanel,而不是JTable子类),我正在尝试将其打印出来。如果我只是将整个框架发送到打印机,滚动窗格会被截断,如果我将框架调整为滚动窗格内容的大小,某种内部障碍会阻止JFrame变得超过约1100像素高。

另一种选择是创建对话框的内容面板,而不将其附加到根JFrame上,因为在这种情况下JPanel的大小没有限制。但是要使组件布局并调整为其适当的大小,似乎我需要使面板可显示,这意味着至少将其添加到JFrame并调用JFrame.pack(),但是1100像素限制再次出现。

以下是打印组件的代码:

public static void print(final Component comp) {
    final float SCALE = .5f;
    PrinterJob job = PrinterJob.getPrinterJob();
    job.setPrintable(new Printable() {
        public int print(Graphics g, PageFormat pf, int page)
            throws PrinterException
        {
            if (page * pf.getImageableHeight() >= SCALE * comp.getHeight())
                return NO_SUCH_PAGE;
            ((Graphics2D)g).translate(pf.getImageableX(), pf.getImageableY()
               - page * pf.getImageableHeight());
            ((Graphics2D)g).scale(SCALE, SCALE);
            comp.printAll(g);
            return PAGE_EXISTS;
        }
    });
    if (job.printDialog())
        try { job.print(); }
        catch (PrinterException ex) {}
}

如果我这样做,组件将具有零大小:
JPanel c = createPanel(); // This JPanel has a JScrollPane in it with its
                          // preferredSize equal to that of its viewport component
                          // (which is not what I do to show the dialog normally)
print(c);

如果我这样做,组件的大小就是正确的,但由于子组件未被布局,因此打印出来会是纯灰色。
JPanel c = createPanel();
c.setSize(c.getPeferredSize());
print(c);

这些似乎没有什么区别:

JPanel c = createPanel();
c.validate();
c.revalidate();
c.repaint();
print(c);

这会使面板变得更大,但它会在大约一页半大小(1100像素)时停止:
JPanel c = createPanel();
JFrame f = new JFrame();
f.setContentPane(c);
f.pack();
print(c);

我的排列组合已经用尽了。有没有人知道以下内容中的任意一项:(a)如何更改操作系统的最大帧大小,(b)如何布局和绘制屏幕外组件,或(c)如何直接打印一个Swing组件,而不必绘制它?感谢您的帮助。


为什么要打印这个?JScrollPane可以通过滚动帮助您在屏幕上查看这个大东西。您能定义不同的视图来使您自定义表格的数据更适合打印吗? - Atreys
@Mario C 在右侧我看到一些相关的帖子,其中包含打印2D图形和文本JComponents的链接/教程。如果我正确理解了您的问题,那么它是关于如何从可见容器(具有未知内容的JScrollPane)计算纸张的DPI与像素(始终如此)。 - mKorbel
@mKorbel 我没有任何关于不完整信息的问题 - 我可以很好地获取我的组件的大小。问题在于绘图 - 我无法让任何不可显示的组件进行绘制(毫不奇怪)。 - Mario Carneiro
@Atreys 的想法是,我可以直接拿表单本身并打印它,而不必创建另一个适合打印的版本。无论如何,将 JScrollPane 替换为其内容直接放入面板也行不通,原因相同。替换后,内容面板会变得非常大,在 JFrame 中显示时会被压缩到大小(即使布局中没有任何弹性)。 - Mario Carneiro
4个回答

8
使用您自定义面板的paint()方法,将内容渲染到BufferedImage中。
附加说明:这里有一个更完整的示例,它只是将组件缩小了一半。在实际应用中,您需要保留纵横比。

enter image description here

import java.awt.Dimension;
import java.awt.EventQueue;
import java.awt.Graphics2D;
import java.awt.GridLayout;
import java.awt.geom.AffineTransform;
import java.awt.image.AffineTransformOp;
import java.awt.image.BufferedImage;
import javax.swing.ImageIcon;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JScrollPane;

/** @see https://dev59.com/9VrUa4cB1Zd3GeqPhCnK */
public class PanelPaint extends JPanel {

    private static final double SCALE = 0.5;

    public PanelPaint() {
        super(new GridLayout(0, 1));
        final MyPanel panel = new MyPanel();
        JScrollPane scroll = new JScrollPane(panel);
        scroll.getViewport().setPreferredSize(new Dimension(320, 240));
        this.add(scroll);
        EventQueue.invokeLater(new Runnable() {

            @Override
            public void run() {
                add(new JLabel(new ImageIcon(createImage(panel))));
            }
        });
    }

    private BufferedImage createImage(MyPanel panel) {
        Dimension size = panel.getPreferredSize();
        BufferedImage image = new BufferedImage(
            size.width, size.height, BufferedImage.TYPE_INT_ARGB);
        Graphics2D g2d = image.createGraphics();
        panel.paint(g2d);
        g2d.dispose();
        AffineTransform at = new AffineTransform();
        at.scale(SCALE, SCALE);
        AffineTransformOp scaleOp =
            new AffineTransformOp(at, AffineTransformOp.TYPE_BILINEAR);
        return scaleOp.filter(image, null);
    }

    private static class MyPanel extends JPanel {

        private static final int N = 16;

        public MyPanel() {
            super(true);
            this.setLayout(new GridLayout(N, N));
            for (int i = 0; i < N * N; i++) {
                this.add(new JLabel(String.valueOf(i) + " "));
            }
        }
    }

    private void display() {
        JFrame f = new JFrame("PanelPaint");
        f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        f.add(this);
        f.pack();
        f.setLocationRelativeTo(null);
        f.setVisible(true);
    }

    public static void main(String[] args) {
        EventQueue.invokeLater(new Runnable() {

            @Override
            public void run() {
                new PanelPaint().display();
            }
        });
    }
}

此处所示,您可以将渲染缩放以适应目标的MediaPrintableArea,或者使用getSubimage()根据需要将内容分成页面。


getSize() 方法只有在内容可显示后才能起作用,而一旦这样做了,框架的内部大小限制就会在达到全尺寸之前停止它。如果我手动设置所有子组件的大小(没有使用我的布局管理器),我不确定是否可以使 paint 方法工作,但这是一个令人恐惧的前景。 - Mario Carneiro
对,我应该使用 getPreferredSize(),并在 setVisible()之后安排缩放;我已经扩展了上面的例子。 - trashgod
是的,这看起来可以工作。我缺失的部分是你必须通过JScrollPane将组件连接到主框架。如果您尝试使用任何直接从JFrame继承的组件(例如其内容窗格,就像我之前所做的那样),它将不被允许变得任意大。 - Mario Carneiro
是的,框架的大小受限于主机的重量级对等体可用的内存,可能是视频RAM;相比之下,轻量级组件的大小要大得多。根据您的内容,您可能需要调整插值参数。 - trashgod

1
你需要打印没有滚动条的组件。滚动条只会打印可见内容。
如果你无法访问createPanel()方法,并且该方法返回的面板只包含滚动条,你可以按照以下方式获取滚动条所包含的组件:
JPanel panel = createPanel();
print(((JScrollPane)panel.getComponents()[0]).getViewport());

但如果您的数据以表格格式呈现,您也可以考虑使用JTable。JTable具有自己强大的print()方法。 更多信息请参见http://download.oracle.com/javase/tutorial/uiswing/misc/printtable.html


createPanel() 创建的对话框实际上包含许多其他字段和框,但中心有一个占据大部分空间的滚动窗格。我考虑过提取滚动窗格的视口并打印它,但这样我就失去了对话框的其余部分。但是(测试)它实际上会打印视口中 JPanel 的所有 3000 像素高度,这给了我另一个想法! - Mario Carneiro

1

由于JScrollPane中的组件大小可以是任意的,即使在它被显示后,我的解决方案是尝试这样做:

JPanel c = createPanel();
JFrame f = new JFrame();
f.getContentPane().add(new JScrollPane(c));
f.pack();
print(c);

这样我就可以验证JPanel而不会被限制在JFrame的最大尺寸上。它还具有“无限分辨率”的外观,字体和其他组件直接打印时不需要像trashgod建议的双缓冲。


2
你不需要使用 getContentPane()add() 方法已经被 JFrame 转发了。 - trashgod
同意,但是出于惯例,我更喜欢不使用JFrame自己的add()方法,因为对于那些没有意识到它们是同义词的人来说,这可能会造成困惑(这样可以更清楚地显示实际行为)。 - Mario Carneiro
print() 方法需要 Graphics 作为输入。你如何能够将 JPanel 传递给它? - jacen.garriss
这是我在原始帖中提供的自己的print(Component)方法。 - Mario Carneiro

0

你在问题中提到,可能需要打印一个不可见的组件。这是 这里Java42 演示的,我只是稍微改了一下:

public class PrintInvisible  {
    static class JPanelPrintable extends JPanel implements Printable
    {
        public int print(Graphics g, PageFormat pf, int page) throws PrinterException
        {
            if (page > 0) return Printable.NO_SUCH_PAGE;
            printAll(g);
            return Printable.PAGE_EXISTS;
        }

        public void print() throws PrinterException
        {
            PrinterJob job = PrinterJob.getPrinterJob();
            job.setPrintable(this);
            if (job.printDialog()) job.print();
        }

    };

    public static void main(String args[]) throws PrinterException {
        final JFrame f = new JFrame();
        f.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);
        f.setSize(400,400);
        final JPanelPrintable j = new JPanelPrintable();
        j.setLayout(new BorderLayout());
        j.add(new JButton("1111"),BorderLayout.NORTH);
        j.add(new JButton("2222"),BorderLayout.SOUTH);
        f.add(j);f.repaint();f.pack();
        //f.setVisible(true);
        j.print();
    }
}

如果你想要打印多个页面,可以将此解决方案与我的如何将组件打印到多个页面上的答案相结合。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接