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

Dan*_*odt 11 java swing jframe jna dwm

我在Java窗口上使用Windows Vista/7的DWM功能时遇到问题.我想让我的画面的背景使用Aero风格.Windows API来做到这一点是函数提供DwmExtendFrameIntoClientAreadwmapi库.我已经设法通过JNA正确调用该程序,它完成了应该做的事情(你可以看到,例如在调整框架大小时,在下一次重绘之前,你会看到尚未绘制的区域中的正确的空气动力效果,见附图).

但某处(我无法弄清楚在哪里)背景画在Aero效果上并且效果丢失了.

我已经尝试过的:

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

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

非常感谢您的帮助.

截图

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

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

奇怪的行为

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

请参阅此屏幕截图:http://d-gfx.kognetwork.ch/java_aero_bug.png

示例代码

这是我使用的代码.

需要jna.jarplatform.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);
    }

}
Run Code Online (Sandbox Code Playgroud)

Mar*_*ers 3

很好的问题。

最明显的答案是

WindowUtils.setWindowOpaque(this, false);
Run Code Online (Sandbox Code Playgroud)

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

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

然后我尝试使用反射。反射性地设置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);
    }
}
Run Code Online (Sandbox Code Playgroud)

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

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

编辑:在检查该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));
}
Run Code Online (Sandbox Code Playgroud)