1

我有一小段代码可以更新 a JSlider; 在另一个时间点,JSlider的最大值需要更新。问题是当我调用setMaximum()滑块时,它也会发送一个ChangeEvent. 为避免这种情况,我正在这样做:

slider.removeChangeListener(this);
slider.setMaximum(newMax);
slider.addChangeListener(this);

有没有更清洁/更优雅的方式来做到这一点?

4

4 回答 4

3

一个干净的方法可能是(取决于你实际需要什么)来实现一个自定义的 BoundedRangeModel ,它触发一个可以携带实际更改的属性的自定义 ChangeEvent :

/**
 * Extended model that passes the list of actually changed properties
 * in an extended changeEvent.
 */
public static class MyBoundedRangeModel extends DefaultBoundedRangeModel {

    public MyBoundedRangeModel() {
    }

    public MyBoundedRangeModel(int value, int extent, int min, int max) {
        super(value, extent, min, max);
    }

    @Override
    public void setRangeProperties(int newValue, int newExtent, int newMin,
            int newMax, boolean adjusting) {
        int oldMax = getMaximum();
        int oldMin = getMinimum();
        int oldValue = getValue();
        int oldExtent = getExtent();
        boolean oldAdjusting = getValueIsAdjusting();
        // todo: enforce constraints of new values for all
        List<String> changedProperties = new ArrayList<>();
        if (oldMax != newMax) {
           changedProperties.add("maximum"); 
        }
        if (oldValue != newValue) {
            changedProperties.add("value");
        }
        // todo: check and add other properties 
        changeEvent = changedProperties.size() > 0 ? 
                new MyChangeEvent(this, changedProperties) : null;
        super.setRangeProperties(newValue, newExtent, newMin, newMax, adjusting);
    }


}

/**
 * Extended ChangeEvent that provides a list of actually 
 * changed properties. 
 */
public static class MyChangeEvent extends ChangeEvent {

    private List<String> changedProperties;

    /**
     * @param source
     */
    public MyChangeEvent(Object source, List<String> changedProperties) {
        super(source);
        this.changedProperties = changedProperties;
    }

    public List<String> getChangedProperties() {
        return changedProperties;
    }

}

它的用法类似于:

final JSlider slider = new JSlider();
slider.setModel(new MyBoundedRangeModel(0, 0, -100, 100));
ChangeListener l = new ChangeListener() {

    @Override
    public void stateChanged(ChangeEvent e) {
        if (e instanceof MyChangeEvent) {
            MyChangeEvent me = (MyChangeEvent) e;
            if (me.getChangedProperties().contains("value")) {
               System.out.println("new value: " + 
                    ((BoundedRangeModel) e.getSource()).getValue()); 
            }
            if (me.getChangedProperties().contains("maximum")) {
                System.out.println("new max: " + 
                    ((BoundedRangeModel) e.getSource()).getMaximum()); 
            }
        } else {
            // do something else or nothing
        }
    }
};
slider.getModel().addChangeListener(l);

请注意,您必须将侦听器注册到模型,而不是滑块(原因是滑块创建了普通类型的新 changeEvent)

于 2013-04-24T15:23:56.333 回答
0

您可以检查谁触发了侦听器的更改。它仍然很脏,但是您不必删除更改侦听器。

于 2013-04-24T14:06:32.213 回答
0

从本质上讲,它看起来与这个问题相同。 在这种情况下,我担心答案是否定的

于 2013-04-24T14:07:16.683 回答
0

如果您只需要滑块在值发生变化时触发事件,您可以通过以下方式简化 kleopatra 的答案:

// make sure slider only fires changEvents when value changes
slider.setModel(new DefaultBoundedRangeModel() {
        final ChangeEvent theOne=new ChangeEvent(this);         
        @Override
        public void setRangeProperties(int newValue, int newExtent, int newMin,int newMax, boolean adjusting) 
            changeEvent= (getValue() != newValue ? theOne:null);
            super.setRangeProperties(newValue, newExtent, newMin, newMax, adjusting);
        }
        @Override
        protected void fireStateChanged()
        {
            if(changeEvent==null) return;
            super.fireStateChanged();
        }
    });

然后,您需要做的就是在模型上注册一个“标准”ChangeListener,它只会在值发生变化时被调用,而不是在最大值或最小值发生变化时被调用。

slider.getModel().addChangeListener(new ChangeListener() {
    public void stateChanged(ChangeEvent e) { 
        // do something here when value changes 
    });

});

于 2016-10-27T11:45:47.197 回答