
org.robolectric.shadows.ShadowSettings Maven / Gradle / Ivy
Show all versions of shadows-framework Show documentation
package org.robolectric.shadows;
import static android.os.Build.VERSION_CODES.M;
import static android.os.Build.VERSION_CODES.P;
import static android.os.Build.VERSION_CODES.Q;
import static android.os.Build.VERSION_CODES.R;
import static android.os.Build.VERSION_CODES.TIRAMISU;
import static android.provider.Settings.Secure.LOCATION_MODE_OFF;
import static org.robolectric.util.reflector.Reflector.reflector;
import android.content.ContentResolver;
import android.content.Context;
import android.content.Intent;
import android.location.LocationManager;
import android.provider.Settings;
import android.provider.Settings.Secure;
import android.provider.Settings.SettingNotFoundException;
import android.text.TextUtils;
import com.google.common.collect.ImmutableMap;
import java.util.Arrays;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Optional;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import org.robolectric.RuntimeEnvironment;
import org.robolectric.annotation.Implementation;
import org.robolectric.annotation.Implements;
import org.robolectric.annotation.Resetter;
import org.robolectric.util.reflector.ForType;
import org.robolectric.util.reflector.Static;
import org.robolectric.versioning.AndroidVersions.U;
@SuppressWarnings({"UnusedDeclaration"})
@Implements(Settings.class)
public class ShadowSettings {
@Implements(value = Settings.System.class)
public static class ShadowSystem {
private static final ImmutableMap> DEFAULTS =
ImmutableMap.>builder()
.put(Settings.System.ANIMATOR_DURATION_SCALE, Optional.of(1))
.build();
private static final Map> settings = new ConcurrentHashMap<>(DEFAULTS);
@Implementation
protected static boolean putInt(ContentResolver cr, String name, int value) {
return put(cr, name, value);
}
@Implementation
protected static int getInt(ContentResolver cr, String name, int def) {
return get(Integer.class, name).orElse(def);
}
@Implementation
protected static int getInt(ContentResolver cr, String name) throws SettingNotFoundException {
return get(Integer.class, name).orElseThrow(() -> new SettingNotFoundException(name));
}
@Implementation
protected static boolean putString(ContentResolver cr, String name, String value) {
return put(cr, name, value);
}
@Implementation
protected static String getString(ContentResolver cr, String name) {
return get(String.class, name).orElse(null);
}
@Implementation
protected static String getStringForUser(ContentResolver cr, String name, int userHandle) {
return get(String.class, name).orElse(null);
}
@Implementation
protected static boolean putLong(ContentResolver cr, String name, long value) {
return put(cr, name, value);
}
@Implementation
protected static long getLong(ContentResolver cr, String name, long def) {
return get(Long.class, name).orElse(def);
}
@Implementation
protected static long getLong(ContentResolver cr, String name) throws SettingNotFoundException {
return get(Long.class, name).orElseThrow(() -> new SettingNotFoundException(name));
}
@Implementation
protected static boolean putFloat(ContentResolver cr, String name, float value) {
boolean result = put(cr, name, value);
if (Settings.System.WINDOW_ANIMATION_SCALE.equals(name)) {
ShadowValueAnimator.setDurationScale(value);
}
return result;
}
@Implementation
protected static float getFloat(ContentResolver cr, String name, float def) {
return get(Float.class, name).orElse(def);
}
@Implementation
protected static float getFloat(ContentResolver cr, String name)
throws SettingNotFoundException {
return get(Float.class, name).orElseThrow(() -> new SettingNotFoundException(name));
}
private static boolean put(ContentResolver cr, String name, Object value) {
if (!Objects.equals(
settings.put(name, Optional.ofNullable(value)), Optional.ofNullable(value))) {
if (cr != null) {
cr.notifyChange(Settings.System.getUriFor(name), null);
}
}
return true;
}
private static Optional get(Class type, String name) {
return settings.getOrDefault(name, Optional.empty()).filter(type::isInstance).map(type::cast);
}
@Resetter
public static void reset() {
settings.clear();
settings.putAll(DEFAULTS);
}
}
@Implements(value = Settings.Secure.class)
public static class ShadowSecure {
private static final HashMap> SECURE_DEFAULTS = new HashMap<>();
// source of truth for initial location state
static final boolean INITIAL_GPS_PROVIDER_STATE = true;
static final boolean INITIAL_NETWORK_PROVIDER_STATE = false;
static {
if (INITIAL_GPS_PROVIDER_STATE && INITIAL_NETWORK_PROVIDER_STATE) {
SECURE_DEFAULTS.put(Secure.LOCATION_MODE, Optional.of(Secure.LOCATION_MODE_HIGH_ACCURACY));
SECURE_DEFAULTS.put(Secure.LOCATION_PROVIDERS_ALLOWED, Optional.of("gps,network"));
} else if (INITIAL_GPS_PROVIDER_STATE) {
SECURE_DEFAULTS.put(Secure.LOCATION_MODE, Optional.of(Secure.LOCATION_MODE_SENSORS_ONLY));
SECURE_DEFAULTS.put(Secure.LOCATION_PROVIDERS_ALLOWED, Optional.of("gps"));
} else if (INITIAL_NETWORK_PROVIDER_STATE) {
SECURE_DEFAULTS.put(Secure.LOCATION_MODE, Optional.of(Secure.LOCATION_MODE_BATTERY_SAVING));
SECURE_DEFAULTS.put(Secure.LOCATION_PROVIDERS_ALLOWED, Optional.of("network"));
} else {
SECURE_DEFAULTS.put(Secure.LOCATION_MODE, Optional.of(LOCATION_MODE_OFF));
}
}
private static final Map> dataMap =
new ConcurrentHashMap<>(SECURE_DEFAULTS);
@Implementation(maxSdk = P)
@SuppressWarnings("robolectric.ShadowReturnTypeMismatch")
protected static boolean setLocationProviderEnabledForUser(
ContentResolver cr, String provider, boolean enabled, int uid) {
return updateEnabledProviders(cr, provider, enabled);
}
// only for use locally and by ShadowLocationManager, which requires a tight integration with
// ShadowSettings due to historical weirdness between LocationManager and Settings.
static boolean updateEnabledProviders(ContentResolver cr, String provider, boolean enabled) {
Set providers = new HashSet<>();
String oldProviders =
Settings.Secure.getString(cr, Settings.Secure.LOCATION_PROVIDERS_ALLOWED);
if (!TextUtils.isEmpty(oldProviders)) {
providers.addAll(Arrays.asList(oldProviders.split(",")));
}
if (enabled == oldProviders.contains(provider)) {
return true;
}
if (enabled) {
providers.add(provider);
} else {
providers.remove(provider);
}
String newProviders = TextUtils.join(",", providers.toArray());
boolean r =
Settings.Secure.putString(cr, Settings.Secure.LOCATION_PROVIDERS_ALLOWED, newProviders);
Intent providersBroadcast = new Intent(LocationManager.PROVIDERS_CHANGED_ACTION);
if (RuntimeEnvironment.getApiLevel() >= Q) {
providersBroadcast.putExtra(LocationManager.EXTRA_PROVIDER_NAME, provider);
}
if (RuntimeEnvironment.getApiLevel() >= R) {
providersBroadcast.putExtra(LocationManager.EXTRA_PROVIDER_ENABLED, enabled);
}
RuntimeEnvironment.getApplication().sendBroadcast(providersBroadcast);
return r;
}
@Implementation
protected static boolean putInt(ContentResolver cr, String name, int value) {
boolean changed = !Objects.equals(dataMap.put(name, Optional.of(value)), Optional.of(value));
if (Settings.Secure.LOCATION_MODE.equals(name)) {
if (RuntimeEnvironment.getApiLevel() <= P) {
// do this after setting location mode but before invoking contentobservers, so that
// observers for both settings will see the correct values
boolean gps =
(value == Settings.Secure.LOCATION_MODE_SENSORS_ONLY
|| value == Settings.Secure.LOCATION_MODE_HIGH_ACCURACY);
boolean network =
(value == Settings.Secure.LOCATION_MODE_BATTERY_SAVING
|| value == Settings.Secure.LOCATION_MODE_HIGH_ACCURACY);
Settings.Secure.setLocationProviderEnabled(cr, LocationManager.GPS_PROVIDER, gps);
Settings.Secure.setLocationProviderEnabled(cr, LocationManager.NETWORK_PROVIDER, network);
}
Intent modeBroadcast = new Intent(LocationManager.MODE_CHANGED_ACTION);
if (RuntimeEnvironment.getApiLevel() >= R) {
modeBroadcast.putExtra(
LocationManager.EXTRA_LOCATION_ENABLED, value != LOCATION_MODE_OFF);
}
RuntimeEnvironment.getApplication().sendBroadcast(modeBroadcast);
}
if (changed && cr != null) {
cr.notifyChange(Settings.Secure.getUriFor(name), null);
}
return true;
}
@Implementation
protected static boolean putIntForUser(
ContentResolver cr, String name, int value, int userHandle) {
putInt(cr, name, value);
return true;
}
@Implementation
protected static int getIntForUser(ContentResolver cr, String name, int def, int userHandle) {
// ignore userhandle
return getInt(cr, name, def);
}
@Implementation
protected static int getIntForUser(ContentResolver cr, String name, int userHandle)
throws SettingNotFoundException {
// ignore userhandle
return getInt(cr, name);
}
@Implementation
protected static int getInt(ContentResolver cr, String name) throws SettingNotFoundException {
if (Settings.Secure.LOCATION_MODE.equals(name) && RuntimeEnvironment.getApiLevel() < P) {
// Map from to underlying location provider storage API to location mode
return reflector(SettingsSecureReflector.class).getLocationModeForUser(cr, 0);
}
return get(Integer.class, name).orElseThrow(() -> new SettingNotFoundException(name));
}
@Implementation
protected static int getInt(ContentResolver cr, String name, int def) {
if (Settings.Secure.LOCATION_MODE.equals(name) && RuntimeEnvironment.getApiLevel() < P) {
// Map from to underlying location provider storage API to location mode
return reflector(SettingsSecureReflector.class).getLocationModeForUser(cr, 0);
}
return get(Integer.class, name).orElse(def);
}
@Implementation
protected static boolean putString(ContentResolver cr, String name, String value) {
return put(cr, name, value);
}
@Implementation
protected static String getString(ContentResolver cr, String name) {
return get(String.class, name).orElse(null);
}
@Implementation
protected static String getStringForUser(ContentResolver cr, String name, int userHandle) {
return getString(cr, name);
}
@Implementation
protected static boolean putLong(ContentResolver cr, String name, long value) {
return put(cr, name, value);
}
@Implementation
protected static long getLong(ContentResolver cr, String name, long def) {
return get(Long.class, name).orElse(def);
}
@Implementation
protected static long getLong(ContentResolver cr, String name) throws SettingNotFoundException {
return get(Long.class, name).orElseThrow(() -> new SettingNotFoundException(name));
}
@Implementation
protected static boolean putFloat(ContentResolver cr, String name, float value) {
return put(cr, name, value);
}
@Implementation
protected static float getFloat(ContentResolver cr, String name, float def) {
return get(Float.class, name).orElse(def);
}
@Implementation
protected static float getFloat(ContentResolver cr, String name)
throws SettingNotFoundException {
return get(Float.class, name).orElseThrow(() -> new SettingNotFoundException(name));
}
private static boolean put(ContentResolver cr, String name, Object value) {
if (!Objects.equals(
dataMap.put(name, Optional.ofNullable(value)), Optional.ofNullable(value))) {
if (cr != null) {
cr.notifyChange(Settings.Secure.getUriFor(name), null);
}
}
return true;
}
private static Optional get(Class type, String name) {
return dataMap.getOrDefault(name, Optional.empty()).filter(type::isInstance).map(type::cast);
}
@Resetter
public static void reset() {
dataMap.clear();
dataMap.putAll(SECURE_DEFAULTS);
}
}
@Implements(value = Settings.Global.class)
public static class ShadowGlobal {
private static final ImmutableMap> DEFAULTS =
ImmutableMap.>builder()
.put(Settings.Global.ANIMATOR_DURATION_SCALE, Optional.of(1))
.build();
private static final Map> settings = new ConcurrentHashMap<>(DEFAULTS);
@Implementation
protected static boolean putInt(ContentResolver cr, String name, int value) {
return put(cr, name, value);
}
@Implementation
protected static int getInt(ContentResolver cr, String name, int def) {
return get(Integer.class, name).orElse(def);
}
@Implementation
protected static int getInt(ContentResolver cr, String name) throws SettingNotFoundException {
return get(Integer.class, name).orElseThrow(() -> new SettingNotFoundException(name));
}
@Implementation
protected static boolean putString(ContentResolver cr, String name, String value) {
return put(cr, name, value);
}
@Implementation
protected static String getString(ContentResolver cr, String name) {
return get(String.class, name).orElse(null);
}
@Implementation
protected static String getStringForUser(ContentResolver cr, String name, int userHandle) {
return getString(cr, name);
}
@Implementation
protected static boolean putLong(ContentResolver cr, String name, long value) {
return put(cr, name, value);
}
@Implementation
protected static long getLong(ContentResolver cr, String name, long def) {
return get(Long.class, name).orElse(def);
}
@Implementation
protected static long getLong(ContentResolver cr, String name) throws SettingNotFoundException {
return get(Long.class, name).orElseThrow(() -> new SettingNotFoundException(name));
}
@Implementation
protected static boolean putFloat(ContentResolver cr, String name, float value) {
boolean result = put(cr, name, value);
if (Settings.Global.ANIMATOR_DURATION_SCALE.equals(name)) {
ShadowValueAnimator.setDurationScale(value);
}
return result;
}
@Implementation
protected static float getFloat(ContentResolver cr, String name, float def) {
return get(Float.class, name).orElse(def);
}
@Implementation
protected static float getFloat(ContentResolver cr, String name)
throws SettingNotFoundException {
return get(Float.class, name).orElseThrow(() -> new SettingNotFoundException(name));
}
private static boolean put(ContentResolver cr, String name, Object value) {
if (!Objects.equals(
settings.put(name, Optional.ofNullable(value)), Optional.ofNullable(value))) {
if (cr != null) {
cr.notifyChange(Settings.Global.getUriFor(name), null);
}
}
return true;
}
private static Optional get(Class type, String name) {
return settings.getOrDefault(name, Optional.empty()).filter(type::isInstance).map(type::cast);
}
@Resetter
public static void reset() {
settings.clear();
settings.putAll(DEFAULTS);
}
}
/**
* Sets the value of the {@link Settings.System#AIRPLANE_MODE_ON} setting.
*
* @param isAirplaneMode new status for airplane mode
*/
public static void setAirplaneMode(boolean isAirplaneMode) {
Settings.Global.putInt(
RuntimeEnvironment.getApplication().getContentResolver(),
Settings.Global.AIRPLANE_MODE_ON,
isAirplaneMode ? 1 : 0);
Settings.System.putInt(
RuntimeEnvironment.getApplication().getContentResolver(),
Settings.System.AIRPLANE_MODE_ON,
isAirplaneMode ? 1 : 0);
}
/**
* Non-Android accessor that allows the value of the WIFI_ON setting to be set.
*
* @param isOn new status for wifi mode
*/
public static void setWifiOn(boolean isOn) {
Settings.Global.putInt(
RuntimeEnvironment.getApplication().getContentResolver(),
Settings.Global.WIFI_ON,
isOn ? 1 : 0);
Settings.System.putInt(
RuntimeEnvironment.getApplication().getContentResolver(),
Settings.System.WIFI_ON,
isOn ? 1 : 0);
}
/**
* Sets the value of the {@link Settings.System#TIME_12_24} setting.
*
* @param use24HourTimeFormat new status for the time setting
*/
public static void set24HourTimeFormat(boolean use24HourTimeFormat) {
Settings.System.putString(
RuntimeEnvironment.getApplication().getContentResolver(),
Settings.System.TIME_12_24,
use24HourTimeFormat ? "24" : "12");
}
private static boolean canDrawOverlays = false;
/**
* @return false by default, or the value specified via {@link #setCanDrawOverlays(boolean)}
*/
@Implementation(minSdk = M)
protected static boolean canDrawOverlays(Context context) {
return canDrawOverlays;
}
/** Sets the value returned by {@link #canDrawOverlays(Context)}. */
public static void setCanDrawOverlays(boolean canDrawOverlays) {
ShadowSettings.canDrawOverlays = canDrawOverlays;
}
/**
* Sets the value of the {@link Settings.Global#ADB_ENABLED} setting or {@link
* Settings.Secure#ADB_ENABLED} depending on API level.
*
* @param adbEnabled new value for whether adb is enabled
*/
public static void setAdbEnabled(boolean adbEnabled) {
Settings.Global.putInt(
RuntimeEnvironment.getApplication().getContentResolver(),
Settings.Global.ADB_ENABLED,
adbEnabled ? 1 : 0);
// Support all clients by always setting the Secure version of the setting
Settings.Secure.putInt(
RuntimeEnvironment.getApplication().getContentResolver(),
Settings.Secure.ADB_ENABLED,
adbEnabled ? 1 : 0);
}
/**
* Sets the value of the {@link Settings.Global#INSTALL_NON_MARKET_APPS} setting or {@link
* Settings.Secure#INSTALL_NON_MARKET_APPS} depending on API level.
*
* @param installNonMarketApps new value for whether non-market apps are allowed to be installed
*/
public static void setInstallNonMarketApps(boolean installNonMarketApps) {
// This setting moved from Secure to Global in JELLY_BEAN_MR1 and then moved it back to Global
// in LOLLIPOP. Support all clients by always setting this field on all versions >=
// JELLY_BEAN_MR1.
Settings.Global.putInt(
RuntimeEnvironment.getApplication().getContentResolver(),
Settings.Global.INSTALL_NON_MARKET_APPS,
installNonMarketApps ? 1 : 0);
// Always set the Secure version of the setting
Settings.Secure.putInt(
RuntimeEnvironment.getApplication().getContentResolver(),
Settings.Secure.INSTALL_NON_MARKET_APPS,
installNonMarketApps ? 1 : 0);
}
public static void setLockScreenShowNotifications(boolean lockScreenShowNotifications) {
Settings.Secure.putInt(
RuntimeEnvironment.getApplication().getContentResolver(),
Settings.Secure.LOCK_SCREEN_SHOW_NOTIFICATIONS,
lockScreenShowNotifications ? 1 : 0);
}
public static void setLockScreenAllowPrivateNotifications(
boolean lockScreenAllowPrivateNotifications) {
Settings.Secure.putInt(
RuntimeEnvironment.getApplication().getContentResolver(),
Settings.Secure.LOCK_SCREEN_ALLOW_PRIVATE_NOTIFICATIONS,
lockScreenAllowPrivateNotifications ? 1 : 0);
}
/**
* Shadow for {@link Settings.Config}.
*
* This shadow is primarily to support {@link android.provider.DeviceConfig}, which queries
* {@link Settings.Config}. {@link android.provider.DeviceConfig} is pure Java code so it's not
* necessary to shadow that directly.
*/
@Implements(value = Settings.Config.class, isInAndroidSdk = false, minSdk = Q)
public static class ShadowConfig {
private static final Map settings = new ConcurrentHashMap<>();
@Implementation(maxSdk = Q)
protected static boolean putString(
ContentResolver cr, String name, String value, boolean makeDefault) {
return put(name, value);
}
@Implementation(minSdk = R, maxSdk = TIRAMISU)
protected static boolean putString(
ContentResolver cr, String namespace, String name, String value, boolean makeDefault) {
String key = reflector(SettingsConfigReflector.class).createCompositeName(namespace, name);
return put(key, value);
}
@Implementation(minSdk = U.SDK_INT)
protected static boolean putString(
String namespace, String name, String value, boolean makeDefault) {
String key = reflector(SettingsConfigReflector.class).createCompositeName(namespace, name);
return put(key, value);
}
@Implementation(maxSdk = TIRAMISU)
protected static String getString(ContentResolver cr, String name) {
return get(name);
}
@Implementation(minSdk = U.SDK_INT)
protected static String getString(String name) {
return get(name);
}
@Implementation(minSdk = R)
protected static Map getStrings(
ContentResolver resolver, String namespace, List names) {
Map result = new HashMap<>();
for (Map.Entry entry : settings.entrySet()) {
String key = entry.getKey();
if (!key.startsWith(namespace + "/")) {
continue;
}
String keyWithoutNamespace = key.substring(namespace.length() + 1);
if (names == null || names.isEmpty()) {
result.put(keyWithoutNamespace, entry.getValue());
} else if (names.contains(keyWithoutNamespace)) {
result.put(keyWithoutNamespace, entry.getValue());
}
}
return ImmutableMap.copyOf(result);
}
private static boolean put(String name, String value) {
settings.put(name, value);
return true;
}
@Implementation(minSdk = R)
protected static boolean setStrings(
ContentResolver cr, String namespace, Map keyValues) {
synchronized (settings) {
settings.entrySet().removeIf(entry -> entry.getKey().startsWith(namespace + "/"));
for (Map.Entry entry : keyValues.entrySet()) {
String key =
reflector(SettingsConfigReflector.class)
.createCompositeName(namespace, entry.getKey());
put(key, entry.getValue());
}
}
return true;
}
@Implementation(minSdk = U.SDK_INT)
protected static boolean deleteString(String namespace, String name) {
String key = reflector(SettingsConfigReflector.class).createCompositeName(namespace, name);
settings.remove(key);
return true;
}
@Implementation(minSdk = TIRAMISU, maxSdk = TIRAMISU)
protected static boolean deleteString(ContentResolver resolver, String namespace, String name) {
return deleteString(namespace, name);
}
private static String get(String name) {
return settings.get(name);
}
@Resetter
public static void reset() {
settings.clear();
}
}
@Resetter
public static void reset() {
canDrawOverlays = false;
}
@ForType(Settings.Secure.class)
interface SettingsSecureReflector {
@Static
int getLocationModeForUser(ContentResolver cr, int userId);
}
@ForType(Settings.Config.class)
interface SettingsConfigReflector {
@Static
String createCompositeName(String namespace, String name);
}
}