是否可以将 JComponent 及其所有子组件绘制到另一个组件



我有一个名为 GridPanel 的扩展JPanel类。它使您可以将图像从JList拖放到其中。GridPanel可让您使用鼠标拖动图像并根据需要重新排列它们。我感兴趣的是制作GridPanel组件的缩略图视图。

如果我理解正确,将JScrollPane的视图设置为GridPanel会使GridPanel成为JViewPort的孩子,而成为JScrollPane的孩子。目前GridPanel已经被设定为JScrollPane的观点,我很确定GridPanel不可能有两个父母。所以我不能让两个组件共享同一个视图,但我真的只需要缩略图视图来绘制GridPanel的缩放视觉副本。

这就引出了我的问题。是否可以复制GridPanel绘制的内容,但将其绘制在完全独立的组件上?

这是我尝试过的一个例子,以防我不被理解。

import java.awt.Color;
import java.awt.Dimension;
import java.awt.FlowLayout;
import java.awt.GridLayout;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
public class TestMain {
public static void main(String[] a) {
    Color[] colors = new Color[]{Color.green, Color.red, Color.blue, Color.yellow};

    final JFrame frame = new JFrame();
    frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    frame.setLocationByPlatform(true);
    JPanel content = new JPanel();
    frame.setContentPane(content);
    content.setLayout(new FlowLayout());
    //using variable names to try and help relate to my question
    final JPanel gridPanel = new JPanel();
    gridPanel.setPreferredSize(new Dimension(200,200));
    gridPanel.setLayout(new GridLayout(2, 2));
    JLabel label;
    for (Color c: colors){
        label = new JLabel();
        label.setOpaque(true);
        label.setBackground(c);
        gridPanel.add(label);
    }
    JScrollPane gridScroll = new JScrollPane(gridPanel);
    final JScrollPane thumbnailScroll = new JScrollPane();
    thumbnailScroll.setPreferredSize(new Dimension(200,200));
    JButton tryThumbnailView = new JButton("Activate Thumbnail");
    tryThumbnailView.addActionListener(new ActionListener(){
        @Override
        public void actionPerformed(ActionEvent arg0) {
            thumbnailScroll.setViewportView(gridPanel);
            frame.repaint();
        }
    });
    content.add(tryThumbnailView);
    content.add(gridScroll);
    content.add(thumbnailScroll);
    frame.pack();
    frame.setVisible(true);
    }
}

我希望发生的是两个组件显示同一组彩色JLabels,而不复制这些JLabels

您所

要做的就是"获取一个组件的渲染结果"并重用它。这需要:

  1. 截取渲染过程 - 覆盖油漆(图形(方法
  2. 创建自己的图像以渲染到
  3. 将图像渲染为原始图形
  4. 将图像渲染到其他组件 - 覆盖油漆(图形(方法

前三个步骤可以像这样完成:

@Override
public void paint(Graphics originalGraphics) {
  GraphicsEnvironment e = GraphicsEnvironment.getLocalGraphicsEnvironment();
  GraphicsConfiguration c = e.getDefaultScreenDevice().getDefaultConfiguration();
  //create own image to paint to
  BufferedImage image = c.createCompatibleImage(getWidth(), getHeight());
  Graphics2D reusableGraphics = image.createGraphics();
  //let it paint into our graphics
  super.paint(reusableGraphics);
  // draw image on this component
  originalGraphics.drawImage(image, 0, 0, null);
  // draw image on other component
  otherComponent.setMirrorImage(image);
}

在其他组件中,您必须保存图像并在需要时绘制它:

@Override
public void paint(Graphics g) {
  if (mirroredImage == null) {
    super.paintAll(g);
  } else {
    g.drawImage(mirroredImage, 0, 0, getWidth() * 3 / 4, getHeight() * 3 / 4, null);
  }
}
public void setMirrorImage(BufferedImage mirroredImage) {
  this.mirroredImage = mirroredImage;
  repaint();
}

你可以看看这里的完整例子

您可以

像这样覆盖paintComponent

protected void paintComponent(Graphics g) {
    super.paintComponent(g);
    Graphics2D g2d = (Graphics2D) g;
    g2d.scale(scaleFactor, scaleFactor);
    gridPanel.paintComponenet(g);
}

这假设gridPanel要么覆盖paintComponent作为public,要么此类必须与GridPanel位于同一包中。

最新更新