11

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

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

我已经尝试过的:

  • 使用ContentPane不透明度设置为的自定义false
  • 将 theLayeredPane和 the的不透明度设置RootPane为 false
  • 使用 aFrame而不是 aJFrame
  • 将/的背景颜色设置JFrameContentPane黑色/完全透明
  • 使用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);
    }

}
4

1 回答 1

3

好问题。

最明显的答案是

WindowUtils.setWindowOpaque(this, false);

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

我尝试的第二件事是重写 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);
    }
}

这行得通!窗口仍能正确响应鼠标事件,但未绘制背景。这幅画有点小故障,但应该能让你上路。

显然它很脆弱,因为它依赖于反射。如果我是你,我会看看是什么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));
}
于 2010-11-03T14:46:42.863 回答