【发布时间】:2011-11-16 05:51:26
【问题描述】:
以下 Swing 代码在我的机器或我同事的机器(所有 Windows XP 和 Java 6)上无法正常工作:
public class Test {
public static void main(String[] args) {
final JFrame frame = new JFrame();
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.setSize(800, 600);
frame.setLayout(new FlowLayout());
frame.add(new JButton(new AbstractAction("Maximize") {
@Override
public void actionPerformed(ActionEvent e) {
frame.setExtendedState((frame.getExtendedState() & Frame.MAXIMIZED_BOTH) == Frame.MAXIMIZED_BOTH ? JFrame.NORMAL : JFrame.MAXIMIZED_BOTH);
}
}));
frame.setUndecorated(true);
frame.setVisible(true);
}
}
它最大化窗口,但不考虑窗口任务栏(它填满屏幕)。如果你注释掉“frame.setUndecorated(true);”它似乎工作正常。
Javadoc 似乎暗示这应该有效。这是Java中的错误吗?它是否仅限于特定版本或 Windows 版本?有解决办法吗?
我见过few workarounds,但它们似乎不完整。我不能成为第一个用 Java 编写自己的框架装饰的人。
编辑: 下载 OpenJDK 并挖掘原生代码后,我发现 Java 正在调用 win32 函数 SetWindowPlacement 并在某些情况下更改 MINMAXINFO 以获得正确的窗口大小。我认为我看到的是没有标题或边框的窗口的默认窗口行为(尽管我在任何地方都找不到记录)。我发现调用 JFrame.setMaximizedBounds() 提供了用于更改 win32 MINMAXINFO 结构的边界。因此,通过更改我执行的操作并使用 camickr 建议的窗口大小作为我的 MaximizedBounds,我离得更近了一点:
GraphicsConfiguration graphicsConfiguration = frame.getGraphicsConfiguration();
frame.setMaximizedBounds(graphicsConfiguration.getBounds());
frame.setExtendedState((frame.getExtendedState() & Frame.MAXIMIZED_BOTH) == Frame.MAXIMIZED_BOTH ? JFrame.NORMAL : JFrame.MAXIMIZED_BOTH);
现在最大化窗口不再隐藏菜单栏。但是..现在我有另一个问题。我无法从辅助显示器最大化。框架就消失了。
我正在添加一些 win32 标记,希望能让 C++ 程序员认识到这个问题。
解决方案:我显然还不能回答我自己的问题,所以我将把我的解决方案放在这里。我必须使用 Sun 类,并且我没有在 Windows 以外的任何平台上对其进行过测试,但到目前为止,这似乎适用于多显示器和多任务栏配置:
GraphicsConfiguration config = frame.getGraphicsConfiguration();
Rectangle usableBounds = SunGraphicsEnvironment.getUsableBounds(config.getDevice());
frame.setMaximizedBounds(new Rectangle(0, 0, usableBounds.width, usableBounds.height));
frame.setExtendedState((frame.getExtendedState() & Frame.MAXIMIZED_BOTH) == Frame.MAXIMIZED_BOTH ? JFrame.NORMAL : JFrame.MAXIMIZED_BOTH);
【问题讨论】:
-
谢谢,但我并不想制作全屏应用。我可能应该更清楚我的目标,但我正在尝试用我自己的(如 Google Chrome、Microsoft Office 等)替换操作系统标题栏和控件。 Sun 甚至尝试过一次开箱即用,但他们遇到了与我相同的问题:
frame.getRootPane().setWindowDecorationStyle(JRootPane.FRAME);