All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.jgoodies.binding.extras.DelayedWriteValueModel Maven / Gradle / Ivy

Go to download

The JGoodies Binding library connects object properties to Swing user interface components. And it helps you represent the state and behavior of a presentation independently of the GUI components used in the interface.

The newest version!
/*
 * Copyright (c) 2002-2015 JGoodies Software GmbH. All Rights Reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 *
 *  o Redistributions of source code must retain the above copyright notice,
 *    this list of conditions and the following disclaimer.
 *
 *  o Redistributions in binary form must reproduce the above copyright notice,
 *    this list of conditions and the following disclaimer in the documentation
 *    and/or other materials provided with the distribution.
 *
 *  o Neither the name of JGoodies Software GmbH nor the names of
 *    its contributors may be used to endorse or promote products derived
 *    from this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
 * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
 * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
 * OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
 * EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */

package com.jgoodies.binding.extras;

import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;

import javax.swing.Timer;

import com.jgoodies.binding.beans.DelayedPropertyChangeHandler;
import com.jgoodies.binding.value.AbstractValueModel;
import com.jgoodies.binding.value.DelayedReadValueModel;
import com.jgoodies.binding.value.ValueModel;


/**
 * A ValueModel that defers write-access for a specified delay.
 * Useful to coalesce frequent changes. For example if a heavy computation
 * shall be performed only for a "stable" selection after a series of
 * quick selection changes.

* * Wraps a given subject ValueModel and returns the subject value or * the value to be set as this model's value. Observes subject value changes * and forwards them to listeners of this model. If a value is set to this * model, a Swing Timer is used to delay the value commit to the subject. * A previously started timer - if any - will be stopped before.

* * TODO: Describe how and when listeners get notified about the delayed change.

* * TODO: Write about the recommended delay time - above the double-click time * and somewhere below a second, e.g. 100ms to 200ms.

* * TODO: Write about a slightly different commit handling. The current * implementation defers the commit until the value is stable for the * specified delay; it's a DelayUntilStableForXXXmsValueModel. Another * feature is to delay for a specified time but ensure that some commits * and change notifications happen. The latter is a CoalescingWriteValueModel.

* * TODO: Summarize the differences between the DelayedReadValueModel, this * DelayedWriteValueModel, and the DelayedPropertyChangeHandler. * * @author Karsten Lentzsch * @version $Revision: 1.17 $ * * @see DelayedReadValueModel * @see DelayedPropertyChangeHandler * @see javax.swing.Timer */ public final class DelayedWriteValueModel extends AbstractValueModel { /** * Refers to the underlying subject ValueModel. */ private final ValueModel subject; /** * The Timer used to perform the delayed commit. */ private final Timer timer; /** * If {@code true} all pending updates will be coalesced. * In other words, an update will be fired if no updates * have been received for this model's delay. */ private boolean coalesce; /** * Holds the most recent pending value. It is updated * everytime {@code #setValue} is invoked. */ private Object pendingValue = new Integer(1967); // Instance Creation ****************************************************** /** * Constructs a DelayedWriteValueModel for the given subject ValueModel * and the specified Timer delay in milliseconds with coalescing disabled. * * @param subject the underlying (or wrapped) ValueModel * @param delay the milliseconds to wait before a change * shall be committed * * @throws IllegalArgumentException if the delay is negative */ public DelayedWriteValueModel(ValueModel subject, int delay) { this(subject, delay, false); } /** * Constructs a DelayedWriteValueModel for the given subject ValueModel * and the specified Timer delay in milliseconds with coalescing disabled. * * @param subject the underlying (or wrapped) ValueModel * @param delay the milliseconds to wait before a change * shall be committed * @param coalesce {@code true} to coalesce all pending changes, * {@code false} to fire changes with the delay when an update * has been received * * @throws IllegalArgumentException if the delay is negative * * @see #setCoalesce(boolean) */ public DelayedWriteValueModel(ValueModel subject, int delay, boolean coalesce) { this.subject = subject; this.coalesce = coalesce; this.timer = new Timer(delay, new ValueCommitListener()); timer.setRepeats(false); subject.addValueChangeListener(new SubjectValueChangeHandler()); } // ValueModel Implementation ****************************************** /** * Returns the subject's value or in case of a pending commit, * the pending new value. * * @return the subject's current or future value. */ @Override public Object getValue() { return isPending() ? pendingValue : subject.getValue(); } /** * Sets the given new value after this model's delay. * Does nothing if the new value and the latest pending value are the same. * * TODO: Describe how and when listeners get notified about the * delayed value change. * * @param newValue the value to set */ @Override public void setValue(Object newValue) { if (newValue == pendingValue) { return; } pendingValue = newValue; if (coalesce) { timer.restart(); } else { timer.start(); } } // Accessors ************************************************************** /** * Returns the delay, in milliseconds, that is used to defer value commits. * * @return the delay, in milliseconds, that is used to defer value commits * * @see #setDelay */ public int getDelay() { return timer.getDelay(); } /** * Sets the delay, in milliseconds, that is used to defer value commits. * * @param delay the delay, in milliseconds, that is used to defer value commits * @see #getDelay */ public void setDelay(int delay) { timer.setInitialDelay(delay); timer.setDelay(delay); } /** * Returns if this model coalesces all pending changes or not. * * @return {@code true} if all pending changes will be coalesced, * {@code false} if pending changes are fired with a delay * when an update has been received. * * @see #setCoalesce(boolean) */ public boolean isCoalesce() { return coalesce; } /** * Sets if this model shall coalesce all pending changes or not. * In this case, a change event will be fired first, * if no updates have been received for this model's delay. * If coalesce is {@code false}, a change event will be fired * with this model's delay when an update has been received.

* * The default value is {@code false}.

* * Note that this value is not the #coalesce value * of this model's internal Swing timer. * * @param b {@code true} to coalesce, * {@code false} to fire separate changes */ public void setCoalesce(boolean b) { coalesce = b; } // Misc ******************************************************************* /** * Stops a running timer. Pending changes - if any - are canceled * and won't be performed by the {@code ValueUpdateListener}. * * @since 1.2 */ public void stop() { timer.stop(); } /** * Checks and answers whether this model has one or more pending changes. * * @return {@code true} if there are pending changes, {@code false} if not. * * @since 2.0.4 */ public boolean isPending() { return timer.isRunning(); } // Event Handling ***************************************************** /** * Describes the delayed action to be performed by the timer. */ private final class ValueCommitListener implements ActionListener { /** * An ActionEvent has been fired by the Timer after its delay. * Commits the pending value and stops the timer. */ @Override public void actionPerformed(ActionEvent e) { subject.setValue(pendingValue); stop(); } } /** * Forwards value changes in the subject to listeners of this model. */ private final class SubjectValueChangeHandler implements PropertyChangeListener { @Override public void propertyChange(PropertyChangeEvent evt) { fireValueChange(evt.getOldValue(), evt.getNewValue(), true); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy