使图像在 JFrame 内容窗格中可滚动

2022-09-04 06:55:54

我正在尝试在JFrame的内容窗格中显示一个大图像。我想使图像或内容窗格可滚动,因为图像很大。我试图使用Jscrollpane做到这一点,并将其添加到内容窗格中,但它不起作用。做了一些寻找解决方案,但最终未能找到一个。有人可以指导我吗?我的代码在下面

最终环境.java

package environment;

import java.awt.*;
import java.net.URL;

import javax.swing.*;

public class FinalEnvironment{

public FinalEnvironment(){

    Image Eastlake;
    URL EastlakeURL = null;

    EastlakeURL = FinalEnvironment.class.getResource("/image1/eastlake_night.png");
    Eastlake = Toolkit.getDefaultToolkit().getImage(EastlakeURL);

    JFrame frame = new JFrame("UniCat World");
    frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    frame.setSize(800, 600);
    frame.setResizable(false);
    frame.setLocationRelativeTo(null);

    JMenuBar yellowMenuBar = new JMenuBar();
    Map map = new Map(800, 550, Eastlake);
    yellowMenuBar.setOpaque(true);
    yellowMenuBar.setBackground(Color.YELLOW);
    yellowMenuBar.setPreferredSize(new Dimension(800, 50));
    frame.setJMenuBar(yellowMenuBar);
    JScrollPane scroller = new JScrollPane(map);
    scroller.setAutoscrolls(true);
    scroller.setPreferredSize(new Dimension(800, 550));
    frame.getContentPane().add(scroller, BorderLayout.CENTER);


    frame.setSize(800, 600);
    frame.setVisible(true);
}

public static void main(String[] args){
    FinalEnvironment fe = new FinalEnvironment();
}
}

这是我的地图.java

package environment;

import java.awt.*;

import javax.swing.*;


public class Map extends JPanel{

    private int width;
    private int height;
    private Image img;

    public Map(int width, int height, Image img){

        this.width = width;
        this.height = height;
        this.img = img;
    }

    protected void  paintComponent(Graphics g)
    {
        super.paintComponents(g);
        Graphics2D g2d = (Graphics2D) g;
        g2d.drawImage(img,0,0,2624,1696,null);
    }

}

最后,我想把Jbuttons放在这张图片的顶部。我应该调用一个矩形并将其放在内容窗格中的图像顶部,然后使用Point来定位我的按钮,还是应该立即使用图像或组件本身来执行此操作?我需要该按钮能够在滚动图像时与图像同步,而不是在内容窗格中保持静态。

谢谢


答案 1

我会在这里做什么:

1.有一个面板(画布),唯一负责的是用覆盖的方法绘制独立于真实图像大小的给定图像paintComponent()

    super.paintComponent(g);
    g.drawImage(image, 0, 0, null);

2.确保画布首选大小等于图像实际大小。

3.有第二个面板,它将作为框架的内容窗格。

4.In 它,您将设置一个作为其中心。JScrollPane

5.In 滚动窗格视口将是步骤 1 中的组件。

6.从步骤1开始将按钮添加到画布面板。它将与图像一起滚动。

7.将内容窗格(步骤 3 中的面板)添加到框架,然后运行应用程序。

编辑:将按钮添加到画布的代码示例,该画布始终保持在其位置,与滚动位置或帧大小无关。

import java.awt.BorderLayout;
import java.awt.Dimension;
import java.awt.Graphics;
import java.awt.image.BufferedImage;
import java.io.IOException;
import java.net.URL;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.imageio.ImageIO;
import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.SwingUtilities;

public class ScrollImageTest extends JPanel {
    private static final long serialVersionUID = 1L;
    private BufferedImage image;
    private JPanel canvas;

    public ScrollImageTest() {
        try {
            this.image = ImageIO.read(new URL("http://interviewpenguin.com/wp-content/uploads/2011/06/java-programmers-brain.jpg"));
        }catch(IOException ex) {
            Logger.getLogger(ScrollImageTest.class.getName()).log(Level.SEVERE, null, ex);
        }

        this.canvas = new JPanel() {
            private static final long serialVersionUID = 1L;
            @Override
            protected void paintComponent(Graphics g) {
                super.paintComponent(g);
                g.drawImage(image, 0, 0, null);
            }
        };
        canvas.add(new JButton("Currently I do nothing"));
        canvas.setPreferredSize(new Dimension(image.getWidth(), image.getHeight()));
        JScrollPane sp = new JScrollPane(canvas);
        setLayout(new BorderLayout());
        add(sp, BorderLayout.CENTER);
    }

    public static void main(String[] args) {
        SwingUtilities.invokeLater(new Runnable() {
            @Override
            public void run() {
                JPanel p = new ScrollImageTest();
                JFrame f = new JFrame();
                f.setContentPane(p);
                f.setSize(400, 300);
                f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
                f.setVisible(true);
            }
        });
    }
}

答案 2

如果您使用尺寸设置地图的首选尺寸,该怎么办?例如,给 Map 下述方法:

// method in the Map class
@Override
public Dimension getPreferredSize() {
  return new Dimension(width, height);
}

这样,Map JPanel将占用必要的空间来显示整个图像。另外,为什么您在paintComponent方法中的drawImage方法具有较大的数?为什么不在那里使用宽度和高度呢?编辑1:或者甚至不要像Boro在他的答案中建议的那样指定图像大小(对他来说是1 +)。


推荐