2

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

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

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

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

4

2 回答 2

8

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

  • JFrame
  • ShadowPanel
  • 和内容...

在此处输入图像描述

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();
        }
    }
}
于 2013-10-01T00:34:59.820 回答
0

疯狂程序员,

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 块中处理它。

于 2020-11-02T04:57:46.267 回答