JLayeredPane 中缺少 JPanel

Missing JPanel in JLayeredPane

我正在为 JPanel 制作叠加层,但我遇到了一个小问题,叠加层的透明度可以直接看到 JFrame

import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Dimension;
import java.awt.EventQueue;

import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JLayeredPane;
import javax.swing.JPanel;

@SuppressWarnings("serial")
public class GUI extends JFrame {
    private GUI() {
        super("Recorder Part");
        setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

        setLayout(new BorderLayout());
        JLayeredPane layers = new Overlay();
        add(layers);

        pack();

        getContentPane().setBackground(Color.GREEN);
        setVisible(true);
    }

    public static void main(String[] args) {
        EventQueue.invokeLater(() -> new GUI());
    }

    private class Overlay extends JLayeredPane {
        JPanel base;
        JPanel overlay;

        private Overlay() {
            setPreferredSize(new Dimension(800, 100));

            createBase();
            createOverlay();

            add(base, new Integer(0));
            add(overlay, new Integer(1));
        }


        private void createBase() {
            base = new JPanel();
            base.setPreferredSize(new Dimension(800, 100));
            base.setBounds(0, 0, 800, 100);
            base.setBackground(Color.BLUE);
            base.setOpaque(true);

            base.add(new JLabel("Hello"));
        }

        private void createOverlay() {
            overlay = new JPanel();
            overlay.setPreferredSize(new Dimension(800, 100));
            overlay.setBounds(0, 0, 800, 100);
            overlay.setBackground(new Color(255, 0, 0, 0));
        }
    }
}

我该如何解决这个问题,以便当名为 overlayJPanel 透明时,可以看到名为 baseJpanel 而不是 JFrame?


编辑

我发现只有当覆盖面板的一个尺寸大于或等于基础面板的尺寸时才会出现此问题。这可以通过调整本例中框架的大小来看出。

import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Dimension;
import java.awt.EventQueue;
import java.awt.event.ComponentAdapter;
import java.awt.event.ComponentEvent;

import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JLayeredPane;
import javax.swing.JPanel;

@SuppressWarnings("serial")
public class GUI extends JFrame {
    private GUI() {
        super("Recorder Part");
        setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

        setLayout(new BorderLayout());
        JLayeredPane layers = new Overlay();
        add(layers);

        pack();

        getContentPane().setBackground(Color.GREEN);
        setVisible(true);
    }

    public static void main(String[] args) {
        EventQueue.invokeLater(() -> new GUI());
    }

    private class Overlay extends JLayeredPane {
        JPanel base;
        JPanel overlay;

        private Overlay() {
            addComponentListener(new Resize());
            setPreferredSize(new Dimension(800, 100));

            createBase();
            createOverlay();

            add(base, new Integer(0));
            add(overlay, new Integer(1));
        }


        private void createBase() {
            base = new JPanel();
            base.setLocation(0, 0);
            base.setBackground(Color.BLUE);
            base.setOpaque(true);

            base.add(new JLabel("Hello"));
        }

        private void createOverlay() {
            overlay = new JPanel();
            overlay.setLocation(0, 0);
            overlay.setSize(new Dimension(800, 100));
            overlay.setBackground(new Color(255, 0, 0, 128));
        }

        private class Resize extends ComponentAdapter {
            @Override
            public void componentResized(ComponentEvent e) {
                System.out.println("Resized");
                base.setSize(new Dimension(getParent().getWidth(), getParent().getHeight()));
            }
        }
    }
}

这是不可取的,因为覆盖面板需要与基础面板大小完全相同。如果有任何帮助,我将不胜感激。

我感觉这是一个错误。更改边界,使其不完全与底部重叠,看看会发生什么:

overlay.setBounds(0, 0, 799, 100); // or (1, 0, 800, 100)

可能是某种优化,比如忽略被另一个组件完全遮挡的组件,但不考虑该优化的透明度 [:-|

编辑

这绝对是问题所在 - JComponent 的 paintChildren 会根据子项被其他子项完全遮挡的情况进行某种优化。

我找到了两个解决方案,第一个可能是正确的 - 将叠加层设置为非不透明:

overlay.setOpaque(false);

这样做的缺点是根本没有使用背景色。

第二个更像是一个解决方法 - 使 JLayeredPane return 为 optimizedDrawingEnabled 为真,这将阻止 JComponent 执行此操作:

private class Overlay extends JLayeredPane {

    /// ...

    @Override
    public boolean isOptimizedDrawingEnabled() {
        return true;
    }
}

但不确定这样做会导致什么停止工作,所以我更喜欢第一个解决方案!