未修饰的JFrame阴影

时间:2013-09-30 22:52:10

标签: java swing jframe shadow

如何为未修饰的jframe添加阴影?

根据我在网上找到的内容,您可以将jframe添加到另一个黑色半透明窗口以产生阴影效果。 或者以某种方式将类似的东西应用于JFrame:

    Border loweredBorder = new EtchedBorder(EtchedBorder.LOWERED);
    setBorder(loweredBorder);

无论哪种方式,我只是想知道最好的方法,或者可能是一种完全不同的方式来获得相同的效果,比如从另一个类而不是jframe扩展。 我是Java的新手,所以我可能会走错方向,所以任何建议都值得赞赏。

2 个答案:

答案 0 :(得分:8)

基本上,你需要制作一系列图层。

  • JFrame
  • ShadowPanel
  • 和内容......

enter image description here

import java.awt.AlphaComposite;
import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Dimension;
import java.awt.EventQueue;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.GridBagLayout;
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 ShadowWindow {

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

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

                JFrame frame = new JFrame("Testing");
                frame.setUndecorated(true);
                frame.setBackground(new Color(0, 0, 0, 0));
                frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
                frame.setContentPane(new ShadowPane());

                JPanel panel = new JPanel(new GridBagLayout());
                panel.add(new JLabel("Look ma, no hands"));

                frame.add(panel);
                frame.pack();
                frame.setLocationRelativeTo(null);
                frame.setVisible(true);
            }
        });
    }

    public class ShadowPane extends JPanel {

        public ShadowPane() {
            setLayout(new BorderLayout());
            setOpaque(false);
            setBackground(Color.BLACK);
            setBorder(new EmptyBorder(0, 0, 10, 10));
        }

        @Override
        public Dimension getPreferredSize() {
            return new Dimension(200, 200);
        }

        @Override
        protected void paintComponent(Graphics g) {
            super.paintComponent(g);
            Graphics2D g2d = (Graphics2D) g.create();
            g2d.setComposite(AlphaComposite.SrcOver.derive(0.5f));
            g2d.fillRect(10, 10, getWidth(), getHeight());
            g2d.dispose();
        }
    }
}

答案 1 :(得分:0)

MadProgammer,

frame.setBackground(new Color(0, 0, 0, 0));

如果您也不使用,那就没用了

frame.setOpacity(1.0f);

将默认关闭操作设置为EXIT_ON_CLSE很不好,应用程序应该在两个进程结束时自然关闭,因此DISPOSE_ON_CLOSE是正确的方法,保留另一个来“修复”错误。

frame.setContentPane(new ShadowPanel());

如果您还(和之后)使用,则无用:

frame.add(panel);

ShadowPanel的构造函数应以以下内容开头:

super(new GridBagLayout());

代替:

setLayout(new GridBagLayout());

此外,使用GridBagLayout仅添加单个组件有点不成比例,不是吗? GridLayout(1,1)甚至是惰性的BorderLayout呢?

每次调用时以首选大小返回一个新的Dimension(很多都是由机器完成的,在很多操作中,例如重画等)非常昂贵。您应该返回在构造函数或类中定义的变量。

为什么要覆盖getPreferredSize()而不是getBackground()或isOpaque()?

GraphicsD应该放置在try块中,以将其处置在finally块中。