/* Copyright 2011-2015 Yorba Foundation * * This software is licensed under the GNU Lesser General Public License * (version 2.1 or later). See the COPYING file in this distribution. */ namespace Core { // A TrackerAccumulator is called by Tracker indicating when a DataObject should be included or // unincluded in its accumulated data. All methods return true if their data has changed, // indicating that the Tracker's "updated" signal should be fired. public interface TrackerAccumulator : Object { public abstract bool include(DataObject object); public abstract bool uninclude(DataObject object); public abstract bool altered(DataObject object, Alteration alteration); } // A Tracker monitors a DataCollection and reports to an installed TrackerAccumulator when objects // are available and unavailable. This simplifies connecting to the DataCollection manually to // monitoring availability (or subclassing for similar reasons, which may not always be available). public class Tracker { protected delegate bool IncludeUnincludeObject(DataObject object); private DataCollection collection; private Gee.Collection<DataObject>? initial; private TrackerAccumulator? acc = null; public virtual signal void updated() { } public Tracker(DataCollection collection, Gee.Collection<DataObject>? initial = null) { this.collection = collection; this.initial = initial; } ~Tracker() { if (acc != null) { collection.items_added.disconnect(on_items_added); collection.items_removed.disconnect(on_items_removed); collection.items_altered.disconnect(on_items_altered); } } public void start(TrackerAccumulator acc) { // can only be started once assert(this.acc == null); this.acc = acc; collection.items_added.connect(on_items_added); collection.items_removed.connect(on_items_removed); collection.items_altered.connect(on_items_altered); if (initial != null && initial.size > 0) on_items_added(initial); else if (initial == null) on_items_added(collection.get_all()); initial = null; } public DataCollection get_collection() { return collection; } private void on_items_added(Gee.Iterable<DataObject> added) { include_uninclude(added, acc.include); } private void on_items_removed(Gee.Iterable<DataObject> removed) { include_uninclude(removed, acc.uninclude); } // Subclasses can use this as a utility method. protected void include_uninclude(Gee.Iterable<DataObject> objects, IncludeUnincludeObject cb) { bool fire_updated = false; foreach (DataObject object in objects) fire_updated = cb(object) || fire_updated; if (fire_updated) updated(); } private void on_items_altered(Gee.Map<DataObject, Alteration> map) { bool fire_updated = false; foreach (DataObject object in map.keys) fire_updated = acc.altered(object, map.get(object)) || fire_updated; if (fire_updated) updated(); } } // A ViewTracker is Tracker designed for ViewCollections. It uses an internal mux to route // Tracker's calls to three TrackerAccumulators: all (all objects in the ViewCollection), selected // (only for selected objects) and visible (only for items not hidden or filtered out). public class ViewTracker : Tracker { private class Mux : Object, TrackerAccumulator { public TrackerAccumulator? all; public TrackerAccumulator? visible; public TrackerAccumulator? selected; public Mux(TrackerAccumulator? all, TrackerAccumulator? visible, TrackerAccumulator? selected) { this.all = all; this.visible = visible; this.selected = selected; } public bool include(DataObject object) { DataView view = (DataView) object; bool fire_updated = false; if (all != null) fire_updated = all.include(view) || fire_updated; if (visible != null && view.is_visible()) fire_updated = visible.include(view) || fire_updated; if (selected != null && view.is_selected()) fire_updated = selected.include(view) || fire_updated; return fire_updated; } public bool uninclude(DataObject object) { DataView view = (DataView) object; bool fire_updated = false; if (all != null) fire_updated = all.uninclude(view) || fire_updated; if (visible != null && view.is_visible()) fire_updated = visible.uninclude(view) || fire_updated; if (selected != null && view.is_selected()) fire_updated = selected.uninclude(view) || fire_updated; return fire_updated; } public bool altered(DataObject object, Alteration alteration) { DataView view = (DataView) object; bool fire_updated = false; if (all != null) fire_updated = all.altered(view, alteration) || fire_updated; if (visible != null && view.is_visible()) fire_updated = visible.altered(view, alteration) || fire_updated; if (selected != null && view.is_selected()) fire_updated = selected.altered(view, alteration) || fire_updated; return fire_updated; } } private Mux? mux = null; public ViewTracker(ViewCollection collection) { base (collection, collection.get_all_unfiltered()); } ~ViewTracker() { if (mux != null) { ViewCollection? collection = get_collection() as ViewCollection; assert(collection != null); collection.items_shown.disconnect(on_items_shown); collection.items_hidden.disconnect(on_items_hidden); collection.items_selected.disconnect(on_items_selected); collection.items_unselected.disconnect(on_items_unselected); } } public new void start(TrackerAccumulator? all, TrackerAccumulator? visible, TrackerAccumulator? selected) { assert(mux == null); mux = new Mux(all, visible, selected); ViewCollection? collection = get_collection() as ViewCollection; assert(collection != null); collection.items_shown.connect(on_items_shown); collection.items_hidden.connect(on_items_hidden); collection.items_selected.connect(on_items_selected); collection.items_unselected.connect(on_items_unselected); base.start(mux); } private void on_items_shown(Gee.Collection<DataView> shown) { if (mux.visible != null) include_uninclude(shown, mux.visible.include); } private void on_items_hidden(Gee.Collection<DataView> hidden) { if (mux.visible != null) include_uninclude(hidden, mux.visible.uninclude); } private void on_items_selected(Gee.Iterable<DataView> selected) { if (mux.selected != null) include_uninclude(selected, mux.selected.include); } private void on_items_unselected(Gee.Iterable<DataView> unselected) { if (mux.selected != null) include_uninclude(unselected, mux.selected.uninclude); } } }