我的笔记本有4k显示屏。我的外接显示器是1920.当我将Java Swing应用程序窗口从4k显示器拖到外部显示器上时,除了窗框外,它会变成全黑。无论我是否在外接显示器上最大化它都无关紧要。
以下是我从Oracle.com获得的HelloWorldSwing
示例。它显示了完全相同的行为。
import javax.swing.*;
public class HelloWorldSwing {
/**
* Create the GUI and show it. For thread safety,
* this method should be invoked from the
* event-dispatching thread.
*/
private static void createAndShowGUI() {
//Create and set up the window.
JFrame frame = new JFrame("HelloWorldSwing");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
//Add the ubiquitous "Hello World" label.
JLabel label = new JLabel("Hello World");
frame.getContentPane().add(label);
//Display the window.
frame.pack();
frame.setVisible(true);
}
public static void main(String[] args) {
//Schedule a job for the event-dispatching thread:
//creating and showing this application's GUI.
javax.swing.SwingUtilities.invokeLater(new Runnable() {
public void run() {
createAndShowGUI();
}
});
}
}
它看起来像是Swing中的一个错误。无论如何,我必须找到一个解决方案/解决方案。有什么想法吗?
我正在运行jdk / jre 1.8.0 121。
答案 0 :(得分:1)
虽然我在任何多屏幕环境中都没有遇到过这样的问题,但到目前为止我还没有尝试使用Swing - 我可能会帮助调试它。
尝试运行此代码示例:
public class HelloWorldSwing
{
private static void createAndShowGUI ( final GraphicsDevice device )
{
final GraphicsConfiguration conf = device.getDefaultConfiguration ();
final JFrame frame = new JFrame ( "HelloWorldSwing", conf );
frame.setDefaultCloseOperation ( JFrame.EXIT_ON_CLOSE );
final JLabel label = new JLabel ( "Hello World", SwingConstants.CENTER );
frame.getContentPane ().add ( label );
final Rectangle sb = conf.getBounds ();
frame.setBounds ( sb.x + sb.width / 2 - 100, sb.y + sb.height / 2 - 100, 200, 200 );
frame.setVisible ( true );
}
public static void main ( final String[] args )
{
SwingUtilities.invokeLater ( new Runnable ()
{
@Override
public void run ()
{
final GraphicsEnvironment env = GraphicsEnvironment.getLocalGraphicsEnvironment ();
final GraphicsDevice[] devices = env.getScreenDevices ();
for ( final GraphicsDevice device : devices )
{
if ( device.getType () == GraphicsDevice.TYPE_RASTER_SCREEN )
{
createAndShowGUI ( device );
}
}
}
} );
}
}
这基本上会在您系统中可用的每个屏幕设备上放置一个帧。它还将为每个帧提供默认的屏幕图形配置。
检查是否在将框架拖到笔记本电脑屏幕时遇到同样的问题,以及是否还有其他问题。
Edit1:此外,了解运行代码示例所使用的JDK / JRE可能非常重要,因为存在一些相关的内部更改。如果您运行的是旧的JDK,那么在较新的Mac OS X版本上可能不是最佳选择。
编辑2:我会尝试做更多的事情来了解更多信息:
在显示框架之前设置Nimbus Look and Feel,然后尝试使用安装的自定义L& F拖动它们。
将系统(本机)外观设为shown in the tutorial here - 这可能会对Mac OS X产生一些影响。
尝试运行Java FX example并在显示之间拖动它,看看你是否遇到同样的问题 - 你已经在使用JDK 8了,所以它应该很容易实现。 Java FX只是前一段时间推出的新UI框架。
总的来说,我可以说Swing对高分辨率显示器的支持很差,所以即使一切运行正常并且你没有得到黑屏,你也会看到其他问题,比如UI模糊或低质量文字呈现。
答案 1 :(得分:0)
我找到了解决方案。基本上我添加了一个组件监听器到我的主框架并监听移动事件。每当移动框架时,我都会检查它所在的显示器。一旦框架移动到另一个显示器,我将其设置为不可见,然后设置位于“新”显示器上的新边界,将扩展状态设置为最大化,最后再次使框架可见。
在最后几分钟,我疯狂地将窗户从一个显示器拖到另一个显示器,同时欢呼雀跃。如果你在生活中也想要那么多的乐趣,那就是代码:
mainFrame.addComponentListener(new ComponentListener()
{
@Override
public void componentMoved(ComponentEvent evt)
{
GraphicsConfiguration conf = mainFrame.getGraphicsConfiguration();
GraphicsDevice curDisplay = conf.getDevice();
GraphicsEnvironment env = GraphicsEnvironment.getLocalGraphicsEnvironment();
GraphicsDevice[] allDisplays = env.getScreenDevices();
for (int i = 0; i < allDisplays.length; i++)
{
if (allDisplays[i].equals(curDisplay))
{
//the window has been dragged to another display
if (i != _currentDisplayIndex)
{
final Rectangle sb = conf.getBounds();
mainFrame.setVisible(false);
//it is important to set the bounds somewhere on the new display before setting the extended state to maximized
mainFrame.setBounds(sb.x, sb.y, 1000, 800);
mainFrame.setExtendedState(Frame.MAXIMIZED_BOTH);
mainFrame.setVisible(true);
_currentDisplayIndex = i;
}
}
}
}
@Override
public void componentShown(ComponentEvent evt)
{
}
@Override
public void componentResized(ComponentEvent evt)
{
}
@Override
public void componentHidden(ComponentEvent evt)
{
}
});
开心辞典!
旁注:我之前尝试拍摄黑色窗口的屏幕截图。但是,截图不是黑色,而是通常显示所有挥杆组件。这让我相信,它不仅是一个摇摆问题,而且还有某种窗口/显示驱动程序问题。按“printscreen”不会触发java中的任何内容(我猜),但只刷新显示。