/* Copyright (C) 2024 Damien Gaignon, Martin.JM, Vitalii Tomin 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.huawei; import android.content.Context; import android.content.SharedPreferences; import java.nio.ByteBuffer; import java.util.ArrayList; import java.util.List; import java.util.Map; import java.util.TreeMap; import org.slf4j.LoggerFactory; import org.slf4j.Logger; import nodomain.freeyourgadget.gadgetbridge.GBApplication; import nodomain.freeyourgadget.gadgetbridge.R; import nodomain.freeyourgadget.gadgetbridge.activities.devicesettings.DeviceSpecificSettings; import nodomain.freeyourgadget.gadgetbridge.activities.devicesettings.DeviceSpecificSettingsScreen; import nodomain.freeyourgadget.gadgetbridge.devices.huawei.packets.Notifications; import nodomain.freeyourgadget.gadgetbridge.devices.huawei.packets.Notifications.NotificationConstraintsType; import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice; import nodomain.freeyourgadget.gadgetbridge.util.GB; import static nodomain.freeyourgadget.gadgetbridge.activities.devicesettings.DeviceSettingsPreferenceConst.*; public class HuaweiCoordinator { Logger LOG = LoggerFactory.getLogger(HuaweiCoordinator.class); TreeMap commandsPerService = new TreeMap<>(); // Each byte of expandCapabilities represent a "service" // Each bit in a "service" represent a feature so 1 or 0 is used to check is support or not byte[] expandCapabilities = null; byte notificationCapabilities = -0x01; ByteBuffer notificationConstraints = null; private final HuaweiCoordinatorSupplier parent; private boolean transactionCrypted=true; public HuaweiCoordinator(HuaweiCoordinatorSupplier parent) { this.parent = parent; for (String key : getCapabilitiesSharedPreferences().getAll().keySet()) { int service; try { service = Integer.parseInt(key); byte[] commands = GB.hexStringToByteArray(getCapabilitiesSharedPreferences().getString(key, "00")); this.commandsPerService.put(service, commands); } catch (NumberFormatException e) { if (key.equals("expandCapabilities")) this.expandCapabilities = GB.hexStringToByteArray(getCapabilitiesSharedPreferences().getString(key, "00")); if (key.equals("notificationCapabilities")) this.notificationCapabilities = (byte)getCapabilitiesSharedPreferences().getInt(key, -0x01); if (key.equals("notificationConstraints")) this.notificationConstraints = ByteBuffer.wrap(GB.hexStringToByteArray( getCapabilitiesSharedPreferences().getString( key, GB.hexdump(Notifications.defaultConstraints) ))); } } } private SharedPreferences getCapabilitiesSharedPreferences() { return GBApplication.getContext().getSharedPreferences("huawei_coordinator_capatilities" + parent.getDeviceType().name(), Context.MODE_PRIVATE); } private SharedPreferences getDeviceSpecificSharedPreferences(GBDevice gbDevice) { return GBApplication.getDeviceSpecificSharedPrefs(gbDevice.getAddress()); } public boolean getForceOption(GBDevice gbDevice, String option) { return getDeviceSpecificSharedPreferences(gbDevice).getBoolean(option, false); } private void saveCommandsForService(int service, byte[] commands) { commandsPerService.put(service, commands); getCapabilitiesSharedPreferences().edit().putString(String.valueOf(service), GB.hexdump(commands)).apply(); } public void saveExpandCapabilities(byte[] capabilities) { expandCapabilities = capabilities; getCapabilitiesSharedPreferences().edit().putString("expandCapabilities", GB.hexdump(capabilities)).apply(); } public void saveNotificationCapabilities(byte capabilities) { notificationCapabilities = capabilities; getCapabilitiesSharedPreferences().edit().putInt("notificationCapabilities", (int)capabilities).apply(); } public void saveNotificationConstraints(ByteBuffer constraints) { notificationConstraints = constraints; getCapabilitiesSharedPreferences().edit().putString("notificationConstraints", GB.hexdump(constraints.array())).apply(); } public void addCommandsForService(int service, byte[] commands) { if (!commandsPerService.containsKey(service)) { saveCommandsForService(service, commands); return; } byte[] saved = commandsPerService.get(service); if (saved == null) { saveCommandsForService(service, commands); return; } if (saved.length != commands.length) { saveCommandsForService(service, commands); return; } boolean changed = false; for (int i = 0; i < saved.length; i++) { if (saved[i] != commands[i]) { changed = true; break; } } if (changed) saveCommandsForService(service, commands); } public byte[] getCommandsForService(int service) { return commandsPerService.get(service); } // Print all Services ID and Commands ID public void printCommandsPerService() { StringBuilder msg = new StringBuilder(); for(Map.Entry entry : commandsPerService.entrySet()) { msg.append("ServiceID: ").append(Integer.toHexString(entry.getKey())).append(" => Commands: "); for (byte b: entry.getValue()) { msg.append(Integer.toHexString(b)).append(" "); } msg.append("\n"); } LOG.info(msg.toString()); } private boolean supportsCommandForService(int service, int command) { byte[] commands = commandsPerService.get(service); if (commands == null) return false; for (byte b : commands) if (b == (byte) command) return true; return false; } private boolean supportsExpandCapability(int which) { // capability is a number containing : // - the index of the "service" // - the real capability number if (which >= expandCapabilities.length * 8) { LOG.debug("Capability is not supported"); return false; } int capability = 1 << (which % 8); if ((expandCapabilities[which / 8] & capability) == capability) return true; return false; } private boolean supportsNotificationConstraint(byte which) { return notificationConstraints.get(which) == 0x01; } private int getNotificationConstraint(byte which) { return (int)notificationConstraints.getShort(which); } public DeviceSpecificSettings getDeviceSpecificSettings(final GBDevice device) { final DeviceSpecificSettings deviceSpecificSettings = new DeviceSpecificSettings(); // Health if (supportsInactivityWarnings()) deviceSpecificSettings.addRootScreen(DeviceSpecificSettingsScreen.HEALTH, R.xml.devicesettings_inactivity_sheduled); if (supportsTruSleep()) deviceSpecificSettings.addRootScreen(DeviceSpecificSettingsScreen.HEALTH, R.xml.devicesettings_trusleep); if (supportsHeartRate()) deviceSpecificSettings.addRootScreen(DeviceSpecificSettingsScreen.HEALTH, R.xml.devicesettings_heartrate_automatic_enable); if (supportsSPo2()) deviceSpecificSettings.addRootScreen(DeviceSpecificSettingsScreen.HEALTH, R.xml.devicesettings_spo_automatic_enable); // Notifications final List notifications = deviceSpecificSettings.addRootScreen(DeviceSpecificSettingsScreen.NOTIFICATIONS); notifications.add(R.xml.devicesettings_notifications_enable); if (supportsNotificationOnBluetoothLoss()) notifications.add(R.xml.devicesettings_disconnectnotification_noshed); if (supportsDoNotDisturb(device)) notifications.add(R.xml.devicesettings_donotdisturb_allday_liftwirst_notwear); // Workout if (supportsSendingGps()) deviceSpecificSettings.addRootScreen(DeviceSpecificSettingsScreen.WORKOUT, R.xml.devicesettings_workout_send_gps_to_band); // Other deviceSpecificSettings.addRootScreen(R.xml.devicesettings_find_phone); deviceSpecificSettings.addRootScreen(R.xml.devicesettings_disable_find_phone_with_dnd); deviceSpecificSettings.addRootScreen(R.xml.devicesettings_allow_accept_reject_calls); // Time if (supportsDateFormat()) { final List dateTime = deviceSpecificSettings.addRootScreen(DeviceSpecificSettingsScreen.DATE_TIME); dateTime.add(R.xml.devicesettings_dateformat); dateTime.add(R.xml.devicesettings_timeformat); } // Display if (supportsWearLocation(device)) deviceSpecificSettings.addRootScreen(DeviceSpecificSettingsScreen.DISPLAY, R.xml.devicesettings_wearlocation); if (supportsAutoWorkMode()) deviceSpecificSettings.addRootScreen(DeviceSpecificSettingsScreen.DISPLAY, R.xml.devicesettings_workmode); if (supportsActivateOnLift()) deviceSpecificSettings.addRootScreen(DeviceSpecificSettingsScreen.DISPLAY, R.xml.devicesettings_liftwrist_display_noshed); if (supportsRotateToCycleInfo()) deviceSpecificSettings.addRootScreen(DeviceSpecificSettingsScreen.DISPLAY, R.xml.devicesettings_rotatewrist_cycleinfo); // Currently on main setting menu. /*if (supportsLanguageSetting()) deviceSpecificSettings.addRootScreen(DeviceSpecificSettingsScreen.DISPLAY, R.xml.devicesettings_language_generic);*/ // Developer final List developer = deviceSpecificSettings.addRootScreen(DeviceSpecificSettingsScreen.DEVELOPER); developer.add(R.xml.devicesettings_huawei_debug); return deviceSpecificSettings; } public boolean supportsDateFormat() { return supportsCommandForService(0x01, 0x04); } public boolean supportsActivateOnLift() { return supportsCommandForService(0x01, 0x09); } public boolean supportsDoNotDisturb() { return supportsCommandForService(0x01, 0x0a); } public boolean supportsDoNotDisturb(GBDevice gbDevice) { return supportsDoNotDisturb() || getForceOption(gbDevice, PREF_FORCE_DND_SUPPORT); } public boolean supportsActivityType() { return supportsCommandForService(0x01, 0x12); } public boolean supportsWearLocation() { return supportsCommandForService(0x01, 0x1a); } public boolean supportsWearLocation(GBDevice gbDevice) { return supportsWearLocation() || getForceOption(gbDevice, PREF_FORCE_ENABLE_WEAR_LOCATION); } public boolean supportsRotateToCycleInfo() { return supportsCommandForService(0x01, 0x1b); } public boolean supportsQueryDndLiftWristDisturbType() { return supportsCommandForService(0x01, 0x1d); } public boolean supportsAcceptAgreement() { return supportsCommandForService(0x01, 0x30); } public boolean supportsSettingRelated() { return supportsCommandForService(0x01, 0x31); } public boolean supportsTimeAndZoneId() { return supportsCommandForService(0x01, 0x32); } public boolean supportsConnectStatus() { return supportsCommandForService(0x01, 0x35); } public boolean supportsExpandCapability() { return supportsCommandForService(0x01, 0x37); } public boolean supportsNotificationAlert() { return supportsCommandForService(0x02, 0x01); } public boolean supportsNotification() { return supportsCommandForService(0x02, 0x04); } public boolean supportsWearMessagePush() { return supportsCommandForService(0x02, 0x08); } public boolean supportsMotionGoal() { return supportsCommandForService(0x07, 0x01); } public boolean supportsInactivityWarnings() { return supportsCommandForService(0x07, 0x06); } public boolean supportsActivityReminder() { return supportsCommandForService(0x07, 0x07); } public boolean supportsTruSleep() { return supportsCommandForService(0x07, 0x16); } public boolean supportsHeartRate() { // TODO: this is not correct return supportsCommandForService(0x07, 0x17); } public boolean supportsHeartRate(GBDevice gbDevice) { return supportsHeartRate() || getForceOption(gbDevice, PREF_FORCE_ENABLE_HEARTRATE_SUPPORT); } public boolean supportsFitnessRestHeartRate() { return supportsCommandForService(0x07, 0x23); } public boolean supportsSPo2() { return supportsCommandForService(0x07, 0x24); } public boolean supportsSPo2(GBDevice gbDevice) { return supportsSPo2() || getForceOption(gbDevice, PREF_FORCE_ENABLE_SPO2_SUPPORT); } public boolean supportsFitnessThresholdValue() { return supportsCommandForService(0x07, 0x29); } public boolean supportsEventAlarm() { return supportsCommandForService(0x08, 0x01); } public boolean supportsSmartAlarm() { return supportsCommandForService(0x08, 0x02) ; } public boolean supportsSmartAlarm(GBDevice gbDevice) { return supportsSmartAlarm() || getForceOption(gbDevice, PREF_FORCE_ENABLE_SMART_ALARM); } public boolean supportsSmartAlarm(GBDevice gbDevice, int alarmPosition) { return supportsSmartAlarm(gbDevice) && alarmPosition == 0; } public boolean forcedSmartWakeup(GBDevice device, int alarmPosition) { return supportsSmartAlarm(device, alarmPosition) && alarmPosition == 0; } /** * @return True if alarms can be changed on the device, false otherwise */ public boolean supportsChangingAlarm() { return supportsCommandForService(0x08, 0x03); } public boolean supportsNotificationOnBluetoothLoss() { return supportsCommandForService(0x0b, 0x03); } public boolean supportsLanguageSetting() { return supportsCommandForService(0x0c, 0x01); } public boolean supportsWeather() { return supportsCommandForService(0x0f, 0x01); } public boolean supportsWeatherUnit() { return supportsCommandForService(0x0f, 0x05); } public boolean supportsWeatherExtended() { return supportsCommandForService(0x0f, 0x06); } public boolean supportsWeatherForecasts() { return supportsCommandForService(0x0f, 0x08); } public boolean supportsWeatherMoonRiseSet() { return supportsCommandForService(0x0f, 0x0a); } public boolean supportsWeatherTides() { return supportsCommandForService(0x0f, 0x0b); } public boolean supportsWorkouts() { return supportsCommandForService(0x17, 0x01); } public boolean supportsWorkoutsTrustHeartRate() { return supportsCommandForService(0x17, 0x17); } public boolean supportsSendingGps() { return supportsCommandForService(0x18, 0x02); } public boolean supportsAccount() { return supportsCommandForService(0x1A, 0x01); } public boolean supportsAccountJudgment() { return supportsCommandForService(0x1A, 0x05); } public boolean supportsAccountSwitch() { return supportsCommandForService(0x1A, 0x06); } public boolean supportsDiffAccountPairingOptimization() { if (supportsExpandCapability()) return supportsExpandCapability(0xac); return false; } public boolean supportsMusic() { return supportsCommandForService(0x25, 0x02); } public boolean supportsAutoWorkMode() { return supportsCommandForService(0x26, 0x02); } public boolean supportsMenstrual() { return supportsCommandForService(0x32, 0x01); } public boolean supportsMultiDevice() { if (supportsExpandCapability()) return supportsExpandCapability(109); return false; } public boolean supportsPromptPushMessage () { // do not ask for capabilities under specific condition // if (deviceType == 10 && deviceVersion == 73617766697368 && deviceSoftVersion == 372E312E31) -> leo device // if V1V0Device // if (serviceId = 0x01 && commandId = 0x03) && productType == 3 return (((notificationCapabilities >> 1) & 1) == 0); } public boolean supportsOutgoingCall () { return (((notificationCapabilities >> 2) & 1) == 0); } public boolean supportsYellowPages() { return supportsNotificationConstraint(NotificationConstraintsType.yellowPagesSupport); } public boolean supportsContentSIgn() { return supportsNotificationConstraint(NotificationConstraintsType.contentSignSupport); } public boolean supportsIncomingNumber() { return supportsNotificationConstraint(NotificationConstraintsType.incomingNumberSupport); } public int getContentFormat() { return getNotificationConstraint(NotificationConstraintsType.contentFormat); } public int getYellowPagesFormat() { return getNotificationConstraint(NotificationConstraintsType.yellowPagesFormat); } public int getContentSignFormat() { return getNotificationConstraint(NotificationConstraintsType.contentSignFormat); } public int getIncomingFormatFormat() { return getNotificationConstraint(NotificationConstraintsType.incomingNumberFormat); } public int getContentLength() { return getNotificationConstraint(NotificationConstraintsType.contentLength); } public int getYellowPagesLength() { return getNotificationConstraint(NotificationConstraintsType.yellowPagesLength); } public int getContentSignLength() { return getNotificationConstraint(NotificationConstraintsType.contentSignLength); } public int getIncomingNumberLength() { return getNotificationConstraint(NotificationConstraintsType.incomingNumberLength); } public int getAlarmSlotCount(GBDevice gbDevice) { int alarmCount = 0; if (supportsEventAlarm()) alarmCount += 5; // Always five event alarms if (supportsSmartAlarm(gbDevice)) alarmCount += 1; // Always a single smart alarm return alarmCount; } public void setTransactionCrypted(boolean crypted) { this.transactionCrypted = crypted; } public boolean isTransactionCrypted() { return this.transactionCrypted; } public String[] getSupportedLanguageSettings(GBDevice device) { return new String[]{ "auto", "cs_CZ", "de_DE", "en_US", "es_ES", "fr_FR", "it_IT", "pt_BR", "ru_RU", "tr_TR", "zh_CN", "zh_TW", }; } }