7

I want to listen to the changes of the value of the SpinnerNumberModel for a JSpinner.
I create a PropertyChangeSupport and put the model into it.

I need the propertyChangeListener, because it shows me the old and new value of the property.

The snippet doesn't work: the propertyChange method prints nothing, when I click on the JSpinner.
A simple ChangeListener give only the new value, but I need also the old value, how can I get it?

package de.unikassel.jung;

import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.beans.PropertyChangeSupport;

import javax.swing.JFrame;
import javax.swing.JSpinner;
import javax.swing.SpinnerNumberModel;

public class PropertyChangeTest implements PropertyChangeListener {

    public static void main(String[] args) {
        new PropertyChangeTest();
    }

    public PropertyChangeTest() {
        JFrame frame = new JFrame();
        frame.setBounds(100, 100, 450, 300);
        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

        int value = 1;
        int min = 0;
        int max = 10;
        int step = 1;
        SpinnerNumberModel spinnerModel = new SpinnerNumberModel(value, min, max, step);

        PropertyChangeSupport pcs = new PropertyChangeSupport(spinnerModel);
        pcs.addPropertyChangeListener("value", this);

        JSpinner spinner = new JSpinner(spinnerModel);
        frame.getContentPane().add(spinner);
        frame.setVisible(true);
    }

    @Override
    public void propertyChange(PropertyChangeEvent evt) {
        System.out.println(evt);
        System.out.println(evt.getSource());
    }

}
4

4 回答 4

6

不要听模型,而是听编辑的JFormattedTextField,如下所示。

JSpinner spinner = new JSpinner(new SpinnerNumberModel(1, 0, 10, 1));
JSpinner.DefaultEditor editor = (JSpinner.DefaultEditor) spinner.getEditor();
editor.getTextField().addPropertyChangeListener("value", this);
于 2011-08-07T01:53:48.723 回答
5

星期一早上...不抗拒几条评论的经典时间:-)

@timaschew

  • 需要 propertyChangeListener,因为它向我展示了该属性的旧值和新值。 ” - (挑剔 - 但总是有强烈的冲动将需求和解决方案分开 :),我认为这是相反的:在更改通知时你需要访问旧值和新值,propertyChangeEvent/Listener 是支持它的通知类型,可能还有其他
  • PropertyChangeSupport 不应该用于观察代码的一部分,它应该用在obervable的一侧(就像@Hovercraft 在他的示例中所做的那样):它的唯一责任是管理并通知注册到 observable 的侦听器
  • 偶尔,accessibleContext 为黑客提供了一个钩子——尽管如此,它是一个钩子(除非你真的需要支持可访问性,这很可能是这种情况:-) 与所有黑客一样,这是一个脆弱的解决方案,很可能会导致在未来的某个时候疼痛。跟随关于 Action 和 AbstractButton 如何交互的链接更加稳定

@气垫船

  • 使用更丰富的更改通知来增强模型是可行的方法(如:我最喜欢的 :-)
  • 只是一个小细节:如果你有一个奴隶让他做所有的工作——PropertyChangeSupport 具有采用旧/新值的方法,无需在可观察对象上创建事件。当新旧相等时,它无论如何都会被扔掉
  • 对于通知事件中的newValue,不要使用参数,而是再次使用getValue(超级可能已经拒绝了更改)

@垃圾神

哈哈 - 你已经猜到我不喜欢解决方案:它破坏了封装,因为它依赖于实现细节,所以除非完全控制 JSpinner 创建并且绝对确定它的编辑器永远不会改变,否则不要这样做

于 2011-08-08T08:10:36.560 回答
4

要使 PropertyChangeSupport 工作,您需要调用它的firePropertyChange方法,但更重要的是,支持对象需要能够访问它正在侦听的属性的 setXXX 方法,并且在该方法中它需要调用 PropertyChangeSupport 的 firePropertyChange 方法。所以我认为要让你的想法奏效,你需要扩展模型的类,给它一个 PropertyChangeSupport 对象,给它添加和删除侦听器方法,并确保监听模型的 setValue 方法中所做的更改钥匙。在我的示例中,该方法如下所示:

   @Override
   public void setValue(Object newValue) {
      // store old value and set the new one
      Object oldValue = getValue();
      super.setValue(newValue);

      // construct the event object using these saved values
      PropertyChangeEvent evt = new PropertyChangeEvent(this, VALUE, oldValue,
               newValue);

      // notify all of the listeners
      pcs.firePropertyChange(evt);
   }

这是我使用 PropertyChangeSupport 的示例模型类:

import java.beans.*;
import javax.swing.*;
import javax.swing.event.*;

@SuppressWarnings("serial")
class MySpinnerNumberModel extends SpinnerNumberModel {
   public static final String VALUE = "value";
   private SwingPropertyChangeSupport pcs = new SwingPropertyChangeSupport(this);

   // you will likely need to create multiple constructors to match
   // the ones available to the SpinnerNumberModel class
   public MySpinnerNumberModel(int value, int min, int max, int step) {
      super(value, min, max, step);
   }

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

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

   @Override
   public void setValue(Object newValue) {
      // store old value and set the new one
      Object oldValue = getValue();
      super.setValue(newValue);

      // construct the event object using these saved values
      PropertyChangeEvent evt = new PropertyChangeEvent(this, VALUE, oldValue,
               newValue);

      // notify all of the listeners
      pcs.firePropertyChange(evt);
   }
}

最后是测试类来测试上面的类,看看它是否正常工作:

import java.beans.*;
import javax.swing.*;

public class TestSpinnerPropChange {

   private static void createAndShowUI() {
      final MySpinnerNumberModel myModel = new MySpinnerNumberModel(1, 0, 10, 1);
      final JSpinner spinner = new JSpinner(myModel);

      final JTextField oldValueField = new JTextField(10);
      final JTextField newValueField = new JTextField(10);

      JPanel panel = new JPanel();
      panel.add(spinner);
      panel.add(new JLabel("old value:"));
      panel.add(oldValueField);
      panel.add(new JLabel("new value:"));
      panel.add(newValueField);

      myModel.addPropertyChangeListener(new PropertyChangeListener() {
         public void propertyChange(PropertyChangeEvent evt) {
            // checking the property name is overkill here, but is a good habit
            // to get into, especially if listening to more than one property.
            if (evt.getPropertyName().equals(MySpinnerNumberModel.VALUE)) {
               oldValueField.setText(evt.getOldValue().toString());
               newValueField.setText(evt.getNewValue().toString());
            }
         }
      });

      JFrame frame = new JFrame("TestSpinnerPropChange");
      frame.getContentPane().add(panel);
      frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
      frame.pack();
      frame.setLocationRelativeTo(null);
      frame.setVisible(true);
   }

   public static void main(String[] args) {
      java.awt.EventQueue.invokeLater(new Runnable() {
         public void run() {
            createAndShowUI();
         }
      });
   }
}
于 2011-08-07T04:01:47.417 回答
1

http://docs.oracle.com/javase/tutorial/javabeans/writing/properties.html#bound 您必须在设置器中触发ThePropertyChange。

于 2012-05-03T15:58:39.533 回答