How do you hide a Swing Popup when you click somewhere else

前端 未结 6 2195
走了就别回头了
走了就别回头了 2021-01-04 06:26

I have a Popup that is shown when a user clicks on a button. I would like to hide the popup when any of the following events occur:

  1. The user clicks somewhere e
相关标签:
6条回答
  • 2021-01-04 06:49

    As pajton noted in a previous comment, Popup is not a JComponent to which listeners can be readily bound. But, as its documentation states, "implementations of Popup are responsible for creating and maintaining their own Components to render [its subject] to the user."

    So in using it as your presentation mechanism, your Popup is going to have to present itself as an actual Swing component anyway. Have it register itself to that component. Have it hide itself when the component loses focus.

    import java.awt.FlowLayout;
    import java.awt.Frame;
    import java.awt.Point;
    import java.awt.event.ActionEvent;
    import java.awt.event.WindowEvent;
    import java.awt.event.WindowFocusListener;
    import javax.swing.JButton;
    import javax.swing.JDialog;
    import javax.swing.JFrame;
    import javax.swing.JLabel;
    import javax.swing.JOptionPane;
    import javax.swing.Popup;
    
    public class PopupTester extends JFrame {
        private static class MessagePopup extends Popup
            implements WindowFocusListener
        {
            private final JDialog dialog;
    
            public MessagePopup(Frame base, String message) {
                super();
                dialog = new JOptionPane().createDialog( base, "Message" );
                dialog.setModal( false );
                dialog.setContentPane( new JLabel( message ) );
            }
            @Override public void show() {
                dialog.addWindowFocusListener( this );
                dialog.setVisible( true );
            }
            @Override public void hide() {
                dialog.setVisible( false );
                dialog.removeWindowFocusListener( this );
            }
            public void windowGainedFocus( WindowEvent e ) {
                // NO-OP
            }
    
            public void windowLostFocus( WindowEvent e ) {
                hide();
            }
        }
    
        public static void main(String[] args) {
        final PopupTester popupTester = new PopupTester();
        popupTester.setLayout(new FlowLayout());
        popupTester.setSize(300, 100);
        popupTester.add(new JButton("Click Me") {
          @Override
          protected void fireActionPerformed(ActionEvent event) {
            Point location = getLocationOnScreen();
              MessagePopup popup = new MessagePopup( popupTester, "Howdy" );
              popup.show();
            }
          });
          popupTester.add(new JButton("No Click Me"));
          popupTester.setVisible(true);
          popupTester.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        }
    }
    
    0 讨论(0)
  • 2021-01-04 06:58

    You could add a FocusListener to your popup-window, and dispose it when it loses focus. However, that will cause you some troubles when the focus loss is due to some other application (new windows comes to the foreground, you switch virtual desktops, etc.)

    But perhaps you (a) know that that cannot happen in your case or (b) would want to close the popup in such cases anyway, a focus-based approach may still be interesting to you.

    0 讨论(0)
  • 2021-01-04 06:59

    You can add MouseListener to your background panel and hide the popup when somebody clicks on the panel.

    To react on application minimization, use WindowListener attached to a JFrame.

    Etc, etc. May seem tedious, but surely will work.

    0 讨论(0)
  • 2021-01-04 07:02

    I know this is an old question but I really needed the Popup to work in my case. So I tried a few things and the following is my solution.

    Add a FocusListener to the component you add to the popup and program the focusLost event on that component to hide the popup when focus is lost. Call the requestFocus method on your component just after showing the popup.

    import java.awt.*;
    import java.awt.event.ActionEvent;
    import java.awt.event.ActionListener;
    import java.awt.event.FocusAdapter;
    import java.awt.event.FocusEvent;
    
    import javax.swing.*;
    
    public class PopupTester extends JFrame {
        JButton myButton = new JButton("Click Me");
        JLabel myComponent = new JLabel("Howdy");
        Popup popup = null;
    
        public PopupTester() {
            setLayout(new FlowLayout());
            setSize(300, 100);
            add(myButton);
            add(new JButton("No Click Me"));
            setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    
            myComponent.addFocusListener(new FocusAdapter() {
                public void focusLost(FocusEvent e) {
                    if (popup != null) {
                        popup.hide();
                    }
                }
            });
    
            myButton.addActionListener(new ActionListener() {
                @Override
                public void actionPerformed(ActionEvent arg0) {
                    if (popup != null) {
                        popup.hide();
                        popup = null;
                    }
    
                    Point location = myButton.getLocationOnScreen();
                    int y = (int) (location.getY() + myButton.getHeight());
                    int x = (int) location.getX();
                    popup = PopupFactory.getSharedInstance().getPopup(PopupTester.this, myComponent, x, y);
                    popup.show();
                    myComponent.requestFocus();
                }
            });
        }
    
        public static void main(String[] args) {
            PopupTester popupTester = new PopupTester();
            popupTester.setVisible(true);
        }
    }
    
    0 讨论(0)
  • Use a JPopupMenu. You can add any component to it, not just menu items.

    0 讨论(0)
  • 2021-01-04 07:06

    Thanks pajton and Noel Ang for getting me pointed in the right direction! Here is the solution that I ended up with. I'm just including it here so that others may benefit from it.

    I ended up going with a JWindow since it doesn't get the window decorations but does get focus events.

    import java.awt.*;
    import java.awt.event.*;
    
    import javax.swing.*;
    
    public class PopupTester extends JFrame {
      private static class MessagePopup extends Popup implements WindowFocusListener {
        private final JWindow dialog;
    
        public MessagePopup(Frame base, JLabel component, int x, int y) {
          super();
          dialog = new JWindow(base);
          dialog.setFocusable(true);
          dialog.setLocation(x, y);
          dialog.setContentPane(component);
          component.setBorder(new JPopupMenu().getBorder());
          dialog.setSize(component.getPreferredSize());
          dialog.addKeyListener(new KeyAdapter() {
            @Override
            public void keyPressed(KeyEvent e) {
              if (e.getKeyCode() == KeyEvent.VK_ESCAPE) {
                dialog.setVisible(false);
              }
            }
          });
        }
    
        @Override
        public void show() {
          dialog.addWindowFocusListener(this);
          dialog.setVisible(true);
        }
    
        @Override
        public void hide() {
          dialog.setVisible(false);
          dialog.removeWindowFocusListener(this);
        }
    
        public void windowGainedFocus(WindowEvent e) {
          // NO-OP
        }
    
        public void windowLostFocus(WindowEvent e) {
          hide();
        }
      }
    
      public static void main(String[] args) {
        final PopupTester popupTester = new PopupTester();
        popupTester.setLayout(new FlowLayout());
        popupTester.setSize(300, 100);
        popupTester.add(new JButton("Click Me") {
          @Override
          protected void fireActionPerformed(ActionEvent event) {
            Point location = getLocationOnScreen();
            int x = (int) location.getX();
            int y = (int) (location.getY() + getHeight());
            JLabel myComponent = new JLabel("Howdy");
    
            MessagePopup popup = new MessagePopup(popupTester, myComponent, x, y);
            popup.show();
          }
        });
        popupTester.add(new JButton("No Click Me"));
        popupTester.setVisible(true);
        popupTester.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
      }
    }
    
    0 讨论(0)
提交回复
热议问题