/** * * Funf: Open Sensing Framework * Copyright (C) 2010-2011 Nadav Aharony, Wei Pan, Alex Pentland. * Acknowledgments: Alan Gardner * Contact: nadav@media.mit.edu * * This file is part of Funf. * * Funf is free software: you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License as * published by the Free Software Foundation, either version 3 of * the License, or (at your option) any later version. * * Funf 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 Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with Funf. If not, see <http://www.gnu.org/licenses/>. * */ package edu.mit.media.funf.probe.builtin; import java.util.Collections; import java.util.HashSet; import java.util.Set; import edu.mit.media.funf.probe.Probe; import edu.mit.media.funf.probe.Probe.Base; /** * A probe that runs once returns data and then automatically unregisters its probes when the data stream is done. * While this probe is running, other DataListeners that register will be queued up for the next run, which will be when the current run stops. * * Extending this class is useful when your probe returns a data stream which has an end, like with ContentProvider probes. */ public abstract class ImpulseProbe extends Base { private Set<DataListener> queuedListeners = Collections.synchronizedSet(new HashSet<DataListener>()); protected Set<DataListener> getQueuedDataListeners() { return queuedListeners; } private StateListener listenerQueueProcessor = new StateListener() { @Override public void onStateChanged(Probe probe, State previousState) { if (previousState == State.RUNNING && getState() != State.RUNNING) { Set<DataListener> listeners = getDataListeners(); if (!listeners.isEmpty()) { DataListener[] listenerArray = new DataListener[listeners.size()]; listeners.toArray(listenerArray); unregisterListener(listenerArray); } synchronized (queuedListeners) { DataListener[] queuedListenerArray = new DataListener[queuedListeners.size()]; queuedListeners.toArray(queuedListenerArray); registerListener(queuedListenerArray); queuedListeners.clear(); } } } }; public ImpulseProbe() { addStateListener(listenerQueueProcessor); } @Override public void registerListener(DataListener... listeners) { if (listeners != null) { // TODO: may want to introduce a small delay in running the probe, in case register listener is called multiple times in a row if (getState() == State.RUNNING) { synchronized (queuedListeners) { Set<DataListener> currentListeners = getDataListeners(); for (DataListener listener : listeners) { if (!currentListeners.contains(listener)) { queuedListeners.add(listener); } } } } else { super.registerListener(listeners); } } } @Override public void unregisterListener(DataListener... listeners) { if (listeners != null) { synchronized (queuedListeners) { for (DataListener listener : listeners) { queuedListeners.remove(listener); } } } super.unregisterListener(listeners); } }