【问题标题】:Printing a large Swing component打印大型 Swing 组件
【发布时间】:2011-10-25 00:06:40
【问题描述】:

我有一个带有 JScrollPane 内的自定义表格的 Swing 表单(它只是一个 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 组件,而不必绘制它(?) .感谢您的帮助。

【问题讨论】:

  • d) 为什么要打印这个? JScrollPane 通过滚动帮助您在屏幕上 看到这个大东西。您能否为自定义表格的数据定义不同的视图,使其便于打印?
  • @Mario C 在右侧我看到一些相关的帖子,也有链接/教程如何打印 2D 图形和文本 JComponents,如果我理解正确你的问题是计算纸张的 DPI 与像素(总是) 来自可见容器(JScrollPane 内容未知)
  • @mKorbel 我对不完整的信息没有任何问题 - 我可以很好地获得组件的大小。问题在于 painting - 我无法绘制任何无法显示的组件(不足为奇)。
  • @Atreys 的想法是我可以直接获取表单并打印它,而不必创建另一个对打印机友好的表单版本。在任何情况下,出于同样的原因,将 JScrollPane 的内容直接替换到面板中也不起作用。替换后,内容窗格会很大,当它在 JFrame 中可显示时会被压缩到一定大小(即使布局中没有给出任何内容)。

标签: java swing printing


【解决方案1】:

使用自定义面板的paint() 方法,将内容呈现为BufferedImage

附录:这是一个更完整的方法示例,它只是将组件缩放一半。您需要在实际应用中保留纵横比。

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://stackoverflow.com/questions/7026822 */
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();
            }
        });
    }
}

如图here 所示,您可以缩放渲染以适应目标的MediaPrintableArea,或使用getSubimage() 将内容按需要分成页面。

【讨论】:

  • getSize() 方法不起作用,除非内容已可显示,一旦显示,框架的内部大小限制在达到完整大小之前将其停止。如果我手动设置所有子组件的大小(没有我的布局管理器的好处),我不确定是否可以让绘制工作,但这是一个可怕的前景。
  • 对,我应该使用getPreferredSize() 并安排缩放之后 setVisible();我已经扩展了上面的例子。
  • 是的,这看起来可行。我缺少的部分是您必须通过JScrollPane 对连接到主框架的组件进行映像。如果您尝试使用 JFrame 的直接后代的任何组件(就像它的内容窗格,就像我之前做的那样),它不会被允许变得任意大。
  • 是的,帧的大小仅限于主机的重量级对等方可用的内存,可能在视频 RAM 中;相比之下,轻量级组件的尺寸要大得多。根据您的内容,您可能会修改插值参数。
【解决方案2】:

您必须在没有滚动窗格的情况下打印组件。滚动窗格将始终只打印可见内容。

如果您没有访问 createPanel() 方法,并且该方法的面板仅包含滚动窗格,则可以通过以下方式获取滚动窗格包含的组件:

JPanel panel = createPanel();
print(((JScrollPane)panel.getComponents()[0]).getViewport());

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

【讨论】:

  • createPanel() 创建的对话框实际上包含了许多其他的字段和框,但是中间有一个滚动窗格占据了大部分空间。我想过提取滚动窗格的视口并打印它,但随后我失去了对话框的其余部分。但是(测试)它确实实际上在 viewpot 中打印了 JPanel 的所有 3000 像素高度,这给了我另一个想法!
【解决方案3】:

由于 JScrollPane 中的组件可以具有任意大小,即使它可以显示,我的解决方案是尝试这个:

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

这样我就可以验证 JPanel 而不会将其大小限制为 JFrame 的最大大小。它还具有“无限分辨率”外观,您可以直接打印组件获得的字体和内容,而无需像垃圾神建议的那样进行双重缓冲。

【讨论】:

  • 你不需要getContentPane(); add()JFrame 转发。
  • 当然,但我不喜欢使用 JFrame 自己的 add() 方法作为惯例,因为这可能会让那些没有意识到它们是同义词的人感到困惑(这更清楚地显示实际行为)。
  • print() 方法需要 Graphics 作为输入。你怎么能把它传递给 JPanel?
  • 这是我自己的 print(Component) 方法,在 OP 中给出。
【解决方案4】:

您在问题中提到,也许您需要打印一个不可见的组件。这是如何工作的,shown here 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();
    }
}

如果您还需要打印多页,可以将此解决方案与我的answer on how to print a Component to multiple pages结合使用

【讨论】:

    猜你喜欢
    • 2011-12-07
    • 1970-01-01
    • 1970-01-01
    • 2012-11-04
    • 2017-04-08
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多