mirror of
https://codeberg.org/Freeyourgadget/Gadgetbridge.git
synced 2025-01-27 09:01:38 +01:00
Merge branch 'master' into hplus-preferences
This commit is contained in:
commit
475426c0ed
@ -2,6 +2,8 @@
|
|||||||
|
|
||||||
####Version next
|
####Version next
|
||||||
* Better integration with android music players
|
* Better integration with android music players
|
||||||
|
* Pebble: Implement notification and incoming call privacy modes
|
||||||
|
* Pebble: Support weather for Obisdian watchface
|
||||||
|
|
||||||
####Version 0.17.3
|
####Version 0.17.3
|
||||||
* HPlus: Improve display of new messages and phone calls
|
* HPlus: Improve display of new messages and phone calls
|
||||||
|
@ -204,6 +204,9 @@ public class PBWReader {
|
|||||||
}
|
}
|
||||||
app = new GBDeviceApp(appUUID, appName, appCreator, appVersion, appType);
|
app = new GBDeviceApp(appUUID, appName, appCreator, appVersion, appType);
|
||||||
}
|
}
|
||||||
|
else if (!isFirmware) {
|
||||||
|
isValid = false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -355,9 +355,6 @@ public class NotificationListener extends NotificationListenerService {
|
|||||||
MediaController c;
|
MediaController c;
|
||||||
try {
|
try {
|
||||||
c = new MediaController(getApplicationContext(), (MediaSession.Token) extras.get(Notification.EXTRA_MEDIA_SESSION));
|
c = new MediaController(getApplicationContext(), (MediaSession.Token) extras.get(Notification.EXTRA_MEDIA_SESSION));
|
||||||
} catch (NullPointerException e) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
PlaybackState s = c.getPlaybackState();
|
PlaybackState s = c.getPlaybackState();
|
||||||
stateSpec.position = (int) (s.getPosition() / 1000);
|
stateSpec.position = (int) (s.getPosition() / 1000);
|
||||||
@ -389,17 +386,20 @@ public class NotificationListener extends NotificationListenerService {
|
|||||||
if (d.containsKey(MediaMetadata.METADATA_KEY_TITLE))
|
if (d.containsKey(MediaMetadata.METADATA_KEY_TITLE))
|
||||||
musicSpec.track = d.getString(MediaMetadata.METADATA_KEY_TITLE);
|
musicSpec.track = d.getString(MediaMetadata.METADATA_KEY_TITLE);
|
||||||
if (d.containsKey(MediaMetadata.METADATA_KEY_DURATION))
|
if (d.containsKey(MediaMetadata.METADATA_KEY_DURATION))
|
||||||
musicSpec.duration = (int)d.getLong(MediaMetadata.METADATA_KEY_DURATION) / 1000;
|
musicSpec.duration = (int) d.getLong(MediaMetadata.METADATA_KEY_DURATION) / 1000;
|
||||||
if (d.containsKey(MediaMetadata.METADATA_KEY_NUM_TRACKS))
|
if (d.containsKey(MediaMetadata.METADATA_KEY_NUM_TRACKS))
|
||||||
musicSpec.trackCount = (int)d.getLong(MediaMetadata.METADATA_KEY_NUM_TRACKS);
|
musicSpec.trackCount = (int) d.getLong(MediaMetadata.METADATA_KEY_NUM_TRACKS);
|
||||||
if (d.containsKey(MediaMetadata.METADATA_KEY_TRACK_NUMBER))
|
if (d.containsKey(MediaMetadata.METADATA_KEY_TRACK_NUMBER))
|
||||||
musicSpec.trackNr = (int)d.getLong(MediaMetadata.METADATA_KEY_TRACK_NUMBER);
|
musicSpec.trackNr = (int) d.getLong(MediaMetadata.METADATA_KEY_TRACK_NUMBER);
|
||||||
|
|
||||||
// finally, tell the device about it
|
// finally, tell the device about it
|
||||||
GBApplication.deviceService().onSetMusicInfo(musicSpec);
|
GBApplication.deviceService().onSetMusicInfo(musicSpec);
|
||||||
GBApplication.deviceService().onSetMusicState(stateSpec);
|
GBApplication.deviceService().onSetMusicState(stateSpec);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
} catch (NullPointerException e) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -0,0 +1,160 @@
|
|||||||
|
package nodomain.freeyourgadget.gadgetbridge.service.devices.pebble;
|
||||||
|
|
||||||
|
import android.util.Pair;
|
||||||
|
import android.widget.Toast;
|
||||||
|
|
||||||
|
import org.json.JSONException;
|
||||||
|
import org.json.JSONObject;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Iterator;
|
||||||
|
import java.util.UUID;
|
||||||
|
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.deviceevents.GBDeviceEvent;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.deviceevents.GBDeviceEventSendBytes;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.model.Weather;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.model.WeatherSpec;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.util.GB;
|
||||||
|
|
||||||
|
class AppMessageHandlerObsidian extends AppMessageHandler {
|
||||||
|
|
||||||
|
/*
|
||||||
|
"appKeys": {
|
||||||
|
"CONFIG_WEATHER_REFRESH": 35,
|
||||||
|
"CONFIG_WEATHER_UNIT_LOCAL": 31,
|
||||||
|
"MSG_KEY_WEATHER_TEMP": 100,
|
||||||
|
|
||||||
|
"CONFIG_WEATHER_EXPIRATION": 36,
|
||||||
|
"MSG_KEY_FETCH_WEATHER": 102,
|
||||||
|
"MSG_KEY_WEATHER_ICON": 101,
|
||||||
|
"MSG_KEY_WEATHER_FAILED": 104,
|
||||||
|
"CONFIG_WEATHER_MODE_LOCAL": 30,
|
||||||
|
"CONFIG_WEATHER_APIKEY_LOCAL": 33,
|
||||||
|
"CONFIG_WEATHER_LOCAL": 28,
|
||||||
|
"CONFIG_COLOR_WEATHER": 29,
|
||||||
|
"CONFIG_WEATHER_LOCATION_LOCAL": 34,
|
||||||
|
"CONFIG_WEATHER_SOURCE_LOCAL": 32
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
private static final String ICON_01d = "a"; //night icons are just uppercase
|
||||||
|
private static final String ICON_02d = "b";
|
||||||
|
private static final String ICON_03d = "c";
|
||||||
|
private static final String ICON_04d = "d";
|
||||||
|
private static final String ICON_09d = "e";
|
||||||
|
private static final String ICON_10d = "f";
|
||||||
|
private static final String ICON_11d = "g";
|
||||||
|
private static final String ICON_13d = "h";
|
||||||
|
private static final String ICON_50d = "i";
|
||||||
|
|
||||||
|
|
||||||
|
AppMessageHandlerObsidian(UUID uuid, PebbleProtocol pebbleProtocol) {
|
||||||
|
super(uuid, pebbleProtocol);
|
||||||
|
messageKeys = new HashMap<>();
|
||||||
|
try {
|
||||||
|
JSONObject appKeys = getAppKeys();
|
||||||
|
Iterator<String> appKeysIterator = appKeys.keys();
|
||||||
|
while (appKeysIterator.hasNext()) {
|
||||||
|
String current = appKeysIterator.next();
|
||||||
|
switch (current) {
|
||||||
|
case "CONFIG_WEATHER_REFRESH":
|
||||||
|
case "CONFIG_WEATHER_UNIT_LOCAL":
|
||||||
|
case "MSG_KEY_WEATHER_TEMP":
|
||||||
|
case "MSG_KEY_WEATHER_ICON":
|
||||||
|
messageKeys.put(current, appKeys.getInt(current));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (JSONException e) {
|
||||||
|
GB.toast("There was an error accessing the timestyle watchface configuration.", Toast.LENGTH_LONG, GB.ERROR);
|
||||||
|
} catch (IOException ignore) {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private String getIconForConditionCode(int conditionCode, boolean isNight) {
|
||||||
|
|
||||||
|
int generalCondition = conditionCode / 100;
|
||||||
|
String iconToLoad;
|
||||||
|
// determine the correct icon
|
||||||
|
switch (generalCondition) {
|
||||||
|
case 2: //thunderstorm
|
||||||
|
iconToLoad = ICON_11d;
|
||||||
|
break;
|
||||||
|
case 3: //drizzle
|
||||||
|
iconToLoad = ICON_09d;
|
||||||
|
break;
|
||||||
|
case 5: //rain
|
||||||
|
if (conditionCode == 500) {
|
||||||
|
iconToLoad = ICON_09d;
|
||||||
|
} else if (conditionCode < 505) {
|
||||||
|
iconToLoad = ICON_10d;
|
||||||
|
} else if (conditionCode == 511) {
|
||||||
|
iconToLoad = ICON_10d;
|
||||||
|
} else {
|
||||||
|
iconToLoad = ICON_09d;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 6: //snow
|
||||||
|
if (conditionCode == 600 || conditionCode == 620) {
|
||||||
|
iconToLoad = ICON_13d;
|
||||||
|
} else if (conditionCode > 610 && conditionCode < 620) {
|
||||||
|
iconToLoad = ICON_13d;
|
||||||
|
} else {
|
||||||
|
iconToLoad = ICON_13d;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 7: // fog, dust, etc
|
||||||
|
iconToLoad = ICON_03d;
|
||||||
|
break;
|
||||||
|
case 8: // clouds
|
||||||
|
if (conditionCode == 800) {
|
||||||
|
iconToLoad = ICON_01d;
|
||||||
|
} else if (conditionCode < 803) {
|
||||||
|
iconToLoad = ICON_02d;
|
||||||
|
} else {
|
||||||
|
iconToLoad = ICON_04d;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
iconToLoad = ICON_02d;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (!isNight) ? iconToLoad : iconToLoad.toUpperCase();
|
||||||
|
}
|
||||||
|
|
||||||
|
private byte[] encodeObisdianWeather(WeatherSpec weatherSpec) {
|
||||||
|
|
||||||
|
if (weatherSpec == null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
ArrayList<Pair<Integer, Object>> pairs = new ArrayList<>();
|
||||||
|
boolean isNight = false; //TODO: use the night icons when night
|
||||||
|
pairs.add(new Pair<>(messageKeys.get("CONFIG_WEATHER_REFRESH"), (Object) 60));
|
||||||
|
pairs.add(new Pair<>(messageKeys.get("CONFIG_WEATHER_UNIT_LOCAL"), (Object) 1)); //celsius
|
||||||
|
pairs.add(new Pair<>(messageKeys.get("MSG_KEY_WEATHER_ICON"), (Object) getIconForConditionCode(weatherSpec.currentConditionCode, isNight))); //celsius
|
||||||
|
pairs.add(new Pair<>(messageKeys.get("MSG_KEY_WEATHER_TEMP"), (Object) (weatherSpec.currentTemp - 273)));
|
||||||
|
|
||||||
|
return mPebbleProtocol.encodeApplicationMessagePush(PebbleProtocol.ENDPOINT_APPLICATIONMESSAGE, mUUID, pairs);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public GBDeviceEvent[] onAppStart() {
|
||||||
|
WeatherSpec weatherSpec = Weather.getInstance().getWeatherSpec();
|
||||||
|
if (weatherSpec == null) {
|
||||||
|
return new GBDeviceEvent[]{null};
|
||||||
|
}
|
||||||
|
GBDeviceEventSendBytes sendBytes = new GBDeviceEventSendBytes();
|
||||||
|
sendBytes.encodedBytes = encodeObisdianWeather(weatherSpec);
|
||||||
|
return new GBDeviceEvent[]{sendBytes};
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public byte[] encodeUpdateWeather(WeatherSpec weatherSpec) {
|
||||||
|
return encodeObisdianWeather(weatherSpec);
|
||||||
|
}
|
||||||
|
}
|
@ -368,6 +368,7 @@ public class PebbleProtocol extends GBDeviceProtocol {
|
|||||||
private static final UUID UUID_ZALEWSZCZAK_CROWEX = UUID.fromString("a88b3151-2426-43c6-b1d0-9b288b3ec47e");
|
private static final UUID UUID_ZALEWSZCZAK_CROWEX = UUID.fromString("a88b3151-2426-43c6-b1d0-9b288b3ec47e");
|
||||||
private static final UUID UUID_ZALEWSZCZAK_FANCY = UUID.fromString("014e17bf-5878-4781-8be1-8ef998cee1ba");
|
private static final UUID UUID_ZALEWSZCZAK_FANCY = UUID.fromString("014e17bf-5878-4781-8be1-8ef998cee1ba");
|
||||||
private static final UUID UUID_ZALEWSZCZAK_TALLY = UUID.fromString("abb51965-52e2-440a-b93c-843eeacb697d");
|
private static final UUID UUID_ZALEWSZCZAK_TALLY = UUID.fromString("abb51965-52e2-440a-b93c-843eeacb697d");
|
||||||
|
private static final UUID UUID_OBSIDIAN = UUID.fromString("ef42caba-0c65-4879-ab23-edd2bde68824");
|
||||||
|
|
||||||
private static final UUID UUID_ZERO = new UUID(0, 0);
|
private static final UUID UUID_ZERO = new UUID(0, 0);
|
||||||
|
|
||||||
@ -390,6 +391,7 @@ public class PebbleProtocol extends GBDeviceProtocol {
|
|||||||
mAppMessageHandlers.put(UUID_ZALEWSZCZAK_CROWEX, new AppMessageHandlerZalewszczak(UUID_ZALEWSZCZAK_CROWEX, PebbleProtocol.this));
|
mAppMessageHandlers.put(UUID_ZALEWSZCZAK_CROWEX, new AppMessageHandlerZalewszczak(UUID_ZALEWSZCZAK_CROWEX, PebbleProtocol.this));
|
||||||
mAppMessageHandlers.put(UUID_ZALEWSZCZAK_FANCY, new AppMessageHandlerZalewszczak(UUID_ZALEWSZCZAK_FANCY, PebbleProtocol.this));
|
mAppMessageHandlers.put(UUID_ZALEWSZCZAK_FANCY, new AppMessageHandlerZalewszczak(UUID_ZALEWSZCZAK_FANCY, PebbleProtocol.this));
|
||||||
mAppMessageHandlers.put(UUID_ZALEWSZCZAK_TALLY, new AppMessageHandlerZalewszczak(UUID_ZALEWSZCZAK_TALLY, PebbleProtocol.this));
|
mAppMessageHandlers.put(UUID_ZALEWSZCZAK_TALLY, new AppMessageHandlerZalewszczak(UUID_ZALEWSZCZAK_TALLY, PebbleProtocol.this));
|
||||||
|
mAppMessageHandlers.put(UUID_OBSIDIAN, new AppMessageHandlerObsidian(UUID_OBSIDIAN, PebbleProtocol.this));
|
||||||
}
|
}
|
||||||
|
|
||||||
private final HashMap<Byte, DatalogSession> mDatalogSessions = new HashMap<>();
|
private final HashMap<Byte, DatalogSession> mDatalogSessions = new HashMap<>();
|
||||||
|
@ -12,6 +12,7 @@ import java.util.Iterator;
|
|||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
|
|
||||||
import nodomain.freeyourgadget.gadgetbridge.GBApplication;
|
import nodomain.freeyourgadget.gadgetbridge.GBApplication;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.R;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
|
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.model.Alarm;
|
import nodomain.freeyourgadget.gadgetbridge.model.Alarm;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.model.CalendarEventSpec;
|
import nodomain.freeyourgadget.gadgetbridge.model.CalendarEventSpec;
|
||||||
@ -113,6 +114,16 @@ public class PebbleSupport extends AbstractSerialDeviceSupport {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onNotification(NotificationSpec notificationSpec) {
|
public void onNotification(NotificationSpec notificationSpec) {
|
||||||
|
String currentPrivacyMode = GBApplication.getPrefs().getString("pebble_pref_privacy_mode", getContext().getString(R.string.p_pebble_privacy_mode_off));
|
||||||
|
if (getContext().getString(R.string.p_pebble_privacy_mode_complete).equals(currentPrivacyMode)) {
|
||||||
|
notificationSpec.body = null;
|
||||||
|
notificationSpec.sender = null;
|
||||||
|
notificationSpec.subject = null;
|
||||||
|
notificationSpec.title = null;
|
||||||
|
notificationSpec.phoneNumber = null;
|
||||||
|
} else if (getContext().getString(R.string.p_pebble_privacy_mode_content).equals(currentPrivacyMode)) {
|
||||||
|
notificationSpec.sender = "\n\n\n\n\n" + notificationSpec.sender;
|
||||||
|
}
|
||||||
if (reconnect()) {
|
if (reconnect()) {
|
||||||
super.onNotification(notificationSpec);
|
super.onNotification(notificationSpec);
|
||||||
}
|
}
|
||||||
@ -120,6 +131,14 @@ public class PebbleSupport extends AbstractSerialDeviceSupport {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onSetCallState(CallSpec callSpec) {
|
public void onSetCallState(CallSpec callSpec) {
|
||||||
|
String currentPrivacyMode = GBApplication.getPrefs().getString("pebble_pref_privacy_mode", getContext().getString(R.string.p_pebble_privacy_mode_off));
|
||||||
|
if (getContext().getString(R.string.p_pebble_privacy_mode_complete).equals(currentPrivacyMode)) {
|
||||||
|
callSpec.name = null;
|
||||||
|
callSpec.number = null;
|
||||||
|
} else if (getContext().getString(R.string.p_pebble_privacy_mode_content).equals(currentPrivacyMode)) {
|
||||||
|
callSpec.name = null;
|
||||||
|
}
|
||||||
|
|
||||||
if (reconnect()) {
|
if (reconnect()) {
|
||||||
if ((callSpec.command != CallSpec.CALL_OUTGOING) || GBApplication.getPrefs().getBoolean("pebble_enable_outgoing_call", true)) {
|
if ((callSpec.command != CallSpec.CALL_OUTGOING) || GBApplication.getPrefs().getBoolean("pebble_enable_outgoing_call", true)) {
|
||||||
super.onSetCallState(callSpec);
|
super.onSetCallState(callSpec);
|
||||||
|
@ -1,5 +1,7 @@
|
|||||||
package nodomain.freeyourgadget.gadgetbridge.util;
|
package nodomain.freeyourgadget.gadgetbridge.util;
|
||||||
|
|
||||||
|
import org.apache.commons.lang3.text.WordUtils;
|
||||||
|
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.text.Normalizer;
|
import java.text.Normalizer;
|
||||||
@ -67,7 +69,7 @@ public class LanguageUtils {
|
|||||||
|
|
||||||
if (lowerChar != c)
|
if (lowerChar != c)
|
||||||
{
|
{
|
||||||
return replace.toUpperCase();
|
return WordUtils.capitalize(replace);
|
||||||
}
|
}
|
||||||
|
|
||||||
return replace;
|
return replace;
|
||||||
|
@ -113,6 +113,19 @@
|
|||||||
<item>3</item>
|
<item>3</item>
|
||||||
<item>1</item>
|
<item>1</item>
|
||||||
</string-array>
|
</string-array>
|
||||||
|
|
||||||
|
<string-array name="pebble_privacymode">
|
||||||
|
<item name="off">@string/pref_pebble_privacy_mode_off</item>
|
||||||
|
<item name="content">@string/pref_pebble_privacy_mode_content</item>
|
||||||
|
<item name="complete">@string/pref_pebble_privacy_mode_complete</item>
|
||||||
|
</string-array>
|
||||||
|
|
||||||
|
<string-array name="pebble_privacymode_values">
|
||||||
|
<item>@string/p_pebble_privacy_mode_off</item>
|
||||||
|
<item>@string/p_pebble_privacy_mode_content</item>
|
||||||
|
<item>@string/p_pebble_privacy_mode_complete</item>
|
||||||
|
</string-array>
|
||||||
|
|
||||||
<string-array name="mi2_dateformats">
|
<string-array name="mi2_dateformats">
|
||||||
<item>@string/dateformat_time</item>
|
<item>@string/dateformat_time</item>
|
||||||
<item>@string/dateformat_date_time</item>
|
<item>@string/dateformat_date_time</item>
|
||||||
|
@ -119,6 +119,11 @@
|
|||||||
<string name="pref_title_autoremove_notifications">Autoremove dismissed Notifications</string>
|
<string name="pref_title_autoremove_notifications">Autoremove dismissed Notifications</string>
|
||||||
<string name="pref_summary_autoremove_notifications">Notifications are automatically removed from the Pebble when dismissed from the Android device</string>
|
<string name="pref_summary_autoremove_notifications">Notifications are automatically removed from the Pebble when dismissed from the Android device</string>
|
||||||
|
|
||||||
|
<string name="pref_title_pebble_privacy_mode">Privacy mode</string>
|
||||||
|
<string name="pref_pebble_privacy_mode_off">Normal notifications and incoming calls display.</string>
|
||||||
|
<string name="pref_pebble_privacy_mode_content">Shift the notification text off-screen. Hide the caller\'s name on incoming calls.</string>
|
||||||
|
<string name="pref_pebble_privacy_mode_complete">Show only the notification icon. Hide the caller\'s name and number on incoming calls.</string>
|
||||||
|
|
||||||
<string name="pref_header_location">Location</string>
|
<string name="pref_header_location">Location</string>
|
||||||
<string name="pref_title_location_aquire">Acquire Location</string>
|
<string name="pref_title_location_aquire">Acquire Location</string>
|
||||||
<string name="pref_title_location_latitude">Latitude</string>
|
<string name="pref_title_location_latitude">Latitude</string>
|
||||||
|
@ -18,4 +18,8 @@
|
|||||||
<item name="p_timeformat_24h" type="string">24h</item>
|
<item name="p_timeformat_24h" type="string">24h</item>
|
||||||
<item name="p_timeformat_am_pm" type="string">am/pm</item>
|
<item name="p_timeformat_am_pm" type="string">am/pm</item>
|
||||||
|
|
||||||
|
<item name="p_pebble_privacy_mode_off" type="string">off</item>
|
||||||
|
<item name="p_pebble_privacy_mode_content" type="string">content</item>
|
||||||
|
<item name="p_pebble_privacy_mode_complete" type="string">complete</item>
|
||||||
|
|
||||||
</resources>
|
</resources>
|
||||||
|
@ -2,8 +2,9 @@
|
|||||||
<changelog>
|
<changelog>
|
||||||
<release version="next">
|
<release version="next">
|
||||||
<change>Better integration with android music players</change>
|
<change>Better integration with android music players</change>
|
||||||
|
<change>Pebble: Implement notification and incoming call privacy modes</change>
|
||||||
|
<change>Pebble: Support weather for Obisdian watchface</change>
|
||||||
</release>
|
</release>
|
||||||
|
|
||||||
<release version="0.17.3" versioncode="84">
|
<release version="0.17.3" versioncode="84">
|
||||||
<change>HPlus: Improve display of new messages and phone calls</change>
|
<change>HPlus: Improve display of new messages and phone calls</change>
|
||||||
<change>HPlus: Fix bug related to steps and heart rate</change>
|
<change>HPlus: Fix bug related to steps and heart rate</change>
|
||||||
|
@ -184,6 +184,13 @@
|
|||||||
android:key="autoremove_notifications"
|
android:key="autoremove_notifications"
|
||||||
android:summary="@string/pref_summary_autoremove_notifications"
|
android:summary="@string/pref_summary_autoremove_notifications"
|
||||||
android:title="@string/pref_title_autoremove_notifications" />
|
android:title="@string/pref_title_autoremove_notifications" />
|
||||||
|
<ListPreference
|
||||||
|
android:key="pebble_pref_privacy_mode"
|
||||||
|
android:title="@string/pref_title_pebble_privacy_mode"
|
||||||
|
android:entries="@array/pebble_privacymode"
|
||||||
|
android:entryValues="@array/pebble_privacymode_values"
|
||||||
|
android:defaultValue="@string/p_pebble_privacy_mode_off"
|
||||||
|
android:summary="%s" />
|
||||||
</PreferenceCategory>
|
</PreferenceCategory>
|
||||||
<PreferenceCategory android:title="@string/pref_header_activitytrackers">
|
<PreferenceCategory android:title="@string/pref_header_activitytrackers">
|
||||||
<ListPreference
|
<ListPreference
|
||||||
|
Loading…
Reference in New Issue
Block a user