/* * Copyright (c) 2013, Will Szumski * Copyright (c) 2013, Doug Szumski * * This file is part of Cyclismo. * * Cyclismo 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. * * Cyclismo 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 Cyclismo. If not, see <http://www.gnu.org/licenses/>. */ /* * Copyright 2011 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ package org.cowboycoders.cyclismo.content; import android.content.SharedPreferences; import android.content.SharedPreferences.OnSharedPreferenceChangeListener; import android.database.ContentObserver; import android.hardware.Sensor; import android.hardware.SensorEvent; import android.hardware.SensorEventListener; import android.location.Location; import android.location.LocationListener; import android.location.LocationManager; import android.location.LocationProvider; import android.os.Bundle; import android.os.Handler; import android.util.Log; import com.google.common.annotations.VisibleForTesting; import java.util.EnumSet; import java.util.Set; /** * Data source manager. Creates observers/listeners and manages their * registration with {@link DataSource}. The observers/listeners calls * {@link DataSourceListener} when data changes. * * @author Rodrigo Damazio */ public class DataSourceManager { private static final String TAG = DataSourceManager.class.getSimpleName(); /** * Observer when the tracks table is updated. * * @author Jimmy Shih */ private class TracksTableObserver extends ContentObserver { public TracksTableObserver() { super(handler); } @Override public void onChange(boolean selfChange) { dataSourceListener.notifyTracksTableUpdated(); } } /** * Observer when the waypoints table is updated. * * @author Jimmy Shih */ private class WaypointsTableObserver extends ContentObserver { public WaypointsTableObserver() { super(handler); } @Override public void onChange(boolean selfChange) { dataSourceListener.notifyWaypointsTableUpdated(); } } /** * Observer when the track points table is updated. * * @author Jimmy Shih */ private class TrackPointsTableObserver extends ContentObserver { public TrackPointsTableObserver() { super(handler); } @Override public void onChange(boolean selfChange) { dataSourceListener.notifyTrackPointsTableUpdated(); } } /** * Listener for location changes. * * @author Jimmy Shih */ @VisibleForTesting class CurrentLocationListener implements LocationListener { @Override public void onLocationChanged(Location location) { dataSourceListener.notifyLocationChanged(location); } @Override public void onProviderDisabled(String provider) { if (!LocationManager.GPS_PROVIDER.equals(provider)) { return; } dataSourceListener.notifyLocationProviderEnabled(false); } @Override public void onProviderEnabled(String provider) { if (!dataSource.isAllowed() || !LocationManager.GPS_PROVIDER.equals(provider)) { return; } dataSourceListener.notifyLocationProviderEnabled(true); } @Override public void onStatusChanged(String provider, int status, Bundle extras) { if (!dataSource.isAllowed() || !LocationManager.GPS_PROVIDER.equals(provider)) { return; } dataSourceListener.notifyLocationProviderAvailable(status == LocationProvider.AVAILABLE); } } /** * Listener for heading changes. * * @author Jimmy Shih */ @VisibleForTesting class HeadingListener implements SensorEventListener { @Override public void onAccuracyChanged(Sensor sensor, int accuracy) { // Do nothing } @Override public void onSensorChanged(SensorEvent event) { dataSourceListener.notifyHeadingChanged(event.values[0]); } } /** * Listener for preference changes. * * @author Jimmy Shih */ private class PreferenceListener implements OnSharedPreferenceChangeListener { @Override public void onSharedPreferenceChanged(SharedPreferences sharedPreferences, String key) { dataSourceListener.notifyPreferenceChanged(key); } } private final DataSource dataSource; private final DataSourceListener dataSourceListener; // Registered listeners private final Set<TrackDataType> registeredListeners = EnumSet.noneOf(TrackDataType.class); private final Handler handler; private final TracksTableObserver tracksTableObserver; private final WaypointsTableObserver waypointsTableObserver; private final TrackPointsTableObserver trackPointsTableObserver; private final CurrentLocationListener currentLocationListener; private final HeadingListener headingListener; private final PreferenceListener preferenceListener; public DataSourceManager(DataSource dataSource, DataSourceListener dataSourceListener) { this.dataSource = dataSource; this.dataSourceListener = dataSourceListener; handler = new Handler(); tracksTableObserver = new TracksTableObserver(); waypointsTableObserver = new WaypointsTableObserver(); trackPointsTableObserver = new TrackPointsTableObserver(); currentLocationListener = new CurrentLocationListener(); headingListener = new HeadingListener(); preferenceListener = new PreferenceListener(); } /** * Updates listeners with data source. * * @param listeners the listeners */ public void updateListeners(EnumSet<TrackDataType> listeners) { EnumSet<TrackDataType> neededListeners = EnumSet.copyOf(listeners); /* * Map SAMPLED_OUT_POINT_UPDATES to POINT_UPDATES since they correspond to * the same internal listener */ if (neededListeners.contains(TrackDataType.SAMPLED_OUT_TRACK_POINTS_TABLE)) { neededListeners.remove(TrackDataType.SAMPLED_OUT_TRACK_POINTS_TABLE); neededListeners.add(TrackDataType.SAMPLED_IN_TRACK_POINTS_TABLE); } Log.d(TAG, "Updating listeners " + neededListeners); // Unnecessary = registered - needed Set<TrackDataType> unnecessaryListeners = EnumSet.copyOf(registeredListeners); unnecessaryListeners.removeAll(neededListeners); // Missing = needed - registered Set<TrackDataType> missingListeners = EnumSet.copyOf(neededListeners); missingListeners.removeAll(registeredListeners); // Remove unnecessary listeners for (TrackDataType trackDataType : unnecessaryListeners) { unregisterListener(trackDataType); } // Add missing listeners for (TrackDataType trackDataType : missingListeners) { registerListener(trackDataType); } // Update registered listeners registeredListeners.clear(); registeredListeners.addAll(neededListeners); } /** * Registers a listener with data source. * * @param trackDataType the listener data type */ private void registerListener(TrackDataType trackDataType) { switch (trackDataType) { case SELECTED_TRACK: // Do nothing break; case TRACKS_TABLE: dataSource.registerContentObserver(TracksColumns.CONTENT_URI, tracksTableObserver); break; case WAYPOINTS_TABLE: dataSource.registerContentObserver(WaypointsColumns.CONTENT_URI, waypointsTableObserver); break; case SAMPLED_IN_TRACK_POINTS_TABLE: dataSource.registerContentObserver( TrackPointsColumns.CONTENT_URI, trackPointsTableObserver); break; case SAMPLED_OUT_TRACK_POINTS_TABLE: // Do nothing. SAMPLED_OUT_POINT_UPDATES is mapped to POINT_UPDATES. break; case LOCATION: dataSource.registerSimulatedLocationListener(currentLocationListener); break; case HEADING: dataSource.registerHeadingListener(headingListener); break; case PREFERENCE: dataSource.registerOnSharedPreferenceChangeListener(preferenceListener); break; default: break; } } /** * Unregisters a listener with data source. * * @param trackDataType listener data type */ private void unregisterListener(TrackDataType trackDataType) { switch (trackDataType) { case SELECTED_TRACK: // Do nothing break; case TRACKS_TABLE: dataSource.unregisterContentObserver(tracksTableObserver); break; case WAYPOINTS_TABLE: dataSource.unregisterContentObserver(waypointsTableObserver); break; case SAMPLED_IN_TRACK_POINTS_TABLE: dataSource.unregisterContentObserver(trackPointsTableObserver); break; case SAMPLED_OUT_TRACK_POINTS_TABLE: // Do nothing. SAMPLED_OUT_POINT_UPDATES is mapped to POINT_UPDATES. break; case LOCATION: dataSource.unregisterLocationListener(currentLocationListener); break; case HEADING: dataSource.unregisterHeadingListener(headingListener); break; case PREFERENCE: dataSource.unregisterOnSharedPreferenceChangeListener(preferenceListener); break; default: break; } } /** * Unregisters all listeners with data source. */ public void unregisterAllListeners() { for (TrackDataType trackDataType : TrackDataType.values()) { unregisterListener(trackDataType); } } }