mirror of
https://codeberg.org/Freeyourgadget/Gadgetbridge.git
synced 2025-01-25 08:05:55 +01:00
Compare commits
40 Commits
0258905b4a
...
a7c4a733a4
Author | SHA1 | Date | |
---|---|---|---|
|
a7c4a733a4 | ||
|
e30a6b4aaf | ||
|
1d26ed1688 | ||
|
0107e60046 | ||
|
9fc0131328 | ||
|
725fadc455 | ||
|
b2161835a8 | ||
|
0a24320aa0 | ||
|
7501c93c2a | ||
|
883409af1c | ||
|
e16504431e | ||
|
f9da037958 | ||
|
acfdeea44d | ||
|
8e69b1a149 | ||
|
8986cde243 | ||
|
001f29bc55 | ||
|
c6734f5428 | ||
|
e9e23d215a | ||
|
d2705ebff9 | ||
|
2c5d5bdad2 | ||
|
82ece1d540 | ||
|
ba41e16420 | ||
|
6a790c8b2c | ||
|
3765c34b9c | ||
|
e444ed67b9 | ||
|
a46c4fa8de | ||
|
2d79cfab2b | ||
|
5a3a2a4a6a | ||
|
79106736d6 | ||
|
9f74454b53 | ||
|
1c813d53f8 | ||
|
27fbb6ca0b | ||
|
94b855995c | ||
|
f0974f49fa | ||
|
489ccb1986 | ||
|
cb4227cef7 | ||
|
996fa059d8 | ||
|
85623d682e | ||
|
05d1ce6869 | ||
|
8a32162100 |
@ -49,6 +49,8 @@ public class GBDaoGenerator {
|
||||
private static final String SAMPLE_TEMPERATURE = "temperature";
|
||||
private static final String SAMPLE_TEMPERATURE_TYPE = "temperatureType";
|
||||
private static final String SAMPLE_WEIGHT_KG = "weightKg";
|
||||
private static final String SAMPLE_BLOOD_PRESSURE_SYSTOLIC = "bpSystolic";
|
||||
private static final String SAMPLE_BLOOD_PRESSURE_DIASTOLIC = "bpDiastolic";
|
||||
private static final String TIMESTAMP_FROM = "timestampFrom";
|
||||
private static final String TIMESTAMP_TO = "timestampTo";
|
||||
|
||||
@ -151,6 +153,10 @@ public class GBDaoGenerator {
|
||||
addColmiHrvValueSample(schema, user, device);
|
||||
addColmiHrvSummarySample(schema, user, device);
|
||||
addColmiTemperatureSample(schema, user, device);
|
||||
addMoyoungActivitySample(schema, user, device);
|
||||
addMoyoungHeartRateSample(schema, user, device);
|
||||
addMoyoungSpo2Sample(schema, user, device);
|
||||
addMoyoungBloodPressureSample(schema, user, device);
|
||||
|
||||
addHuaweiActivitySample(schema, user, device);
|
||||
|
||||
@ -603,6 +609,11 @@ public class GBDaoGenerator {
|
||||
activitySample.addIntProperty(SAMPLE_HEART_RATE).notNull().codeBeforeGetterAndSetter(OVERRIDE);
|
||||
}
|
||||
|
||||
private static void addBloodPressureProperies(Entity activitySample) {
|
||||
activitySample.addIntProperty(SAMPLE_BLOOD_PRESSURE_SYSTOLIC).notNull();
|
||||
activitySample.addIntProperty(SAMPLE_BLOOD_PRESSURE_DIASTOLIC).notNull();
|
||||
}
|
||||
|
||||
private static Entity addPebbleHealthActivitySample(Schema schema, Entity user, Entity device) {
|
||||
Entity activitySample = addEntity(schema, "PebbleHealthActivitySample");
|
||||
addCommonActivitySampleProperties("AbstractPebbleHealthActivitySample", activitySample, user, device);
|
||||
@ -1057,6 +1068,41 @@ public class GBDaoGenerator {
|
||||
return activitySample;
|
||||
}
|
||||
|
||||
private static Entity addMoyoungActivitySample(Schema schema, Entity user, Entity device) {
|
||||
Entity activitySample = addEntity(schema, "MoyoungActivitySample");
|
||||
activitySample.implementsSerializable();
|
||||
addCommonActivitySampleProperties("AbstractActivitySample", activitySample, user, device);
|
||||
activitySample.addIntProperty(SAMPLE_STEPS).notNull().codeBeforeGetterAndSetter(OVERRIDE);
|
||||
activitySample.addIntProperty(SAMPLE_RAW_KIND).notNull().codeBeforeGetterAndSetter(OVERRIDE);
|
||||
activitySample.addIntProperty("dataSource").notNull();
|
||||
activitySample.addIntProperty("caloriesBurnt").notNull();
|
||||
activitySample.addIntProperty("distanceMeters").notNull();
|
||||
addHeartRateProperties(activitySample);
|
||||
return activitySample;
|
||||
}
|
||||
|
||||
private static Entity addMoyoungHeartRateSample(Schema schema, Entity user, Entity device) {
|
||||
Entity heartRateSample = addEntity(schema, "MoyoungHeartRateSample");
|
||||
heartRateSample.implementsSerializable();
|
||||
addCommonTimeSampleProperties("AbstractHeartRateSample", heartRateSample, user, device);
|
||||
heartRateSample.addIntProperty(SAMPLE_HEART_RATE).notNull();
|
||||
return heartRateSample;
|
||||
}
|
||||
|
||||
private static Entity addMoyoungSpo2Sample(Schema schema, Entity user, Entity device) {
|
||||
Entity spo2sample = addEntity(schema, "MoyoungSpo2Sample");
|
||||
addCommonTimeSampleProperties("AbstractSpo2Sample", spo2sample, user, device);
|
||||
spo2sample.addIntProperty("spo2").notNull().codeBeforeGetter(OVERRIDE);
|
||||
return spo2sample;
|
||||
}
|
||||
|
||||
private static Entity addMoyoungBloodPressureSample(Schema schema, Entity user, Entity device) {
|
||||
Entity bpSample = addEntity(schema, "MoyoungBloodPressureSample");
|
||||
addCommonTimeSampleProperties("AbstractBloodPressureSample", bpSample, user, device);
|
||||
addBloodPressureProperies(bpSample);
|
||||
return bpSample;
|
||||
}
|
||||
|
||||
private static void addCommonActivitySampleProperties(String superClass, Entity activitySample, Entity user, Entity device) {
|
||||
activitySample.setSuperclass(superClass);
|
||||
activitySample.addImport(MAIN_PACKAGE + ".devices.SampleProvider");
|
||||
|
@ -192,6 +192,7 @@ public class DeviceSettingsPreferenceConst {
|
||||
public static final String PREF_INACTIVITY_DND = "inactivity_warnings_dnd";
|
||||
public static final String PREF_INACTIVITY_DND_START = "inactivity_warnings_dnd_start";
|
||||
public static final String PREF_INACTIVITY_DND_END = "inactivity_warnings_dnd_end";
|
||||
public static final String PREF_INACTIVITY_STEPS = "inactivity_warnings_steps";
|
||||
|
||||
public static final String PREF_HEARTRATE_USE_FOR_SLEEP_DETECTION = "heartrate_sleep_detection";
|
||||
public static final String PREF_HEARTRATE_MEASUREMENT_INTERVAL = "heartrate_measurement_interval";
|
||||
@ -226,6 +227,8 @@ public class DeviceSettingsPreferenceConst {
|
||||
public static final String PREF_DO_NOT_DISTURB_END = "do_not_disturb_end";
|
||||
public static final String PREF_DO_NOT_DISTURB_LIFT_WRIST = "do_not_disturb_lift_wrist";
|
||||
public static final String PREF_DO_NOT_DISTURB_NOT_WEAR = "do_not_disturb_not_wear";
|
||||
public static final String PREF_DO_NOT_DISTURB_BOOL = "do_not_disturb_on_off";
|
||||
public static final String PREF_DO_NOT_DISTURB_FOLLOW_PHONE = "do_not_disturb_follow_phone";
|
||||
public static final String PREF_DO_NOT_DISTURB_OFF = "off";
|
||||
public static final String PREF_DO_NOT_DISTURB_AUTOMATIC = "automatic";
|
||||
public static final String PREF_DO_NOT_DISTURB_ALWAYS = "always";
|
||||
|
@ -35,6 +35,8 @@ import static nodomain.freeyourgadget.gadgetbridge.devices.miband.MiBandConst.PR
|
||||
import static nodomain.freeyourgadget.gadgetbridge.devices.miband.MiBandConst.PREF_NIGHT_MODE_SCHEDULED;
|
||||
import static nodomain.freeyourgadget.gadgetbridge.devices.miband.MiBandConst.PREF_NIGHT_MODE_START;
|
||||
import static nodomain.freeyourgadget.gadgetbridge.devices.miband.MiBandConst.PREF_SWIPE_UNLOCK;
|
||||
import static nodomain.freeyourgadget.gadgetbridge.devices.moyoung.MoyoungConstants.PREF_MOYOUNG_DEVICE_VERSION;
|
||||
import static nodomain.freeyourgadget.gadgetbridge.devices.moyoung.MoyoungConstants.PREF_MOYOUNG_WATCH_FACE;
|
||||
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
@ -570,6 +572,8 @@ public class DeviceSpecificSettingsFragment extends AbstractPreferenceFragment i
|
||||
addPreferenceHandlerFor(PREF_DO_NOT_DISTURB_SU);
|
||||
addPreferenceHandlerFor(PREF_DO_NOT_DISTURB_LIFT_WRIST);
|
||||
addPreferenceHandlerFor(PREF_DO_NOT_DISTURB_NOT_WEAR);
|
||||
addPreferenceHandlerFor(PREF_DO_NOT_DISTURB_BOOL);
|
||||
addPreferenceHandlerFor(PREF_DO_NOT_DISTURB_FOLLOW_PHONE);
|
||||
addPreferenceHandlerFor(PREF_FIND_PHONE);
|
||||
addPreferenceHandlerFor(PREF_FIND_PHONE_DURATION);
|
||||
addPreferenceHandlerFor(PREF_AUTOLIGHT);
|
||||
@ -814,6 +818,9 @@ public class DeviceSpecificSettingsFragment extends AbstractPreferenceFragment i
|
||||
|
||||
addPreferenceHandlerFor(PREF_FEMOMETER_MEASUREMENT_MODE);
|
||||
|
||||
addPreferenceHandlerFor(PREF_MOYOUNG_WATCH_FACE);
|
||||
addPreferenceHandlerFor(PREF_MOYOUNG_DEVICE_VERSION);
|
||||
|
||||
addPreferenceHandlerFor(PREF_QC35_NOISE_CANCELLING_LEVEL);
|
||||
addPreferenceHandlerFor(PREF_USER_FITNESS_GOAL);
|
||||
addPreferenceHandlerFor(PREF_USER_FITNESS_GOAL_NOTIFICATION);
|
||||
|
@ -45,6 +45,7 @@ public class HeartRateCapability {
|
||||
MINUTES_5(300, R.string.interval_five_minutes),
|
||||
MINUTES_10(600, R.string.interval_ten_minutes),
|
||||
MINUTES_15(900, R.string.interval_fifteen_minutes),
|
||||
MINUTES_20(1200, R.string.interval_twenty_minutes),
|
||||
MINUTES_30(1800, R.string.interval_thirty_minutes),
|
||||
MINUTES_45(2700, R.string.interval_forty_five_minutes),
|
||||
HOUR_1(3600, R.string.interval_one_hour),
|
||||
|
@ -0,0 +1,260 @@
|
||||
/* Copyright (C) 2019 krzys_h
|
||||
|
||||
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 <http://www.gnu.org/licenses/>. */
|
||||
package nodomain.freeyourgadget.gadgetbridge.devices.moyoung;
|
||||
|
||||
import android.annotation.TargetApi;
|
||||
import android.bluetooth.le.ScanFilter;
|
||||
import android.os.Build;
|
||||
import android.os.ParcelUuid;
|
||||
|
||||
import androidx.annotation.NonNull;
|
||||
|
||||
import java.util.Arrays;
|
||||
import java.util.Collection;
|
||||
import java.util.Collections;
|
||||
import java.util.List;
|
||||
|
||||
import de.greenrobot.dao.query.QueryBuilder;
|
||||
import nodomain.freeyourgadget.gadgetbridge.GBException;
|
||||
import nodomain.freeyourgadget.gadgetbridge.R;
|
||||
import nodomain.freeyourgadget.gadgetbridge.activities.devicesettings.DeviceSpecificSettings;
|
||||
import nodomain.freeyourgadget.gadgetbridge.activities.devicesettings.DeviceSpecificSettingsScreen;
|
||||
import nodomain.freeyourgadget.gadgetbridge.capabilities.HeartRateCapability;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.AbstractBLEDeviceCoordinator;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.SampleProvider;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.TimeSampleProvider;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.moyoung.samples.MoyoungActivitySampleProvider;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.moyoung.samples.MoyoungSpo2SampleProvider;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.moyoung.settings.MoyoungEnumDeviceVersion;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.moyoung.settings.MoyoungEnumMetricSystem;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.moyoung.settings.MoyoungEnumTimeSystem;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.moyoung.settings.MoyoungSetting;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.moyoung.settings.MoyoungSettingBool;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.moyoung.settings.MoyoungSettingByte;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.moyoung.settings.MoyoungSettingEnum;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.moyoung.settings.MoyoungSettingInt;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.moyoung.settings.MoyoungSettingLanguage;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.moyoung.settings.MoyoungSettingRemindersToMove;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.moyoung.settings.MoyoungSettingTimeRange;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.moyoung.settings.MoyoungSettingUserInfo;
|
||||
import nodomain.freeyourgadget.gadgetbridge.entities.DaoSession;
|
||||
import nodomain.freeyourgadget.gadgetbridge.entities.Device;
|
||||
import nodomain.freeyourgadget.gadgetbridge.entities.MoyoungActivitySampleDao;
|
||||
import nodomain.freeyourgadget.gadgetbridge.entities.MoyoungBloodPressureSampleDao;
|
||||
import nodomain.freeyourgadget.gadgetbridge.entities.MoyoungHeartRateSampleDao;
|
||||
import nodomain.freeyourgadget.gadgetbridge.entities.MoyoungSpo2SampleDao;
|
||||
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
|
||||
import nodomain.freeyourgadget.gadgetbridge.model.ActivitySample;
|
||||
import nodomain.freeyourgadget.gadgetbridge.model.Spo2Sample;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.DeviceSupport;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.moyoung.MoyoungDeviceSupport;
|
||||
|
||||
public abstract class AbstractMoyoungDeviceCoordinator extends AbstractBLEDeviceCoordinator {
|
||||
|
||||
@NonNull
|
||||
@Override
|
||||
@TargetApi(Build.VERSION_CODES.LOLLIPOP)
|
||||
public Collection<? extends ScanFilter> createBLEScanFilters() {
|
||||
ParcelUuid service = new ParcelUuid(MoyoungConstants.UUID_SERVICE_MOYOUNG);
|
||||
ScanFilter filter = new ScanFilter.Builder().setServiceUuid(service).build();
|
||||
return Collections.singletonList(filter);
|
||||
}
|
||||
|
||||
@NonNull
|
||||
@Override
|
||||
public Class<? extends DeviceSupport> getDeviceSupportClass() {
|
||||
return MoyoungDeviceSupport.class;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getBondingStyle() {
|
||||
return BONDING_STYLE_LAZY;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void deleteDevice(@NonNull GBDevice gbDevice, @NonNull Device device, @NonNull DaoSession session) throws GBException {
|
||||
Long deviceId = device.getId();
|
||||
QueryBuilder<?> qb;
|
||||
|
||||
qb = session.getMoyoungActivitySampleDao().queryBuilder();
|
||||
qb.where(MoyoungActivitySampleDao.Properties.DeviceId.eq(deviceId)).buildDelete().executeDeleteWithoutDetachingEntities();
|
||||
qb = session.getMoyoungHeartRateSampleDao().queryBuilder();
|
||||
qb.where(MoyoungHeartRateSampleDao.Properties.DeviceId.eq(deviceId)).buildDelete().executeDeleteWithoutDetachingEntities();
|
||||
qb = session.getMoyoungSpo2SampleDao().queryBuilder();
|
||||
qb.where(MoyoungSpo2SampleDao.Properties.DeviceId.eq(deviceId)).buildDelete().executeDeleteWithoutDetachingEntities();
|
||||
qb = session.getMoyoungBloodPressureSampleDao().queryBuilder();
|
||||
qb.where(MoyoungBloodPressureSampleDao.Properties.DeviceId.eq(deviceId)).buildDelete().executeDeleteWithoutDetachingEntities();
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean supportsActivityDataFetching() {
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean supportsActivityTracking() {
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean supportsSpo2(GBDevice device) {
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public SampleProvider<? extends ActivitySample> getSampleProvider(GBDevice device, DaoSession session) {
|
||||
return new MoyoungActivitySampleProvider(device, session);
|
||||
}
|
||||
|
||||
@Override
|
||||
public TimeSampleProvider<? extends Spo2Sample> getSpo2SampleProvider(GBDevice device, DaoSession session) {
|
||||
return new MoyoungSpo2SampleProvider(device, session);
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getAlarmSlotCount(GBDevice device) {
|
||||
return 3;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean supportsHeartRateMeasurement(GBDevice device) {
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean supportsCalendarEvents() {
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean supportsRealtimeData() {
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean supportsWeather() {
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean supportsFindDevice() {
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean supportsActivityTracks() {
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean supportsMusicInfo() {
|
||||
return true;
|
||||
}
|
||||
|
||||
private static final MoyoungSetting[] MOYOUNG_SETTINGS = {
|
||||
new MoyoungSettingUserInfo("USER_INFO", MoyoungConstants.CMD_SET_USER_INFO),
|
||||
new MoyoungSettingByte("STEP_LENGTH", (byte)-1, MoyoungConstants.CMD_SET_STEP_LENGTH),
|
||||
// (*) new MoyoungSettingEnum<>("DOMINANT_HAND", MoyoungConstants.CMD_QUERY_DOMINANT_HAND, MoyoungConstants.CMD_SET_DOMINANT_HAND, MoyoungEnumDominantHand.class),
|
||||
new MoyoungSettingInt("GOAL_STEP", MoyoungConstants.CMD_QUERY_GOAL_STEP, MoyoungConstants.CMD_SET_GOAL_STEP),
|
||||
new MoyoungSettingByte("HR_AUTO_INTERVAL", MoyoungConstants.CMD_QUERY_TIMING_MEASURE_HEART_RATE, MoyoungConstants.CMD_SET_TIMING_MEASURE_HEART_RATE),
|
||||
|
||||
new MoyoungSettingEnum<>("DEVICE_VERSION", MoyoungConstants.CMD_QUERY_DEVICE_VERSION, MoyoungConstants.CMD_SET_DEVICE_VERSION, MoyoungEnumDeviceVersion.class),
|
||||
new MoyoungSettingLanguage("DEVICE_LANGUAGE", MoyoungConstants.CMD_QUERY_DEVICE_LANGUAGE, MoyoungConstants.CMD_SET_DEVICE_LANGUAGE),
|
||||
new MoyoungSettingEnum<>("TIME_SYSTEM", MoyoungConstants.CMD_QUERY_TIME_SYSTEM, MoyoungConstants.CMD_SET_TIME_SYSTEM, MoyoungEnumTimeSystem.class),
|
||||
new MoyoungSettingEnum<>("METRIC_SYSTEM", MoyoungConstants.CMD_QUERY_METRIC_SYSTEM, MoyoungConstants.CMD_SET_METRIC_SYSTEM, MoyoungEnumMetricSystem.class),
|
||||
|
||||
// (*) new MoyoungSetting("DISPLAY_DEVICE_FUNCTION", MoyoungConstants.CMD_QUERY_DISPLAY_DEVICE_FUNCTION, MoyoungConstants.CMD_SET_DISPLAY_DEVICE_FUNCTION),
|
||||
// (*) new MoyoungSetting("SUPPORT_WATCH_FACE", MoyoungConstants.CMD_QUERY_SUPPORT_WATCH_FACE, (byte)-1),
|
||||
// (*) new MoyoungSetting("WATCH_FACE_LAYOUT", MoyoungConstants.CMD_QUERY_WATCH_FACE_LAYOUT, MoyoungConstants.CMD_SET_WATCH_FACE_LAYOUT),
|
||||
new MoyoungSettingByte("DISPLAY_WATCH_FACE", MoyoungConstants.CMD_QUERY_DISPLAY_WATCH_FACE, MoyoungConstants.CMD_SET_DISPLAY_WATCH_FACE),
|
||||
new MoyoungSettingBool("OTHER_MESSAGE_STATE", MoyoungConstants.CMD_QUERY_OTHER_MESSAGE_STATE, MoyoungConstants.CMD_SET_OTHER_MESSAGE_STATE),
|
||||
|
||||
new MoyoungSettingBool("QUICK_VIEW", MoyoungConstants.CMD_QUERY_QUICK_VIEW, MoyoungConstants.CMD_SET_QUICK_VIEW),
|
||||
new MoyoungSettingTimeRange("QUICK_VIEW_TIME", MoyoungConstants.CMD_QUERY_QUICK_VIEW_TIME, MoyoungConstants.CMD_SET_QUICK_VIEW_TIME),
|
||||
new MoyoungSettingBool("SEDENTARY_REMINDER", MoyoungConstants.CMD_QUERY_SEDENTARY_REMINDER, MoyoungConstants.CMD_SET_SEDENTARY_REMINDER),
|
||||
new MoyoungSettingRemindersToMove("REMINDERS_TO_MOVE_PERIOD", MoyoungConstants.CMD_QUERY_REMINDERS_TO_MOVE_PERIOD, MoyoungConstants.CMD_SET_REMINDERS_TO_MOVE_PERIOD),
|
||||
new MoyoungSettingTimeRange("DO_NOT_DISTURB_TIME", MoyoungConstants.CMD_QUERY_DO_NOT_DISTURB_TIME, MoyoungConstants.CMD_SET_DO_NOT_DISTURB_TIME),
|
||||
new MoyoungSettingBool("DO_NOT_DISTURB_ONOFF", MoyoungConstants.CMD_QUERY_DO_NOT_DISTURB_TIME, MoyoungConstants.CMD_SET_DO_NOT_DISTURB_TIME),
|
||||
// (*) new MoyoungSetting("PSYCHOLOGICAL_PERIOD", MoyoungConstants.CMD_QUERY_PSYCHOLOGICAL_PERIOD, MoyoungConstants.CMD_SET_PSYCHOLOGICAL_PERIOD),
|
||||
|
||||
new MoyoungSettingBool("BREATHING_LIGHT", MoyoungConstants.CMD_QUERY_BREATHING_LIGHT, MoyoungConstants.CMD_SET_BREATHING_LIGHT),
|
||||
new MoyoungSettingBool("POWER_SAVING", MoyoungConstants.CMD_QUERY_POWER_SAVING, MoyoungConstants.CMD_SET_POWER_SAVING)
|
||||
};
|
||||
|
||||
|
||||
@Override
|
||||
public DeviceSpecificSettings getDeviceSpecificSettings(final GBDevice device) {
|
||||
final DeviceSpecificSettings deviceSpecificSettings = new DeviceSpecificSettings();
|
||||
final List<Integer> generic = deviceSpecificSettings.addRootScreen(DeviceSpecificSettingsScreen.GENERIC);
|
||||
generic.add(R.xml.devicesettings_moyoung_device_version);
|
||||
generic.add(R.xml.devicesettings_timeformat);
|
||||
generic.add(R.xml.devicesettings_moyoung_watchface);
|
||||
generic.add(R.xml.devicesettings_power_saving);
|
||||
generic.add(R.xml.devicesettings_liftwrist_display);
|
||||
// generic.add(R.xml.devicesettings_donotdisturb_no_auto); // not supported by Colmi i28 Ultra
|
||||
generic.add(R.xml.devicesettings_donotdisturb_on_off_follow);
|
||||
generic.add(R.xml.devicesettings_world_clocks);
|
||||
generic.add(R.xml.devicesettings_sync_calendar);
|
||||
final List<Integer> health = deviceSpecificSettings.addRootScreen(DeviceSpecificSettingsScreen.HEALTH);
|
||||
health.add(R.xml.devicesettings_heartrate_interval);
|
||||
health.add(R.xml.devicesettings_inactivity_with_steps);
|
||||
return deviceSpecificSettings;
|
||||
}
|
||||
|
||||
@Override
|
||||
public String[] getSupportedLanguageSettings(final GBDevice device) {
|
||||
// TODO: use settings customizer to display the languages
|
||||
// retrieved from the watch instead of this fixed list
|
||||
return new String[]{
|
||||
"ar_SA",
|
||||
"cs_CZ",
|
||||
"de_DE",
|
||||
"en_US",
|
||||
"es_ES",
|
||||
"fr_FR",
|
||||
"it_IT",
|
||||
"ja_JP",
|
||||
"ko_KO",
|
||||
"nl_NL",
|
||||
"pl_PL",
|
||||
"pt_PT",
|
||||
"ro_RO",
|
||||
"ru_RU",
|
||||
"uk_UA",
|
||||
"zh_CN",
|
||||
};
|
||||
};
|
||||
|
||||
@Override
|
||||
public List<HeartRateCapability.MeasurementInterval> getHeartRateMeasurementIntervals() {
|
||||
return Arrays.asList(
|
||||
HeartRateCapability.MeasurementInterval.OFF,
|
||||
HeartRateCapability.MeasurementInterval.MINUTES_5,
|
||||
HeartRateCapability.MeasurementInterval.MINUTES_10,
|
||||
HeartRateCapability.MeasurementInterval.MINUTES_20,
|
||||
HeartRateCapability.MeasurementInterval.MINUTES_30
|
||||
);
|
||||
}
|
||||
|
||||
public MoyoungSetting[] getSupportedSettings() {
|
||||
return MOYOUNG_SETTINGS;
|
||||
}
|
||||
|
||||
public int getMtu() {
|
||||
return 20;
|
||||
}
|
||||
}
|
@ -0,0 +1,78 @@
|
||||
/* Copyright (C) 2024 Arjan Schrijver
|
||||
|
||||
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.moyoung;
|
||||
|
||||
import androidx.annotation.DrawableRes;
|
||||
|
||||
import org.slf4j.Logger;
|
||||
import org.slf4j.LoggerFactory;
|
||||
|
||||
import java.util.regex.Pattern;
|
||||
|
||||
import nodomain.freeyourgadget.gadgetbridge.R;
|
||||
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
|
||||
|
||||
public class ColmiI28UltraCoordinator extends AbstractMoyoungDeviceCoordinator {
|
||||
private static final Logger LOG = LoggerFactory.getLogger(ColmiI28UltraCoordinator.class);
|
||||
|
||||
@Override
|
||||
protected Pattern getSupportedDeviceName() {
|
||||
return Pattern.compile("i28 Ultra");
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getDeviceNameResource() {
|
||||
return R.string.devicetype_colmi_i28_ultra;
|
||||
}
|
||||
|
||||
@Override
|
||||
@DrawableRes
|
||||
public int getDefaultIconResource() {
|
||||
return R.drawable.ic_device_miwatch;
|
||||
}
|
||||
|
||||
@Override
|
||||
@DrawableRes
|
||||
public int getDisabledIconResource() {
|
||||
return R.drawable.ic_device_miwatch_disabled;
|
||||
}
|
||||
|
||||
@Override
|
||||
public String getManufacturer() {
|
||||
return "Colmi";
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getMtu() {
|
||||
return 508;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getAlarmSlotCount(GBDevice device) {
|
||||
return 8;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getWorldClocksSlotCount() {
|
||||
return 6;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getWorldClocksLabelLength() {
|
||||
return 30;
|
||||
}
|
||||
}
|
@ -0,0 +1,63 @@
|
||||
/* Copyright (C) 2024 Arjan Schrijver
|
||||
|
||||
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.moyoung;
|
||||
|
||||
import androidx.annotation.DrawableRes;
|
||||
|
||||
import org.slf4j.Logger;
|
||||
import org.slf4j.LoggerFactory;
|
||||
|
||||
import java.util.regex.Pattern;
|
||||
|
||||
import nodomain.freeyourgadget.gadgetbridge.R;
|
||||
|
||||
public class MisirunC17Coordinator extends AbstractMoyoungDeviceCoordinator {
|
||||
private static final Logger LOG = LoggerFactory.getLogger(MisirunC17Coordinator.class);
|
||||
|
||||
@Override
|
||||
protected Pattern getSupportedDeviceName() {
|
||||
return Pattern.compile("C17");
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getDeviceNameResource() {
|
||||
return R.string.devicetype_misirun_c17;
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
@DrawableRes
|
||||
public int getDefaultIconResource() {
|
||||
return R.drawable.ic_device_banglejs;
|
||||
}
|
||||
|
||||
@Override
|
||||
@DrawableRes
|
||||
public int getDisabledIconResource() {
|
||||
return R.drawable.ic_device_banglejs_disabled;
|
||||
}
|
||||
|
||||
@Override
|
||||
public String getManufacturer() {
|
||||
return "Misirun";
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getMtu() {
|
||||
return 508;
|
||||
}
|
||||
}
|
@ -0,0 +1,435 @@
|
||||
/* Copyright (C) 2019 krzys_h
|
||||
|
||||
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 <http://www.gnu.org/licenses/>. */
|
||||
package nodomain.freeyourgadget.gadgetbridge.devices.moyoung;
|
||||
|
||||
import java.text.ParseException;
|
||||
import java.text.SimpleDateFormat;
|
||||
import java.util.Date;
|
||||
import java.util.TimeZone;
|
||||
import java.util.UUID;
|
||||
|
||||
import nodomain.freeyourgadget.gadgetbridge.model.NotificationType;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.btle.AbstractBTLEDeviceSupport;
|
||||
|
||||
public class MoyoungConstants {
|
||||
// (*) - based only on static reverse engineering of the original app code,
|
||||
// not supported by my watch so not implemented
|
||||
// (or at least I didn't manage to get any response out of it)
|
||||
|
||||
// (?) - not checked
|
||||
|
||||
|
||||
// The device communicates by sending packets by writing to UUID_CHARACTERISTIC_DATA_OUT
|
||||
// in MTU-sized chunks. The value of MTU seems to be somehow changeable (?), but the default
|
||||
// is 20. Responses are received via notify on UUID_CHARACTERISTIC_DATA_IN in similar format.
|
||||
// The write success notification comes AFTER the responses.
|
||||
|
||||
// Packet format:
|
||||
// packet[0] = 0xFE;
|
||||
// packet[1] = 0xEA;
|
||||
// if (MTU == 20) // could be a protocol version check?
|
||||
// {
|
||||
// packet[2] = 16;
|
||||
// packet[3] = packet.length;
|
||||
// }
|
||||
// else
|
||||
// {
|
||||
// packet[2] = 32 + (packet.length >> 8) & 0xFF;
|
||||
// packet[3] = packet.length & 0xFF;
|
||||
// }
|
||||
// packet[4] = packetType;
|
||||
// packet[5:] = payload;
|
||||
|
||||
// Protocol version is determined by reading manufacturer name. MOYOUNG for old fixed-size
|
||||
// or MOYOUNG-V2 for MTU. The non-MTU version uses packets of size 256
|
||||
// for firmware >= 1.6.5, and 64 otherwise.
|
||||
|
||||
// The firmware version is also used to detect availability of some features.
|
||||
|
||||
// Additionally, there seems to be a trace of special packets with cmd 1 and 2, that are sent
|
||||
// to UUID_CHARACTERISTIC_DATA_SPECIAL_1 and UUID_CHARACTERISTIC_DATA_SPECIAL_2 instead.
|
||||
// They don't appear on my watch though.
|
||||
|
||||
// The response to CMD_ECG is special and is returned using UUID_CHARACTERISTIC_DATA_ECG_OLD
|
||||
// or UUID_CHARACTERISTIC_DATA_ECG_NEW. The old version is clearly labeled as old in the
|
||||
// unobfuscated part of the code. If both of them exist, old is used (but I presume only one
|
||||
// of them is supposed to exist at a time). They also don't appear on my watch as it doesn't
|
||||
// support ECG.
|
||||
|
||||
// In addition to the proprietary protocol described above, the following standard BLE services
|
||||
// are used:
|
||||
// * org.bluetooth.service.generic_access for device name
|
||||
// * org.bluetooth.service.device_information for manufacturer, model, serial number and
|
||||
// firmware version
|
||||
// * org.bluetooth.service.battery_service for battery level
|
||||
// * org.bluetooth.service.heart_rate is exposed, but doesn't seem to work
|
||||
// * org.bluetooth.service.human_interface_device is exposed, but not even mentioned
|
||||
// in the official app (?) - needs further research
|
||||
// * the custom UUID_CHARACTERISTIC_STEPS is used to sync the pedometer data in real time
|
||||
// via READ or NOTIFY - it's identical to the "sync past data" packet
|
||||
// ({distance:uint24, steps:uint24, calories:uint24})
|
||||
// * (?) 0000FEE7-0000-1000-8000-00805F9B34FB another custom service
|
||||
// (NOT UUID_CHARACTERISTIC_DATA_ECG_OLD!!!) not mentioned anywhere in the official app,
|
||||
// containing the following characteristics:
|
||||
// * 0000FEA1-0000-1000-8000-00805F9B34FB - READ, NOTIFY
|
||||
// * 0000FEC9-0000-1000-8000-00805F9B34FB - READ
|
||||
|
||||
// The above standard services are internally handled by the app using the following
|
||||
// "packet numbers":
|
||||
// * 16 - query steps
|
||||
// * 17 - firmware version
|
||||
// * 18 - query battery
|
||||
// * 19 - DFU status (queries model number, looks for the string DFU and a number == 0 or != 0)
|
||||
// * 20 - protocol version (queries manufacturer name, see description above)
|
||||
|
||||
|
||||
public static final UUID UUID_SERVICE_MOYOUNG = UUID.fromString(String.format(AbstractBTLEDeviceSupport.BASE_UUID, "feea"));
|
||||
public static final UUID UUID_CHARACTERISTIC_STEPS = UUID.fromString(String.format(AbstractBTLEDeviceSupport.BASE_UUID, "fee1"));
|
||||
public static final UUID UUID_CHARACTERISTIC_DATA_OUT = UUID.fromString(String.format(AbstractBTLEDeviceSupport.BASE_UUID, "fee2"));
|
||||
public static final UUID UUID_CHARACTERISTIC_DATA_IN = UUID.fromString(String.format(AbstractBTLEDeviceSupport.BASE_UUID, "fee3"));
|
||||
public static final UUID UUID_CHARACTERISTIC_DATA_SPECIAL_1 = UUID.fromString(String.format(AbstractBTLEDeviceSupport.BASE_UUID, "fee5")); // (*)
|
||||
public static final UUID UUID_CHARACTERISTIC_DATA_SPECIAL_2 = UUID.fromString(String.format(AbstractBTLEDeviceSupport.BASE_UUID, "fee6")); // (*)
|
||||
public static final UUID UUID_CHARACTERISTIC_DATA_ECG_OLD = UUID.fromString(String.format(AbstractBTLEDeviceSupport.BASE_UUID, "fee7")); // (*)
|
||||
public static final UUID UUID_CHARACTERISTIC_DATA_ECG_NEW = UUID.fromString(String.format(AbstractBTLEDeviceSupport.BASE_UUID, "fee8")); // (*)
|
||||
|
||||
|
||||
// Special
|
||||
public static final byte CMD_SHUTDOWN = 81; // {-1}
|
||||
public static final byte CMD_FIND_MY_WATCH = 97; // {}
|
||||
public static final byte CMD_FIND_MY_PHONE = 98; // (*) outgoing {-1} to stop, incoming {0} start, {!=0} stop
|
||||
public static final byte CMD_HS_DFU = 99; // (?) {1} - enableHsDfu(), {0} - queryHsDfuAddress()
|
||||
|
||||
|
||||
// Activity/training tracking
|
||||
|
||||
// CMD_QUERY_LAST_DYNAMIC_RATE is triggered immediately after a training recording is finished on the watch.
|
||||
// The watch sends CMD_QUERY_LAST_DYNAMIC_RATE command to the phone with the first part of the data, and then
|
||||
// the phone is supposed to respond with empty CMD_QUERY_LAST_DYNAMIC_RATE to retrieve the next part.
|
||||
// There seems to be no way to query this data later, or to start communication from phone side.
|
||||
// The data format is uint32 date_recorded, uint8 heart_rate[] (where 0 is invalid measurement and
|
||||
// data is recorded every 1 minute)
|
||||
|
||||
// CMD_QUERY_MOVEMENT_HEART_RATE returns the summary of last 3 trainings recorded on the watch.
|
||||
// This is a cyclic buffer, so the watch will first overwrite entry number 0, then 1, then 2, then 0 again
|
||||
|
||||
// CMD_QUERY_PAST_HEART_RATE_1 and CMD_QUERY_PAST_HEART_RATE_2 don't seem to work at all on my watch.
|
||||
|
||||
// All "date recorded" values are in the hardcoded GMT+8 watch timezone
|
||||
|
||||
public static final byte CMD_QUERY_LAST_DYNAMIC_RATE = 52; // TRANSMISSION TRIGGERED FROM WATCH SIDE AFTER FINISHED TRAINING. Does custom packet splitting. The packet takes no data as input. Send the query repeatedly until you get all the data. THE FIRST PACKET IS SENT BY THE WATCH - THE PHONE QUERIES THIS COMMAND TO GET THE NEXT PART. The response starts with one byte: 0 for first packet, 1 for continuation packet, 2 for end of data. 0,time:uint32,measurement:uint8[] 1,measurement:uint8[] 1,measurement:uint8[] 2
|
||||
public static final byte CMD_QUERY_PAST_HEART_RATE_1 = 53; // (*) Two arrays built of 4 packets each. See below. todayHeartRate(1) starts at 0 and ends at 3, yesterdayHeartRate() starts at 4 and ends at 7. Sampled every 5 minutes.
|
||||
public static final byte CMD_QUERY_PAST_HEART_RATE_2 = 54; // (*) An array built of 20 packets. The packet takes the index as input. i.e. {x} -> {data[N*x], data[N*x+1], ..., data[N*x+N-1]} for x in 0-19 -- todayHeartRate(2). Sampled every 1 minute.
|
||||
public static final byte CMD_QUERY_MOVEMENT_HEART_RATE = 55; // {} -> One packet with 3 entries of 24 bytes each {startTime:uint32, endTime:uint32, validTime:uint16, entry_number:uint8, type:uint8, steps:uint32, distance:uint32, calories:uint16}, everything little endian
|
||||
|
||||
// first byte for CMD_QUERY_LAST_DYNAMIC_RATE packets
|
||||
public static final byte ARG_TRANSMISSION_FIRST = 0;
|
||||
public static final byte ARG_TRANSMISSION_NEXT = 1;
|
||||
public static final byte ARG_TRANSMISSION_LAST = 2; // note: last packet always empty
|
||||
|
||||
// Health measurements
|
||||
public static final byte CMD_QUERY_TIMING_MEASURE_HEART_RATE = 47; // (*) {} -> ???
|
||||
public static final byte CMD_SET_TIMING_MEASURE_HEART_RATE = 31; // (*) {i}, i >= 0, 0 is disabled
|
||||
public static final byte CMD_START_STOP_MEASURE_DYNAMIC_RATE = 104; // (*) {enabled ? 0 : -1}
|
||||
|
||||
public static final byte HR_INTERVAL_OFF = 0;
|
||||
public static final byte HR_INTERVAL_5MIN = 1;
|
||||
public static final byte HR_INTERVAL_10MIN = 2;
|
||||
public static final byte HR_INTERVAL_20MIN = 4;
|
||||
public static final byte HR_INTERVAL_30MIN = 6;
|
||||
|
||||
public static final byte CMD_TRIGGER_MEASURE_BLOOD_PRESSURE = 105; // (?) {0, 0, 0} to start, {-1, -1, -1} to stop -> {unused?, num1, num2}
|
||||
public static final byte CMD_TRIGGER_MEASURE_BLOOD_OXYGEN = 107; // (?) {start ? 0 : -1} -> {num}
|
||||
public static final byte CMD_TRIGGER_MEASURE_HEARTRATE = 109; // {start ? 0 : -1} -> {bpm}
|
||||
public static final byte CMD_ECG = 111; // (?) {heart_rate} or {1} to start or {0} to stop or {2} to query
|
||||
// ECG data is special and comes from UUID_CHARACTERISTIC_DATA_ECG_OLD or UUID_CHARACTERISTIC_DATA_ECG_NEW
|
||||
|
||||
|
||||
// Functionality
|
||||
public static final byte CMD_SYNC_TIME = 49; // {time >> 24, time >> 16, time >> 8, time, 8}, time is a timestamp in seconds in GMT+8
|
||||
|
||||
public static final byte CMD_SYNC_SLEEP = 50; // {} -> {type, start_h, start_m}, repeating, type is SOBER(0),LIGHT(1),RESTFUL(2)
|
||||
public static final byte CMD_SYNC_PAST_SLEEP_AND_STEP = 51; // {b (see below)} -> {x<=2, distance:uint24, steps:uint24, calories:uint24} or {x>2, (sleep data like above)} - two functions same CMD
|
||||
|
||||
// NOTE: these names are as specified in the original app. They do NOT match what my watch actually does. See note in FetchDataOperation.
|
||||
public static final byte ARG_SYNC_YESTERDAY_STEPS = 1;
|
||||
public static final byte ARG_SYNC_DAY_BEFORE_YESTERDAY_STEPS = 2;
|
||||
public static final byte ARG_SYNC_YESTERDAY_SLEEP = 3;
|
||||
public static final byte ARG_SYNC_DAY_BEFORE_YESTERDAY_SLEEP = 4;
|
||||
|
||||
public static final byte SLEEP_SOBER = 0;
|
||||
public static final byte SLEEP_LIGHT = 1;
|
||||
public static final byte SLEEP_RESTFUL = 2;
|
||||
|
||||
public static final byte CMD_QUERY_SLEEP_ACTION = 58; // (*) {i} -> {hour, x[60]}
|
||||
|
||||
public static final byte CMD_SEND_MESSAGE = 65; // {type, message[]}, message is encoded with manual splitting by String.valueOf(0x2080)
|
||||
// CMD_SEND_CALL_OFF_HOOK = 65; // {-1} - the same ID as above, different arguments
|
||||
|
||||
public static final byte CMD_SET_WEATHER_FUTURE = 66; // {weatherId, low_temp, high_temp} * 7
|
||||
public static final byte CMD_SET_WEATHER_TODAY = 67; // {have_pm25 ? 1 : 0, weatherId, temp[, pm25 >> 8, pm25], lunar_or_festival[8], city[8]}, names are UTF-16BE encoded (4 characters each!)
|
||||
public static final byte CMD_SET_WEATHER_LOCATION = 69; // {string utf8}
|
||||
public static final byte CMD_SET_SUNRISE_SUNSET = -75; // {5 bytes unknown, sunrise hour, sunrise min, sunset hour, sunset min, string (location utf8)}
|
||||
|
||||
public static final byte CMD_SET_MUSIC_INFO = 68; // {artist=1/track=0, string}
|
||||
public static final byte CMD_SET_MUSIC_STATE = 123; // {is_playing ? 1 : 0}
|
||||
|
||||
public static final byte CMD_GSENSOR_CALIBRATION = 82; // (?) {}
|
||||
|
||||
public static final byte CMD_QUERY_STEPS_CATEGORY = 89; // (*) {i} -> {0, data:uint16[*]}, {1}, {2, data:uint16[*]}, {3}, query 0+1 together and 2+3 together
|
||||
//public static final byte ARG_QUERY_STEPS_CATEGORY_TODAY_STEPS = 0;
|
||||
//public static final byte ARG_QUERY_STEPS_CATEGORY_YESTERDAY_STEPS = 2;
|
||||
|
||||
public static final byte CMD_SWITCH_CAMERA_VIEW = 102; // {} -> {}, outgoing open screen, incoming take photo
|
||||
|
||||
public static final byte CMD_NOTIFY_PHONE_OPERATION = 103; // ONLY INCOMING! -> {x}, x -> 0 = play/pause, 1 = prev, 2 = next, 3 = reject incoming call)
|
||||
public static final byte CMD_NOTIFY_WEATHER_CHANGE = 100; // ONLY INCOMING! -> {} - when the watch really wants us to retransmit the weather again (it seems to often happen after stopping training - running the training blocks access to main menu so I guess it restarts afterwards or something). Will repeat whenever navigating the menu where the weather should be, and weather won't be visible on watch screen until that happens.
|
||||
|
||||
public static final byte ARG_OPERATION_PLAY_PAUSE = 0;
|
||||
public static final byte ARG_OPERATION_PREV_SONG = 1;
|
||||
public static final byte ARG_OPERATION_NEXT_SONG = 2;
|
||||
public static final byte ARG_OPERATION_DROP_INCOMING_CALL = 3;
|
||||
public static final byte ARG_OPERATION_VOLUME_UP = 4;
|
||||
public static final byte ARG_OPERATION_VOLUME_DOWN = 5;
|
||||
public static final byte ARG_OPERATION_PLAY = 6;
|
||||
public static final byte ARG_OPERATION_PAUSE = 7;
|
||||
public static final byte ARG_OPERATION_SEND_CURRENT_VOLUME = 12; // {0x00-0x10}
|
||||
|
||||
public static final byte CMD_QUERY_ALARM_CLOCK = 33; // (?) {} -> a list of entries like below
|
||||
public static final byte CMD_SET_ALARM_CLOCK = 17; // (?) {id, enable ? 1 : 0, repeat, hour, minute, i >> 8, i, repeatMode}, repeatMode is 0(SINGLE), 127(EVERYDAY), or bitmask of 1,2,4,8,16,32,64(SUNDAY-SATURDAY) is 0,1,2, i is ((year << 12) + (month << 8) + day) where year is 2015-based, month and day start at 1 for repeatMode=SINGLE and 0 otherwise, repeat is 0(SINGLE),1(EVERYDAY),2(OTHER)
|
||||
|
||||
public static final byte CMD_ADVANCED_QUERY = (byte) 0xb9;
|
||||
public static final byte CMD_ADVANCED_CMD = (byte) 0xbb;
|
||||
|
||||
public static final byte CMD_QUERY_POWER_SAVING = (byte) 0xa4;
|
||||
public static final byte CMD_SET_POWER_SAVING = (byte) 0x94;
|
||||
|
||||
public static final byte ARG_ADVANCED_SET_ALARM = 0x05;
|
||||
public static final byte ARG_ADVANCED_SET_CALENDAR = 0x08;
|
||||
public static final byte ARG_ADVANCED_QUERY_STOCKS = 0x0e;
|
||||
public static final byte ARG_ADVANCED_QUERY_ALARMS = 0x15;
|
||||
|
||||
public static final byte ARG_ALARM_SET = 0x00;
|
||||
public static final byte ARG_ALARM_DELETE = 0x02;
|
||||
public static final byte ARG_ALARM_FROM_WATCH = 0x04;
|
||||
|
||||
public static final byte ARG_CALENDAR_ADD_ITEM = 0x00;
|
||||
public static final byte ARG_CALENDAR_DISABLE = 0x04;
|
||||
public static final byte ARG_CALENDAR_FINISHED = 0x05;
|
||||
public static final byte ARG_CALENDAR_CLEAR = 0x06;
|
||||
|
||||
public static final int MAX_CALENDAR_ITEMS = 12; // Tested only on Colmi i28 Ultra, move to coordinator if different on other devices
|
||||
|
||||
// Settings
|
||||
public static final byte CMD_SET_USER_INFO = 18; // (?) {height, weight, age, gender}, MALE = 0, FEMALE = 1
|
||||
|
||||
public static final byte CMD_QUERY_DOMINANT_HAND = 36; // (*) {} -> {value}
|
||||
public static final byte CMD_SET_DOMINANT_HAND = 20; // (*) {value}
|
||||
|
||||
public static final byte CMD_QUERY_DISPLAY_DEVICE_FUNCTION = 37; // (*) {} - current, {-1} - list all supported -> {[-1, ], ...} (prefixed with -1 if lists supported, nothing otherwise)
|
||||
public static final byte CMD_SET_DISPLAY_DEVICE_FUNCTION = 21; // (*) {..., 0} - null terminated list of functions to enable
|
||||
|
||||
public static final byte CMD_QUERY_GOAL_STEP = 38; // {} -> {value, value >> 8, value >> 16, value >> 24} // this has the endianness swapped between query and set
|
||||
public static final byte CMD_SET_GOAL_STEP = 22; // {value >> 24, value >> 16, value >> 8, value} // yes, really
|
||||
|
||||
public static final byte CMD_QUERY_TIME_SYSTEM = 39; // {} -> {value}
|
||||
public static final byte CMD_SET_TIME_SYSTEM = 23; // {value}
|
||||
|
||||
// quick view = enable display when wrist is lifted
|
||||
public static final byte CMD_QUERY_QUICK_VIEW = 40; // {} -> {value}
|
||||
public static final byte CMD_SET_QUICK_VIEW = 24; // {enabled ? 1 : 0}
|
||||
|
||||
public static final byte CMD_QUERY_DISPLAY_WATCH_FACE = 41; // {} -> {value}
|
||||
public static final byte CMD_SET_DISPLAY_WATCH_FACE = 25; // {value}
|
||||
|
||||
public static final byte CMD_QUERY_METRIC_SYSTEM = 42; // {} -> {value}
|
||||
public static final byte CMD_SET_METRIC_SYSTEM = 26; // {value}
|
||||
|
||||
public static final byte CMD_QUERY_DEVICE_LANGUAGE = 43; // {} -> {value, bitmask_of_supported_langs:uint32}
|
||||
public static final byte CMD_SET_DEVICE_LANGUAGE = 27; // {new_value}
|
||||
|
||||
// enables "other" (as in "not a messaging app") on the notifications configuration screen in the official app
|
||||
// seems to be used only in the app, not sure why they even store it on the watch
|
||||
public static final byte CMD_QUERY_OTHER_MESSAGE_STATE = 44; // {} -> {value}
|
||||
public static final byte CMD_SET_OTHER_MESSAGE_STATE = 28; // {enabled ? 1 : 0}
|
||||
|
||||
public static final byte CMD_QUERY_SEDENTARY_REMINDER = 45; // {} -> {value}
|
||||
public static final byte CMD_SET_SEDENTARY_REMINDER = 29; // {enabled ? 1 : 0}
|
||||
|
||||
public static final byte CMD_QUERY_DEVICE_VERSION = 46; // {} -> {value}
|
||||
public static final byte CMD_SET_DEVICE_VERSION = 30; // {new_value}
|
||||
|
||||
public static final byte CMD_QUERY_WATCH_FACE_LAYOUT = 57; // (*) {} -> {time_position, time_top_content, time_bottom_content, text_color >> 8, text_color, background_picture_md5[32]}
|
||||
public static final byte CMD_SET_WATCH_FACE_LAYOUT = 56; // (*) {time_position, time_top_content, time_bottom_content, text_color >> 8, text_color, background_picture_md5[32]}, text_color is R5G6B5, background_picture is stored as hex digits (numbers 0-15 not chars '0'-'F' !)
|
||||
|
||||
public static final byte CMD_SET_STEP_LENGTH = 84; // (?) {value}
|
||||
|
||||
public static final byte CMD_QUERY_DO_NOT_DISTURB_TIME = -127; // {} -> {start >> 8, start, end >> 8, end} these are 16-bit values (somebody was drunk while writing this or what?)
|
||||
public static final byte CMD_SET_DO_NOT_DISTURB_TIME = 113; // {start_hour, start_min, end_hour, end_min}
|
||||
|
||||
public static final byte CMD_QUERY_QUICK_VIEW_TIME = -126; // {} -> {start >> 8, start, end >> 8, end} these are 16-bit values (somebody was drunk while writing this or what?)
|
||||
public static final byte CMD_SET_QUICK_VIEW_TIME = 114; // {start_hour, start_min, end_hour, end_min}
|
||||
|
||||
public static final byte CMD_QUERY_REMINDERS_TO_MOVE_PERIOD = -125; // {} -> {period, steps, start_hour, end_hour}
|
||||
public static final byte CMD_SET_REMINDERS_TO_MOVE_PERIOD = 115; // {period, steps, start_hour, end_hour}
|
||||
|
||||
public static final byte CMD_QUERY_SUPPORT_WATCH_FACE = -124; // (*) {} -> {count >> 8, count, ...}
|
||||
|
||||
public static final byte CMD_QUERY_PSYCHOLOGICAL_PERIOD = -123; // (*) {} -> ??? (too lazy to check, sorry :P)
|
||||
public static final byte CMD_SET_PSYCHOLOGICAL_PERIOD = 117; // (*) {encodeConfiguredReminders(info), 15, info.getPhysiologcalPeriod(), info.getMenstrualPeriod(), info.startDate.get(Calendar.MONTH), info.startDate.get(Calendar.DATE), info.getReminderHour(), info.getReminderMinute(), info.getReminderHour(), info.getReminderMinute(), info.getReminderHour(), info.getReminderMinute(), info.getReminderHour(), info.getReminderMinute()}
|
||||
// encodeConfiguredReminders(CRPPhysiologcalPeriodInfo info) {
|
||||
// int i = info.isMenstrualReminder() ? 241 : 240;
|
||||
// if (info.isOvulationReminder())
|
||||
// i += 2;
|
||||
// if (info.isOvulationDayReminder())
|
||||
// i += 4;
|
||||
// if (info.isOvulationEndReminder())
|
||||
// i += 8;
|
||||
// return (byte) i;
|
||||
// }
|
||||
|
||||
// no idea what this does
|
||||
public static final byte CMD_QUERY_BREATHING_LIGHT = -120; // {} -> {value}
|
||||
public static final byte CMD_SET_BREATHING_LIGHT = 120; // {enabled ? 1 : 0}
|
||||
|
||||
public static final byte TRAINING_TYPE_WALK = 0;
|
||||
public static final byte TRAINING_TYPE_RUN = 1;
|
||||
public static final byte TRAINING_TYPE_BIKING = 2;
|
||||
public static final byte TRAINING_TYPE_ROPE = 3;
|
||||
public static final byte TRAINING_TYPE_BADMINTON = 4;
|
||||
public static final byte TRAINING_TYPE_BASKETBALL = 5;
|
||||
public static final byte TRAINING_TYPE_FOOTBALL = 6;
|
||||
public static final byte TRAINING_TYPE_SWIM = 7;
|
||||
public static final byte TRAINING_TYPE_MOUNTAINEERING = 8;
|
||||
public static final byte TRAINING_TYPE_TENNIS = 9;
|
||||
public static final byte TRAINING_TYPE_RUGBY = 10;
|
||||
public static final byte TRAINING_TYPE_GOLF = 11;
|
||||
|
||||
// The watch stores all dates in GMT+8 time zone with seconds resolution
|
||||
// These helper functions convert between the watch time representation and local system representation
|
||||
|
||||
public static int LocalTimeToWatchTime(Date localTime)
|
||||
{
|
||||
SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss.SSS");
|
||||
simpleDateFormat.setTimeZone(TimeZone.getDefault());
|
||||
String format = simpleDateFormat.format(localTime);
|
||||
simpleDateFormat.setTimeZone(TimeZone.getTimeZone("GMT+8"));
|
||||
try {
|
||||
return (int)(simpleDateFormat.parse(format).getTime() / 1000);
|
||||
} catch (ParseException e) {
|
||||
throw new IllegalArgumentException(e);
|
||||
}
|
||||
}
|
||||
|
||||
public static Date WatchTimeToLocalTime(int watchTime)
|
||||
{
|
||||
SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss.SSS");
|
||||
simpleDateFormat.setTimeZone(TimeZone.getTimeZone("GMT+8"));
|
||||
String format = simpleDateFormat.format(new Date((long)watchTime * 1000));
|
||||
simpleDateFormat.setTimeZone(TimeZone.getDefault());
|
||||
try {
|
||||
return simpleDateFormat.parse(format);
|
||||
} catch (ParseException e) {
|
||||
throw new IllegalArgumentException(e);
|
||||
}
|
||||
}
|
||||
|
||||
// The notification types used by CMD_SEND_MESSAGE
|
||||
public static final byte NOTIFICATION_TYPE_CALL_OFF_HOOK = -1;
|
||||
public static final byte NOTIFICATION_TYPE_CALL = 0;
|
||||
public static final byte NOTIFICATION_TYPE_MESSAGE_SMS = 1;
|
||||
public static final byte NOTIFICATION_TYPE_MESSAGE_WECHAT = 2;
|
||||
public static final byte NOTIFICATION_TYPE_MESSAGE_QQ = 3;
|
||||
public static final byte NOTIFICATION_TYPE_MESSAGE_FACEBOOK = 4;
|
||||
public static final byte NOTIFICATION_TYPE_MESSAGE_TWITTER = 5;
|
||||
public static final byte NOTIFICATION_TYPE_MESSAGE_INSTAGRAM = 6;
|
||||
public static final byte NOTIFICATION_TYPE_MESSAGE_SKYPE = 7;
|
||||
public static final byte NOTIFICATION_TYPE_MESSAGE_WHATSAPP = 8;
|
||||
public static final byte NOTIFICATION_TYPE_MESSAGE_LINE = 9;
|
||||
public static final byte NOTIFICATION_TYPE_MESSAGE_KAKAO = 10;
|
||||
public static final byte NOTIFICATION_TYPE_MESSAGE_OTHER = 11;
|
||||
|
||||
public static byte notificationType(NotificationType type)
|
||||
{
|
||||
switch(type)
|
||||
{
|
||||
case FACEBOOK:
|
||||
case FACEBOOK_MESSENGER:
|
||||
return NOTIFICATION_TYPE_MESSAGE_FACEBOOK;
|
||||
case GENERIC_SMS:
|
||||
return NOTIFICATION_TYPE_MESSAGE_SMS;
|
||||
case INSTAGRAM:
|
||||
return NOTIFICATION_TYPE_MESSAGE_INSTAGRAM;
|
||||
case KAKAO_TALK:
|
||||
return NOTIFICATION_TYPE_MESSAGE_KAKAO;
|
||||
case LINE:
|
||||
return NOTIFICATION_TYPE_MESSAGE_LINE;
|
||||
case SKYPE:
|
||||
return NOTIFICATION_TYPE_MESSAGE_SKYPE;
|
||||
case TWITTER:
|
||||
return NOTIFICATION_TYPE_MESSAGE_TWITTER;
|
||||
case WECHAT:
|
||||
return NOTIFICATION_TYPE_MESSAGE_WECHAT;
|
||||
case WHATSAPP:
|
||||
return NOTIFICATION_TYPE_MESSAGE_WHATSAPP;
|
||||
default:
|
||||
return NOTIFICATION_TYPE_MESSAGE_OTHER;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
// Weather types
|
||||
public static final byte WEATHER_CLOUDY = 0;
|
||||
public static final byte WEATHER_FOGGY = 1;
|
||||
public static final byte WEATHER_OVERCAST = 2;
|
||||
public static final byte WEATHER_RAINY = 3;
|
||||
public static final byte WEATHER_SNOWY = 4;
|
||||
public static final byte WEATHER_SUNNY = 5;
|
||||
public static final byte WEATHER_SANDSTORM = 6; // aka "wind", according to the image
|
||||
public static final byte WEATHER_HAZE = 7; // it's basically very big fog :P
|
||||
// NOTE: values > 7 give random glitchy crap as images :D
|
||||
|
||||
public static byte openWeatherConditionToMoyoungConditionId(int openWeatherMapCondition) {
|
||||
int openWeatherMapGroup = openWeatherMapCondition / 100;
|
||||
switch (openWeatherMapGroup) {
|
||||
case 2: // thunderstorm
|
||||
case 3: // drizzle
|
||||
case 5: // rain
|
||||
return MoyoungConstants.WEATHER_RAINY;
|
||||
case 6: // snow
|
||||
return MoyoungConstants.WEATHER_SNOWY;
|
||||
case 7: // fog
|
||||
return MoyoungConstants.WEATHER_FOGGY;
|
||||
case 8: // clear / clouds
|
||||
if (openWeatherMapCondition <= 801) // few clouds
|
||||
return MoyoungConstants.WEATHER_SUNNY;
|
||||
if (openWeatherMapCondition >= 804) // overcast clouds
|
||||
return MoyoungConstants.WEATHER_CLOUDY;
|
||||
return MoyoungConstants.WEATHER_OVERCAST;
|
||||
case 9: // extreme
|
||||
default:
|
||||
if (openWeatherMapCondition == 905) // windy
|
||||
return MoyoungConstants.WEATHER_SANDSTORM;
|
||||
return MoyoungConstants.WEATHER_HAZE;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
public static final String PREF_MOYOUNG_WATCH_FACE = "moyoung_watch_face";
|
||||
public static final String PREF_LANGUAGE = "moyoung_language";
|
||||
public static final String PREF_LANGUAGE_SUPPORT = "moyoung_language_supported";
|
||||
public static final String PREF_MOYOUNG_DEVICE_VERSION = "moyoung_device_version";
|
||||
}
|
@ -0,0 +1,38 @@
|
||||
/* Copyright (C) 2019 krzys_h
|
||||
|
||||
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 <http://www.gnu.org/licenses/>. */
|
||||
package nodomain.freeyourgadget.gadgetbridge.devices.moyoung;
|
||||
|
||||
import nodomain.freeyourgadget.gadgetbridge.model.WeatherSpec;
|
||||
|
||||
public class MoyoungWeatherForecast {
|
||||
public final byte conditionId;
|
||||
public final byte minTemp;
|
||||
public final byte maxTemp;
|
||||
|
||||
public MoyoungWeatherForecast(byte conditionId, byte minTemp, byte maxTemp) {
|
||||
this.conditionId = conditionId;
|
||||
this.minTemp = minTemp;
|
||||
this.maxTemp = maxTemp;
|
||||
}
|
||||
|
||||
public MoyoungWeatherForecast(WeatherSpec.Daily forecast)
|
||||
{
|
||||
conditionId = MoyoungConstants.openWeatherConditionToMoyoungConditionId(forecast.conditionCode);
|
||||
minTemp = (byte)(forecast.minTemp - 273); // Kelvin -> Celcius
|
||||
maxTemp = (byte)(forecast.maxTemp - 273); // Kelvin -> Celcius
|
||||
}
|
||||
}
|
@ -0,0 +1,52 @@
|
||||
/* Copyright (C) 2019 krzys_h
|
||||
|
||||
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 <http://www.gnu.org/licenses/>. */
|
||||
package nodomain.freeyourgadget.gadgetbridge.devices.moyoung;
|
||||
|
||||
import androidx.annotation.NonNull;
|
||||
import androidx.annotation.Nullable;
|
||||
|
||||
import nodomain.freeyourgadget.gadgetbridge.model.WeatherSpec;
|
||||
import nodomain.freeyourgadget.gadgetbridge.util.StringUtils;
|
||||
|
||||
public class MoyoungWeatherToday {
|
||||
public final byte conditionId;
|
||||
public final byte currentTemp;
|
||||
public final Short pm25; // (*)
|
||||
public final String lunar_or_festival; // (*)
|
||||
public final String city; // (*)
|
||||
|
||||
public MoyoungWeatherToday(byte conditionId, byte currentTemp, @Nullable Short pm25, @NonNull String lunar_or_festival, @NonNull String city) {
|
||||
if (lunar_or_festival.length() != 4)
|
||||
throw new IllegalArgumentException("lunar_or_festival");
|
||||
if (city.length() != 4)
|
||||
throw new IllegalArgumentException("city");
|
||||
this.conditionId = conditionId;
|
||||
this.currentTemp = currentTemp;
|
||||
this.pm25 = pm25;
|
||||
this.lunar_or_festival = lunar_or_festival;
|
||||
this.city = city;
|
||||
}
|
||||
|
||||
public MoyoungWeatherToday(WeatherSpec weatherSpec)
|
||||
{
|
||||
conditionId = MoyoungConstants.openWeatherConditionToMoyoungConditionId(weatherSpec.currentConditionCode);
|
||||
currentTemp = (byte)(weatherSpec.currentTemp - 273); // Kelvin -> Celcius
|
||||
pm25 = null;
|
||||
lunar_or_festival = StringUtils.pad("", 4);
|
||||
city = StringUtils.pad(weatherSpec.location.substring(0, 4), 4);
|
||||
}
|
||||
}
|
@ -0,0 +1,275 @@
|
||||
/* Copyright (C) 2019 krzys_h
|
||||
|
||||
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 <http://www.gnu.org/licenses/>. */
|
||||
package nodomain.freeyourgadget.gadgetbridge.devices.moyoung.samples;
|
||||
|
||||
import androidx.annotation.NonNull;
|
||||
import androidx.annotation.Nullable;
|
||||
|
||||
import org.slf4j.Logger;
|
||||
import org.slf4j.LoggerFactory;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.Collections;
|
||||
import java.util.HashMap;
|
||||
import java.util.List;
|
||||
import java.util.ListIterator;
|
||||
import java.util.Map;
|
||||
|
||||
import de.greenrobot.dao.AbstractDao;
|
||||
import de.greenrobot.dao.Property;
|
||||
import de.greenrobot.dao.internal.SqlUtils;
|
||||
import de.greenrobot.dao.query.WhereCondition;
|
||||
import nodomain.freeyourgadget.gadgetbridge.database.DBHelper;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.AbstractSampleProvider;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.moyoung.MoyoungConstants;
|
||||
import nodomain.freeyourgadget.gadgetbridge.entities.DaoSession;
|
||||
import nodomain.freeyourgadget.gadgetbridge.entities.Device;
|
||||
import nodomain.freeyourgadget.gadgetbridge.entities.MoyoungActivitySample;
|
||||
import nodomain.freeyourgadget.gadgetbridge.entities.MoyoungActivitySampleDao;
|
||||
import nodomain.freeyourgadget.gadgetbridge.entities.MoyoungHeartRateSample;
|
||||
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
|
||||
import nodomain.freeyourgadget.gadgetbridge.model.ActivityKind;
|
||||
import nodomain.freeyourgadget.gadgetbridge.model.ActivitySample;
|
||||
|
||||
public class MoyoungActivitySampleProvider extends AbstractSampleProvider<MoyoungActivitySample> {
|
||||
private static final Logger LOG = LoggerFactory.getLogger(MoyoungActivitySampleProvider.class);
|
||||
|
||||
public static final int SOURCE_NOT_MEASURED = -1;
|
||||
public static final int SOURCE_STEPS_REALTIME = 1; // steps gathered at realtime from the steps characteristic
|
||||
public static final int SOURCE_STEPS_SUMMARY = 2; // steps gathered from the daily summary
|
||||
public static final int SOURCE_STEPS_IDLE = 3; // idle sample inserted because the user was not moving (to differentiate from missing data because watch not connected)
|
||||
public static final int SOURCE_SLEEP_SUMMARY = 4; // data collected from the sleep function
|
||||
public static final int SOURCE_SINGLE_MEASURE = 5; // heart rate / blood data gathered from the "single measurement" function
|
||||
public static final int SOURCE_TRAINING_HEARTRATE = 6; // heart rate data collected from the training function
|
||||
public static final int SOURCE_BATTERY = 7; // battery report
|
||||
|
||||
public static final int ACTIVITY_NOT_MEASURED = -1;
|
||||
public static final int ACTIVITY_TRAINING_WALK = MoyoungConstants.TRAINING_TYPE_WALK;
|
||||
public static final int ACTIVITY_TRAINING_RUN = MoyoungConstants.TRAINING_TYPE_RUN;
|
||||
public static final int ACTIVITY_TRAINING_BIKING = MoyoungConstants.TRAINING_TYPE_BIKING;
|
||||
public static final int ACTIVITY_TRAINING_ROPE = MoyoungConstants.TRAINING_TYPE_ROPE;
|
||||
public static final int ACTIVITY_TRAINING_BADMINTON = MoyoungConstants.TRAINING_TYPE_BADMINTON;
|
||||
public static final int ACTIVITY_TRAINING_BASKETBALL = MoyoungConstants.TRAINING_TYPE_BASKETBALL;
|
||||
public static final int ACTIVITY_TRAINING_FOOTBALL = MoyoungConstants.TRAINING_TYPE_FOOTBALL;
|
||||
public static final int ACTIVITY_TRAINING_SWIM = MoyoungConstants.TRAINING_TYPE_SWIM;
|
||||
public static final int ACTIVITY_TRAINING_MOUNTAINEERING = MoyoungConstants.TRAINING_TYPE_MOUNTAINEERING;
|
||||
public static final int ACTIVITY_TRAINING_TENNIS = MoyoungConstants.TRAINING_TYPE_TENNIS;
|
||||
public static final int ACTIVITY_TRAINING_RUGBY = MoyoungConstants.TRAINING_TYPE_RUGBY;
|
||||
public static final int ACTIVITY_TRAINING_GOLF = MoyoungConstants.TRAINING_TYPE_GOLF;
|
||||
public static final int ACTIVITY_SLEEP_LIGHT = 16;
|
||||
public static final int ACTIVITY_SLEEP_RESTFUL = 17;
|
||||
public static final int ACTIVITY_SLEEP_START = 18;
|
||||
public static final int ACTIVITY_SLEEP_END = 19;
|
||||
|
||||
public MoyoungActivitySampleProvider(GBDevice device, DaoSession session) {
|
||||
super(device, session);
|
||||
}
|
||||
|
||||
@Override
|
||||
public AbstractDao<MoyoungActivitySample, ?> getSampleDao() {
|
||||
return getSession().getMoyoungActivitySampleDao();
|
||||
}
|
||||
|
||||
@NonNull
|
||||
@Override
|
||||
protected Property getTimestampSampleProperty() {
|
||||
return MoyoungActivitySampleDao.Properties.Timestamp;
|
||||
}
|
||||
|
||||
@Nullable
|
||||
@Override
|
||||
protected Property getRawKindSampleProperty() {
|
||||
return MoyoungActivitySampleDao.Properties.RawKind;
|
||||
}
|
||||
|
||||
@NonNull
|
||||
@Override
|
||||
protected Property getDeviceIdentifierSampleProperty() {
|
||||
return MoyoungActivitySampleDao.Properties.DeviceId;
|
||||
}
|
||||
|
||||
@Override
|
||||
public MoyoungActivitySample createActivitySample() {
|
||||
return new MoyoungActivitySample();
|
||||
}
|
||||
|
||||
@Override
|
||||
public ActivityKind normalizeType(int rawType) {
|
||||
if (rawType == ACTIVITY_NOT_MEASURED)
|
||||
return ActivityKind.NOT_MEASURED;
|
||||
else if (rawType == ACTIVITY_SLEEP_LIGHT)
|
||||
return ActivityKind.LIGHT_SLEEP;
|
||||
else if (rawType == ACTIVITY_SLEEP_RESTFUL)
|
||||
return ActivityKind.DEEP_SLEEP;
|
||||
else if (rawType == ACTIVITY_SLEEP_START || rawType == ACTIVITY_SLEEP_END)
|
||||
return ActivityKind.NOT_MEASURED;
|
||||
else if (rawType == ACTIVITY_TRAINING_WALK)
|
||||
return ActivityKind.WALKING;
|
||||
else if (rawType == ACTIVITY_TRAINING_RUN)
|
||||
return ActivityKind.RUNNING;
|
||||
else if (rawType == ACTIVITY_TRAINING_BIKING)
|
||||
return ActivityKind.CYCLING;
|
||||
else if (rawType == ACTIVITY_TRAINING_SWIM)
|
||||
return ActivityKind.SWIMMING;
|
||||
else if (rawType == ACTIVITY_TRAINING_ROPE || rawType == ACTIVITY_TRAINING_BADMINTON ||
|
||||
rawType == ACTIVITY_TRAINING_BASKETBALL || rawType == ACTIVITY_TRAINING_FOOTBALL ||
|
||||
rawType == ACTIVITY_TRAINING_MOUNTAINEERING || rawType == ACTIVITY_TRAINING_TENNIS ||
|
||||
rawType == ACTIVITY_TRAINING_RUGBY || rawType == ACTIVITY_TRAINING_GOLF)
|
||||
return ActivityKind.EXERCISE;
|
||||
else
|
||||
return ActivityKind.ACTIVITY;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int toRawActivityKind(ActivityKind activityKind) {
|
||||
if (activityKind == ActivityKind.NOT_MEASURED)
|
||||
return ACTIVITY_NOT_MEASURED;
|
||||
else if (activityKind == ActivityKind.LIGHT_SLEEP)
|
||||
return ACTIVITY_SLEEP_LIGHT;
|
||||
else if (activityKind == ActivityKind.DEEP_SLEEP)
|
||||
return ACTIVITY_SLEEP_RESTFUL;
|
||||
else if (activityKind == ActivityKind.ACTIVITY)
|
||||
return ACTIVITY_NOT_MEASURED; // TODO: ?
|
||||
else
|
||||
throw new IllegalArgumentException("Invalid Gadgetbridge activity kind: " + activityKind);
|
||||
}
|
||||
|
||||
@Override
|
||||
public float normalizeIntensity(int rawIntensity) {
|
||||
if (rawIntensity == ActivitySample.NOT_MEASURED)
|
||||
return Float.NEGATIVE_INFINITY;
|
||||
else
|
||||
return rawIntensity;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected List<MoyoungActivitySample> getGBActivitySamples(final int timestamp_from, final int timestamp_to) {
|
||||
LOG.trace(
|
||||
"Getting Moyoung activity samples between {} and {}",
|
||||
timestamp_from,
|
||||
timestamp_to
|
||||
);
|
||||
final long nanoStart = System.nanoTime();
|
||||
|
||||
final List<MoyoungActivitySample> samples = fillGaps(
|
||||
super.getGBActivitySamples(timestamp_from, timestamp_to),
|
||||
timestamp_from,
|
||||
timestamp_to
|
||||
);
|
||||
|
||||
final Map<Integer, MoyoungActivitySample> sampleByTs = new HashMap<>();
|
||||
for (final MoyoungActivitySample sample : samples) {
|
||||
sampleByTs.put(sample.getTimestamp(), sample);
|
||||
}
|
||||
|
||||
overlayHeartRate(sampleByTs, timestamp_from, timestamp_to);
|
||||
// overlaySleep(sampleByTs, timestamp_from, timestamp_to);
|
||||
|
||||
// Add empty dummy samples every 5 min to make sure the charts and stats aren't too malformed
|
||||
// This is necessary due to the Colmi rings just reporting steps/calories/distance aggregates per hour
|
||||
// for (int i=timestamp_from; i<=timestamp_to; i+=300) {
|
||||
// MoyoungActivitySample sample = sampleByTs.get(i);
|
||||
// if (sample == null) {
|
||||
// sample = new MoyoungActivitySample();
|
||||
// sample.setTimestamp(i);
|
||||
// sample.setProvider(this);
|
||||
// sample.setRawKind(ActivitySample.NOT_MEASURED);
|
||||
// sampleByTs.put(i, sample);
|
||||
// }
|
||||
// }
|
||||
|
||||
final List<MoyoungActivitySample> finalSamples = new ArrayList<>(sampleByTs.values());
|
||||
Collections.sort(finalSamples, (a, b) -> Integer.compare(a.getTimestamp(), b.getTimestamp()));
|
||||
|
||||
final long nanoEnd = System.nanoTime();
|
||||
final long executionTime = (nanoEnd - nanoStart) / 1000000;
|
||||
LOG.trace("Getting Moyoung samples took {}ms", executionTime);
|
||||
|
||||
return finalSamples;
|
||||
}
|
||||
|
||||
private void overlayHeartRate(final Map<Integer, MoyoungActivitySample> sampleByTs, final int timestamp_from, final int timestamp_to) {
|
||||
final MoyoungHeartRateSampleProvider heartRateSampleProvider = new MoyoungHeartRateSampleProvider(getDevice(), getSession());
|
||||
final List<MoyoungHeartRateSample> hrSamples = heartRateSampleProvider.getAllSamples(timestamp_from * 1000L, timestamp_to * 1000L);
|
||||
|
||||
for (final MoyoungHeartRateSample hrSample : hrSamples) {
|
||||
// round to the nearest minute, we don't need per-second granularity
|
||||
final int tsSeconds = (int) ((hrSample.getTimestamp() / 1000) / 60) * 60;
|
||||
MoyoungActivitySample sample = sampleByTs.get(tsSeconds);
|
||||
if (sample == null) {
|
||||
sample = new MoyoungActivitySample();
|
||||
sample.setTimestamp(tsSeconds);
|
||||
sample.setProvider(this);
|
||||
sampleByTs.put(tsSeconds, sample);
|
||||
}
|
||||
|
||||
sample.setHeartRate(hrSample.getHeartRate());
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Set the activity kind from NOT_MEASURED to new_raw_activity_kind on the given range
|
||||
* @param timestamp_from the start timestamp
|
||||
* @param timestamp_to the end timestamp
|
||||
* @param new_raw_activity_kind the activity kind to set
|
||||
*/
|
||||
public void updateActivityInRange(int timestamp_from, int timestamp_to, int new_raw_activity_kind)
|
||||
{
|
||||
// greenDAO does not provide a bulk update functionality, and manual update fails because
|
||||
// of no primary key
|
||||
|
||||
Property timestampProperty = getTimestampSampleProperty();
|
||||
Device dbDevice = DBHelper.findDevice(getDevice(), getSession());
|
||||
if (dbDevice == null)
|
||||
throw new IllegalStateException();
|
||||
Property deviceProperty = getDeviceIdentifierSampleProperty();
|
||||
|
||||
/*QueryBuilder<MoyoungActivitySample> qb = getSampleDao().queryBuilder();
|
||||
qb.where(deviceProperty.eq(dbDevice.getId()))
|
||||
.where(timestampProperty.ge(timestamp_from), timestampProperty.le(timestamp_to))
|
||||
.where(getRawKindSampleProperty().eq(ACTIVITY_NOT_MEASURED));
|
||||
List<MoyoungActivitySample> samples = qb.build().list();
|
||||
for (MoyoungActivitySample sample : samples) {
|
||||
sample.setProvider(this);
|
||||
sample.setRawKind(new_raw_activity_kind);
|
||||
sample.update();
|
||||
}*/
|
||||
|
||||
String tablename = getSampleDao().getTablename();
|
||||
String baseSql = SqlUtils.createSqlUpdate(tablename, new String[] { getRawKindSampleProperty().columnName }, new String[] { });
|
||||
StringBuilder builder = new StringBuilder(baseSql);
|
||||
|
||||
List<Object> values = new ArrayList<>();
|
||||
values.add(new_raw_activity_kind);
|
||||
List<WhereCondition> whereConditions = new ArrayList<>();
|
||||
whereConditions.add(deviceProperty.eq(dbDevice.getId()));
|
||||
whereConditions.add(timestampProperty.ge(timestamp_from));
|
||||
whereConditions.add(timestampProperty.le(timestamp_to));
|
||||
whereConditions.add(getRawKindSampleProperty().eq(ACTIVITY_NOT_MEASURED));
|
||||
|
||||
ListIterator<WhereCondition> iter = whereConditions.listIterator();
|
||||
while (iter.hasNext()) {
|
||||
if (iter.hasPrevious()) {
|
||||
builder.append(" AND ");
|
||||
}
|
||||
WhereCondition condition = iter.next();
|
||||
condition.appendTo(builder, tablename);
|
||||
condition.appendValuesTo(values);
|
||||
}
|
||||
getSampleDao().getDatabase().execSQL(builder.toString(), values.toArray());
|
||||
}
|
||||
}
|
@ -0,0 +1,56 @@
|
||||
/* Copyright (C) 2024 Arjan Schrijver
|
||||
|
||||
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.moyoung.samples;
|
||||
|
||||
import androidx.annotation.NonNull;
|
||||
|
||||
import de.greenrobot.dao.AbstractDao;
|
||||
import de.greenrobot.dao.Property;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.AbstractTimeSampleProvider;
|
||||
import nodomain.freeyourgadget.gadgetbridge.entities.DaoSession;
|
||||
import nodomain.freeyourgadget.gadgetbridge.entities.MoyoungBloodPressureSample;
|
||||
import nodomain.freeyourgadget.gadgetbridge.entities.MoyoungBloodPressureSampleDao;
|
||||
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
|
||||
|
||||
public class MoyoungBloodPressureSampleProvider extends AbstractTimeSampleProvider<MoyoungBloodPressureSample> {
|
||||
public MoyoungBloodPressureSampleProvider(final GBDevice device, final DaoSession session) {
|
||||
super(device, session);
|
||||
}
|
||||
|
||||
@NonNull
|
||||
@Override
|
||||
public AbstractDao<MoyoungBloodPressureSample, ?> getSampleDao() {
|
||||
return getSession().getMoyoungBloodPressureSampleDao();
|
||||
}
|
||||
|
||||
@NonNull
|
||||
@Override
|
||||
protected Property getTimestampSampleProperty() {
|
||||
return MoyoungBloodPressureSampleDao.Properties.Timestamp;
|
||||
}
|
||||
|
||||
@NonNull
|
||||
@Override
|
||||
protected Property getDeviceIdentifierSampleProperty() {
|
||||
return MoyoungBloodPressureSampleDao.Properties.DeviceId;
|
||||
}
|
||||
|
||||
@Override
|
||||
public MoyoungBloodPressureSample createSample() {
|
||||
return new MoyoungBloodPressureSample();
|
||||
}
|
||||
}
|
@ -0,0 +1,56 @@
|
||||
/* Copyright (C) 2024 Arjan Schrijver
|
||||
|
||||
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.moyoung.samples;
|
||||
|
||||
import androidx.annotation.NonNull;
|
||||
|
||||
import de.greenrobot.dao.AbstractDao;
|
||||
import de.greenrobot.dao.Property;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.AbstractTimeSampleProvider;
|
||||
import nodomain.freeyourgadget.gadgetbridge.entities.MoyoungHeartRateSample;
|
||||
import nodomain.freeyourgadget.gadgetbridge.entities.MoyoungHeartRateSampleDao;
|
||||
import nodomain.freeyourgadget.gadgetbridge.entities.DaoSession;
|
||||
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
|
||||
|
||||
public class MoyoungHeartRateSampleProvider extends AbstractTimeSampleProvider<MoyoungHeartRateSample> {
|
||||
public MoyoungHeartRateSampleProvider(final GBDevice device, final DaoSession session) {
|
||||
super(device, session);
|
||||
}
|
||||
|
||||
@NonNull
|
||||
@Override
|
||||
public AbstractDao<MoyoungHeartRateSample, ?> getSampleDao() {
|
||||
return getSession().getMoyoungHeartRateSampleDao();
|
||||
}
|
||||
|
||||
@NonNull
|
||||
@Override
|
||||
protected Property getTimestampSampleProperty() {
|
||||
return MoyoungHeartRateSampleDao.Properties.Timestamp;
|
||||
}
|
||||
|
||||
@NonNull
|
||||
@Override
|
||||
protected Property getDeviceIdentifierSampleProperty() {
|
||||
return MoyoungHeartRateSampleDao.Properties.DeviceId;
|
||||
}
|
||||
|
||||
@Override
|
||||
public MoyoungHeartRateSample createSample() {
|
||||
return new MoyoungHeartRateSample();
|
||||
}
|
||||
}
|
@ -0,0 +1,56 @@
|
||||
/* Copyright (C) 2024 Arjan Schrijver
|
||||
|
||||
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.moyoung.samples;
|
||||
|
||||
import androidx.annotation.NonNull;
|
||||
|
||||
import de.greenrobot.dao.AbstractDao;
|
||||
import de.greenrobot.dao.Property;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.AbstractTimeSampleProvider;
|
||||
import nodomain.freeyourgadget.gadgetbridge.entities.MoyoungSpo2Sample;
|
||||
import nodomain.freeyourgadget.gadgetbridge.entities.MoyoungSpo2SampleDao;
|
||||
import nodomain.freeyourgadget.gadgetbridge.entities.DaoSession;
|
||||
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
|
||||
|
||||
public class MoyoungSpo2SampleProvider extends AbstractTimeSampleProvider<MoyoungSpo2Sample> {
|
||||
public MoyoungSpo2SampleProvider(final GBDevice device, final DaoSession session) {
|
||||
super(device, session);
|
||||
}
|
||||
|
||||
@NonNull
|
||||
@Override
|
||||
public AbstractDao<MoyoungSpo2Sample, ?> getSampleDao() {
|
||||
return getSession().getMoyoungSpo2SampleDao();
|
||||
}
|
||||
|
||||
@NonNull
|
||||
@Override
|
||||
protected Property getTimestampSampleProperty() {
|
||||
return MoyoungSpo2SampleDao.Properties.Timestamp;
|
||||
}
|
||||
|
||||
@NonNull
|
||||
@Override
|
||||
protected Property getDeviceIdentifierSampleProperty() {
|
||||
return MoyoungSpo2SampleDao.Properties.DeviceId;
|
||||
}
|
||||
|
||||
@Override
|
||||
public MoyoungSpo2Sample createSample() {
|
||||
return new MoyoungSpo2Sample();
|
||||
}
|
||||
}
|
@ -0,0 +1,21 @@
|
||||
/* Copyright (C) 2019 krzys_h
|
||||
|
||||
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 <http://www.gnu.org/licenses/>. */
|
||||
package nodomain.freeyourgadget.gadgetbridge.devices.moyoung.settings;
|
||||
|
||||
public interface MoyoungEnum {
|
||||
byte value();
|
||||
}
|
@ -0,0 +1,33 @@
|
||||
/* Copyright (C) 2019 krzys_h
|
||||
|
||||
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 <http://www.gnu.org/licenses/>. */
|
||||
package nodomain.freeyourgadget.gadgetbridge.devices.moyoung.settings;
|
||||
|
||||
public enum MoyoungEnumDeviceVersion implements MoyoungEnum {
|
||||
CHINESE_EDITION((byte)0),
|
||||
INTERNATIONAL_EDITION((byte)1);
|
||||
|
||||
public final byte value;
|
||||
|
||||
MoyoungEnumDeviceVersion(byte value) {
|
||||
this.value = value;
|
||||
}
|
||||
|
||||
@Override
|
||||
public byte value() {
|
||||
return value;
|
||||
}
|
||||
}
|
@ -0,0 +1,33 @@
|
||||
/* Copyright (C) 2019 krzys_h
|
||||
|
||||
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 <http://www.gnu.org/licenses/>. */
|
||||
package nodomain.freeyourgadget.gadgetbridge.devices.moyoung.settings;
|
||||
|
||||
public enum MoyoungEnumDominantHand implements MoyoungEnum {
|
||||
LEFT_HAND((byte)0),
|
||||
RIGHT_HAND((byte)1);
|
||||
|
||||
public final byte value;
|
||||
|
||||
MoyoungEnumDominantHand(byte value) {
|
||||
this.value = value;
|
||||
}
|
||||
|
||||
@Override
|
||||
public byte value() {
|
||||
return value;
|
||||
}
|
||||
}
|
@ -0,0 +1,56 @@
|
||||
/* Copyright (C) 2019 krzys_h
|
||||
|
||||
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 <http://www.gnu.org/licenses/>. */
|
||||
package nodomain.freeyourgadget.gadgetbridge.devices.moyoung.settings;
|
||||
|
||||
public enum MoyoungEnumLanguage implements MoyoungEnum {
|
||||
LANGUAGE_ENGLISH((byte)0),
|
||||
LANGUAGE_CHINESE((byte)1),
|
||||
LANGUAGE_JAPANESE((byte)2),
|
||||
LANGUAGE_KOREAN((byte)3),
|
||||
LANGUAGE_GERMAN((byte)4),
|
||||
LANGUAGE_FRENCH((byte)5),
|
||||
LANGUAGE_SPANISH((byte)6),
|
||||
LANGUAGE_ARABIC((byte)7),
|
||||
LANGUAGE_RUSSIAN((byte)8),
|
||||
LANGUAGE_TRADITIONAL((byte)9),
|
||||
LANGUAGE_UKRAINIAN((byte)10),
|
||||
LANGUAGE_ITALIAN((byte)11),
|
||||
LANGUAGE_PORTUGUESE((byte)12),
|
||||
LANGUAGE_DUTCH((byte)13),
|
||||
LANGUAGE_POLISH((byte)14),
|
||||
LANGUAGE_SWEDISH((byte)15),
|
||||
LANGUAGE_FINNISH((byte)16),
|
||||
LANGUAGE_DANISH((byte)17),
|
||||
LANGUAGE_NORWEGIAN((byte)18),
|
||||
LANGUAGE_HUNGARIAN((byte)19),
|
||||
LANGUAGE_CZECH((byte)20),
|
||||
LANGUAGE_BULGARIAN((byte)21),
|
||||
LANGUAGE_ROMANIAN((byte)22),
|
||||
LANGUAGE_SLOVAK_LANGUAGE((byte)23),
|
||||
LANGUAGE_LATVIAN((byte)24);
|
||||
|
||||
public final byte value;
|
||||
|
||||
MoyoungEnumLanguage(byte value) {
|
||||
this.value = value;
|
||||
}
|
||||
|
||||
@Override
|
||||
public byte value() {
|
||||
return value;
|
||||
}
|
||||
}
|
@ -0,0 +1,33 @@
|
||||
/* Copyright (C) 2019 krzys_h
|
||||
|
||||
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 <http://www.gnu.org/licenses/>. */
|
||||
package nodomain.freeyourgadget.gadgetbridge.devices.moyoung.settings;
|
||||
|
||||
public enum MoyoungEnumMetricSystem implements MoyoungEnum {
|
||||
METRIC_SYSTEM((byte)0),
|
||||
IMPERIAL_SYSTEM((byte)1);
|
||||
|
||||
public final byte value;
|
||||
|
||||
MoyoungEnumMetricSystem(byte value) {
|
||||
this.value = value;
|
||||
}
|
||||
|
||||
@Override
|
||||
public byte value() {
|
||||
return value;
|
||||
}
|
||||
}
|
@ -0,0 +1,33 @@
|
||||
/* Copyright (C) 2019 krzys_h
|
||||
|
||||
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 <http://www.gnu.org/licenses/>. */
|
||||
package nodomain.freeyourgadget.gadgetbridge.devices.moyoung.settings;
|
||||
|
||||
public enum MoyoungEnumTimeSystem implements MoyoungEnum {
|
||||
TIME_SYSTEM_12((byte)0),
|
||||
TIME_SYSTEM_24((byte)1);
|
||||
|
||||
public final byte value;
|
||||
|
||||
MoyoungEnumTimeSystem(byte value) {
|
||||
this.value = value;
|
||||
}
|
||||
|
||||
@Override
|
||||
public byte value() {
|
||||
return value;
|
||||
}
|
||||
}
|
@ -0,0 +1,32 @@
|
||||
/* Copyright (C) 2019 krzys_h
|
||||
|
||||
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 <http://www.gnu.org/licenses/>. */
|
||||
package nodomain.freeyourgadget.gadgetbridge.devices.moyoung.settings;
|
||||
|
||||
public abstract class MoyoungSetting<T> {
|
||||
public final String name;
|
||||
public final byte cmdQuery;
|
||||
public final byte cmdSet;
|
||||
|
||||
public MoyoungSetting(String name, byte cmdQuery, byte cmdSet) {
|
||||
this.name = name;
|
||||
this.cmdQuery = cmdQuery;
|
||||
this.cmdSet = cmdSet;
|
||||
}
|
||||
|
||||
public abstract byte[] encode(T value);
|
||||
public abstract T decode(byte[] data);
|
||||
}
|
@ -0,0 +1,37 @@
|
||||
/* Copyright (C) 2019 krzys_h
|
||||
|
||||
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 <http://www.gnu.org/licenses/>. */
|
||||
package nodomain.freeyourgadget.gadgetbridge.devices.moyoung.settings;
|
||||
|
||||
public class MoyoungSettingBool extends MoyoungSetting<Boolean> {
|
||||
public MoyoungSettingBool(String name, byte cmdQuery, byte cmdSet) {
|
||||
super(name, cmdQuery, cmdSet);
|
||||
}
|
||||
|
||||
@Override
|
||||
public byte[] encode(Boolean value) {
|
||||
return new byte[] { value ? (byte)1 : (byte)0 };
|
||||
}
|
||||
|
||||
@Override
|
||||
public Boolean decode(byte[] data) {
|
||||
if (data.length != 1)
|
||||
throw new IllegalArgumentException("Wrong data length, should be 1, was " + data.length);
|
||||
if (data[0] != 0 && data[0] != 1)
|
||||
throw new IllegalArgumentException("Expected a boolean, got " + data[0]);
|
||||
return data[0] != 0;
|
||||
}
|
||||
}
|
@ -0,0 +1,35 @@
|
||||
/* Copyright (C) 2019 krzys_h
|
||||
|
||||
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 <http://www.gnu.org/licenses/>. */
|
||||
package nodomain.freeyourgadget.gadgetbridge.devices.moyoung.settings;
|
||||
|
||||
public class MoyoungSettingByte extends MoyoungSetting<Byte> {
|
||||
public MoyoungSettingByte(String name, byte cmdQuery, byte cmdSet) {
|
||||
super(name, cmdQuery, cmdSet);
|
||||
}
|
||||
|
||||
@Override
|
||||
public byte[] encode(Byte value) {
|
||||
return new byte[] { value };
|
||||
}
|
||||
|
||||
@Override
|
||||
public Byte decode(byte[] data) {
|
||||
if (data.length != 1)
|
||||
throw new IllegalArgumentException("Wrong data length, should be 1, was " + data.length);
|
||||
return data[0];
|
||||
}
|
||||
}
|
@ -0,0 +1,54 @@
|
||||
/* Copyright (C) 2019 krzys_h
|
||||
|
||||
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 <http://www.gnu.org/licenses/>. */
|
||||
package nodomain.freeyourgadget.gadgetbridge.devices.moyoung.settings;
|
||||
|
||||
public class MoyoungSettingEnum<T extends Enum <?> & MoyoungEnum> extends MoyoungSetting<T> {
|
||||
protected final Class<T> clazz;
|
||||
|
||||
public MoyoungSettingEnum(String name, byte cmdQuery, byte cmdSet, Class<T> clazz) {
|
||||
super(name, cmdQuery, cmdSet);
|
||||
this.clazz = clazz;
|
||||
}
|
||||
|
||||
public T findByValue(byte value)
|
||||
{
|
||||
for (T e : clazz.getEnumConstants()) {
|
||||
if (e.value() == value) {
|
||||
return e;
|
||||
}
|
||||
}
|
||||
|
||||
throw new IllegalArgumentException("No enum value for " + value);
|
||||
}
|
||||
|
||||
@Override
|
||||
public byte[] encode(T value) {
|
||||
return new byte[] { value.value() };
|
||||
}
|
||||
|
||||
@Override
|
||||
public T decode(byte[] data) {
|
||||
if (data.length < 1)
|
||||
throw new IllegalArgumentException("Wrong data length, should be at least 1, was " + data.length);
|
||||
|
||||
return findByValue(data[0]);
|
||||
}
|
||||
|
||||
public T[] decodeSupportedValues(byte[] data) {
|
||||
return clazz.getEnumConstants();
|
||||
}
|
||||
}
|
@ -0,0 +1,43 @@
|
||||
/* Copyright (C) 2019 krzys_h
|
||||
|
||||
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 <http://www.gnu.org/licenses/>. */
|
||||
package nodomain.freeyourgadget.gadgetbridge.devices.moyoung.settings;
|
||||
|
||||
import java.nio.ByteBuffer;
|
||||
import java.nio.ByteOrder;
|
||||
|
||||
public class MoyoungSettingInt extends MoyoungSetting<Integer> {
|
||||
public MoyoungSettingInt(String name, byte cmdQuery, byte cmdSet) {
|
||||
super(name, cmdQuery, cmdSet);
|
||||
}
|
||||
|
||||
@Override
|
||||
public byte[] encode(Integer value) {
|
||||
ByteBuffer buffer = ByteBuffer.allocate(4);
|
||||
buffer.order(ByteOrder.BIG_ENDIAN); // <- this is what happens when somebody in China designs a communication protocol
|
||||
buffer.putInt(value);
|
||||
return buffer.array();
|
||||
}
|
||||
|
||||
@Override
|
||||
public Integer decode(byte[] data) {
|
||||
if (data.length != 4)
|
||||
throw new IllegalArgumentException("Wrong data length, should be 4, was " + data.length);
|
||||
ByteBuffer buffer = ByteBuffer.wrap(data);
|
||||
buffer.order(ByteOrder.LITTLE_ENDIAN); // <- yes, it's different here
|
||||
return buffer.getInt();
|
||||
}
|
||||
}
|
@ -0,0 +1,71 @@
|
||||
/* Copyright (C) 2019 krzys_h
|
||||
|
||||
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 <http://www.gnu.org/licenses/>. */
|
||||
package nodomain.freeyourgadget.gadgetbridge.devices.moyoung.settings;
|
||||
|
||||
import android.util.Pair;
|
||||
|
||||
import org.slf4j.Logger;
|
||||
import org.slf4j.LoggerFactory;
|
||||
|
||||
import java.nio.ByteBuffer;
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.moyoung.QuerySettingsOperation;
|
||||
|
||||
public class MoyoungSettingLanguage extends MoyoungSettingEnum<MoyoungEnumLanguage> {
|
||||
private static final Logger LOG = LoggerFactory.getLogger(MoyoungSettingLanguage.class);
|
||||
|
||||
public MoyoungSettingLanguage(String name, byte cmdQuery, byte cmdSet) {
|
||||
super(name, cmdQuery, cmdSet, MoyoungEnumLanguage.class);
|
||||
}
|
||||
|
||||
private Pair<MoyoungEnumLanguage, MoyoungEnumLanguage[]> decodeData(byte[] data) {
|
||||
if (data.length < 5)
|
||||
throw new IllegalArgumentException("Wrong data length, should be at least 5, was " + data.length);
|
||||
|
||||
byte[] current = new byte[] { data[0] };
|
||||
byte[] supported = new byte[] { data[1], data[2], data[3], data[4] };
|
||||
|
||||
ByteBuffer buffer = ByteBuffer.wrap(supported);
|
||||
int supportedNum = buffer.getInt();
|
||||
String supportedStr = new StringBuffer(Integer.toBinaryString(supportedNum)).reverse().toString();
|
||||
|
||||
MoyoungEnumLanguage currentLanguage = super.decode(current);
|
||||
List<MoyoungEnumLanguage> supportedLanguages = new ArrayList<>();
|
||||
for (MoyoungEnumLanguage e : clazz.getEnumConstants()) {
|
||||
if (e.value() >= supportedStr.length())
|
||||
continue;
|
||||
if (Integer.parseInt(supportedStr.substring(e.value(), e.value() + 1)) != 0)
|
||||
supportedLanguages.add(e);
|
||||
}
|
||||
|
||||
MoyoungEnumLanguage[] supportedLanguagesArr = new MoyoungEnumLanguage[supportedLanguages.size()];
|
||||
LOG.debug("Supported languages: {}", supportedLanguages);
|
||||
return Pair.create(currentLanguage, supportedLanguages.toArray(supportedLanguagesArr));
|
||||
}
|
||||
|
||||
@Override
|
||||
public MoyoungEnumLanguage decode(byte[] data) {
|
||||
return decodeData(data).first;
|
||||
}
|
||||
|
||||
@Override
|
||||
public MoyoungEnumLanguage[] decodeSupportedValues(byte[] data) {
|
||||
return decodeData(data).second;
|
||||
}
|
||||
}
|
@ -0,0 +1,74 @@
|
||||
/* Copyright (C) 2019 krzys_h
|
||||
|
||||
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 <http://www.gnu.org/licenses/>. */
|
||||
package nodomain.freeyourgadget.gadgetbridge.devices.moyoung.settings;
|
||||
|
||||
import java.nio.ByteBuffer;
|
||||
|
||||
public class MoyoungSettingRemindersToMove extends MoyoungSetting<MoyoungSettingRemindersToMove.RemindersToMove> {
|
||||
public static class RemindersToMove {
|
||||
public byte period;
|
||||
public byte steps;
|
||||
public byte start_h;
|
||||
public byte end_h;
|
||||
|
||||
public RemindersToMove() {
|
||||
}
|
||||
|
||||
public RemindersToMove(byte period, byte steps, byte start_h, byte end_h) {
|
||||
this.period = period;
|
||||
this.steps = steps;
|
||||
this.start_h = start_h;
|
||||
this.end_h = end_h;
|
||||
}
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
return "RemindersToMove{" +
|
||||
"period=" + period +
|
||||
", steps=" + steps +
|
||||
", start_h=" + start_h +
|
||||
", end_h=" + end_h +
|
||||
'}';
|
||||
}
|
||||
}
|
||||
|
||||
public MoyoungSettingRemindersToMove(String name, byte cmdQuery, byte cmdSet) {
|
||||
super(name, cmdQuery, cmdSet);
|
||||
}
|
||||
|
||||
@Override
|
||||
public byte[] encode(RemindersToMove value) {
|
||||
ByteBuffer buffer = ByteBuffer.allocate(4);
|
||||
buffer.put(value.period);
|
||||
buffer.put(value.steps);
|
||||
buffer.put(value.start_h);
|
||||
buffer.put(value.end_h);
|
||||
return buffer.array();
|
||||
}
|
||||
|
||||
@Override
|
||||
public RemindersToMove decode(byte[] data) {
|
||||
if (data.length != 4)
|
||||
throw new IllegalArgumentException("Wrong data length, should be 4, was " + data.length);
|
||||
ByteBuffer buffer = ByteBuffer.wrap(data);
|
||||
byte period = buffer.get();
|
||||
byte steps = buffer.get();
|
||||
byte start_h = buffer.get();
|
||||
byte end_h = buffer.get();
|
||||
return new RemindersToMove(period, steps, start_h, end_h);
|
||||
}
|
||||
}
|
@ -0,0 +1,76 @@
|
||||
/* Copyright (C) 2019 krzys_h
|
||||
|
||||
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 <http://www.gnu.org/licenses/>. */
|
||||
package nodomain.freeyourgadget.gadgetbridge.devices.moyoung.settings;
|
||||
|
||||
import java.nio.ByteBuffer;
|
||||
import java.nio.ByteOrder;
|
||||
|
||||
public class MoyoungSettingTimeRange extends MoyoungSetting<MoyoungSettingTimeRange.TimeRange> {
|
||||
public static class TimeRange {
|
||||
public byte start_h;
|
||||
public byte start_m;
|
||||
public byte end_h;
|
||||
public byte end_m;
|
||||
|
||||
public TimeRange() {
|
||||
}
|
||||
|
||||
public TimeRange(byte start_h, byte start_m, byte end_h, byte end_m) {
|
||||
this.start_h = start_h;
|
||||
this.start_m = start_m;
|
||||
this.end_h = end_h;
|
||||
this.end_m = end_m;
|
||||
}
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
return "TimeRange{" +
|
||||
"start_h=" + start_h +
|
||||
", start_m=" + start_m +
|
||||
", end_h=" + end_h +
|
||||
", end_m=" + end_m +
|
||||
'}';
|
||||
}
|
||||
}
|
||||
|
||||
public MoyoungSettingTimeRange(String name, byte cmdQuery, byte cmdSet) {
|
||||
super(name, cmdQuery, cmdSet);
|
||||
}
|
||||
|
||||
// Yes, these are different. Was somebody drunk when designing this?
|
||||
|
||||
@Override
|
||||
public byte[] encode(TimeRange value) {
|
||||
ByteBuffer buffer = ByteBuffer.allocate(4);
|
||||
buffer.put(value.start_h);
|
||||
buffer.put(value.start_m);
|
||||
buffer.put(value.end_h);
|
||||
buffer.put(value.end_m);
|
||||
return buffer.array();
|
||||
}
|
||||
|
||||
@Override
|
||||
public TimeRange decode(byte[] data) {
|
||||
if (data.length != 4)
|
||||
throw new IllegalArgumentException("Wrong data length, should be 4, was " + data.length);
|
||||
ByteBuffer buffer = ByteBuffer.wrap(data);
|
||||
buffer.order(ByteOrder.LITTLE_ENDIAN);
|
||||
short start = buffer.getShort();
|
||||
short end = buffer.getShort();
|
||||
return new TimeRange((byte)(start / 60), (byte)(start % 60), (byte)(end / 60), (byte)(start % 60));
|
||||
}
|
||||
}
|
@ -0,0 +1,44 @@
|
||||
/* Copyright (C) 2019 krzys_h
|
||||
|
||||
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 <http://www.gnu.org/licenses/>. */
|
||||
package nodomain.freeyourgadget.gadgetbridge.devices.moyoung.settings;
|
||||
|
||||
import org.apache.commons.lang3.NotImplementedException;
|
||||
|
||||
import java.nio.ByteBuffer;
|
||||
|
||||
import nodomain.freeyourgadget.gadgetbridge.model.ActivityUser;
|
||||
|
||||
public class MoyoungSettingUserInfo extends MoyoungSetting<ActivityUser> {
|
||||
public MoyoungSettingUserInfo(String name, byte cmdSet) {
|
||||
super(name, (byte)-1, cmdSet);
|
||||
}
|
||||
|
||||
@Override
|
||||
public byte[] encode(ActivityUser value) {
|
||||
ByteBuffer buffer = ByteBuffer.allocate(4);
|
||||
buffer.put((byte)value.getHeightCm());
|
||||
buffer.put((byte)value.getWeightKg());
|
||||
buffer.put((byte)value.getAge());
|
||||
buffer.put((byte)value.getGender());
|
||||
return buffer.array();
|
||||
}
|
||||
|
||||
@Override
|
||||
public ActivityUser decode(byte[] data) {
|
||||
throw new NotImplementedException("decode");
|
||||
}
|
||||
}
|
@ -0,0 +1,36 @@
|
||||
/* Copyright (C) 2024 Arjan Schrijver
|
||||
|
||||
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.entities;
|
||||
|
||||
import androidx.annotation.NonNull;
|
||||
|
||||
import nodomain.freeyourgadget.gadgetbridge.model.BloodPressureSample;
|
||||
import nodomain.freeyourgadget.gadgetbridge.util.DateTimeUtils;
|
||||
|
||||
public abstract class AbstractBloodPressureSample extends AbstractTimeSample implements BloodPressureSample {
|
||||
@NonNull
|
||||
@Override
|
||||
public String toString() {
|
||||
return getClass().getSimpleName() + "{" +
|
||||
"timestamp=" + DateTimeUtils.formatDateTime(DateTimeUtils.parseTimestampMillis(getTimestamp())) +
|
||||
", bpSystolic=" + getBpSystolic() +
|
||||
", bpDiastolic=" + getBpDiastolic() +
|
||||
", userId=" + getUserId() +
|
||||
", deviceId=" + getDeviceId() +
|
||||
"}";
|
||||
}
|
||||
}
|
@ -0,0 +1,22 @@
|
||||
/* Copyright (C) 2024 Arjan Schrijver
|
||||
|
||||
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.model;
|
||||
|
||||
public interface BloodPressureSample extends TimeSample {
|
||||
int getBpSystolic();
|
||||
int getBpDiastolic();
|
||||
}
|
@ -228,6 +228,8 @@ import nodomain.freeyourgadget.gadgetbridge.devices.mijia_lywsd.MijiaXmwsdj04Coo
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.miscale.MiCompositionScaleCoordinator;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.miscale.MiSmartScaleCoordinator;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.moondrop.MoondropSpaceTravelCoordinator;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.moyoung.ColmiI28UltraCoordinator;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.moyoung.MisirunC17Coordinator;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.no1f1.No1F1Coordinator;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.nothing.CmfBudsPro2Coordinator;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.nothing.Ear1Coordinator;
|
||||
@ -586,6 +588,8 @@ public enum DeviceType {
|
||||
COLMI_R06(ColmiR06Coordinator.class),
|
||||
COLMI_R09(ColmiR09Coordinator.class),
|
||||
COLMI_R10(ColmiR10Coordinator.class),
|
||||
COLMI_I28_ULTRA(ColmiI28UltraCoordinator.class),
|
||||
MISIRUN_C17(MisirunC17Coordinator.class),
|
||||
B_AND_W_P_SERIES(BandWPSeriesDeviceCoordinator.class),
|
||||
SCANNABLE(ScannableDeviceCoordinator.class),
|
||||
CYCLING_SENSOR(CyclingSensorCoordinator.class),
|
||||
|
@ -212,6 +212,14 @@ public class BLETypeConversions {
|
||||
return (short) (bytes[0] & 0xff | ((bytes[1] & 0xff) << 8));
|
||||
}
|
||||
|
||||
public static int toUint24(byte... bytes) {
|
||||
return (bytes[0] & 0xff) | ((bytes[1] & 0xff) << 8) | ((bytes[2] & 0xff) << 16);
|
||||
}
|
||||
|
||||
public static int toUint24(byte[] bytes, int offset) {
|
||||
return (bytes[offset + 0] & 0xff) | ((bytes[offset + 1] & 0xff) << 8) | ((bytes[offset + 2] & 0xff) << 16);
|
||||
}
|
||||
|
||||
public static int toUint32(byte... bytes) {
|
||||
return (bytes[0] & 0xff) | ((bytes[1] & 0xff) << 8) | ((bytes[2] & 0xff) << 16) | ((bytes[3] & 0xff) << 24);
|
||||
}
|
||||
|
@ -0,0 +1,212 @@
|
||||
/* Copyright (C) 2019 krzys_h
|
||||
|
||||
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 <http://www.gnu.org/licenses/>. */
|
||||
package nodomain.freeyourgadget.gadgetbridge.service.devices.moyoung;
|
||||
|
||||
import android.bluetooth.BluetoothGatt;
|
||||
import android.bluetooth.BluetoothGattCharacteristic;
|
||||
import android.util.Pair;
|
||||
|
||||
import org.slf4j.Logger;
|
||||
import org.slf4j.LoggerFactory;
|
||||
|
||||
import java.io.IOException;
|
||||
import java.util.UUID;
|
||||
|
||||
import nodomain.freeyourgadget.gadgetbridge.Logging;
|
||||
import nodomain.freeyourgadget.gadgetbridge.R;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.moyoung.MoyoungConstants;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.btle.AbstractBTLEOperation;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.btle.TransactionBuilder;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.miband.operations.OperationStatus;
|
||||
import nodomain.freeyourgadget.gadgetbridge.util.GB;
|
||||
|
||||
public class FetchDataOperation extends AbstractBTLEOperation<MoyoungDeviceSupport> {
|
||||
|
||||
private static final Logger LOG = LoggerFactory.getLogger(FetchDataOperation.class);
|
||||
|
||||
private boolean[] receivedSteps = new boolean[3];
|
||||
private boolean[] receivedSleep = new boolean[3];
|
||||
private boolean receivedTrainingData = false;
|
||||
|
||||
private MoyoungPacketIn packetIn = new MoyoungPacketIn();
|
||||
|
||||
public FetchDataOperation(MoyoungDeviceSupport support) {
|
||||
super(support);
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void prePerform() {
|
||||
getDevice().setBusyTask(getContext().getString(R.string.busy_task_fetch_activity_data));
|
||||
getDevice().sendDeviceUpdateIntent(getContext());
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void doPerform() throws IOException {
|
||||
TransactionBuilder builder = performInitialized("FetchDataOperation");
|
||||
getSupport().sendPacket(builder, MoyoungPacketOut.buildPacket(getSupport().getMtu(), MoyoungConstants.CMD_SYNC_PAST_SLEEP_AND_STEP, new byte[] { MoyoungConstants.ARG_SYNC_YESTERDAY_SLEEP }));
|
||||
getSupport().sendPacket(builder, MoyoungPacketOut.buildPacket(getSupport().getMtu(), MoyoungConstants.CMD_SYNC_PAST_SLEEP_AND_STEP, new byte[] { MoyoungConstants.ARG_SYNC_DAY_BEFORE_YESTERDAY_SLEEP }));
|
||||
getSupport().sendPacket(builder, MoyoungPacketOut.buildPacket(getSupport().getMtu(), MoyoungConstants.CMD_SYNC_SLEEP, new byte[0]));
|
||||
getSupport().sendPacket(builder, MoyoungPacketOut.buildPacket(getSupport().getMtu(), MoyoungConstants.CMD_SYNC_PAST_SLEEP_AND_STEP, new byte[] { MoyoungConstants.ARG_SYNC_YESTERDAY_STEPS }));
|
||||
getSupport().sendPacket(builder, MoyoungPacketOut.buildPacket(getSupport().getMtu(), MoyoungConstants.CMD_SYNC_PAST_SLEEP_AND_STEP, new byte[] { MoyoungConstants.ARG_SYNC_DAY_BEFORE_YESTERDAY_STEPS }));
|
||||
builder.read(getCharacteristic(MoyoungConstants.UUID_CHARACTERISTIC_STEPS));
|
||||
getSupport().sendPacket(builder, MoyoungPacketOut.buildPacket(getSupport().getMtu(), MoyoungConstants.CMD_QUERY_MOVEMENT_HEART_RATE, new byte[] { }));
|
||||
getSupport().sendPacket(builder, MoyoungPacketOut.buildPacket(getSupport().getMtu(), MoyoungConstants.CMD_QUERY_PAST_HEART_RATE_1, new byte[] { 0x00 }));
|
||||
builder.queue(getQueue());
|
||||
|
||||
updateProgressAndCheckFinish();
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean onCharacteristicRead(BluetoothGatt gatt, BluetoothGattCharacteristic characteristic, int status) {
|
||||
if (!isOperationRunning())
|
||||
{
|
||||
LOG.error("onCharacteristicRead but operation is not running!");
|
||||
}
|
||||
else
|
||||
{
|
||||
UUID charUuid = characteristic.getUuid();
|
||||
if (charUuid.equals(MoyoungConstants.UUID_CHARACTERISTIC_STEPS)) {
|
||||
byte[] data = characteristic.getValue();
|
||||
LOG.info("TODAY STEPS data: " + Logging.formatBytes(data));
|
||||
decodeSteps(0, data);
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
return super.onCharacteristicRead(gatt, characteristic, status);
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean onCharacteristicChanged(BluetoothGatt gatt, BluetoothGattCharacteristic characteristic) {
|
||||
if (!isOperationRunning())
|
||||
{
|
||||
LOG.error("onCharacteristicChanged but operation is not running!");
|
||||
}
|
||||
else
|
||||
{
|
||||
UUID charUuid = characteristic.getUuid();
|
||||
if (charUuid.equals(MoyoungConstants.UUID_CHARACTERISTIC_DATA_IN))
|
||||
{
|
||||
if (packetIn.putFragment(characteristic.getValue())) {
|
||||
Pair<Byte, byte[]> packet = MoyoungPacketIn.parsePacket(packetIn.getPacket());
|
||||
packetIn = new MoyoungPacketIn();
|
||||
if (packet != null) {
|
||||
byte packetType = packet.first;
|
||||
byte[] payload = packet.second;
|
||||
|
||||
if (handlePacket(packetType, payload))
|
||||
return true;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return super.onCharacteristicChanged(gatt, characteristic);
|
||||
}
|
||||
|
||||
private boolean handlePacket(byte packetType, byte[] payload) {
|
||||
if (packetType == MoyoungConstants.CMD_SYNC_SLEEP) {
|
||||
LOG.info("TODAY SLEEP data: " + Logging.formatBytes(payload));
|
||||
decodeSleep(0, payload);
|
||||
return true;
|
||||
}
|
||||
if (packetType == MoyoungConstants.CMD_SYNC_PAST_SLEEP_AND_STEP) {
|
||||
byte dataType = payload[0];
|
||||
byte[] data = new byte[payload.length - 1];
|
||||
System.arraycopy(payload, 1, data, 0, data.length);
|
||||
|
||||
// NOTE: Does this seem swapped to you? That's because IT IS! I took the constant names
|
||||
// from the official app, but as it turns out, the official app has a bug.
|
||||
// (and yes, you can see that data from yesterday appears as two days ago
|
||||
// in the app itself and all past data is getting messed up because of it)
|
||||
|
||||
if (dataType == MoyoungConstants.ARG_SYNC_YESTERDAY_STEPS) {
|
||||
LOG.info("2 DAYS AGO STEPS data: " + Logging.formatBytes(data));
|
||||
decodeSteps(2, data);
|
||||
return true;
|
||||
}
|
||||
else if (dataType == MoyoungConstants.ARG_SYNC_DAY_BEFORE_YESTERDAY_STEPS) {
|
||||
LOG.info("YESTERDAY STEPS data: " + Logging.formatBytes(data));
|
||||
decodeSteps(1, data);
|
||||
return true;
|
||||
}
|
||||
else if (dataType == MoyoungConstants.ARG_SYNC_YESTERDAY_SLEEP) {
|
||||
LOG.info("2 DAYS AGO SLEEP data: " + Logging.formatBytes(data));
|
||||
decodeSleep(2, data);
|
||||
return true;
|
||||
}
|
||||
else if (dataType == MoyoungConstants.ARG_SYNC_DAY_BEFORE_YESTERDAY_SLEEP) {
|
||||
LOG.info("YESTERDAY SLEEP data: " + Logging.formatBytes(data));
|
||||
decodeSleep(1, data);
|
||||
return true;
|
||||
}
|
||||
}
|
||||
if (packetType == MoyoungConstants.CMD_QUERY_MOVEMENT_HEART_RATE) {
|
||||
decodeTrainingData(payload);
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
private void decodeSteps(int daysAgo, byte[] data)
|
||||
{
|
||||
getSupport().handleStepsHistory(daysAgo, data, false);
|
||||
receivedSteps[daysAgo] = true;
|
||||
updateProgressAndCheckFinish();
|
||||
}
|
||||
|
||||
private void decodeSleep(int daysAgo, byte[] data)
|
||||
{
|
||||
getSupport().handleSleepHistory(daysAgo, data);
|
||||
receivedSleep[daysAgo] = true;
|
||||
updateProgressAndCheckFinish();
|
||||
}
|
||||
|
||||
private void decodeTrainingData(byte[] data)
|
||||
{
|
||||
getSupport().handleTrainingData(data);
|
||||
receivedTrainingData = true;
|
||||
updateProgressAndCheckFinish();
|
||||
}
|
||||
|
||||
private void updateProgressAndCheckFinish()
|
||||
{
|
||||
int count = 0;
|
||||
int total = receivedSteps.length + receivedSleep.length;
|
||||
for(int i = 0; i < receivedSteps.length; i++)
|
||||
if (receivedSteps[i])
|
||||
++count;
|
||||
for(int i = 0; i < receivedSleep.length; i++)
|
||||
if (receivedSleep[i])
|
||||
++count;
|
||||
if (receivedTrainingData)
|
||||
++count;
|
||||
GB.updateTransferNotification(null, getContext().getString(R.string.busy_task_fetch_activity_data), true, 100 * count / total, getContext());
|
||||
LOG.debug("Fetching activity data status: {} out of {}", count, total);
|
||||
if (count == total)
|
||||
operationFinished();
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void operationFinished() {
|
||||
operationStatus = OperationStatus.FINISHED;
|
||||
if (getDevice() != null && getDevice().isConnected()) {
|
||||
unsetBusy();
|
||||
GB.signalActivityDataFinish(getDevice());
|
||||
}
|
||||
}
|
||||
}
|
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,22 @@
|
||||
/* Copyright (C) 2019 krzys_h
|
||||
|
||||
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 <http://www.gnu.org/licenses/>. */
|
||||
package nodomain.freeyourgadget.gadgetbridge.service.devices.moyoung;
|
||||
|
||||
public class MoyoungPacket {
|
||||
protected byte[] packet;
|
||||
protected int position = 0;
|
||||
}
|
@ -0,0 +1,135 @@
|
||||
/* Copyright (C) 2019 krzys_h
|
||||
|
||||
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 <http://www.gnu.org/licenses/>. */
|
||||
package nodomain.freeyourgadget.gadgetbridge.service.devices.moyoung;
|
||||
|
||||
import android.util.Pair;
|
||||
|
||||
import androidx.annotation.NonNull;
|
||||
|
||||
import org.slf4j.Logger;
|
||||
import org.slf4j.LoggerFactory;
|
||||
|
||||
import nodomain.freeyourgadget.gadgetbridge.Logging;
|
||||
|
||||
/**
|
||||
* A class for handling fragmentation of incoming packets<br>
|
||||
* <br>
|
||||
* Usage:
|
||||
* <pre>
|
||||
* {@code
|
||||
* if(packetIn.putFragment(fragment)) {
|
||||
* Pair<Byte, byte[]> packet = MoyoungPacketIn.parsePacket(packetIn.getPacket());
|
||||
* packetIn = new MoyoungPacketIn();
|
||||
* if (packet != null) {
|
||||
* byte packetType = packet.first;
|
||||
* byte[] payload = packet.second;
|
||||
* // ...
|
||||
* }
|
||||
* }
|
||||
* </pre>
|
||||
*/
|
||||
public class MoyoungPacketIn extends MoyoungPacket {
|
||||
private static final Logger LOG = LoggerFactory.getLogger(MoyoungPacketIn.class);
|
||||
|
||||
public MoyoungPacketIn()
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* Store the incoming fragment and try to reconstruct packet
|
||||
*
|
||||
* @param fragment The incoming fragment
|
||||
* @return true if the packet is complete
|
||||
*/
|
||||
public boolean putFragment(byte[] fragment)
|
||||
{
|
||||
if (packet == null)
|
||||
{
|
||||
int len = parsePacketLength(fragment);
|
||||
if (len < 0)
|
||||
return false; // corrupted packet
|
||||
packet = new byte[len];
|
||||
}
|
||||
|
||||
int toCopy = Math.min(fragment.length, packet.length - position);
|
||||
if (fragment.length > toCopy)
|
||||
{
|
||||
LOG.warn("Got fragment with more data than expected!");
|
||||
}
|
||||
|
||||
System.arraycopy(fragment, 0, packet, position, toCopy);
|
||||
position += fragment.length;
|
||||
return position >= packet.length;
|
||||
}
|
||||
|
||||
public byte[] getPacket()
|
||||
{
|
||||
if (packet == null || position < packet.length)
|
||||
throw new IllegalStateException("Packet is not complete yet");
|
||||
return packet;
|
||||
}
|
||||
|
||||
/**
|
||||
* Parse the packet header and return the length
|
||||
* @param packetOrFragment The entire packet or it's first fragment
|
||||
* @return The packet length, or -1 if packet is corrupted
|
||||
*/
|
||||
private static int parsePacketLength(@NonNull byte[] packetOrFragment)
|
||||
{
|
||||
if (packetOrFragment[0] != (byte)0xFE || packetOrFragment[1] != (byte)0xEA)
|
||||
{
|
||||
LOG.warn("Invalid packet header, ignoring! Fragment: " + Logging.formatBytes(packetOrFragment));
|
||||
return -1;
|
||||
}
|
||||
|
||||
int len_h = 0;
|
||||
if (packetOrFragment[2] != 16)
|
||||
{
|
||||
if ((packetOrFragment[2] & 0xFF) < 32)
|
||||
{
|
||||
LOG.warn("Corrupted packet, unable to parse length");
|
||||
return -1;
|
||||
}
|
||||
len_h = (packetOrFragment[2] & 0xFF) - 32;
|
||||
}
|
||||
int len_l = (packetOrFragment[3] & 0xFF);
|
||||
|
||||
return (len_h << 8) | len_l;
|
||||
}
|
||||
|
||||
/**
|
||||
* Parse the packet
|
||||
* @param packet The complete packet
|
||||
* @return A pair containing the packet type and payload
|
||||
*/
|
||||
public static Pair<Byte, byte[]> parsePacket(@NonNull byte[] packet)
|
||||
{
|
||||
int len = parsePacketLength(packet);
|
||||
if (len < 0)
|
||||
return null;
|
||||
if (len != packet.length)
|
||||
{
|
||||
LOG.warn("Invalid packet length!");
|
||||
return null;
|
||||
}
|
||||
byte packetType = packet[4];
|
||||
byte[] payload = new byte[packet.length - 5];
|
||||
System.arraycopy(packet, 5, payload, 0, payload.length);
|
||||
return Pair.create(packetType, payload);
|
||||
}
|
||||
}
|
@ -0,0 +1,81 @@
|
||||
/* Copyright (C) 2019 krzys_h
|
||||
|
||||
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 <http://www.gnu.org/licenses/>. */
|
||||
package nodomain.freeyourgadget.gadgetbridge.service.devices.moyoung;
|
||||
|
||||
import androidx.annotation.NonNull;
|
||||
|
||||
/**
|
||||
* A class for handling fragmentation of outgoing packets<br>
|
||||
* <br>
|
||||
* Usage:
|
||||
* <pre>
|
||||
* {@code
|
||||
* MoyoungPacketOut packetOut = new MoyoungPacketOut(MoyoungPacketOut.buildPacket(type, payload));
|
||||
* byte[] fragment = new byte[MTU];
|
||||
* while(packetOut.getFragment(fragment))
|
||||
* send(fragment);
|
||||
* }
|
||||
* </pre>
|
||||
*/
|
||||
public class MoyoungPacketOut extends MoyoungPacket {
|
||||
public MoyoungPacketOut(byte[] packet)
|
||||
{
|
||||
this.packet = packet;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the next fragment of this packet to be sent
|
||||
*
|
||||
* @param fragmentBuffer The buffer to store the output in, of desired size (i.e. == MTU)
|
||||
* @return true if there is more data to be sent, false otherwise
|
||||
*/
|
||||
public boolean getFragment(byte[] fragmentBuffer)
|
||||
{
|
||||
if (position >= packet.length)
|
||||
return false;
|
||||
int remainingToTransfer = Math.min(fragmentBuffer.length, packet.length - position);
|
||||
System.arraycopy(packet, position, fragmentBuffer, 0, remainingToTransfer);
|
||||
position += remainingToTransfer;
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Encode the packet
|
||||
* @param packetType The packet type
|
||||
* @param payload The packet payload
|
||||
* @return The encoded packet
|
||||
*/
|
||||
public static byte[] buildPacket(int mtu, byte packetType, @NonNull byte[] payload)
|
||||
{
|
||||
byte[] packet = new byte[payload.length + 5];
|
||||
packet[0] = (byte)0xFE;
|
||||
packet[1] = (byte)0xEA;
|
||||
if (mtu == 20)
|
||||
{
|
||||
packet[2] = 16;
|
||||
packet[3] = (byte)(packet.length & 0xFF);
|
||||
}
|
||||
else
|
||||
{
|
||||
packet[2] = (byte)(32 + (packet.length >> 8) & 0xFF);
|
||||
packet[3] = (byte)(packet.length & 0xFF);
|
||||
}
|
||||
packet[4] = packetType;
|
||||
System.arraycopy(payload, 0, packet, 5, payload.length);
|
||||
return packet;
|
||||
}
|
||||
}
|
@ -0,0 +1,139 @@
|
||||
/* Copyright (C) 2019 krzys_h
|
||||
|
||||
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 <http://www.gnu.org/licenses/>. */
|
||||
package nodomain.freeyourgadget.gadgetbridge.service.devices.moyoung;
|
||||
|
||||
import android.bluetooth.BluetoothGatt;
|
||||
import android.bluetooth.BluetoothGattCharacteristic;
|
||||
import android.util.Pair;
|
||||
|
||||
import org.slf4j.Logger;
|
||||
import org.slf4j.LoggerFactory;
|
||||
|
||||
import java.io.IOException;
|
||||
import java.util.UUID;
|
||||
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.moyoung.AbstractMoyoungDeviceCoordinator;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.moyoung.MoyoungConstants;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.moyoung.settings.MoyoungSetting;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.btle.AbstractBTLEOperation;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.btle.TransactionBuilder;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.miband.operations.OperationStatus;
|
||||
|
||||
public class QuerySettingsOperation extends AbstractBTLEOperation<MoyoungDeviceSupport> {
|
||||
|
||||
private static final Logger LOG = LoggerFactory.getLogger(QuerySettingsOperation.class);
|
||||
|
||||
private final MoyoungSetting[] settingsToQuery;
|
||||
private boolean[] received;
|
||||
|
||||
private MoyoungPacketIn packetIn = new MoyoungPacketIn();
|
||||
|
||||
public QuerySettingsOperation(MoyoungDeviceSupport support, MoyoungSetting[] settingsToQuery) {
|
||||
super(support);
|
||||
this.settingsToQuery = settingsToQuery;
|
||||
}
|
||||
|
||||
public QuerySettingsOperation(MoyoungDeviceSupport support) {
|
||||
super(support);
|
||||
AbstractMoyoungDeviceCoordinator coordinator = (AbstractMoyoungDeviceCoordinator) getDevice().getDeviceCoordinator();
|
||||
this.settingsToQuery = coordinator.getSupportedSettings();
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void prePerform() {
|
||||
getDevice().setBusyTask("Querying settings"); // mark as busy quickly to avoid interruptions from the outside
|
||||
getDevice().sendDeviceUpdateIntent(getContext());
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void doPerform() throws IOException {
|
||||
received = new boolean[settingsToQuery.length];
|
||||
TransactionBuilder builder = performInitialized("QuerySettingsOperation");
|
||||
for (MoyoungSetting setting : settingsToQuery)
|
||||
{
|
||||
if (setting.cmdQuery == -1)
|
||||
continue;
|
||||
|
||||
getSupport().sendPacket(builder, MoyoungPacketOut.buildPacket(getSupport().getMtu(), setting.cmdQuery, new byte[0]));
|
||||
}
|
||||
builder.queue(getQueue());
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean onCharacteristicChanged(BluetoothGatt gatt, BluetoothGattCharacteristic characteristic) {
|
||||
if (!isOperationRunning())
|
||||
{
|
||||
LOG.error("onCharacteristicChanged but operation is not running!");
|
||||
}
|
||||
else
|
||||
{
|
||||
UUID charUuid = characteristic.getUuid();
|
||||
if (charUuid.equals(MoyoungConstants.UUID_CHARACTERISTIC_DATA_IN))
|
||||
{
|
||||
if (packetIn.putFragment(characteristic.getValue())) {
|
||||
Pair<Byte, byte[]> packet = MoyoungPacketIn.parsePacket(packetIn.getPacket());
|
||||
packetIn = new MoyoungPacketIn();
|
||||
if (packet != null) {
|
||||
byte packetType = packet.first;
|
||||
byte[] payload = packet.second;
|
||||
|
||||
if (handlePacket(packetType, payload))
|
||||
return true;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return super.onCharacteristicChanged(gatt, characteristic);
|
||||
}
|
||||
|
||||
private boolean handlePacket(byte packetType, byte[] payload) {
|
||||
boolean handled = false;
|
||||
boolean receivedEverything = true;
|
||||
for(int i = 0; i < settingsToQuery.length; i++)
|
||||
{
|
||||
MoyoungSetting setting = settingsToQuery[i];
|
||||
if (setting.cmdQuery == -1)
|
||||
continue;
|
||||
if (setting.cmdQuery == packetType)
|
||||
{
|
||||
try {
|
||||
Object value = setting.decode(payload);
|
||||
LOG.info("SETTING QUERY " + setting.name + " = " + value.toString());
|
||||
} catch (Exception e) {
|
||||
LOG.error("Parse error in packet for setting " + setting.name + ": ", e);
|
||||
}
|
||||
received[i] = true;
|
||||
handled = true;
|
||||
}
|
||||
else if (!received[i])
|
||||
receivedEverything = false;
|
||||
}
|
||||
if (receivedEverything)
|
||||
operationFinished();
|
||||
|
||||
return handled;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void operationFinished() {
|
||||
operationStatus = OperationStatus.FINISHED;
|
||||
if (getDevice() != null && getDevice().isConnected()) {
|
||||
unsetBusy();
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,221 @@
|
||||
/* Copyright (C) 2019 krzys_h
|
||||
|
||||
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 <http://www.gnu.org/licenses/>. */
|
||||
package nodomain.freeyourgadget.gadgetbridge.service.devices.moyoung;
|
||||
|
||||
import android.bluetooth.BluetoothGatt;
|
||||
import android.bluetooth.BluetoothGattCharacteristic;
|
||||
import android.util.Pair;
|
||||
import android.widget.Toast;
|
||||
|
||||
import org.slf4j.Logger;
|
||||
import org.slf4j.LoggerFactory;
|
||||
|
||||
import java.io.ByteArrayOutputStream;
|
||||
import java.io.IOException;
|
||||
import java.nio.ByteBuffer;
|
||||
import java.nio.ByteOrder;
|
||||
import java.util.Calendar;
|
||||
import java.util.UUID;
|
||||
|
||||
import nodomain.freeyourgadget.gadgetbridge.GBApplication;
|
||||
import nodomain.freeyourgadget.gadgetbridge.Logging;
|
||||
import nodomain.freeyourgadget.gadgetbridge.R;
|
||||
import nodomain.freeyourgadget.gadgetbridge.database.DBHandler;
|
||||
import nodomain.freeyourgadget.gadgetbridge.database.DBHelper;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.moyoung.MoyoungConstants;
|
||||
import nodomain.freeyourgadget.gadgetbridge.devices.moyoung.samples.MoyoungHeartRateSampleProvider;
|
||||
import nodomain.freeyourgadget.gadgetbridge.entities.Device;
|
||||
import nodomain.freeyourgadget.gadgetbridge.entities.MoyoungHeartRateSample;
|
||||
import nodomain.freeyourgadget.gadgetbridge.entities.User;
|
||||
import nodomain.freeyourgadget.gadgetbridge.model.ActivitySample;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.btle.AbstractBTLEOperation;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.btle.TransactionBuilder;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.miband.operations.OperationStatus;
|
||||
import nodomain.freeyourgadget.gadgetbridge.util.GB;
|
||||
|
||||
public class TrainingFinishedDataOperation extends AbstractBTLEOperation<MoyoungDeviceSupport> {
|
||||
|
||||
private static final Logger LOG = LoggerFactory.getLogger(TrainingFinishedDataOperation.class);
|
||||
|
||||
private final byte[] firstPacketData;
|
||||
private final long firstPacketTimeInMillis;
|
||||
ByteArrayOutputStream data = new ByteArrayOutputStream();
|
||||
|
||||
private MoyoungPacketIn packetIn = new MoyoungPacketIn();
|
||||
|
||||
public TrainingFinishedDataOperation(MoyoungDeviceSupport support, byte[] firstPacketData) {
|
||||
super(support);
|
||||
this.firstPacketData = firstPacketData;
|
||||
this.firstPacketTimeInMillis = Calendar.getInstance().getTimeInMillis();
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void prePerform() {
|
||||
getDevice().setBusyTask(getContext().getString(R.string.busy_task_fetch_training_data));
|
||||
getDevice().sendDeviceUpdateIntent(getContext());
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void doPerform() {
|
||||
GB.updateTransferNotification(null, getContext().getString(R.string.busy_task_fetch_training_data), true, 0, getContext());
|
||||
handleTrainingHealthRatePacket(firstPacketData, true);
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean onCharacteristicChanged(BluetoothGatt gatt, BluetoothGattCharacteristic characteristic) {
|
||||
if (!isOperationRunning())
|
||||
{
|
||||
LOG.error("onCharacteristicChanged but operation is not running!");
|
||||
}
|
||||
else
|
||||
{
|
||||
UUID charUuid = characteristic.getUuid();
|
||||
if (charUuid.equals(MoyoungConstants.UUID_CHARACTERISTIC_DATA_IN))
|
||||
{
|
||||
if (packetIn.putFragment(characteristic.getValue())) {
|
||||
Pair<Byte, byte[]> packet = MoyoungPacketIn.parsePacket(packetIn.getPacket());
|
||||
packetIn = new MoyoungPacketIn();
|
||||
if (packet != null) {
|
||||
byte packetType = packet.first;
|
||||
byte[] payload = packet.second;
|
||||
|
||||
if (handlePacket(packetType, payload))
|
||||
return true;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
return super.onCharacteristicChanged(gatt, characteristic);
|
||||
}
|
||||
|
||||
private boolean handlePacket(byte packetType, byte[] payload) {
|
||||
if (packetType == MoyoungConstants.CMD_QUERY_LAST_DYNAMIC_RATE) {
|
||||
handleTrainingHealthRatePacket(payload, false);
|
||||
return true;
|
||||
}
|
||||
if (packetType == MoyoungConstants.CMD_QUERY_MOVEMENT_HEART_RATE) {
|
||||
handleTrainingSummaryDataPacket(payload);
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
|
||||
private void handleTrainingHealthRatePacket(byte[] payload, boolean isFirst) {
|
||||
LOG.info("TRAINING DATA: " + Logging.formatBytes(payload));
|
||||
byte sequenceType = payload[0];
|
||||
if (isFirst != (sequenceType == MoyoungConstants.ARG_TRANSMISSION_FIRST))
|
||||
throw new IllegalArgumentException("Expected packet to be " + (isFirst ? "first" : "continued") + " but got packet of type " + sequenceType);
|
||||
if (sequenceType == MoyoungConstants.ARG_TRANSMISSION_LAST && payload.length > 1)
|
||||
throw new IllegalArgumentException("Last packet shouldn't have any data");
|
||||
|
||||
data.write(payload, 1, payload.length - 1);
|
||||
|
||||
if (sequenceType != MoyoungConstants.ARG_TRANSMISSION_LAST)
|
||||
queryMoreData();
|
||||
else
|
||||
processAllData();
|
||||
}
|
||||
|
||||
private void queryMoreData() {
|
||||
try {
|
||||
TransactionBuilder builder = performInitialized("TrainingFinishedDataOperation");
|
||||
getSupport().sendPacket(builder, MoyoungPacketOut.buildPacket(getSupport().getMtu(), MoyoungConstants.CMD_QUERY_LAST_DYNAMIC_RATE, new byte[0]));
|
||||
builder.queue(getQueue());
|
||||
} catch (IOException e) {
|
||||
LOG.error("Error fetching training data: ", e);
|
||||
GB.toast(getContext(), "Error fetching training data: " + e.getLocalizedMessage(), Toast.LENGTH_LONG, GB.ERROR);
|
||||
GB.updateTransferNotification(null, "Data transfer failed", false, 0, getContext());
|
||||
operationFinished();
|
||||
}
|
||||
}
|
||||
|
||||
private void processAllData() {
|
||||
byte[] completeData = data.toByteArray();
|
||||
LOG.info("HAVE COMPLETE DATA: " + Logging.formatBytes(completeData));
|
||||
ByteBuffer dataBuffer = ByteBuffer.wrap(completeData);
|
||||
dataBuffer.order(ByteOrder.LITTLE_ENDIAN);
|
||||
|
||||
Calendar dateRecorded = Calendar.getInstance();
|
||||
dateRecorded.setTime(MoyoungConstants.WatchTimeToLocalTime(dataBuffer.getInt()));
|
||||
|
||||
// NOTE: The first sample always matches dateRecorded (which is aligned to the minute)
|
||||
// The last sample is saved at the moment the recording is stopped (and this code starts executing)
|
||||
|
||||
try (DBHandler dbHandler = GBApplication.acquireDB()) {
|
||||
User user = DBHelper.getUser(dbHandler.getDaoSession());
|
||||
Device device = DBHelper.getDevice(getDevice(), dbHandler.getDaoSession());
|
||||
|
||||
MoyoungHeartRateSampleProvider provider = new MoyoungHeartRateSampleProvider(getDevice(), dbHandler.getDaoSession());
|
||||
|
||||
LOG.info("START DATE: " + dateRecorded.getTime().toString());
|
||||
while (dataBuffer.hasRemaining())
|
||||
{
|
||||
int measurement = dataBuffer.get() & 0xFF;
|
||||
if (!dataBuffer.hasRemaining())
|
||||
dateRecorded.setTimeInMillis(firstPacketTimeInMillis); // the last sample is captured exactly at the end of measurement
|
||||
|
||||
LOG.info("MEASUREMENT: at " + dateRecorded.getTime().toString() + " was " + measurement);
|
||||
|
||||
MoyoungHeartRateSample sample = new MoyoungHeartRateSample();
|
||||
sample.setDevice(device);
|
||||
sample.setUser(user);
|
||||
sample.setTimestamp((int)(dateRecorded.getTimeInMillis() / 1000));
|
||||
sample.setHeartRate(measurement != 0 ? measurement : ActivitySample.NOT_MEASURED);
|
||||
|
||||
provider.addSample(sample);
|
||||
LOG.info("Adding a training sample: " + sample.toString());
|
||||
|
||||
dateRecorded.add(Calendar.MINUTE, 1);
|
||||
}
|
||||
} catch (Exception ex) {
|
||||
LOG.error("Error saving samples: ", ex);
|
||||
GB.toast(getContext(), "Error saving samples: " + ex.getLocalizedMessage(), Toast.LENGTH_LONG, GB.ERROR);
|
||||
GB.updateTransferNotification(null, "Data transfer failed", false, 0, getContext());
|
||||
}
|
||||
|
||||
try {
|
||||
TransactionBuilder builder = performInitialized("TrainingFinishedDataOperation fetch training type");
|
||||
getSupport().sendPacket(builder, MoyoungPacketOut.buildPacket(getSupport().getMtu(), MoyoungConstants.CMD_QUERY_MOVEMENT_HEART_RATE, new byte[] { }));
|
||||
builder.queue(getQueue());
|
||||
} catch (IOException e) {
|
||||
LOG.error("Error fetching training data: ", e);
|
||||
GB.toast(getContext(), "Error fetching training data: " + e.getLocalizedMessage(), Toast.LENGTH_LONG, GB.ERROR);
|
||||
GB.updateTransferNotification(null, "Data transfer failed", false, 0, getContext());
|
||||
operationFinished();
|
||||
}
|
||||
}
|
||||
|
||||
private void handleTrainingSummaryDataPacket(byte[] payload)
|
||||
{
|
||||
getSupport().handleTrainingData(payload);
|
||||
|
||||
GB.updateTransferNotification(null, getContext().getString(R.string.busy_task_fetch_training_data_finished), false, 0, getContext());
|
||||
operationFinished();
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void operationFinished() {
|
||||
operationStatus = OperationStatus.FINISHED;
|
||||
if (getDevice() != null && getDevice().isConnected()) {
|
||||
unsetBusy();
|
||||
GB.signalActivityDataFinish(getDevice());
|
||||
}
|
||||
}
|
||||
}
|
@ -120,4 +120,31 @@ public class ArrayUtils {
|
||||
}
|
||||
return -1;
|
||||
}
|
||||
|
||||
/**
|
||||
* Check if a byte array contains all zeros
|
||||
* @param array The array to check
|
||||
* @param startIndex The starting position
|
||||
* @param length Number of elements to check
|
||||
* @return true if all checked elements were == 0, false otherwise
|
||||
*/
|
||||
public static boolean isAllZeros(byte[] array, int startIndex, int length)
|
||||
{
|
||||
for(int i = startIndex; i < startIndex + length; i++)
|
||||
{
|
||||
if (array[i] != 0)
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Check if a byte array contains all zeros
|
||||
* @param array The array to check
|
||||
* @return true if all checked elements were == 0, false otherwise
|
||||
*/
|
||||
public static boolean isAllZeros(byte[] array)
|
||||
{
|
||||
return isAllZeros(array, 0, array.length);
|
||||
}
|
||||
}
|
||||
|
5
app/src/main/res/drawable/ic_measurement_system.xml
Normal file
5
app/src/main/res/drawable/ic_measurement_system.xml
Normal file
@ -0,0 +1,5 @@
|
||||
<vector android:height="24dp" android:tint="#7E7E7E"
|
||||
android:viewportHeight="24.0" android:viewportWidth="24.0"
|
||||
android:width="24dp" xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
<path android:fillColor="#FF000000" android:pathData="M8,19h3v4h2v-4h3l-4,-4 -4,4zM16,5h-3L13,1h-2v4L8,5l4,4 4,-4zM4,11v2h16v-2L4,11z"/>
|
||||
</vector>
|
5
app/src/main/res/drawable/ic_sitting.xml
Normal file
5
app/src/main/res/drawable/ic_sitting.xml
Normal file
@ -0,0 +1,5 @@
|
||||
<vector android:height="24dp" android:tint="#7E7E7E"
|
||||
android:viewportHeight="24.0" android:viewportWidth="24.0"
|
||||
android:width="24dp" xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
<path android:fillColor="#FF000000" android:pathData="M4,18v3h3v-3h10v3h3v-6L4,15zM19,10h3v3h-3zM2,10h3v3L2,13zM17,13L7,13L7,5c0,-1.1 0.9,-2 2,-2h6c1.1,0 2,0.9 2,2v8z"/>
|
||||
</vector>
|
@ -366,6 +366,16 @@
|
||||
<item>@string/p_scheduled</item>
|
||||
</string-array>
|
||||
|
||||
<string-array name="sedentary_reminder">
|
||||
<item>@string/off</item>
|
||||
<item>@string/on</item>
|
||||
</string-array>
|
||||
|
||||
<string-array name="sedentary_reminder_values">
|
||||
<item>@string/p_off</item>
|
||||
<item>@string/p_on</item>
|
||||
</string-array>
|
||||
|
||||
<string-array name="mi2_do_not_disturb">
|
||||
<item>@string/mi2_dnd_off</item>
|
||||
<item>@string/mi2_dnd_automatic</item>
|
||||
@ -2624,6 +2634,44 @@
|
||||
<item>ru_RU</item>
|
||||
</string-array>
|
||||
|
||||
<string-array name="pref_moyoung_watch_face">
|
||||
<item>Watch face 1</item>
|
||||
<item>Watch face 2</item>
|
||||
<item>Watch face 3</item>
|
||||
<item>Watch face 4</item>
|
||||
<item>Watch face 5</item>
|
||||
<item>Watch face 6</item>
|
||||
<item>Watch face 7</item>
|
||||
<item>Watch face 8</item>
|
||||
<item>Watch face 9</item>
|
||||
<item>Watch face 10</item>
|
||||
<item>Watch face 11</item>
|
||||
</string-array>
|
||||
|
||||
<string-array name="pref_moyoung_watch_face_values">
|
||||
<item>1</item>
|
||||
<item>2</item>
|
||||
<item>3</item>
|
||||
<item>4</item>
|
||||
<item>5</item>
|
||||
<item>6</item>
|
||||
<item>7</item>
|
||||
<item>8</item>
|
||||
<item>9</item>
|
||||
<item>10</item>
|
||||
<item>11</item>
|
||||
</string-array>
|
||||
|
||||
<string-array name="pref_moyoung_device_version">
|
||||
<item>Chinese edition</item>
|
||||
<item>International edition</item>
|
||||
</string-array>
|
||||
|
||||
<string-array name="pref_moyoung_device_version_values">
|
||||
<item>0</item>
|
||||
<item>1</item>
|
||||
</string-array>
|
||||
|
||||
<string-array name="prefs_heartrate_measurement_interval">
|
||||
<!-- This will be filled dynamically by HeartRateCapability -->
|
||||
<item name="0">@string/off</item>
|
||||
|
@ -436,6 +436,8 @@
|
||||
<string name="pref_display_add_device_fab_off">Visible only if no device is added</string>
|
||||
<string name="pref_title_huami_force_new_protocol">New Auth Protocol</string>
|
||||
<string name="pref_summary_huami_force_new_protocol">Enable if your device no longer connects after a firmware upgrade</string>
|
||||
<string name="pref_watch_face">Watch face</string>
|
||||
<string name="pref_device_version">Device version</string>
|
||||
<!-- HPlus Preferences -->
|
||||
<string name="pref_title_unit_system">Units</string>
|
||||
<string name="pref_title_timeformat">Time format</string>
|
||||
@ -702,6 +704,8 @@
|
||||
<string name="busy_task_fetch_sleep_respiratory_rate_data">Fetching sleep respiratory rate data</string>
|
||||
<string name="busy_task_fetch_temperature">Fetching temperature data</string>
|
||||
<string name="busy_task_fetch_statistics">Fetching statistics</string>
|
||||
<string name="busy_task_fetch_training_data">Fetching last training data</string>
|
||||
<string name="busy_task_fetch_training_data_finished">Training data recieved!</string>
|
||||
<string name="sleep_activity_date_range">From %1$s to %2$s</string>
|
||||
<string name="prefs_wearside">Wearing left or right?</string>
|
||||
<string name="prefs_weardirection">Wearing direction</string>
|
||||
@ -742,6 +746,7 @@
|
||||
<string name="interval_five_minutes">every 5 minutes</string>
|
||||
<string name="interval_ten_minutes">every 10 minutes</string>
|
||||
<string name="interval_fifteen_minutes">every 15 minutes</string>
|
||||
<string name="interval_twenty_minutes">every 20 minutes</string>
|
||||
<string name="interval_thirty_minutes">every 30 minutes</string>
|
||||
<string name="interval_forty_five_minutes">every 45 minutes</string>
|
||||
<string name="heartrate_bpm_40">40 bpm</string>
|
||||
@ -1907,6 +1912,8 @@
|
||||
<string name="devicetype_colmi_r06">Colmi R06</string>
|
||||
<string name="devicetype_colmi_r09">Colmi R09</string>
|
||||
<string name="devicetype_colmi_r10">Colmi R10</string>
|
||||
<string name="devicetype_colmi_i28_ultra">Colmi i28 Ultra</string>
|
||||
<string name="devicetype_misirun_c17">Misirun C17</string>
|
||||
<string name="devicetype_bandw_pseries">Bowers and Wilkins P series</string>
|
||||
<string name="choose_auto_export_location">Choose export location</string>
|
||||
<string name="notification_channel_name">General</string>
|
||||
|
@ -0,0 +1,20 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<androidx.preference.PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
|
||||
<SwitchPreferenceCompat
|
||||
android:defaultValue="false"
|
||||
android:icon="@drawable/ic_block"
|
||||
android:key="do_not_disturb_on_off"
|
||||
android:layout="@layout/preference_checkbox"
|
||||
android:title="@string/mi2_prefs_do_not_disturb"
|
||||
android:summary="@string/mi2_prefs_do_not_disturb_summary" />
|
||||
|
||||
<SwitchPreferenceCompat
|
||||
android:defaultValue="false"
|
||||
android:icon="@drawable/ic_action_find_lost_device"
|
||||
android:key="do_not_disturb_follow_phone"
|
||||
android:layout="@layout/preference_checkbox"
|
||||
android:title="@string/pref_dnd_follow_phone_title"
|
||||
android:summary="@string/pref_dnd_follow_phone_summary" />
|
||||
|
||||
</androidx.preference.PreferenceScreen>
|
11
app/src/main/res/xml/devicesettings_heartrate_interval.xml
Normal file
11
app/src/main/res/xml/devicesettings_heartrate_interval.xml
Normal file
@ -0,0 +1,11 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<androidx.preference.PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
<ListPreference
|
||||
android:defaultValue="0"
|
||||
android:entries="@array/prefs_heartrate_measurement_interval"
|
||||
android:entryValues="@array/prefs_heartrate_measurement_interval_values"
|
||||
android:icon="@drawable/ic_heartrate"
|
||||
android:key="heartrate_measurement_interval"
|
||||
android:summary="%s"
|
||||
android:title="@string/prefs_title_heartrate_measurement_interval" />
|
||||
</androidx.preference.PreferenceScreen>
|
@ -0,0 +1,41 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<androidx.preference.PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
<PreferenceScreen
|
||||
android:icon="@drawable/ic_chair"
|
||||
android:key="screen_inactivity"
|
||||
android:persistent="false"
|
||||
android:summary="@string/mi2_prefs_inactivity_warnings_summary"
|
||||
android:title="@string/mi2_prefs_inactivity_warnings">
|
||||
|
||||
<SwitchPreferenceCompat
|
||||
android:defaultValue="false"
|
||||
android:key="inactivity_warnings_enable"
|
||||
android:layout="@layout/preference_checkbox"
|
||||
android:title="@string/mi2_prefs_inactivity_warnings"
|
||||
android:summary="@string/mi2_prefs_inactivity_warnings_summary" />
|
||||
<EditTextPreference
|
||||
android:defaultValue="60"
|
||||
android:dependency="inactivity_warnings_enable"
|
||||
android:key="inactivity_warnings_threshold"
|
||||
android:summary="@string/mi2_prefs_inactivity_warnings_summary"
|
||||
android:title="@string/mi2_prefs_inactivity_warnings_threshold"/>
|
||||
<EditTextPreference
|
||||
android:defaultValue="60"
|
||||
android:dependency="inactivity_warnings_enable"
|
||||
android:key="inactivity_warnings_steps"
|
||||
android:title="@string/inactivity_warnings_minimum_steps_title"
|
||||
android:summary="@string/inactivity_warnings_minimum_steps_summary"/>
|
||||
<nodomain.freeyourgadget.gadgetbridge.util.XTimePreference
|
||||
android:defaultValue="06:00"
|
||||
android:dependency="inactivity_warnings_enable"
|
||||
android:key="inactivity_warnings_start"
|
||||
android:title="@string/mi2_prefs_do_not_disturb_start" />
|
||||
<nodomain.freeyourgadget.gadgetbridge.util.XTimePreference
|
||||
android:defaultValue="23:00"
|
||||
android:dependency="inactivity_warnings_enable"
|
||||
android:key="inactivity_warnings_end"
|
||||
android:title="@string/mi2_prefs_do_not_disturb_end" />
|
||||
|
||||
</PreferenceScreen>
|
||||
|
||||
</androidx.preference.PreferenceScreen>
|
@ -0,0 +1,11 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<androidx.preference.PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
<ListPreference
|
||||
android:defaultValue="1"
|
||||
android:entries="@array/pref_moyoung_device_version"
|
||||
android:entryValues="@array/pref_moyoung_device_version_values"
|
||||
android:key="moyoung_device_version"
|
||||
android:summary="%s"
|
||||
android:title="@string/pref_device_version"
|
||||
android:icon="@drawable/ic_language" />
|
||||
</androidx.preference.PreferenceScreen>
|
11
app/src/main/res/xml/devicesettings_moyoung_watchface.xml
Normal file
11
app/src/main/res/xml/devicesettings_moyoung_watchface.xml
Normal file
@ -0,0 +1,11 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<androidx.preference.PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
<ListPreference
|
||||
android:icon="@drawable/ic_widgets"
|
||||
android:defaultValue="1"
|
||||
android:entries="@array/pref_moyoung_watch_face"
|
||||
android:entryValues="@array/pref_moyoung_watch_face_values"
|
||||
android:key="moyoung_watch_face"
|
||||
android:summary="%s"
|
||||
android:title="@string/pref_watch_face" />
|
||||
</androidx.preference.PreferenceScreen>
|
Loading…
Reference in New Issue
Block a user