在JLayeredPane中缺少JPanel

时间:2017-04-04 11:06:40

标签: java swing jpanel 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));
        }
    }
}

如何解决此问题,以便JPanel调用overlay透明时,可以看到Jpanel base,而不是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()));
            }
        }
    }
}

这是不合需要的,因为覆盖面板需要与基板的尺寸完全相同。我将不胜感激任何帮助。

1 个答案:

答案 0 :(得分:1)

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

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

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

编辑

这肯定是问题--JComponent的paintChildren根据被其他孩子完全遮挡的孩子进行某种优化。

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

overlay.setOpaque(false);

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

第二个更像是workarround - 使得JLayeredPane对optimizeDrawingEnabled返回true,这将阻止JComponent执行:

private class Overlay extends JLayeredPane {

    /// ...

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

但不确定可能会因此而停止工作,所以我更倾向于第一个解决方案!