Печать большого компонента Swing
у меня есть форма 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);
у меня заканчиваются перестановки. Кто-нибудь знает (а) как изменить максимальный размер кадра ОС, (Б) как раскрасить и раскрасить компонент вне экрана или (в) как распечатать Компонент качания сразу, без покрасить его (?). Помощь приветствуется.
4 ответов
использование пользовательских панелей 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();
}
});
}
}
как показано здесь, вы можете масштабировать рендеринг, чтобы соответствовать MediaPrintableArea
, или использовать getSubimage()
в разделите содержимое на страницы по желанию.
вы должны распечатать компонент без scrollpane. scrollpane всегда будет печатать только видимое содержимое.
если у вас нет доступа к методу createPanel (), а панель из этого метода содержит только scrollpane, вы можете получить компонент, который содержит scrollpane, следующим образом:
JPanel panel = createPanel();
print(((JScrollPane)panel.getComponents()[0]).getViewport());
но если ваши данные форматируются как таблица, вы также можете использовать JTable. JTable имеет собственный мощный метод print (). более подробная информация по http://download.oracle.com/javase/tutorial/uiswing/misc/printtable.html
поскольку компонент в JScrollPane может иметь произвольный размер, даже после того, как он сделан отображаемым, мое решение-попробовать это:
JPanel c = createPanel();
JFrame f = new JFrame();
f.getContentPane().add(new JScrollPane(c));
f.pack();
print(c);
чтобы я мог проверить JPanel без ограничения размера до максимального размера JFrame. Он также имеет" неограниченное разрешение " смотреть на шрифты и вещи, которые вы получаете от печати компонентов непосредственно, без двойной буферизации, как trashgod предложил.
Вы упомянули в своем вопросе, что, возможно, вам нужно будет распечатать невидимый компонент. Как это работает показано здесь by 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();
}
}
Если вам еще нужно распечатать несколько страниц, вы можете объединить это решение с my ответ о том, как печатать Component
на нескольких страницах