Gadgetbridge/app/src/main/java/nodomain/freeyourgadget/gadgetbridge/devices/huawei/HuaweiCoordinator.java

539 lines
19 KiB
Java

/* 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 <https://www.gnu.org/licenses/>. */
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<Integer, byte[]> 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<Integer, byte[]> 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<Integer> 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
// 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<Integer> 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<Integer> 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 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",
};
}
}