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

dk.clanie.properties.TemporalProperty Maven / Gradle / Ivy

The newest version!
/**
 * Copyright (C) 2010, Claus Nielsen, [email protected]
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 3 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License along
 * with this program; if not, write to the Free Software Foundation, Inc.,
 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
 */
package dk.clanie.properties;

import static dk.clanie.collections.CollectionFactory.newConcurrentSkipListMap;

import java.util.Collections;
import java.util.Map.Entry;
import java.util.NavigableMap;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;

import org.joda.time.Instant;

/**
 * Temporal Property.
 * 
 * Keeps track of the different values assigned to the property over time.
 * An implementation of the "Temporal Property" pattern described by
 * Martin Fowler (and others).
 * 
 * @author Claus Nielsen
 *
 * @param  the type of the property.
 */
public class TemporalProperty implements ObservableProperty {

	NavigableMap values = newConcurrentSkipListMap();
	Set> listeners = Collections.newSetFromMap(new ConcurrentHashMap, Boolean>());

	/* (non-Javadoc)
	 * @see dk.clanie.properties.Property#set(java.lang.Object)
	 */
	@Override
	public void set(T value) {
		set(new Instant(), value);
	}

	/**
	 * Sets the value of the property at the specified time.
	 * 
	 * For internal and test use only.

* Values are assumed to be set in chronological order. * * @param effectiveFrom * @param value */ void set(Instant effectiveFrom, T value) { T oldValue = get(); values.put(effectiveFrom, value); notifyListeners(oldValue, value); } /** * Gets the value effective at the specified instant in time. * * Returns null if the property had no value at the specified time. * * @param effectivAt * @return the value of the property at the specified time. */ public T get(Instant effectivAt) { Entry entry = values.floorEntry(effectivAt); if (entry == null) return null; return entry.getValue(); } /* (non-Javadoc) * @see dk.clanie.properties.Property#get() */ @Override public T get() { return get(new Instant()); } /** * Removes values no longer in effect at the specified point in time. * * @param limit */ public void purge(Instant limit) { Instant firstKeeper = values.floorKey(limit); if (firstKeeper == null) return; Set purgables = values.headMap(firstKeeper).keySet(); if (purgables == null) return; for (Instant purgable : purgables) { values.remove(purgable); } } private void notifyListeners(T oldValue, T newValue) { for (PropertyChangeListener pcl : listeners) { pcl.propertyChanged(oldValue, newValue); } } @Override public void addChangeListener(PropertyChangeListener listener) { listeners.add(listener); } @Override public void removeChangeListener(PropertyChangeListener listener) { listeners.remove(listener); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy