1
0
mirror of https://codeberg.org/Freeyourgadget/Gadgetbridge synced 2025-01-26 09:37:33 +01:00

Merge remote-tracking branch 'freeyourgadget/master' into freeyourgadget_master

This commit is contained in:
Gordon Williams 2022-06-13 08:26:53 +01:00
commit bdcaeae177
13 changed files with 122 additions and 37 deletions

View File

@ -143,6 +143,8 @@ public class Widget extends AppWidgetProvider {
if (sleep < 1) { if (sleep < 1) {
views.setViewVisibility(R.id.todaywidget_sleep_layout, View.GONE); views.setViewVisibility(R.id.todaywidget_sleep_layout, View.GONE);
} else {
views.setViewVisibility(R.id.todaywidget_sleep_layout, View.VISIBLE);
} }
views.setTextViewText(R.id.todaywidget_steps, String.format("%1s", steps)); views.setTextViewText(R.id.todaywidget_steps, String.format("%1s", steps));

View File

@ -57,6 +57,7 @@ import android.widget.Toast;
import androidx.core.app.NavUtils; import androidx.core.app.NavUtils;
import androidx.core.app.NotificationCompat; import androidx.core.app.NotificationCompat;
import androidx.core.app.RemoteInput; import androidx.core.app.RemoteInput;
import androidx.core.content.FileProvider;
import androidx.localbroadcastmanager.content.LocalBroadcastManager; import androidx.localbroadcastmanager.content.LocalBroadcastManager;
import org.slf4j.Logger; import org.slf4j.Logger;
@ -681,10 +682,17 @@ public class DebugActivity extends AbstractGBActivity {
return; return;
} }
final Uri providerUri = FileProvider.getUriForFile(
this,
getApplicationContext().getPackageName() + ".screenshot_provider",
logFile
);
Intent emailIntent = new Intent(android.content.Intent.ACTION_SEND); Intent emailIntent = new Intent(android.content.Intent.ACTION_SEND);
emailIntent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
emailIntent.setType("*/*"); emailIntent.setType("*/*");
emailIntent.putExtra(EXTRA_SUBJECT, "Gadgetbridge log file"); emailIntent.putExtra(EXTRA_SUBJECT, "Gadgetbridge log file");
emailIntent.putExtra(Intent.EXTRA_STREAM, Uri.fromFile(logFile)); emailIntent.putExtra(Intent.EXTRA_STREAM, providerUri);
startActivity(Intent.createChooser(emailIntent, "Share File")); startActivity(Intent.createChooser(emailIntent, "Share File"));
} }
} }

View File

@ -198,6 +198,8 @@ public class DiscoveryActivity extends AbstractGBActivity implements AdapterView
} }
} }
}; };
int CallbackType = android.bluetooth.le.ScanSettings.CALLBACK_TYPE_ALL_MATCHES;
int MatchMode = android.bluetooth.le.ScanSettings.MATCH_MODE_STICKY;
public void logMessageContent(byte[] value) { public void logMessageContent(byte[] value) {
if (value != null) { if (value != null) {
@ -654,17 +656,17 @@ public class DiscoveryActivity extends AbstractGBActivity implements AdapterView
private ScanSettings getScanSettings() { private ScanSettings getScanSettings() {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
return new ScanSettings.Builder() return new ScanSettings.Builder()
.setCallbackType(android.bluetooth.le.ScanSettings.CALLBACK_TYPE_ALL_MATCHES) .setCallbackType(CallbackType)
.setScanMode(android.bluetooth.le.ScanSettings.SCAN_MODE_LOW_LATENCY) .setScanMode(android.bluetooth.le.ScanSettings.SCAN_MODE_LOW_LATENCY)
.setMatchMode(android.bluetooth.le.ScanSettings.MATCH_MODE_AGGRESSIVE) .setMatchMode(MatchMode)
.setPhy(android.bluetooth.le.ScanSettings.PHY_LE_ALL_SUPPORTED) .setPhy(android.bluetooth.le.ScanSettings.PHY_LE_ALL_SUPPORTED)
.setNumOfMatches(android.bluetooth.le.ScanSettings.MATCH_NUM_ONE_ADVERTISEMENT) .setNumOfMatches(android.bluetooth.le.ScanSettings.MATCH_NUM_ONE_ADVERTISEMENT)
.build(); .build();
} else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { } else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
return new ScanSettings.Builder() return new ScanSettings.Builder()
.setCallbackType(android.bluetooth.le.ScanSettings.CALLBACK_TYPE_ALL_MATCHES) .setCallbackType(CallbackType)
.setScanMode(android.bluetooth.le.ScanSettings.SCAN_MODE_LOW_LATENCY) .setScanMode(android.bluetooth.le.ScanSettings.SCAN_MODE_LOW_LATENCY)
.setMatchMode(android.bluetooth.le.ScanSettings.MATCH_MODE_AGGRESSIVE) .setMatchMode(MatchMode)
.setNumOfMatches(android.bluetooth.le.ScanSettings.MATCH_NUM_ONE_ADVERTISEMENT) .setNumOfMatches(android.bluetooth.le.ScanSettings.MATCH_NUM_ONE_ADVERTISEMENT)
.build(); .build();
} else { } else {
@ -910,6 +912,26 @@ public class DiscoveryActivity extends AbstractGBActivity implements AdapterView
if (oldBleScanning) { if (oldBleScanning) {
LOG.info("New BLE scanning disabled via settings, using old method"); LOG.info("New BLE scanning disabled via settings, using old method");
} }
int level = prefs.getInt("scanning_intensity", 1);
switch (level) {
case 0:
CallbackType = android.bluetooth.le.ScanSettings.CALLBACK_TYPE_FIRST_MATCH;
MatchMode = android.bluetooth.le.ScanSettings.MATCH_MODE_STICKY;
break;
case 1:
CallbackType = android.bluetooth.le.ScanSettings.CALLBACK_TYPE_FIRST_MATCH;
MatchMode = android.bluetooth.le.ScanSettings.MATCH_MODE_AGGRESSIVE;
break;
case 2:
CallbackType = android.bluetooth.le.ScanSettings.CALLBACK_TYPE_ALL_MATCHES;
MatchMode = android.bluetooth.le.ScanSettings.MATCH_MODE_STICKY;
break;
case 3:
CallbackType = android.bluetooth.le.ScanSettings.CALLBACK_TYPE_ALL_MATCHES;
MatchMode = android.bluetooth.le.ScanSettings.MATCH_MODE_AGGRESSIVE;
break;
}
LOG.debug("Device discovery - scanning level: " + level + " " + CallbackType + " " + MatchMode);
} }
@Override @Override

View File

@ -18,14 +18,33 @@
package nodomain.freeyourgadget.gadgetbridge.activities; package nodomain.freeyourgadget.gadgetbridge.activities;
import android.os.Bundle; import android.os.Bundle;
import android.preference.Preference;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import nodomain.freeyourgadget.gadgetbridge.GBApplication;
import nodomain.freeyourgadget.gadgetbridge.R; import nodomain.freeyourgadget.gadgetbridge.R;
import nodomain.freeyourgadget.gadgetbridge.util.Prefs;
public class DiscoveryPairingPreferenceActivity extends AbstractSettingsActivity { public class DiscoveryPairingPreferenceActivity extends AbstractSettingsActivity {
private static final Logger LOG = LoggerFactory.getLogger(AbstractSettingsActivity.class);
@Override @Override
protected void onCreate(Bundle savedInstanceState) { protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState); super.onCreate(savedInstanceState);
addPreferencesFromResource(R.xml.discovery_pairing_preferences); addPreferencesFromResource(R.xml.discovery_pairing_preferences);
final Prefs prefs = GBApplication.getPrefs();
final Preference pref = findPreference("scanning_intensity");
pref.setSummary(String.valueOf(prefs.getInt("scanning_intensity", 1)));
pref.setOnPreferenceChangeListener(new Preference.OnPreferenceChangeListener() {
@Override
public boolean onPreferenceChange(Preference preference, Object newVal) {
preference.setSummary(newVal.toString());
return true;
}
});
} }
@Override @Override
protected void onPostCreate(Bundle savedInstanceState) { protected void onPostCreate(Bundle savedInstanceState) {

View File

@ -141,7 +141,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, false, calendar); Alarm alarm = AlarmUtils.createSingleShot(0, true, true, 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);

View File

@ -144,7 +144,7 @@ public class MiBand5Coordinator extends HuamiCoordinator {
"id_ID", "id_ID",
"it_IT", "it_IT",
"nl_NL", "nl_NL",
"pt_PT", "pt_BR",
"pl_PL", "pl_PL",
"ro_RO", "ro_RO",
"ru_RU", "ru_RU",

View File

@ -956,33 +956,9 @@ public class FitProDeviceSupport extends AbstractBTLEDeviceSupport {
// this device doesn't have concept of on-off alarm, so use the last slot for this and store // this device doesn't have concept of on-off alarm, so use the last slot for this and store
// this alarm in the database so the user knows what is going on and can disable it // this alarm in the database so the user knows what is going on and can disable it
if (alarms.toArray().length == 1 && alarms.get(0).getRepetition() == 0) { if (alarms.toArray().length == 1 && alarms.get(0).getRepetition() == 0) { //single shot?
Alarm oneshot = alarms.get(0); Alarm oneshot = alarms.get(0);
DBHandler db = null; alarms = (ArrayList<? extends Alarm>) AlarmUtils.mergeOneshotToDeviceAlarms(gbDevice, (nodomain.freeyourgadget.gadgetbridge.entities.Alarm) oneshot, 7);
try {
db = GBApplication.acquireDB();
DaoSession daoSession = db.getDaoSession();
Device device = DBHelper.getDevice(gbDevice, daoSession);
User user = DBHelper.getUser(daoSession);
nodomain.freeyourgadget.gadgetbridge.entities.Alarm tmpAlarm =
new nodomain.freeyourgadget.gadgetbridge.entities.Alarm(
device.getId(),
user.getId(),
7,
true,
false,
false,
0,
oneshot.getHour(),
oneshot.getMinute(),
true, //kind of indicate the specialty of this alarm
"",
"");
daoSession.insertOrReplace(tmpAlarm);
GBApplication.releaseDB();
} catch (GBException e) {
LOG.error("error storing one shot quick alarm");
}
} }
try { try {

View File

@ -78,6 +78,7 @@ import nodomain.freeyourgadget.gadgetbridge.service.devices.qhybrid.requests.mis
import nodomain.freeyourgadget.gadgetbridge.service.devices.qhybrid.requests.misfit.ReleaseHandsControlRequest; import nodomain.freeyourgadget.gadgetbridge.service.devices.qhybrid.requests.misfit.ReleaseHandsControlRequest;
import nodomain.freeyourgadget.gadgetbridge.service.devices.qhybrid.requests.misfit.RequestHandControlRequest; import nodomain.freeyourgadget.gadgetbridge.service.devices.qhybrid.requests.misfit.RequestHandControlRequest;
import nodomain.freeyourgadget.gadgetbridge.service.devices.qhybrid.requests.misfit.SaveCalibrationRequest; import nodomain.freeyourgadget.gadgetbridge.service.devices.qhybrid.requests.misfit.SaveCalibrationRequest;
import nodomain.freeyourgadget.gadgetbridge.util.AlarmUtils;
import nodomain.freeyourgadget.gadgetbridge.util.FileUtils; import nodomain.freeyourgadget.gadgetbridge.util.FileUtils;
import nodomain.freeyourgadget.gadgetbridge.util.GB; import nodomain.freeyourgadget.gadgetbridge.util.GB;
import nodomain.freeyourgadget.gadgetbridge.util.UriHelper; import nodomain.freeyourgadget.gadgetbridge.util.UriHelper;
@ -516,6 +517,16 @@ public class FossilWatchAdapter extends WatchAdapter {
@Override @Override
public void onSetAlarms(ArrayList<? extends Alarm> alarms) { public void onSetAlarms(ArrayList<? extends Alarm> alarms) {
// handle one-shot alarm from the widget:
// this device doesn't have concept of on-off alarm, so use the last slot for this and store
// this alarm in the database so the user knows what is going on and can disable it
if (alarms.toArray().length == 1 && alarms.get(0).getRepetition() == 0) { //single shot?
Alarm oneshot = alarms.get(0);
alarms = (ArrayList<? extends Alarm>) AlarmUtils.mergeOneshotToDeviceAlarms(getDeviceSupport().getDevice(), (nodomain.freeyourgadget.gadgetbridge.entities.Alarm) oneshot, 5);
}
// throw new RuntimeException("noope"); // throw new RuntimeException("noope");
ArrayList<nodomain.freeyourgadget.gadgetbridge.service.devices.qhybrid.requests.fossil.alarm.Alarm> activeAlarms = new ArrayList<>(); ArrayList<nodomain.freeyourgadget.gadgetbridge.service.devices.qhybrid.requests.fossil.alarm.Alarm> activeAlarms = new ArrayList<>();
for (Alarm alarm : alarms) { for (Alarm alarm : alarms) {

View File

@ -1380,7 +1380,9 @@ public class FossilHRWatchAdapter extends FossilWatchAdapter {
outerLoop: outerLoop:
for (ButtonConfiguration config : configs) { for (ButtonConfiguration config : configs) {
for (ApplicationInformation installedApp : installedApplications) { for (ApplicationInformation installedApp : installedApplications) {
if (installedApp.getAppName().equals(config.getAction())) { if (installedApp.getAppName().equals(config.getAction()) ||
config.getAction().equals("workoutApp") //workoutApp is part of internal firmware
) {
availableConfigs.add(config); availableConfigs.add(config);
continue outerLoop; continue outerLoop;
} }

View File

@ -28,6 +28,7 @@ import java.util.List;
import java.util.Set; import java.util.Set;
import nodomain.freeyourgadget.gadgetbridge.GBApplication; import nodomain.freeyourgadget.gadgetbridge.GBApplication;
import nodomain.freeyourgadget.gadgetbridge.R;
import nodomain.freeyourgadget.gadgetbridge.database.DBHandler; import nodomain.freeyourgadget.gadgetbridge.database.DBHandler;
import nodomain.freeyourgadget.gadgetbridge.database.DBHelper; import nodomain.freeyourgadget.gadgetbridge.database.DBHelper;
import nodomain.freeyourgadget.gadgetbridge.devices.miband.MiBandConst; import nodomain.freeyourgadget.gadgetbridge.devices.miband.MiBandConst;
@ -50,7 +51,7 @@ public class AlarmUtils {
* @return * @return
*/ */
public static nodomain.freeyourgadget.gadgetbridge.model.Alarm createSingleShot(int index, boolean smartWakeup, boolean snooze, 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, snooze, Alarm.ALARM_ONCE, calendar.get(Calendar.HOUR_OF_DAY), calendar.get(Calendar.MINUTE), false, null, null); return new Alarm(-1, -1, index, true, smartWakeup, snooze, Alarm.ALARM_ONCE, calendar.get(Calendar.HOUR_OF_DAY), calendar.get(Calendar.MINUTE), false, GBApplication.getContext().getString(R.string.quick_alarm), GBApplication.getContext().getString(R.string.quick_alarm_description));
} }
/** /**
@ -144,4 +145,23 @@ public class AlarmUtils {
} }
}; };
} }
public static List<Alarm> mergeOneshotToDeviceAlarms(GBDevice gbDevice, Alarm oneshot, int position) {
List<Alarm> all_alarms = new ArrayList<>();
try {
DBHandler db = GBApplication.acquireDB();
DaoSession daoSession = db.getDaoSession();
Device device = DBHelper.getDevice(gbDevice, daoSession);
User user = DBHelper.getUser(daoSession);
oneshot.setPosition(position);
oneshot.setDeviceId(device.getId());
oneshot.setUserId(user.getId());
daoSession.insertOrReplace(oneshot);
all_alarms = DBHelper.getAlarms(gbDevice);
GBApplication.releaseDB();
} catch (Exception e) {
GB.log("error storing one shot quick alarm", GB.ERROR, e);
}
return all_alarms;
}
} }

View File

@ -405,6 +405,7 @@
<item>@string/menuitem_status</item> <item>@string/menuitem_status</item>
<item>@string/menuitem_pai</item> <item>@string/menuitem_pai</item>
<item>@string/menuitem_hr</item> <item>@string/menuitem_hr</item>
<item>@string/menuitem_breathing</item>
<item>@string/menuitem_stress</item> <item>@string/menuitem_stress</item>
<item>@string/menuitem_eventreminder</item> <item>@string/menuitem_eventreminder</item>
<item>@string/menuitem_dnd</item> <item>@string/menuitem_dnd</item>
@ -427,6 +428,7 @@
<item>@string/p_menuitem_status</item> <item>@string/p_menuitem_status</item>
<item>@string/p_menuitem_pai</item> <item>@string/p_menuitem_pai</item>
<item>@string/p_menuitem_hr</item> <item>@string/p_menuitem_hr</item>
<item>@string/p_menuitem_breathing</item>
<item>@string/p_menuitem_stress</item> <item>@string/p_menuitem_stress</item>
<item>@string/p_menuitem_eventreminder</item> <item>@string/p_menuitem_eventreminder</item>
<item>@string/p_menuitem_dnd</item> <item>@string/p_menuitem_dnd</item>
@ -726,6 +728,7 @@
<item>@string/menuitem_pai</item> <item>@string/menuitem_pai</item>
<item>@string/menuitem_hr</item> <item>@string/menuitem_hr</item>
<item>@string/menuitem_spo2</item> <item>@string/menuitem_spo2</item>
<item>@string/menuitem_breathing</item>
<item>@string/menuitem_stress</item> <item>@string/menuitem_stress</item>
<item>@string/menuitem_eventreminder</item> <item>@string/menuitem_eventreminder</item>
<item>@string/menuitem_dnd</item> <item>@string/menuitem_dnd</item>
@ -749,6 +752,7 @@
<item>@string/p_menuitem_pai</item> <item>@string/p_menuitem_pai</item>
<item>@string/p_menuitem_hr</item> <item>@string/p_menuitem_hr</item>
<item>@string/p_menuitem_spo2</item> <item>@string/p_menuitem_spo2</item>
<item>@string/p_menuitem_breathing</item>
<item>@string/p_menuitem_stress</item> <item>@string/p_menuitem_stress</item>
<item>@string/p_menuitem_eventreminder</item> <item>@string/p_menuitem_eventreminder</item>
<item>@string/p_menuitem_dnd</item> <item>@string/p_menuitem_dnd</item>
@ -1431,7 +1435,8 @@
<item name="de_DE">@string/german</item> <item name="de_DE">@string/german</item>
<item name="it_IT">@string/italian</item> <item name="it_IT">@string/italian</item>
<item name="fr_FR">@string/french</item> <item name="fr_FR">@string/french</item>
<item name="pt_BR">@string/portuguese</item> <item name="pt_BR">@string/portuguese_br</item>
<item name="pt_PT">@string/portuguese_pt</item>
<item name="nl_NL">@string/dutch</item> <item name="nl_NL">@string/dutch</item>
<item name="pl_PL">@string/polish</item> <item name="pl_PL">@string/polish</item>
<item name="cs_CZ">@string/czesh</item> <item name="cs_CZ">@string/czesh</item>
@ -1458,6 +1463,7 @@
<item>it_IT</item> <item>it_IT</item>
<item>fr_FR</item> <item>fr_FR</item>
<item>pt_BR</item> <item>pt_BR</item>
<item>pt_PT</item>
<item>nl_NL</item> <item>nl_NL</item>
<item>pl_PL</item> <item>pl_PL</item>
<item>cs_CZ</item> <item>cs_CZ</item>

View File

@ -387,6 +387,8 @@
<string name="pref_auto_fetch_limit_fetches">Minimum time between fetches</string> <string name="pref_auto_fetch_limit_fetches">Minimum time between fetches</string>
<string name="pref_auto_fetch_limit_fetches_summary">Fetches every %d minutes</string> <string name="pref_auto_fetch_limit_fetches_summary">Fetches every %d minutes</string>
<!-- developer/debug preferences--> <!-- developer/debug preferences-->
<string name="discovery_scanning_intensity">Scanning intensity</string>
<string name="discovery_scanning_intensity_warning">If you experience freezing or unresponsiveness, try to set Scanning intensity to lower level. If your device is not being discovered, try to set Scanning intensity to higher level.</string>
<string name="pref_disable_new_ble_scanning">Disable new BLE scanning</string> <string name="pref_disable_new_ble_scanning">Disable new BLE scanning</string>
<string name="pref_summary_disable_new_ble_scanning">Check this option if your device cannot be found during discovery</string> <string name="pref_summary_disable_new_ble_scanning">Check this option if your device cannot be found during discovery</string>
<string name="not_connected">Not connected</string> <string name="not_connected">Not connected</string>
@ -755,6 +757,8 @@
<string name="thai">Thai</string> <string name="thai">Thai</string>
<string name="vietnamese">Vietnamese</string> <string name="vietnamese">Vietnamese</string>
<string name="portuguese">Portuguese</string> <string name="portuguese">Portuguese</string>
<string name="portuguese_br">Portuguese (Brazil)</string>
<string name="portuguese_pt">Portuguese (Portugal)</string>
<string name="romanian">Romanian</string> <string name="romanian">Romanian</string>
<string name="hungarian">Hungarian</string> <string name="hungarian">Hungarian</string>
<string name="greek">Greek</string> <string name="greek">Greek</string>
@ -1131,6 +1135,8 @@
<string name="widget_10_minutes">10 minutes</string> <string name="widget_10_minutes">10 minutes</string>
<string name="widget_20_minutes">20 minutes</string> <string name="widget_20_minutes">20 minutes</string>
<string name="widget_1_hour">1 hour</string> <string name="widget_1_hour">1 hour</string>
<string name="quick_alarm">Quick alarm</string>
<string name="quick_alarm_description">Alarm from widget</string>
<string name="icon_placeholder" translatable="false">Icon</string> <string name="icon_placeholder" translatable="false">Icon</string>
<string name="watch_not_connected">Watch not connected</string> <string name="watch_not_connected">Watch not connected</string>
<string name="qhybrid_vibration_strength">vibration strength:</string> <string name="qhybrid_vibration_strength">vibration strength:</string>

View File

@ -1,5 +1,6 @@
<?xml version="1.0" encoding="utf-8"?> <?xml version="1.0" encoding="utf-8"?>
<PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android"> <PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto">
<PreferenceCategory <PreferenceCategory
android:key="pref_discovery_pairing" android:key="pref_discovery_pairing"
android:title="@string/activity_prefs_discovery_pairing"> android:title="@string/activity_prefs_discovery_pairing">
@ -27,5 +28,17 @@
android:layout="@layout/preference_checkbox" android:layout="@layout/preference_checkbox"
android:summary="@string/discover_unsupported_devices_description" android:summary="@string/discover_unsupported_devices_description"
android:title="@string/discover_unsupported_devices" /> android:title="@string/discover_unsupported_devices" />
<SeekBarPreference
android:key="scanning_intensity"
android:max="3"
android:min="0"
android:title="@string/discovery_scanning_intensity"
android:defaultValue="1"
app:showSeekBarValue="true" />
<Preference
android:selectable="false"
android:persistent="false"
android:summary="@string/discovery_scanning_intensity_warning" />
</PreferenceCategory> </PreferenceCategory>
</PreferenceScreen> </PreferenceScreen>