我有一个JPanel,我使用通常的'paintComponent(Graphics g)'方法绘制了许多自定义编写的JComponents。我使用JLayeredPane来控制自定义组件的显示顺序,如下所示:
public Class MyPanel extends JPanel {
private JLayeredPane layeredPane = new JLayeredPane();
private JComponent component1;
private JComponent component2;
public MyPanel() {
super();
setLayout(new BoxLayout(this, BoxLayout.PAGE_AXIS));
component1 = new CustomComponent1();
layeredPane.add (component1, new Integer(0));
component2 = new CustomComponent2();
layeredPane.add (component2, new Integer(1));
add (layeredPane);
}
public void resizePanel(Graphics g, int newWidth, int newHeight) {
component1.setBounds (f(x), f(y), f(newWidth), f(newHeight));
component2.setBounds (f(x), f(y), f(newWidth), f(newHeight));
}
public void paintComponent(Graphics g) {
if ((getWidth() != oldWidth) || (getHeight() != oldHeight)) {
oldWidth = getWidth();
oldHeight = getHeight();
resizePanel (g, getWidth(), getHeight());
}
super.paintComponent(g);
}
现在,我想将此面板导出为JPEG文件,但大小不同。当我使用以下代码时,它成功创建/导出所需大小的JPEG文件,但它也将我的屏幕图像版本的面板更新为新的大小!糟糕!
public void export(File file, int width, int height)
throws IOException
{
BufferedImage scaledImage = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
Graphics2D g2 = scaledImage.createGraphics();
resizePanel (g2, width, height);
super.paintComponent (g2);
try {
OutputStream out = new FileOutputStream(file);
JPEGImageEncoder encoder = JPEGCodec.createJPEGEncoder(out);
encoder.encode(scaledImage);
out.close();
} catch (FileNotFoundException e) {
throw new IOException ("Unable to export chart to ("
+ file.getAbsolutePath() + "): " + e.getLocalizedMessage());
} finally {
g2.dispose();
}
}
如何“绘制”适合导出的图像,但实际上不会导致显示此新图像?
谢谢!
好吧,我又回到了这个问题.....
我正在绘制的场景包含一些文本,现在最终用户希望以“纵向”宽高比导出图形。由于我不是在新尺寸中重新绘制场景而只是缩放图像,因此会导致文本被水平压扁。
无论如何?
答案 0 :(得分:1)
根本不要调整面板的大小;只需创建一个BufferedImage
,它是面板的当前大小并绘制它。获得快照后,使用BufferedImage
或drawImage()
在AffineTransformOp
秒内将其缩放到所需尺寸。
附录:AffineTransformOp
的一个优点是控制旋转,插值,比例和纵横比:
BufferedImage image = ...
int w = image.getWidth();
int h = image.getHeight();
AffineTransform scaleTransform = new AffineTransform();
// last-in-first-applied: rotate, scale
scaleTransform.scale(scaleX, scaleY);
scaleTransform.rotate(Math.PI / 2, w / 2, h / 2);
AffineTransformOp scaleOp = new AffineTransformOp(
scaleTransform, AffineTransformOp.TYPE_BILINEAR);
BufferedImage new = scaleOp.filter(image, null);
答案 1 :(得分:1)
看起来是第二个出色的解决方案。我的最终解决方案如下:
public void export(File file, int width, int height)
throws IOException
{
BufferedImage scaledImage = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
Graphics2D g2 = scaledImage.createGraphics();
g2.scale(((double) width)/((double)getWidth()), ((double) height)/((double)getHeight()));
paintComponents(g2);
try {
ImageIO.write(scaledImage, "jpg", file);
} catch (FileNotFoundException e) {
throw new IOException ("Unable to export chart to ("
+ file.getAbsolutePath() + "): " + e.getLocalizedMessage());
} finally {
g2.dispose();
}
}
答案 2 :(得分:0)
嗯,唯一可以让所有内容看起来都不错的方法(例如,在光栅化之后没有缩放文本)就是使用新的大小在新的图形上下文中绘制图像。这是我的新'export()'方法。
public void export(File file, final int width, final int height)
throws IOException
{
BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_ARGB);
final Graphics2D g2 = image.createGraphics();
// Must wait for the bloody image to be drawn as Swing 'paint()' methods
// merely schedule painting events. The 'draw()' below may not complete
// the painting process before the 'write()' of the image is performed.
// thus, we wait....
try {
SwingUtilities.invokeAndWait(new Runnable() {
public void run() {
draw (g2, new Rectangle (0, 0, width, height));
}
});
ImageIO.write(image, "png", file);
} catch (Exception e) {
throw new IOException ("Unable to export chart to ("
+ file.getAbsolutePath() + "): " + e.getLocalizedMessage());
} finally {
g2.dispose();
}
}
上面调用的'draw()'方法在我的所有子组件上调用'setbounds()'方法,将它们绘制到新的缓冲图像上。一点工作,但结果正是我需要的。