JFrameが何らかの理由でバックグラウンドに移動する

Aug 21 2020

昨日ここでこのトピックについて非常に漠然とした質問をした後(私は今閉じることに投票しました)、問題を特定し、この動作を示すMCVEを作成することができました。

シナリオは次のようになります。

一部の操作がバックグラウンドで進行している間、モーダルの「待機」ダイアログがフォアグラウンドで提供され、念のため、JFrameも無効に設定されています。バックグラウンドタスクが終了すると、フレームが再び有効になり、ダイアログが破棄されます。

問題は、JFrameが有効になり、モーダルダイアログが破棄された後、JFrameが突然バックグラウンドに移動することです。「背景」の意味で、JFrameの前にフォーカスがあったウィンドウの後ろに移動しています。なぜこれが起こるのですか?

このコードは問題を再現する必要があります。

private static JFrame frame;
private static JDialog dialog;

public static void main(String[] args) {
    try {
        SwingUtilities.invokeAndWait(new Runnable() {

            @Override
            public void run() {
                buildFrame();
                buildDialog();
            }
        });
    } catch (InvocationTargetException | InterruptedException e) {
        e.printStackTrace();
    }
}

protected static void buildDialog() {
    dialog = new JDialog(frame);
    dialog.getContentPane().add(new JLabel("This is the dialog"));
    dialog.setLocationRelativeTo(frame);
    javax.swing.Timer t = new javax.swing.Timer(1000, new ActionListener() {

        @Override
        public void actionPerformed(ActionEvent e) {
            dialog.dispose();
            frame.setEnabled(true);
        }
    });
    t.setRepeats(false);
    t.start();
    dialog.pack();
    dialog.setVisible(true);
}

protected static void buildFrame() {
    frame = new JFrame();
    frame.setMinimumSize(new Dimension(400, 400));
    frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    frame.getContentPane().add(new JLabel("This is the Frame"));
    frame.setEnabled(false);
    frame.pack();
    frame.setVisible(true);
}

なぜこれが起こるのか、そしてこれをどのように防ぐことができるのか誰かが知っていますか?

回答

2 SergiyMedvynskyy Aug 21 2020 at 07:39

問題はメソッドframe.setEnabled()です。理由はわかりませんが、フレームが隠れてしまいます。私の提案は、それを削除してモダリティの概念を使用することです:(dialog.setModal(true)ダイアログが表示されたときに親フレームも使用できなくなります。フレームを使用できなくするには、その上にガラス板を置くことができます。コードは次のとおりです。

import java.awt.Color;
import java.awt.Component;
import java.awt.Dimension;
import java.awt.Graphics;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.FocusEvent;
import java.awt.event.FocusListener;
import java.awt.event.MouseAdapter;
import java.lang.reflect.InvocationTargetException;

import javax.swing.JDialog;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JTextField;
import javax.swing.SwingUtilities;

/**
 * <code>DialogFrameTest</code>.
 */
public class DialogFrameTest {
    private static JFrame frame;

    private static JDialog dialog;

    private static Component oldGlassPane;

    public static void main(String[] args) {
        try {
            SwingUtilities.invokeAndWait(new Runnable() {

                @Override
                public void run() {
                    buildFrame();
                    buildDialog();
                }
            });
        } catch (InvocationTargetException | InterruptedException e) {
            e.printStackTrace();
        }
    }

    protected static void buildDialog() {
        dialog = new JDialog(frame);
        dialog.getContentPane().add(new JTextField("This is the dialog"));
        dialog.setLocationRelativeTo(frame);
        dialog.setModal(true);
        javax.swing.Timer t = new javax.swing.Timer(1000, new ActionListener() {

            @Override
            public void actionPerformed(ActionEvent e) {
                dialog.dispose();
                frame.setGlassPane(oldGlassPane);
                oldGlassPane.setVisible(false);
            }
        });
        t.setRepeats(false);
        t.start();
        dialog.pack();
        dialog.setVisible(true);
    }

    protected static void buildFrame() {
        frame = new JFrame();
        frame.setMinimumSize(new Dimension(400, 400));
        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        frame.getContentPane().add(new JTextField("This is the Frame"));
        oldGlassPane = frame.getGlassPane();
        frame.setGlassPane(new SplashGlassPane());
        frame.getGlassPane().setVisible(true);
        frame.pack();
        frame.setVisible(true);
    }

    private static class SplashGlassPane extends JPanel implements FocusListener {

        /** Holds the id of this panel. The creator of this object can submit this id to determine whether it's the owner of this object. */
        private String typeId;

        /**
         * Creates new GlassPane.
         */
        public SplashGlassPane() {
            addMouseListener(new MouseAdapter() {});
            addMouseMotionListener(new MouseAdapter() {});
            addFocusListener(this);
            setOpaque(false);
            setFocusable(true);
            setBackground(new Color(0, 0, 0, 100));
        }

        @Override
        public final void setVisible(boolean v) {
            // Make sure we grab the focus so that key events don't go astray.
            if (v) {
                requestFocus();
            }
            super.setVisible(v);
        }

        // Once we have focus, keep it if we're visible
        @Override
        public final void focusLost(FocusEvent fe) {
            if (isVisible()) {
                requestFocus();
            }
        }

        /**
         * {@inheritDoc}
         */
        @Override
        public final void paint(Graphics g) {
            final Color old = g.getColor();
            g.setColor(getBackground());
            g.fillRect(0, 0, getSize().width, getSize().height);
            g.setColor(old);
            super.paint(g);
        }

        @Override
        public void focusGained(FocusEvent fe) {
            // nothing to do
        }
    }

}