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
        }
    }

}