mirror of
https://codeberg.org/Freeyourgadget/Gadgetbridge
synced 2025-01-11 18:35:49 +01:00
Control alarm snooze mode on MiBand2
This commit is contained in:
parent
d6fa5d0dd5
commit
10c9b07c69
@ -43,7 +43,7 @@ public class GBDaoGenerator {
|
|||||||
|
|
||||||
|
|
||||||
public static void main(String[] args) throws Exception {
|
public static void main(String[] args) throws Exception {
|
||||||
Schema schema = new Schema(22, MAIN_PACKAGE + ".entities");
|
Schema schema = new Schema(23, MAIN_PACKAGE + ".entities");
|
||||||
|
|
||||||
Entity userAttributes = addUserAttributes(schema);
|
Entity userAttributes = addUserAttributes(schema);
|
||||||
Entity user = addUserInfo(schema, userAttributes);
|
Entity user = addUserInfo(schema, userAttributes);
|
||||||
@ -383,6 +383,7 @@ public class GBDaoGenerator {
|
|||||||
alarm.addIndex(indexUnique);
|
alarm.addIndex(indexUnique);
|
||||||
alarm.addBooleanProperty("enabled").notNull();
|
alarm.addBooleanProperty("enabled").notNull();
|
||||||
alarm.addBooleanProperty("smartWakeup").notNull();
|
alarm.addBooleanProperty("smartWakeup").notNull();
|
||||||
|
alarm.addBooleanProperty("snooze").notNull();
|
||||||
alarm.addIntProperty("repetition").notNull().codeBeforeGetter(
|
alarm.addIntProperty("repetition").notNull().codeBeforeGetter(
|
||||||
"public boolean isRepetitive() { return getRepetition() != ALARM_ONCE; } " +
|
"public boolean isRepetitive() { return getRepetition() != ALARM_ONCE; } " +
|
||||||
"public boolean getRepetition(int dow) { return (this.repetition & dow) > 0; }"
|
"public boolean getRepetition(int dow) { return (this.repetition & dow) > 0; }"
|
||||||
|
@ -121,7 +121,7 @@ public class SleepAlarmWidget extends AppWidgetProvider {
|
|||||||
context.getString(R.string.appwidget_setting_alarm, hours, minutes),
|
context.getString(R.string.appwidget_setting_alarm, hours, minutes),
|
||||||
Toast.LENGTH_SHORT, GB.INFO);
|
Toast.LENGTH_SHORT, GB.INFO);
|
||||||
|
|
||||||
Alarm alarm = AlarmUtils.createSingleShot(0,true, calendar);
|
Alarm alarm = AlarmUtils.createSingleShot(0, true, false, calendar);
|
||||||
ArrayList<Alarm> alarms = new ArrayList<>(1);
|
ArrayList<Alarm> alarms = new ArrayList<>(1);
|
||||||
alarms.add(alarm);
|
alarms.add(alarm);
|
||||||
GBApplication.deviceService().onSetAlarms(alarms);
|
GBApplication.deviceService().onSetAlarms(alarms);
|
||||||
|
@ -38,6 +38,7 @@ public class AlarmDetails extends AbstractGBActivity {
|
|||||||
private Alarm alarm;
|
private Alarm alarm;
|
||||||
private TimePicker timePicker;
|
private TimePicker timePicker;
|
||||||
private CheckedTextView cbSmartWakeup;
|
private CheckedTextView cbSmartWakeup;
|
||||||
|
private CheckedTextView cbSnooze;
|
||||||
private CheckedTextView cbMonday;
|
private CheckedTextView cbMonday;
|
||||||
private CheckedTextView cbTuesday;
|
private CheckedTextView cbTuesday;
|
||||||
private CheckedTextView cbWednesday;
|
private CheckedTextView cbWednesday;
|
||||||
@ -57,6 +58,7 @@ public class AlarmDetails extends AbstractGBActivity {
|
|||||||
|
|
||||||
timePicker = findViewById(R.id.alarm_time_picker);
|
timePicker = findViewById(R.id.alarm_time_picker);
|
||||||
cbSmartWakeup = findViewById(R.id.alarm_cb_smart_wakeup);
|
cbSmartWakeup = findViewById(R.id.alarm_cb_smart_wakeup);
|
||||||
|
cbSnooze = findViewById(R.id.alarm_cb_snooze);
|
||||||
cbMonday = findViewById(R.id.alarm_cb_monday);
|
cbMonday = findViewById(R.id.alarm_cb_monday);
|
||||||
cbTuesday = findViewById(R.id.alarm_cb_tuesday);
|
cbTuesday = findViewById(R.id.alarm_cb_tuesday);
|
||||||
cbWednesday = findViewById(R.id.alarm_cb_wednesday);
|
cbWednesday = findViewById(R.id.alarm_cb_wednesday);
|
||||||
@ -71,6 +73,11 @@ public class AlarmDetails extends AbstractGBActivity {
|
|||||||
((CheckedTextView) v).toggle();
|
((CheckedTextView) v).toggle();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
cbSnooze.setOnClickListener(new View.OnClickListener() {
|
||||||
|
public void onClick(View v) {
|
||||||
|
((CheckedTextView) v).toggle();
|
||||||
|
}
|
||||||
|
});
|
||||||
cbMonday.setOnClickListener(new View.OnClickListener() {
|
cbMonday.setOnClickListener(new View.OnClickListener() {
|
||||||
public void onClick(View v) {
|
public void onClick(View v) {
|
||||||
((CheckedTextView) v).toggle();
|
((CheckedTextView) v).toggle();
|
||||||
@ -115,6 +122,10 @@ public class AlarmDetails extends AbstractGBActivity {
|
|||||||
int smartAlarmVisibility = supportsSmartWakeup() ? View.VISIBLE : View.GONE;
|
int smartAlarmVisibility = supportsSmartWakeup() ? View.VISIBLE : View.GONE;
|
||||||
cbSmartWakeup.setVisibility(smartAlarmVisibility);
|
cbSmartWakeup.setVisibility(smartAlarmVisibility);
|
||||||
|
|
||||||
|
cbSnooze.setChecked(alarm.getSnooze());
|
||||||
|
int snoozeVisibility = supportsSnoozing() ? View.VISIBLE : View.GONE;
|
||||||
|
cbSnooze.setVisibility(snoozeVisibility);
|
||||||
|
|
||||||
cbMonday.setChecked(alarm.getRepetition(Alarm.ALARM_MON));
|
cbMonday.setChecked(alarm.getRepetition(Alarm.ALARM_MON));
|
||||||
cbTuesday.setChecked(alarm.getRepetition(Alarm.ALARM_TUE));
|
cbTuesday.setChecked(alarm.getRepetition(Alarm.ALARM_TUE));
|
||||||
cbWednesday.setChecked(alarm.getRepetition(Alarm.ALARM_WED));
|
cbWednesday.setChecked(alarm.getRepetition(Alarm.ALARM_WED));
|
||||||
@ -133,6 +144,14 @@ public class AlarmDetails extends AbstractGBActivity {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private boolean supportsSnoozing() {
|
||||||
|
if (device != null) {
|
||||||
|
DeviceCoordinator coordinator = DeviceHelper.getInstance().getCoordinator(device);
|
||||||
|
return coordinator.supportsAlarmSnoozing();
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean onOptionsItemSelected(MenuItem item) {
|
public boolean onOptionsItemSelected(MenuItem item) {
|
||||||
switch (item.getItemId()) {
|
switch (item.getItemId()) {
|
||||||
@ -146,6 +165,7 @@ public class AlarmDetails extends AbstractGBActivity {
|
|||||||
|
|
||||||
private void updateAlarm() {
|
private void updateAlarm() {
|
||||||
alarm.setSmartWakeup(supportsSmartWakeup() && cbSmartWakeup.isChecked());
|
alarm.setSmartWakeup(supportsSmartWakeup() && cbSmartWakeup.isChecked());
|
||||||
|
alarm.setSnooze(supportsSnoozing() && cbSnooze.isChecked());
|
||||||
int repetitionMask = AlarmUtils.createRepetitionMassk(cbMonday.isChecked(), cbTuesday.isChecked(), cbWednesday.isChecked(), cbThursday.isChecked(), cbFriday.isChecked(), cbSaturday.isChecked(), cbSunday.isChecked());
|
int repetitionMask = AlarmUtils.createRepetitionMassk(cbMonday.isChecked(), cbTuesday.isChecked(), cbWednesday.isChecked(), cbThursday.isChecked(), cbFriday.isChecked(), cbSaturday.isChecked(), cbSunday.isChecked());
|
||||||
alarm.setRepetition(repetitionMask);
|
alarm.setRepetition(repetitionMask);
|
||||||
alarm.setHour(timePicker.getCurrentHour());
|
alarm.setHour(timePicker.getCurrentHour());
|
||||||
|
@ -146,7 +146,7 @@ public class ConfigureAlarms extends AbstractGBActivity {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private Alarm createDefaultAlarm(@NonNull Device device, @NonNull User user, int position) {
|
private Alarm createDefaultAlarm(@NonNull Device device, @NonNull User user, int position) {
|
||||||
return new Alarm(device.getId(), user.getId(), position, false, false, 0, 6, 30, false);
|
return new Alarm(device.getId(), user.getId(), position, false, false, false, 0, 6, 30, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -135,7 +135,7 @@ public class WidgetAlarmsActivity extends Activity implements View.OnClickListen
|
|||||||
this.getString(R.string.appwidget_setting_alarm, hours, minutes),
|
this.getString(R.string.appwidget_setting_alarm, hours, minutes),
|
||||||
Toast.LENGTH_SHORT, GB.INFO);
|
Toast.LENGTH_SHORT, GB.INFO);
|
||||||
|
|
||||||
Alarm alarm = AlarmUtils.createSingleShot(0, true, calendar);
|
Alarm alarm = AlarmUtils.createSingleShot(0, true, false, calendar);
|
||||||
ArrayList<Alarm> alarms = new ArrayList<>(1);
|
ArrayList<Alarm> alarms = new ArrayList<>(1);
|
||||||
alarms.add(alarm);
|
alarms.add(alarm);
|
||||||
GBApplication.deviceService().onSetAlarms(alarms);
|
GBApplication.deviceService().onSetAlarms(alarms);
|
||||||
|
@ -0,0 +1,40 @@
|
|||||||
|
/* Copyright (C) 2017-2020 Andreas Shimokawa, protomors
|
||||||
|
|
||||||
|
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.database.schema;
|
||||||
|
|
||||||
|
import android.database.sqlite.SQLiteDatabase;
|
||||||
|
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.database.DBHelper;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.database.DBUpdateScript;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.entities.AlarmDao;
|
||||||
|
|
||||||
|
public class GadgetbridgeUpdate_23 implements DBUpdateScript {
|
||||||
|
@Override
|
||||||
|
public void upgradeSchema(SQLiteDatabase db) {
|
||||||
|
if (!DBHelper.existsColumn(AlarmDao.TABLENAME, AlarmDao.Properties.Snooze.columnName, db)) {
|
||||||
|
// Setting default value of SNOOZE column to 1 (true), so that existing MiBand2 alarms
|
||||||
|
// behave as before
|
||||||
|
String ADD_COLUMN_SNOOZE = "ALTER TABLE " + AlarmDao.TABLENAME + " ADD COLUMN "
|
||||||
|
+ AlarmDao.Properties.Snooze.columnName + " INTEGER NOT NULL DEFAULT 1;";
|
||||||
|
db.execSQL(ADD_COLUMN_SNOOZE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void downgradeSchema(SQLiteDatabase db) {
|
||||||
|
}
|
||||||
|
}
|
@ -145,11 +145,17 @@ public abstract class AbstractDeviceCoordinator implements DeviceCoordinator {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean supportsAlarmSnoozing() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean supportsMusicInfo() {
|
public boolean supportsMusicInfo() {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
public boolean supportsLedColor() {
|
public boolean supportsLedColor() {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -201,6 +201,12 @@ public interface DeviceCoordinator {
|
|||||||
*/
|
*/
|
||||||
boolean supportsSmartWakeup(GBDevice device);
|
boolean supportsSmartWakeup(GBDevice device);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns true if this device/coordinator supports alarm snoozing
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
boolean supportsAlarmSnoozing();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns true if the given device supports heart rate measurements.
|
* Returns true if the given device supports heart rate measurements.
|
||||||
* @return
|
* @return
|
||||||
|
@ -65,6 +65,11 @@ public class MiBand2Coordinator extends HuamiCoordinator {
|
|||||||
return handler.isValid() ? handler : null;
|
return handler.isValid() ? handler : null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean supportsAlarmSnoozing() {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean supportsHeartRateMeasurement(GBDevice device) {
|
public boolean supportsHeartRateMeasurement(GBDevice device) {
|
||||||
return true;
|
return true;
|
||||||
|
@ -67,6 +67,11 @@ public class MiBand2HRXCoordinator extends HuamiCoordinator {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean supportsAlarmSnoozing() {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean supportsHeartRateMeasurement(GBDevice device) {
|
public boolean supportsHeartRateMeasurement(GBDevice device) {
|
||||||
return false;
|
return false;
|
||||||
|
@ -41,6 +41,8 @@ public interface Alarm extends Serializable {
|
|||||||
|
|
||||||
boolean getSmartWakeup();
|
boolean getSmartWakeup();
|
||||||
|
|
||||||
|
boolean getSnooze();
|
||||||
|
|
||||||
int getRepetition();
|
int getRepetition();
|
||||||
|
|
||||||
boolean isRepetitive();
|
boolean isRepetitive();
|
||||||
|
@ -1608,10 +1608,14 @@ public class HuamiSupport extends AbstractBTLEDeviceSupport {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
int base = 0;
|
int actionMask = 0;
|
||||||
int daysMask = 0;
|
int daysMask = 0;
|
||||||
if (alarm.getEnabled() && !alarm.getUnused()) {
|
if (alarm.getEnabled() && !alarm.getUnused()) {
|
||||||
base = 128;
|
actionMask = 0x80;
|
||||||
|
|
||||||
|
if (coordinator.supportsAlarmSnoozing() && !alarm.getSnooze()) {
|
||||||
|
actionMask |= 0x40;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (!alarm.getUnused()) {
|
if (!alarm.getUnused()) {
|
||||||
daysMask = alarm.getRepetition();
|
daysMask = alarm.getRepetition();
|
||||||
@ -1622,7 +1626,7 @@ public class HuamiSupport extends AbstractBTLEDeviceSupport {
|
|||||||
|
|
||||||
byte[] alarmMessage = new byte[] {
|
byte[] alarmMessage = new byte[] {
|
||||||
(byte) 0x2, // TODO what is this?
|
(byte) 0x2, // TODO what is this?
|
||||||
(byte) (base + alarm.getPosition()), // 128 is the base, alarm slot is added
|
(byte) (actionMask | alarm.getPosition()), // action mask + alarm slot
|
||||||
(byte) calendar.get(Calendar.HOUR_OF_DAY),
|
(byte) calendar.get(Calendar.HOUR_OF_DAY),
|
||||||
(byte) calendar.get(Calendar.MINUTE),
|
(byte) calendar.get(Calendar.MINUTE),
|
||||||
(byte) daysMask,
|
(byte) daysMask,
|
||||||
@ -1684,7 +1688,7 @@ public class HuamiSupport extends AbstractBTLEDeviceSupport {
|
|||||||
int slotToUse = 2 - iteration;
|
int slotToUse = 2 - iteration;
|
||||||
Calendar calendar = Calendar.getInstance();
|
Calendar calendar = Calendar.getInstance();
|
||||||
calendar.setTimeInMillis(mEvt.getBegin());
|
calendar.setTimeInMillis(mEvt.getBegin());
|
||||||
Alarm alarm = AlarmUtils.createSingleShot(slotToUse, false, calendar);
|
Alarm alarm = AlarmUtils.createSingleShot(slotToUse, false, true, calendar);
|
||||||
queueAlarm(alarm, builder, characteristic);
|
queueAlarm(alarm, builder, characteristic);
|
||||||
iteration++;
|
iteration++;
|
||||||
}
|
}
|
||||||
|
@ -1243,7 +1243,7 @@ public class MiBandSupport extends AbstractBTLEDeviceSupport {
|
|||||||
int slotToUse = 2 - iteration;
|
int slotToUse = 2 - iteration;
|
||||||
Calendar calendar = Calendar.getInstance();
|
Calendar calendar = Calendar.getInstance();
|
||||||
calendar.setTimeInMillis(mEvt.getBegin());
|
calendar.setTimeInMillis(mEvt.getBegin());
|
||||||
Alarm alarm = AlarmUtils.createSingleShot(slotToUse, false, calendar);
|
Alarm alarm = AlarmUtils.createSingleShot(slotToUse, false, false, calendar);
|
||||||
queueAlarm(alarm, builder, characteristic);
|
queueAlarm(alarm, builder, characteristic);
|
||||||
iteration++;
|
iteration++;
|
||||||
}
|
}
|
||||||
|
@ -48,8 +48,8 @@ public class AlarmUtils {
|
|||||||
* @param calendar
|
* @param calendar
|
||||||
* @return
|
* @return
|
||||||
*/
|
*/
|
||||||
public static nodomain.freeyourgadget.gadgetbridge.model.Alarm createSingleShot(int index, boolean smartWakeup, Calendar calendar) {
|
public static nodomain.freeyourgadget.gadgetbridge.model.Alarm createSingleShot(int index, boolean smartWakeup, boolean snooze, Calendar calendar) {
|
||||||
return new Alarm(-1, -1, index, true, smartWakeup, Alarm.ALARM_ONCE, calendar.get(Calendar.HOUR_OF_DAY), calendar.get(Calendar.MINUTE), false);
|
return new Alarm(-1, -1, index, true, smartWakeup, snooze, Alarm.ALARM_ONCE, calendar.get(Calendar.HOUR_OF_DAY), calendar.get(Calendar.MINUTE), false);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -128,7 +128,7 @@ public class AlarmUtils {
|
|||||||
int hour = Integer.parseInt(tokens[4]);
|
int hour = Integer.parseInt(tokens[4]);
|
||||||
int minute = Integer.parseInt(tokens[5]);
|
int minute = Integer.parseInt(tokens[5]);
|
||||||
|
|
||||||
return new Alarm(device.getId(), user.getId(), index, enabled, smartWakeup, repetition, hour, minute, false);
|
return new Alarm(device.getId(), user.getId(), index, enabled, smartWakeup, false, repetition, hour, minute, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
private static Comparator<Alarm> createComparator() {
|
private static Comparator<Alarm> createComparator() {
|
||||||
|
@ -33,6 +33,16 @@
|
|||||||
android:text="@string/alarm_smart_wakeup"
|
android:text="@string/alarm_smart_wakeup"
|
||||||
android:textAppearance="?android:attr/textAppearanceSmall" />
|
android:textAppearance="?android:attr/textAppearanceSmall" />
|
||||||
|
|
||||||
|
<androidx.appcompat.widget.AppCompatCheckedTextView
|
||||||
|
android:id="@+id/alarm_cb_snooze"
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:layout_marginStart="4dp"
|
||||||
|
android:drawableStart="?android:attr/listChoiceIndicatorMultiple"
|
||||||
|
android:gravity="center"
|
||||||
|
android:text="@string/alarm_snooze"
|
||||||
|
android:textAppearance="?android:attr/textAppearanceSmall" />
|
||||||
|
|
||||||
<TimePicker
|
<TimePicker
|
||||||
android:id="@+id/alarm_time_picker"
|
android:id="@+id/alarm_time_picker"
|
||||||
android:layout_width="match_parent"
|
android:layout_width="match_parent"
|
||||||
|
@ -366,6 +366,7 @@
|
|||||||
<string name="alarm_fri_short">Fri</string>
|
<string name="alarm_fri_short">Fri</string>
|
||||||
<string name="alarm_sat_short">Sat</string>
|
<string name="alarm_sat_short">Sat</string>
|
||||||
<string name="alarm_smart_wakeup">Smart wakeup</string>
|
<string name="alarm_smart_wakeup">Smart wakeup</string>
|
||||||
|
<string name="alarm_snooze">Snooze</string>
|
||||||
<string name="user_feedback_miband_set_alarms_failed">There was an error setting the alarms, please try again.</string>
|
<string name="user_feedback_miband_set_alarms_failed">There was an error setting the alarms, please try again.</string>
|
||||||
<string name="user_feedback_miband_set_alarms_ok">Alarms sent to device.</string>
|
<string name="user_feedback_miband_set_alarms_ok">Alarms sent to device.</string>
|
||||||
<string name="chart_no_data_synchronize">No data. Synchronize device?</string>
|
<string name="chart_no_data_synchronize">No data. Synchronize device?</string>
|
||||||
|
Loading…
Reference in New Issue
Block a user