diff options
Diffstat (limited to 'app/src/main/java')
-rw-r--r-- | app/src/main/java/io/trygvis/android/bt/DefaultBtService.java | 4 | ||||
-rw-r--r-- | app/src/main/java/io/trygvis/soilmoisture/DefaultSoilMoistureService.java (renamed from app/src/main/java/io/trygvis/soilmoisture/DefaultSmDevicesManager.java) | 42 | ||||
-rw-r--r-- | app/src/main/java/io/trygvis/soilmoisture/MainActivity.java | 234 | ||||
-rw-r--r-- | app/src/main/java/io/trygvis/soilmoisture/SmDevice.java | 11 | ||||
-rw-r--r-- | app/src/main/java/io/trygvis/soilmoisture/SoilMoistureService.java (renamed from app/src/main/java/io/trygvis/soilmoisture/SmDevicesManager.java) | 10 | ||||
-rw-r--r-- | app/src/main/java/io/trygvis/soilmoisture/SoilMonitor.java | 35 |
6 files changed, 260 insertions, 76 deletions
diff --git a/app/src/main/java/io/trygvis/android/bt/DefaultBtService.java b/app/src/main/java/io/trygvis/android/bt/DefaultBtService.java index 5e13a1f..3c14c1f 100644 --- a/app/src/main/java/io/trygvis/android/bt/DefaultBtService.java +++ b/app/src/main/java/io/trygvis/android/bt/DefaultBtService.java @@ -179,6 +179,10 @@ public class DefaultBtService<A extends BtDevice.BtDeviceWrapper<A>> extends Ser return binder; } + /** + * TODO: move this to initialize or somewhere it can be called so it doesn't block the UI + * thread. + */ @Override public void onCreate() { Bundle data; diff --git a/app/src/main/java/io/trygvis/soilmoisture/DefaultSmDevicesManager.java b/app/src/main/java/io/trygvis/soilmoisture/DefaultSoilMoistureService.java index 301bfc7..8a36476 100644 --- a/app/src/main/java/io/trygvis/soilmoisture/DefaultSmDevicesManager.java +++ b/app/src/main/java/io/trygvis/soilmoisture/DefaultSoilMoistureService.java @@ -25,19 +25,18 @@ import io.trygvis.bluetooth.TrygvisIoUuids; import static io.trygvis.android.bt.BtService.BtServiceListenerBroadcastReceiver; -public class DefaultSmDevicesManager extends Service implements SmDevicesManager { - private final static String TAG = DefaultSmDevicesManager.class.getSimpleName(); +public class DefaultSoilMoistureService extends Service implements SoilMoistureService { + private final static String TAG = DefaultSoilMoistureService.class.getSimpleName(); private final IBinder binder = new LocalBinder<>(this); @SuppressWarnings("UnusedDeclaration") - private final DefaultSmDevicesManager context = DefaultSmDevicesManager.this; + private final DefaultSoilMoistureService context = DefaultSoilMoistureService.this; private ServiceConnection serviceConnection; private BtService<SmDevice> btService; - @Override public IBinder onBind(Intent intent) { return binder; @@ -52,7 +51,7 @@ public class DefaultSmDevicesManager extends Service implements SmDevicesManager @Override public void onServiceConnected(ComponentName componentName, IBinder service) { btService = ((LocalBinder<BtService<SmDevice>>) service).getService(); - boolean ok = btService.initialize(DefaultSmDevicesManager.this::onNewDevice); + boolean ok = btService.initialize(DefaultSoilMoistureService.this::onNewDevice); sendBroadcast(createReady(ok)); } @@ -111,25 +110,20 @@ public class DefaultSmDevicesManager extends Service implements SmDevicesManager BluetoothGattService service = gatt.getService(TrygvisIoUuids.Services.SOIL_MOISTURE_SERVICE); - if (service == null) { - smDevice.setIsUseful(false); - return false; - } + boolean useful = false; + if (service != null) { + BluetoothGattCharacteristic characteristic = service.getCharacteristic(TrygvisIoUuids.Characteristics.SOIL_MOISTURE); - BluetoothGattCharacteristic characteristic = service.getCharacteristic(TrygvisIoUuids.Characteristics.SOIL_MOISTURE); + useful = characteristic != null; - if (characteristic == null) { - smDevice.setIsUseful(false); - return false; + smDevice.setIsUseful(useful); } - smDevice.setIsUseful(true); - sendBroadcast(createNewDevice(address)); - gatt.disconnect(); - return true; - }); + return false; + }). + onFinally(() -> sendBroadcast(createNewDevice(address))); btDevice.connect(executor); } else { @@ -172,7 +166,7 @@ public class DefaultSmDevicesManager extends Service implements SmDevicesManager } // ----------------------------------------------------------------------- - // + // Event creation and dispatching // ----------------------------------------------------------------------- private SmDevice onNewDevice(BtDevice<SmDevice> btDevice) { @@ -180,28 +174,28 @@ public class DefaultSmDevicesManager extends Service implements SmDevicesManager } private Intent createReady(boolean success) { - return new Intent(SmDeviceListener.INTENT_NAME). + return new Intent(SoilMoistureListener.INTENT_NAME). putExtra("event", "ready"). putExtra("success", success); } private Intent createScanStarted() { - return new Intent(SmDeviceListener.INTENT_NAME). + return new Intent(SoilMoistureListener.INTENT_NAME). putExtra("event", "scanStarted"); } private Intent createScanStopped() { - return new Intent(SmDeviceListener.INTENT_NAME). + return new Intent(SoilMoistureListener.INTENT_NAME). putExtra("event", "scanStopped"); } private Intent createNewDevice(String address) { - return new Intent(SmDeviceListener.INTENT_NAME). + return new Intent(SoilMoistureListener.INTENT_NAME). putExtra("event", "newDevice"). putExtra("address", address); } - public static void dispatchEvent(Intent intent, SmDeviceListener listener) { + public static void dispatchEvent(Intent intent, SoilMoistureListener listener) { String event = intent.getStringExtra("event"); Log.i(TAG, "Dispatching event " + intent.getAction() + "/" + event); switch (event) { diff --git a/app/src/main/java/io/trygvis/soilmoisture/MainActivity.java b/app/src/main/java/io/trygvis/soilmoisture/MainActivity.java index 6adc96c..64e8202 100644 --- a/app/src/main/java/io/trygvis/soilmoisture/MainActivity.java +++ b/app/src/main/java/io/trygvis/soilmoisture/MainActivity.java @@ -6,6 +6,8 @@ import android.app.ProgressDialog; import android.content.ComponentName; import android.content.Intent; import android.content.ServiceConnection; +import android.database.DataSetObservable; +import android.database.DataSetObserver; import android.os.Bundle; import android.os.IBinder; import android.util.Log; @@ -14,8 +16,8 @@ import android.view.Menu; import android.view.MenuItem; import android.view.View; import android.view.ViewGroup; -import android.widget.BaseAdapter; import android.widget.Button; +import android.widget.ListAdapter; import android.widget.ListView; import android.widget.ProgressBar; import android.widget.TextView; @@ -31,7 +33,7 @@ import io.trygvis.android.LocalBinder; import io.trygvis.android.bt.BtActivitySupport; import static io.trygvis.soilmoisture.ExceptionHandler.EXCEPTION_HANDLER; -import static io.trygvis.soilmoisture.SmDevicesManager.SmDeviceListener; +import static io.trygvis.soilmoisture.SoilMoistureService.SoilMoistureListener; import static java.lang.String.valueOf; public class MainActivity extends ListActivity { @@ -42,12 +44,12 @@ public class MainActivity extends ListActivity { private static final int REQUEST_ENABLE_BT = 1; private final BtActivitySupport btActivitySupport = new BtActivitySupport(this, REQUEST_ENABLE_BT); - private final SmDeviceListener serviceListener = new MySmDeviceListener(); + private final SoilMoistureListener serviceListener = new MySoilMoistureListener(); private final MainActivity context = this; private DeviceListAdapter deviceList; private ServiceConnection serviceConnection; - private SmDevicesManager smDevicesManager; + private SoilMoistureService soilMoistureService; private ProgressDialog initializing; private boolean ready; @@ -73,22 +75,24 @@ public class MainActivity extends ListActivity { @Override public void onServiceConnected(ComponentName componentName, IBinder service) { Log.i(TAG, "onServiceConnected"); - smDevicesManager = ((LocalBinder<SmDevicesManager>) service).getService(); - registerReceiver(serviceListener, SmDeviceListener.INTENT_FILTER); + soilMoistureService = ((LocalBinder<SoilMoistureService>) service).getService(); + registerReceiver(serviceListener, SoilMoistureListener.INTENT_FILTER); } @Override public void onServiceDisconnected(ComponentName componentName) { Log.i(TAG, "onServiceDisconnected"); - smDevicesManager = null; + soilMoistureService = null; stopScan(); } }; - bindService(new Intent(this, DefaultSmDevicesManager.class), serviceConnection, BIND_AUTO_CREATE); + bindService(new Intent(this, DefaultSoilMoistureService.class), serviceConnection, BIND_AUTO_CREATE); initializing = ProgressDialog. show(this, "Initializing", "Connecting to Bluetooth system.", true); + + setContentView(R.layout.main); } @Override @@ -112,7 +116,7 @@ public class MainActivity extends ListActivity { return; } - registerReceiver(serviceListener, SmDeviceListener.INTENT_FILTER); + registerReceiver(serviceListener, SoilMoistureListener.INTENT_FILTER); } @Override @@ -141,20 +145,30 @@ public class MainActivity extends ListActivity { // Inflate the menu; this adds items to the action bar if it is present. getMenuInflater().inflate(R.menu.main, menu); + MenuItem stop = menu.findItem(R.id.menu_stop); + MenuItem scan = menu.findItem(R.id.menu_scan); + MenuItem refresh = menu.findItem(R.id.menu_refresh); + MenuItem showAll = menu.findItem(R.id.menu_show_all); + MenuItem groupByDevice = menu.findItem(R.id.menu_group_by_device); + if (ready) { - if (!smDevicesManager.isScanning()) { - menu.findItem(R.id.menu_stop).setVisible(false); - menu.findItem(R.id.menu_scan).setVisible(true); - menu.findItem(R.id.menu_refresh).setActionView(null); + if (!soilMoistureService.isScanning()) { + stop.setVisible(false); + scan.setVisible(true); + refresh.setActionView(null); } else { - menu.findItem(R.id.menu_stop).setVisible(true); - menu.findItem(R.id.menu_scan).setVisible(false); - menu.findItem(R.id.menu_refresh).setActionView(R.layout.actionbar_indeterminate_progress); + stop.setVisible(true); + scan.setVisible(false); + refresh.setActionView(R.layout.actionbar_indeterminate_progress); } + showAll.setChecked(deviceList.isShowAll()); + groupByDevice.setChecked(deviceList.isGroupByDevice()); } else { - menu.findItem(R.id.menu_stop).setVisible(false); - menu.findItem(R.id.menu_scan).setVisible(true); - menu.findItem(R.id.menu_refresh).setActionView(null); + stop.setVisible(false); + scan.setVisible(true); + refresh.setActionView(null); + showAll.setVisible(false); + groupByDevice.setVisible(false); } return true; } @@ -163,6 +177,7 @@ public class MainActivity extends ListActivity { public boolean onOptionsItemSelected(MenuItem item) { Log.i(TAG, "onOptionsItemSelected"); + boolean consumed = true; switch (item.getItemId()) { case R.id.menu_scan: startScan(); @@ -170,39 +185,47 @@ public class MainActivity extends ListActivity { case R.id.menu_stop: stopScan(); break; + case R.id.menu_show_all: + item.setChecked(!item.isChecked()); + deviceList.setShowAll(item.isChecked()); + break; + case R.id.menu_group_by_device: + item.setChecked(!item.isChecked()); + deviceList.setGroupByDevice(item.isChecked()); + break; + default: + consumed = super.onOptionsItemSelected(item); } - return super.onOptionsItemSelected(item); + return consumed; } private void startScan() { - smDevicesManager.startScanning(SCAN_PERIOD); + soilMoistureService.startScanning(SCAN_PERIOD); } private void stopScan() { - if (smDevicesManager != null) { - smDevicesManager.stopScanning(); + if (soilMoistureService != null) { + soilMoistureService.stopScanning(); } } @Override protected void onListItemClick(ListView l, View v, int position, long id) { stopScan(); - -// SmDevice state = smDevicesManager.getDevices(SmDevice.deviceComparator).get(position); } // ----------------------------------------------------------------------- // // ----------------------------------------------------------------------- - static class DeviceListItem { + static class DeviceItem { final TextView deviceName; final TextView deviceAddress; final TextView rssi; final ProgressBar spinner; final Button connect; - DeviceListItem(View view) { + DeviceItem(View view) { this.deviceName = (TextView) view.findViewById(R.id.device_name); this.deviceAddress = (TextView) view.findViewById(R.id.device_address); this.rssi = (TextView) view.findViewById(R.id.device_rssi); @@ -211,39 +234,162 @@ public class MainActivity extends ListActivity { } } - private class DeviceListAdapter extends BaseAdapter { + private class DeviceListAdapter implements ListAdapter { + private final DataSetObservable dataSetObservable = new DataSetObservable(); private List<SmDevice> devices = new ArrayList<>(); + private List<Object> current = new ArrayList<>(); private LayoutInflater inflater = MainActivity.this.getLayoutInflater(); + private boolean groupByDevice = true; + private boolean showAll = false; + + public void sort() { + Log.i(TAG, "sort(), groupByDevice=" + groupByDevice + ", showAll=" + showAll); + current = new ArrayList<>(); + + List<SmDevice> usefulDevices = new ArrayList<>(devices.size()); + List<SmDevice> unusefulDevices = new ArrayList<>(devices.size()); + for (SmDevice d : devices) { + (d.isUseful() ? usefulDevices : unusefulDevices).add(d); + } + List<SoilMonitor> monitors = new ArrayList<>(); + for (SmDevice d : devices) { + monitors.addAll(d.getMonitors()); + } + + if (groupByDevice) { + current.addAll(usefulDevices); + if (showAll) { + current.addAll(unusefulDevices); + } + } else { + current.addAll(monitors); + } + dataSetObservable.notifyChanged(); + } + + public void notifyDataSetChanged() { + dataSetObservable.notifyChanged(); + } + + public void setShowAll(boolean showAll) { + if (showAll == this.showAll) { + return; + } + + this.showAll = showAll; + sort(); + } + + public boolean isShowAll() { + return showAll; + } + + public void setGroupByDevice(boolean groupByDevice) { + if (groupByDevice == this.groupByDevice) { + return; + } + + this.groupByDevice = groupByDevice; + sort(); + } + + public boolean isGroupByDevice() { + return groupByDevice; + } + + // ----------------------------------------------------------------------- + // ListAdapter Implementation + // ----------------------------------------------------------------------- + + @Override + public void registerDataSetObserver(DataSetObserver observer) { + dataSetObservable.registerObserver(observer); + } + + @Override + public void unregisterDataSetObserver(DataSetObserver observer) { + dataSetObservable.unregisterObserver(observer); + } + + @Override + public boolean areAllItemsEnabled() { + return true; + } + + @Override + public boolean isEnabled(int position) { + return true; + } + + @Override + public boolean hasStableIds() { + return true; + } + + @Override + public int getItemViewType(int position) { + Object o = current.get(position); + + if (o instanceof SmDevice) { + return 0; + } else if (o instanceof SoilMonitor) { + return 1; + } + + throw new RuntimeException("Unknown kind: " + o.getClass()); + } + + @Override + public int getViewTypeCount() { + return 2; + } + + @Override + public boolean isEmpty() { + return current.isEmpty(); + } + @Override public int getCount() { - return devices.size(); + return current.size(); } @Override - public SmDevice getItem(int i) { - return devices.get(i); + public Object getItem(int position) { + return current.get(position); } @Override - public long getItemId(int i) { - return i; + public long getItemId(int position) { + return position; } @Override - public View getView(int i, View view, ViewGroup viewGroup) { - DeviceListItem item; + public View getView(int position, View view, ViewGroup viewGroup) { + Object o = current.get(position); + if (o instanceof SmDevice) { + return getSmDeviceView((SmDevice) o, view); + } else if (o instanceof SoilMonitor) { + return getSoilMonitorView((SoilMonitor) o, view); + } + + throw new RuntimeException("Not implemented"); + } + + private View getSmDeviceView(SmDevice smDevice, View view) { + + DeviceItem item; if (view == null) { view = inflater.inflate(R.layout.listitem_device, null); - item = new DeviceListItem(view); + item = new DeviceItem(view); view.setTag(item); view.setClickable(false); } else { - item = (DeviceListItem) view.getTag(); + item = (DeviceItem) view.getTag(); } - SmDevice smDevice = getItem(i); if (smDevice.getName() != null) { item.deviceName.setText(smDevice.getName()); } else { @@ -261,9 +407,13 @@ public class MainActivity extends ListActivity { return view; } + + private View getSoilMonitorView(SoilMonitor soilMonitor, View view) { + throw new RuntimeException("Not implemented"); + } } - private class MySmDeviceListener extends SmDeviceListener { + private class MySoilMoistureListener extends SoilMoistureListener { @Override public void onReady(boolean ok) { if (!ok) { @@ -297,11 +447,9 @@ public class MainActivity extends ListActivity { @Override public void onNewDevice(String address) { - SmDevice device = smDevicesManager.getDevice(address); + SmDevice device = soilMoistureService.getDevice(address); deviceList.devices.add(device); - deviceList.notifyDataSetInvalidated(); - - Log.i(TAG, "deviceList.devices.size() = " + deviceList.devices.size()); + deviceList.sort(); } } } diff --git a/app/src/main/java/io/trygvis/soilmoisture/SmDevice.java b/app/src/main/java/io/trygvis/soilmoisture/SmDevice.java index 1ed7ecb..169f4b3 100644 --- a/app/src/main/java/io/trygvis/soilmoisture/SmDevice.java +++ b/app/src/main/java/io/trygvis/soilmoisture/SmDevice.java @@ -2,13 +2,12 @@ package io.trygvis.soilmoisture; import android.util.Log; -import java.util.Comparator; +import java.util.ArrayList; +import java.util.List; import io.trygvis.android.bt.BtDevice; class SmDevice implements BtDevice.BtDeviceWrapper<SmDevice> { - public static final Comparator<SmDevice> deviceComparator = (a, b) -> a.getBtDevice().getAddress().compareTo(b.getBtDevice().getAddress()); - private final static String TAG = SmDevice.class.getSimpleName(); private final BtDevice<SmDevice> btDevice; @@ -17,6 +16,8 @@ class SmDevice implements BtDevice.BtDeviceWrapper<SmDevice> { private Boolean isUseful; + private List<SoilMonitor> monitors = new ArrayList<>(); + public SmDevice(BtDevice<SmDevice> btDevice) { this.btDevice = btDevice; Log.i(TAG, "new device"); @@ -52,4 +53,8 @@ class SmDevice implements BtDevice.BtDeviceWrapper<SmDevice> { public String getName() { return name; } + + public List<SoilMonitor> getMonitors() { + return monitors; + } } diff --git a/app/src/main/java/io/trygvis/soilmoisture/SmDevicesManager.java b/app/src/main/java/io/trygvis/soilmoisture/SoilMoistureService.java index 531061a..8100649 100644 --- a/app/src/main/java/io/trygvis/soilmoisture/SmDevicesManager.java +++ b/app/src/main/java/io/trygvis/soilmoisture/SoilMoistureService.java @@ -8,9 +8,7 @@ import android.content.IntentFilter; import java.util.Comparator; import java.util.List; -import io.trygvis.android.bt.BtDevice; - -public interface SmDevicesManager { +public interface SoilMoistureService { List<SmDevice> getDevices(Comparator<SmDevice> comparator); SmDevice getDevice(String address); @@ -21,9 +19,9 @@ public interface SmDevicesManager { void stopScanning(); - public abstract static class SmDeviceListener extends BroadcastReceiver { + public abstract static class SoilMoistureListener extends BroadcastReceiver { - public static final String INTENT_NAME = SmDeviceListener.class.getName(); + public static final String INTENT_NAME = SoilMoistureListener.class.getName(); public static final IntentFilter INTENT_FILTER = new IntentFilter(INTENT_NAME); @@ -32,7 +30,7 @@ public interface SmDevicesManager { return; } - DefaultSmDevicesManager.dispatchEvent(intent, this); + DefaultSoilMoistureService.dispatchEvent(intent, this); } public void onReady(boolean ok) { diff --git a/app/src/main/java/io/trygvis/soilmoisture/SoilMonitor.java b/app/src/main/java/io/trygvis/soilmoisture/SoilMonitor.java new file mode 100644 index 0000000..564202d --- /dev/null +++ b/app/src/main/java/io/trygvis/soilmoisture/SoilMonitor.java @@ -0,0 +1,35 @@ +package io.trygvis.soilmoisture; + +import java.util.Date; + +class SoilMonitor { + private final SmDevice device; + + private Date timestamp; + + private int lastValue; + + SoilMonitor(SmDevice device) { + this.device = device; + } + + public SmDevice getDevice() { + return device; + } + + public int getLastValue() { + return lastValue; + } + + public void setLastValue(int lastValue) { + this.lastValue = lastValue; + } + + public Date getTimestamp() { + return timestamp; + } + + public void setTimestamp(Date timestamp) { + this.timestamp = timestamp; + } +} |