从4k显示器拖动到FullHD显示器时,应用程序变黑

时间:2017-02-21 12:13:05

标签: java swing screen drag multiple-monitors

我的笔记本有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。

2 个答案:

答案 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:我会尝试做更多的事情来了解更多信息:

  1. 在显示框架之前设置Nimbus Look and Feel,然后尝试使用安装的自定义L& F拖动它们。

  2. 将系统(本机)外观设为shown in the tutorial here - 这可能会对Mac OS X产生一些影响。

  3. 尝试运行Java FX example并在显示之间拖动它,看看你是否遇到同样的问题 - 你已经在使用JDK 8了,所以它应该很容易实现。 Java FX只是前一段时间推出的新UI框架。

  4. 总的来说,我可以说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中的任何内容(我猜),但只刷新显示。