我有一个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);
这使得面板变大,但它停在大约一页半(1100px):
JPanel c = createPanel();
JFrame f = new JFrame();
f.setContentPane(c);
f.pack();
print(c);
我在这里没有排列。有没有人知道(a)如何更改操作系统最大帧大小,(b)如何布局和绘制离屏组件,或(c)如何直接打印Swing组件,而不必绘制它(?) 。感谢帮助。
答案 0 :(得分:8)
使用自定义面板的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()
将内容划分为页面。
答案 1 :(得分:1)
您必须在没有滚动窗格的情况下打印组件。滚动窗格将始终只打印可见内容。
如果您无法访问createPanel()方法,并且此方法的面板仅包含滚动窗格,您可以通过以下方式获取滚动窗格包含的组件:
JPanel panel = createPanel();
print(((JScrollPane)panel.getComponents()[0]).getViewport());
但如果您的数据格式为表格,您还可以使用JTable。 JTable有一个强大的print()方法。 更多信息来自http://download.oracle.com/javase/tutorial/uiswing/misc/printtable.html
答案 2 :(得分:1)
由于JScrollPane中的组件可以具有任意大小,即使它可以显示,我的解决方案是试试这个:
JPanel c = createPanel();
JFrame f = new JFrame();
f.getContentPane().add(new JScrollPane(c));
f.pack();
print(c);
这样我就可以验证JPanel,而不会将其大小限制为JFrame的最大大小。它还具有“无限分辨率”外观,可以直接打印组件所获得的字体和内容,而不需要像trashgod那样的双缓冲。
答案 3 :(得分:0)
您在问题中提到,您可能需要打印一个不可见的组件。这是如何工作的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