Java 创建具有背景图像的表单(JLayeredPane)

Java 创建具有背景图像的表单(JLayeredPane),java,swing,jframe,jlayeredpane,Java,Swing,Jframe,Jlayeredpane,我一直在努力做一些很简单的事情: 我想创建一个带有背景图像的表单(JTextField)。为了使表单不覆盖背景图像,我使用了JLayeredPane。我一直在尝试不同的东西,但似乎没有任何效果:出于某种原因,我要么只显示背景,要么只显示JTextField,但绝不同时显示两者。我的目标是要有一个永远不会改变的背景图像,只需在上面使用我的按钮/文本字段 package gestion; import java.awt.*; import javax.swing.*; @SuppressWar

我一直在努力做一些很简单的事情:

我想创建一个带有背景图像的表单(JTextField)。为了使表单不覆盖背景图像,我使用了JLayeredPane。我一直在尝试不同的东西,但似乎没有任何效果:出于某种原因,我要么只显示背景,要么只显示JTextField,但绝不同时显示两者。我的目标是要有一个永远不会改变的背景图像,只需在上面使用我的按钮/文本字段

package gestion;

import java.awt.*;

import javax.swing.*;

@SuppressWarnings("serial")
public class Main extends JFrame{
JLayeredPane layeredPane;
JPanel board;
JPanel background;


public Main(){
    super("Test");
    background = new JPanel();
    layeredPane = new JLayeredPane();
    board = new JPanel();

    // Creating frame with LayeredPane
    Dimension boardSize = new Dimension(1280, 1024);
    this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    this.setResizable(false);
    this.setSize(boardSize.width, boardSize.height);
    this.setVisible(true);
    this.setLocationRelativeTo(null);

    layeredPane.setPreferredSize( boardSize );
    this.add(layeredPane);


    // Add a background to the Layered Pane
    JLabel picLabel = new JLabel(new ImageIcon("background.jpg"));
    background.add(picLabel);
    background.setPreferredSize(boardSize);
    background.setBounds(0,0,boardSize.width, boardSize.height);
    layeredPane.add(background, JLayeredPane.DEFAULT_LAYER);

    // Add a JTextField
    final JTextField jtf = new JTextField("Default Value");

    Font police = new Font("Arial", Font.BOLD, 14);
    jtf.setFont(police);
    jtf.setPreferredSize(new Dimension(600, 800));
    background.setBounds(0,0,boardSize.width, boardSize.height);
    jtf.setForeground(Color.BLUE);

    board.add(jtf);
    layeredPane.add(board, JLayeredPane.PALETTE_LAYER);

}

public static void main(String[] args)
{
    new Main();
}
}


只有图像似乎出现了,出于某种原因(我最好的选择是黑魔法),JTextField不在那里。任何想法或帮助都将不胜感激!谢谢大家!

每当您依赖任何使用
null
布局的东西时(如
JLayeredPane
),都会遇到问题

你应该使用像

board.setBounds(new Rectangle(new Point(0, 0), board.getPreferredSize()));
layeredPane.add(board, JLayeredPane.PALETTE_LAYER);
设置
板的大小和位置

null
layouts还有一系列其他问题,这些问题只会让它们成为代码中的一大难题,最终浪费的时间比节省的时间还多

坦率地说,一个更简单、更实用的解决方案是创建一个自定义组件,它可以为您绘制背景图像,这样,您就可以使用您需要的任何布局管理器,而不会遇到这些问题。在任何人跳过我之前,使用一个布局管理器,你“可以”使用一个带有
JLayeredPane
,但是这会带来更多的问题,需要让组件重叠,以便背景层可以充当背景…只是更混乱

另外,在任何人跳过我之前,您可以使用
JLabel
作为背景组件,在其上设置布局管理器并将您的组件添加到其中,但是
JLabel
不会根据它包含的子组件计算它所需的大小,而是,使用
图标
文本
属性。如果您的背景图像足够大,这可能不是问题,但似乎始终是一个有待突破的弱点

import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Dimension;
import java.awt.EventQueue;
import java.awt.Graphics;
import java.awt.GridBagLayout;
import java.awt.image.BufferedImage;
import java.io.File;
import java.io.IOException;
import javax.imageio.ImageIO;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.UIManager;
import javax.swing.UnsupportedLookAndFeelException;
import javax.swing.border.EmptyBorder;

public class Test {

    public static void main(String[] args) {
        new Test();
    }

    public Test() {
        EventQueue.invokeLater(new Runnable() {
            @Override
            public void run() {
                try {
                    UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
                } catch (ClassNotFoundException | InstantiationException | IllegalAccessException | UnsupportedLookAndFeelException ex) {
                    ex.printStackTrace();
                }

                JFrame frame = new JFrame("Testing");
                frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
                frame.add(new TestPane());
                frame.pack();
                frame.setLocationRelativeTo(null);
                frame.setVisible(true);
            }
        });
    }

    public class TestPane extends JPanel {

        public TestPane() {
            setLayout(new BorderLayout());
            BackgroundPane bgPane = new BackgroundPane();
            bgPane.setLayout(new GridBagLayout());
            add(bgPane);

            try {
                BufferedImage bg = ImageIO.read(new File("C:\\Users\\shane\\Dropbox\\MegaTokyo\\thumnails\\megatokyo_omnibus_1_3_cover_by_fredrin-d4oupef.jpg"));
                bgPane.setBackgroundImage(bg);
            } catch (IOException ex) {
                ex.printStackTrace();
            }

            JLabel show = new JLabel("Bananas are yellow");
            show.setOpaque(true);
            show.setForeground(Color.RED);
            show.setBackground(Color.YELLOW);
            show.setBorder(new EmptyBorder(20, 20, 20, 20));
            bgPane.add(show);

        }

    }

    public class BackgroundPane extends JPanel {

        private BufferedImage img;

        @Override
        public Dimension getPreferredSize() {
            BufferedImage img = getBackgroundImage();

            Dimension size = super.getPreferredSize();
            if (img != null) {
                size.width = Math.max(size.width, img.getWidth());
                size.height = Math.max(size.height, img.getHeight());
            }

            return size;
        }

        public BufferedImage getBackgroundImage() {
            return img;
        }

        public void setBackgroundImage(BufferedImage value) {
            if (img != value) {
                BufferedImage old = img;
                img = value;
                firePropertyChange("background", old, img);
                revalidate();
                repaint();
            }
        }

        @Override
        protected void paintComponent(Graphics g) {
            super.paintComponent(g);
            BufferedImage bg = getBackgroundImage();
            if (bg != null) {
                int x = (getWidth() - bg.getWidth()) / 2;
                int y = (getHeight() - bg.getHeight()) / 2;
                g.drawImage(bg, x, y, this);
            }
        }

    }

}


此实现缺少自动缩放或重复等功能,但您知道了在哪里设置板组件的大小?使用
JPanel
作为背景(覆盖其
paintComponent
方法),然后向其添加组件会更简单。您可以/不应该调用
super.paintComponent()
在绘制背景之后?或者这不重要?重要的是,在调用
super.paintComponent
后,您需要绘制背景,因为它所做的一件事就是用背景色填充组件(这就是为什么调用它很重要的原因)