禁用 JFrame 中的背景绘制以正确显示 Aero (DWM) 效果

2024-03-28

我在 Java Windows 上使用 Windows Vista/7 的 DWM 功能时遇到问题。我想让我的框架背景使用 Aero 风格。执行此操作的 Windows API 由函数提供DwmExtendFrameIntoClientArea in the dwmapi图书馆。我已经成功地通过 JNA 正确调用了该过程,并且它执行了它应该执行的操作(例如,您可以看到,在调整框架大小时,在下次重新绘制之前,您会在尚未绘制的区域中看到适当的空气动力学效果,参见附图)。

但在某个地方(我不知道在哪里)背景被绘制在 Aero 效果上,并且效果丢失了。

我已经尝试过的:

  • 使用自定义ContentPane不透明度设置为false
  • 设置不透明度LayeredPaneRootPane为假
  • Using a Frame代替JFrame
  • 设置背景颜色JFrame/ContentPane黑色/全透明
  • Use setLayersOpaque及其自定义变体,请参阅第一个答案以获取更多详细信息

到目前为止,我无法成功删除该背景。这是 AWT/Swing 的限制吗?如何删除该背景或正确使用 Aero 效果?

非常感谢您的帮助。

截屏

这是一个没有任何内容的框架的屏幕截图,将 RootPane、LayeredPane 和 ContentPane 的不透明度设置为 false。我在调整大小时做得很快。您会看到该效果已正确应用于 Java 尚未绘制的区域。

http://i55.tinypic.com/v614qo.png http://i55.tinypic.com/v614qo.png(作为新用户,我无法直接发布图像......)

奇怪的行为

经过进一步调查,我发现了以下奇怪的行为。如果窗口大小为 150x150 或以下,内容将透明显示。这对于普通的窗口组件来说是非常棘手的。如果您通过覆盖直接在框架上绘画paint()方法一切都绘制为半透明。另外,坐标系似乎有点偏离,它显示为坐标系的零点JFrame设置为窗口的实际零点。因此,Swing 尝试绘制窗口边框实际所在的区域,而该区域当然是不可见的。

看这个截图:http://d-gfx.kognetwork.ch/java_aero_bug.png http://d-gfx.kognetwork.ch/java_aero_bug.png

示例代码

这是我使用的代码。

需要jna.jar and platform.jar。可从 JNA 主页获取。

import com.sun.jna.Function;
import com.sun.jna.Native;
import com.sun.jna.NativeLibrary;
import com.sun.jna.Structure;
import com.sun.jna.platform.win32.WinDef.HWND;
import com.sun.jna.platform.win32.WinNT.HRESULT;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.UIManager;

public class AeroFrame extends JFrame {

    public AeroFrame() {
        setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

        JLabel label = new JLabel("Testlabel");
        label.setOpaque(false);

        add(label);

        pack();

        enableAeroEffect();
    }

    private void enableAeroEffect() {
        NativeLibrary dwmapi = NativeLibrary.getInstance("dwmapi");
        HWND aeroFrameHWND = new HWND(Native.getWindowPointer(this));
        MARGINS margins = new MARGINS();
        margins.cxLeftWidth = -1;
        margins.cxRightWidth = -1;
        margins.cyBottomHeight = -1;
        margins.cyTopHeight = -1;
        //DwmExtendFrameIntoClientArea(HWND hWnd, MARGINS *pMarInset)
        //http://msdn.microsoft.com/en-us/library/aa969512%28v=VS.85%29.aspx
        Function extendFrameIntoClientArea = dwmapi.getFunction("DwmExtendFrameIntoClientArea");
        HRESULT result = (HRESULT) extendFrameIntoClientArea.invoke(HRESULT.class,
                new Object[] { aeroFrameHWND, margins});
        if(result.intValue()!=0)
            System.err.println("Call to DwmExtendFrameIntoClientArea failed.");
    }

    /**
     * http://msdn.microsoft.com/en-us/library/bb773244%28v=VS.85%29.aspx
     */
    public class MARGINS extends Structure implements Structure.ByReference {
            public int cxLeftWidth;
            public int cxRightWidth;
            public int cyTopHeight;
            public int cyBottomHeight;
    }

    public static void main(String[] args) {
        try {
            UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
            JFrame.setDefaultLookAndFeelDecorated(true);

        } catch (Exception e) {
            e.printStackTrace();
        }
        new AeroFrame().setVisible(true);
    }

}

很好的问题。

最明显的答案是

WindowUtils.setWindowOpaque(this, false);

这为您提供了您想要的视觉效果,但不幸的是您无法单击窗口!

我尝试的第二件事是重写 Paint() 方法以执行与Window.paint()opaque标志设置为 false。那没有做任何事情。

然后我尝试使用反射。反思性设置Window.opaque为 true 给出与使用相同的结果WindowUtils.

最后,我尝试将其添加到enableAeroEffect():

Method m = null;
try {
    m = Window.class.getDeclaredMethod("setLayersOpaque", Component.class, Boolean.TYPE);
    m.setAccessible(true);
    m.invoke(null, this, false);
} catch ( Exception e ) {
    //TODO: handle errors correctly
} finally {
    if ( m != null ) {
        m.setAccessible(false);
    }
}

这有效!窗口仍然可以正确响应鼠标事件,但不会绘制背景。这张图有点小问题,但应该可以让你上路。

显然它很脆弱,因为它依赖于反射。如果我是你,我会看看什么Window.setLayersOpaque() does,并尝试以不依赖反射的方式复制它。

Edit: 在检查时setLayersOpaque方法,它似乎实际上可以归结为禁用透明组件上的双缓冲。从您的enableAeroEffect()方法,你就可以了:

//original source: Sun, java/awt/Window.java, setLayersOpaque(Component, boolean)
private static void setLayersTransparent(JFrame frame) {
    JRootPane root = frame.getRootPane();
    root.setOpaque(false);
    root.setDoubleBuffered(false);

    Container c = root.getContentPane();
    if (c instanceof JComponent) {
        JComponent content = (JComponent) c;
        content.setOpaque(false);
        content.setDoubleBuffered(false);
    }
    frame.setBackground(new Color(0, 0, 0, 0));
}
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

禁用 JFrame 中的背景绘制以正确显示 Aero (DWM) 效果 的相关文章

随机推荐