/* Copyright (C) 2015-2021 Andreas Shimokawa, Carsten Pfeiffer, Christian Fischer, Daniele Gobbetti, José Rebelo, Szymon Tomasz Stefanek This file is part of Gadgetbridge. Gadgetbridge is free software: you can redistribute it and/or modify it under the terms of the GNU Affero General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. Gadgetbridge 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 Affero General Public License for more details. You should have received a copy of the GNU Affero General Public License along with this program. If not, see . */ package nodomain.freeyourgadget.gadgetbridge.devices.miband; import android.annotation.TargetApi; import android.app.Activity; import android.bluetooth.BluetoothDevice; import android.bluetooth.le.ScanFilter; import android.content.Context; import android.net.Uri; import android.os.Build; import android.os.ParcelUuid; import androidx.annotation.NonNull; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import java.util.Collection; import java.util.Collections; import java.util.EnumSet; import de.greenrobot.dao.query.QueryBuilder; import nodomain.freeyourgadget.gadgetbridge.GBApplication; import nodomain.freeyourgadget.gadgetbridge.GBException; import nodomain.freeyourgadget.gadgetbridge.R; import nodomain.freeyourgadget.gadgetbridge.activities.devicesettings.DeviceSettingsPreferenceConst; import nodomain.freeyourgadget.gadgetbridge.devices.AbstractBLEDeviceCoordinator; import nodomain.freeyourgadget.gadgetbridge.devices.InstallHandler; import nodomain.freeyourgadget.gadgetbridge.devices.SampleProvider; import nodomain.freeyourgadget.gadgetbridge.entities.AbstractActivitySample; import nodomain.freeyourgadget.gadgetbridge.entities.DaoSession; import nodomain.freeyourgadget.gadgetbridge.entities.Device; import nodomain.freeyourgadget.gadgetbridge.entities.MiBandActivitySampleDao; import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice; import nodomain.freeyourgadget.gadgetbridge.impl.GBDeviceCandidate; import nodomain.freeyourgadget.gadgetbridge.model.ActivityUser; import nodomain.freeyourgadget.gadgetbridge.model.DeviceType; import nodomain.freeyourgadget.gadgetbridge.service.DeviceSupport; import nodomain.freeyourgadget.gadgetbridge.service.ServiceDeviceSupport; import nodomain.freeyourgadget.gadgetbridge.service.devices.miband.MiBandSupport; import nodomain.freeyourgadget.gadgetbridge.util.Prefs; import static nodomain.freeyourgadget.gadgetbridge.model.ActivityUser.PREF_USER_NAME; public class MiBandCoordinator extends AbstractBLEDeviceCoordinator { private static final Logger LOG = LoggerFactory.getLogger(MiBandCoordinator.class); public MiBandCoordinator() { } @NonNull @Override public Collection createBLEScanFilters() { ParcelUuid mi1Service = new ParcelUuid(MiBandService.UUID_SERVICE_MIBAND_SERVICE); ScanFilter filter = new ScanFilter.Builder().setServiceUuid(mi1Service).build(); return Collections.singletonList(filter); } @NonNull @Override public DeviceType getSupportedType(GBDeviceCandidate candidate) { String macAddress = candidate.getMacAddress().toUpperCase(); if (macAddress.startsWith(MiBandService.MAC_ADDRESS_FILTER_1_1A) || macAddress.startsWith(MiBandService.MAC_ADDRESS_FILTER_1S)) { return DeviceType.MIBAND; } if (candidate.supportsService(MiBandService.UUID_SERVICE_MIBAND_SERVICE) && !candidate.supportsService(MiBandService.UUID_SERVICE_MIBAND2_SERVICE)) { return DeviceType.MIBAND; } // and a heuristic try { BluetoothDevice device = candidate.getDevice(); if (isHealthWearable(device)) { String name = device.getName(); if (name != null && name.toUpperCase().startsWith(MiBandConst.MI_GENERAL_NAME_PREFIX.toUpperCase())) { return DeviceType.MIBAND; } } } catch (Exception ex) { LOG.error("unable to check device support", ex); } return DeviceType.UNKNOWN; } @Override protected void deleteDevice(GBDevice gbDevice, Device device, DaoSession session) throws GBException { Long deviceId = device.getId(); QueryBuilder qb = session.getMiBandActivitySampleDao().queryBuilder(); qb.where(MiBandActivitySampleDao.Properties.DeviceId.eq(deviceId)).buildDelete().executeDeleteWithoutDetachingEntities(); } @Override public DeviceType getDeviceType() { return DeviceType.MIBAND; } @Override public Class getPairingActivity() { return MiBandPairingActivity.class; } @Override public SampleProvider getSampleProvider(GBDevice device, DaoSession session) { return new MiBandSampleProvider(device, session); } @Override public InstallHandler findInstallHandler(Uri uri, Context context) { MiBandFWInstallHandler handler = new MiBandFWInstallHandler(uri, context); return handler.isValid() ? handler : null; } @Override public boolean supportsActivityDataFetching() { return true; } @Override public boolean supportsScreenshots() { return false; } @Override public int getAlarmSlotCount(GBDevice device) { return 3; } @Override public boolean supportsSmartWakeup(GBDevice device) { return true; } @Override public boolean supportsActivityTracking() { return true; } @Override public String getManufacturer() { return "Xiaomi"; } @Override public boolean supportsAppsManagement(final GBDevice device) { return false; } @Override public Class getAppsManagementActivity() { return null; } @Override public boolean supportsCalendarEvents() { return false; } @Override public boolean supportsRealtimeData() { return true; } @Override public boolean supportsWeather() { return false; } @Override public boolean supportsFindDevice() { return true; } public static boolean hasValidUserInfo() { String dummyMacAddress = MiBandService.MAC_ADDRESS_FILTER_1_1A + ":00:00:00"; try { UserInfo userInfo = getConfiguredUserInfo(dummyMacAddress); return true; } catch (IllegalArgumentException ex) { return false; } } /** * Returns the configured user info, or, if that is not available or invalid, * a default user info. * * @param miBandAddress */ public static UserInfo getAnyUserInfo(String miBandAddress) { try { return getConfiguredUserInfo(miBandAddress); } catch (Exception ex) { LOG.error("Error creating user info from settings, using default user instead: " + ex); return UserInfo.getDefault(miBandAddress); } } /** * Returns the user info from the user configured data in the preferences. * * @param miBandAddress * @throws IllegalArgumentException when the user info can not be created */ public static UserInfo getConfiguredUserInfo(String miBandAddress) throws IllegalArgumentException { ActivityUser activityUser = new ActivityUser(); Prefs prefs = GBApplication.getPrefs(); UserInfo info = UserInfo.create( miBandAddress, prefs.getString(PREF_USER_NAME, null), activityUser.getGender(), activityUser.getAge(), activityUser.getHeightCm(), activityUser.getWeightKg(), 0 ); return info; } public static int getWearLocation(String deviceAddress) throws IllegalArgumentException { int location = 0; //left hand Prefs prefs = new Prefs(GBApplication.getDeviceSpecificSharedPrefs(deviceAddress)); if ("right".equals(prefs.getString(DeviceSettingsPreferenceConst.PREF_WEARLOCATION, "left"))) { location = 1; // right hand } return location; } public static int getDeviceTimeOffsetHours(String deviceAddress) throws IllegalArgumentException { Prefs prefs = new Prefs(GBApplication.getDeviceSpecificSharedPrefs(deviceAddress)); return prefs.getInt(MiBandConst.PREF_MIBAND_DEVICE_TIME_OFFSET_HOURS, 0); } public static boolean getHeartrateSleepSupport(String deviceAddress) throws IllegalArgumentException { Prefs prefs = new Prefs(GBApplication.getDeviceSpecificSharedPrefs(deviceAddress)); return prefs.getBoolean(DeviceSettingsPreferenceConst.PREF_HEARTRATE_USE_FOR_SLEEP_DETECTION, false); } public static int getReservedAlarmSlots(String miBandAddress) throws IllegalArgumentException { Prefs prefs = new Prefs(GBApplication.getDeviceSpecificSharedPrefs(miBandAddress)); return prefs.getInt(DeviceSettingsPreferenceConst.PREF_RESERVER_ALARMS_CALENDAR, 0); } @Override public boolean supportsHeartRateMeasurement(GBDevice device) { String hwVersion = device.getModel(); return isMi1S(hwVersion) || isMiPro(hwVersion); } @Override public int[] getSupportedDeviceSpecificSettings(GBDevice device) { return new int[]{ R.xml.devicesettings_wearlocation, R.xml.devicesettings_heartrate_sleep, R.xml.devicesettings_lowlatency_fwupdate, R.xml.devicesettings_reserve_alarms_calendar, R.xml.devicesettings_fake_timeoffset }; } @NonNull @Override public Class getDeviceSupportClass() { return MiBandSupport.class; } private boolean isMi1S(String hardwareVersion) { return MiBandConst.MI_1S.equals(hardwareVersion); } private boolean isMiPro(String hardwareVersion) { return MiBandConst.MI_PRO.equals(hardwareVersion); } @Override public EnumSet getInitialFlags() { return EnumSet.of(ServiceDeviceSupport.Flags.THROTTLING, ServiceDeviceSupport.Flags.BUSY_CHECKING); } }