3

我希望我的 GUI 在出现 JOptionPane 时进行一些检查。因为我找不到任何其他方式,我虽然每次应用程序窗口失去焦点时都可以这样做(它只是检查一个字符串)。出于这个原因,我在我的 JFrame 上添加了以下代码:

appFrame.addWindowListener(new WindowAdapter() {

            @Override
            public void windowLostFocus(WindowEvent e) {
                System.out.println("Focus Lost");

            }
            @Override
            public void windowClosing(WindowEvent e) {
                //some other stuff here that work
            }
});

窗口关闭侦听器工作正常。虽然当 JFrame 没有聚焦时,什么也没有发生。每次我从 JFrame 切换到其他窗口时,不应该打印“Focus Lost”吗?另外,显示 JOptionPane 时会触发此方法吗?

4

3 回答 3

6

对我来说,关键是您希望通过更改字符串变量来触发 GUI 更改。我认为解决此问题的最佳方法是使用 PropertyChangeListenerSupport 使 String 变量成为绑定属性。这样,您可以让 GUI 将 PropertyChangeListener 附加到保存 String 变量的类,然后在它发生更改时收到通知,从而允许您适当地更新 GUI。

如果您走这条路,请考虑为观察到的类提供一个 SwingPropertyChangeSupport 字段,以便在 Swing 事件线程上通知侦听器,并希望避免任何 Swing 并发问题。

这是一个简短的例子:

import java.awt.Dimension;
import java.awt.event.*;
import java.beans.*;
import javax.swing.*;
import javax.swing.event.SwingPropertyChangeSupport;

public class ShowPropertyChangeSupport {
   @SuppressWarnings("serial")
   private static void createAndShowGui() {
      final MainGUI mainGui = new MainGUI("Title");
      final ObservedClass observedClass = new ObservedClass();
      observedClass.addPropertyChangeListener(new PropertyChangeListener() {

         @Override
         public void propertyChange(PropertyChangeEvent pcEvt) {
            if (pcEvt.getPropertyName().equals(ObservedClass.BOUND_PROPERTY)) {
               mainGui.setTitle(pcEvt.getNewValue().toString());
            }
         }
      });

      mainGui.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
      mainGui.pack();
      mainGui.setLocationRelativeTo(null);
      mainGui.setVisible(true);

      int timerDelay = 6000; // every 6 seconds
      new Timer(timerDelay, new ActionListener() {
         @Override
         public void actionPerformed(ActionEvent arg0) {
            String result = JOptionPane.showInputDialog(mainGui,
                  "Please enter a String", "Set GUI title", JOptionPane.PLAIN_MESSAGE);
            if (result != null) {
               observedClass.setBoundProperty(result);
            }
         }
      }){{setInitialDelay(1000);}}.start();
   }

   public static void main(String[] args) {
      SwingUtilities.invokeLater(new Runnable() {
         public void run() {
            createAndShowGui();
         }
      });
   }
}

// ** note that I don't like extending JFrame,
// but will do this for sake of example simplicity
class MainGUI extends JFrame {
   public MainGUI(String title) {
      super(title);
   }

   @Override
   public Dimension getPreferredSize() {
      return new Dimension(400, 300);
   }

}

class ObservedClass {
   public static final String BOUND_PROPERTY = "bound property";
   private String boundProperty = "";
   private SwingPropertyChangeSupport spcSupport = new SwingPropertyChangeSupport(
         this);

   public SwingPropertyChangeSupport getSpcSupport() {
      return spcSupport;
   }

   public void setSpcSupport(SwingPropertyChangeSupport spcSupport) {
      this.spcSupport = spcSupport;
   }

   public String getBoundProperty() {
      return boundProperty;
   }

   public void setBoundProperty(String boundProperty) {
      String oldValue = this.boundProperty;
      String newValue = boundProperty;
      this.boundProperty = newValue;
      spcSupport.firePropertyChange(BOUND_PROPERTY, oldValue, newValue);
   }

   public void addPropertyChangeListener(PropertyChangeListener listener) {
      spcSupport.addPropertyChangeListener(listener);
   }

   public void removePropertyChangeListener(PropertyChangeListener listener) {
      spcSupport.removePropertyChangeListener(listener);
   }

}

在我看来,这一切的关键是使用监听器,以便具有绑定属性的类——被监听的字符串——不知道 GUI、监听器和 GUI,同样不知道具有绑定属性的类。它们是完全解耦的。

于 2012-04-17T21:04:51.013 回答
5

我不会详细说明您为什么要做您正在做的事情,但由于以下原因,它没有按您的预期工作:

WindowAdapter是一个便利类,因此您可以创建一个侦听器并将其注册用于多种类型的事件。您只为一组事件注册了它,您还需要通过以下方式为焦点事件注册它:Window.addWindowFocusListener()

WindowAdapter adapter = new WindowAdapter() {
        @Override
        public void windowLostFocus(WindowEvent e) {
            System.out.println("Focus Lost");
        }
        @Override
        public void windowClosing(WindowEvent e) {
            //some other stuff here that work
        }
    };
appFrame.addWindowListener(adapter);
appFrame.addWindowFocusListener(adapter);
于 2012-04-17T23:48:03.530 回答
2

1)JOptionPane/模态JDialog有模态问题,但如果所有容器都有自己的所有者,模态可能是优势,你需要知道真正的解决方法(我将讨论如何测试它)

2) 恕我直言,我不知道你为什么需要那个,为什么我需要知道这个,关于业务规则,你总是需要知道....,如果在 EDT 上完成

于 2012-04-17T21:47:00.477 回答