Merge "Update the behavior of the Dark theme slice" into qt-qpr1-dev
This commit is contained in:
committed by
Android (Google) Code Review
commit
bc96cf54cd
@@ -17,15 +17,21 @@ package com.android.settings.homepage.contextualcards.slices;
|
|||||||
|
|
||||||
import static androidx.slice.builders.ListBuilder.ICON_IMAGE;
|
import static androidx.slice.builders.ListBuilder.ICON_IMAGE;
|
||||||
|
|
||||||
|
import static android.provider.Settings.Global.LOW_POWER_MODE;
|
||||||
|
|
||||||
import android.annotation.ColorInt;
|
import android.annotation.ColorInt;
|
||||||
import android.app.PendingIntent;
|
import android.app.PendingIntent;
|
||||||
import android.app.UiModeManager;
|
import android.app.UiModeManager;
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
|
import android.content.res.Configuration;
|
||||||
|
import android.database.ContentObserver;
|
||||||
import android.net.Uri;
|
import android.net.Uri;
|
||||||
import android.os.BatteryManager;
|
import android.os.BatteryManager;
|
||||||
import android.os.Handler;
|
import android.os.Handler;
|
||||||
import android.os.Looper;
|
import android.os.Looper;
|
||||||
|
import android.os.PowerManager;
|
||||||
|
import android.provider.Settings;
|
||||||
import android.util.Log;
|
import android.util.Log;
|
||||||
|
|
||||||
import androidx.annotation.VisibleForTesting;
|
import androidx.annotation.VisibleForTesting;
|
||||||
@@ -39,6 +45,9 @@ import com.android.settings.Utils;
|
|||||||
import com.android.settings.overlay.FeatureFactory;
|
import com.android.settings.overlay.FeatureFactory;
|
||||||
import com.android.settings.slices.CustomSliceRegistry;
|
import com.android.settings.slices.CustomSliceRegistry;
|
||||||
import com.android.settings.slices.CustomSliceable;
|
import com.android.settings.slices.CustomSliceable;
|
||||||
|
import com.android.settings.slices.SliceBackgroundWorker;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
public class DarkThemeSlice implements CustomSliceable {
|
public class DarkThemeSlice implements CustomSliceable {
|
||||||
private static final String TAG = "DarkThemeSlice";
|
private static final String TAG = "DarkThemeSlice";
|
||||||
@@ -53,10 +62,12 @@ public class DarkThemeSlice implements CustomSliceable {
|
|||||||
|
|
||||||
private final Context mContext;
|
private final Context mContext;
|
||||||
private final UiModeManager mUiModeManager;
|
private final UiModeManager mUiModeManager;
|
||||||
|
private final PowerManager mPowerManager;
|
||||||
|
|
||||||
public DarkThemeSlice(Context context) {
|
public DarkThemeSlice(Context context) {
|
||||||
mContext = context;
|
mContext = context;
|
||||||
mUiModeManager = context.getSystemService(UiModeManager.class);
|
mUiModeManager = context.getSystemService(UiModeManager.class);
|
||||||
|
mPowerManager = context.getSystemService(PowerManager.class);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@@ -67,15 +78,18 @@ public class DarkThemeSlice implements CustomSliceable {
|
|||||||
sActiveUiSession = currentUiSession;
|
sActiveUiSession = currentUiSession;
|
||||||
sKeepSliceShow = false;
|
sKeepSliceShow = false;
|
||||||
}
|
}
|
||||||
if (!sKeepSliceShow && !isAvailable(mContext)) {
|
// Dark theme slice will disappear when battery saver is ON.
|
||||||
return null;
|
if (mPowerManager.isPowerSaveMode() || (!sKeepSliceShow && !isAvailable(mContext))) {
|
||||||
|
return new ListBuilder(mContext, CustomSliceRegistry.DARK_THEME_SLICE_URI,
|
||||||
|
ListBuilder.INFINITY)
|
||||||
|
.setIsError(true)
|
||||||
|
.build();
|
||||||
}
|
}
|
||||||
sKeepSliceShow = true;
|
sKeepSliceShow = true;
|
||||||
final PendingIntent toggleAction = getBroadcastIntent(mContext);
|
final PendingIntent toggleAction = getBroadcastIntent(mContext);
|
||||||
@ColorInt final int color = Utils.getColorAccentDefaultColor(mContext);
|
@ColorInt final int color = Utils.getColorAccentDefaultColor(mContext);
|
||||||
final IconCompat icon =
|
final IconCompat icon =
|
||||||
IconCompat.createWithResource(mContext, R.drawable.dark_theme);
|
IconCompat.createWithResource(mContext, R.drawable.dark_theme);
|
||||||
final boolean isChecked = mUiModeManager.getNightMode() == UiModeManager.MODE_NIGHT_YES;
|
|
||||||
return new ListBuilder(mContext, CustomSliceRegistry.DARK_THEME_SLICE_URI,
|
return new ListBuilder(mContext, CustomSliceRegistry.DARK_THEME_SLICE_URI,
|
||||||
ListBuilder.INFINITY)
|
ListBuilder.INFINITY)
|
||||||
.setAccentColor(color)
|
.setAccentColor(color)
|
||||||
@@ -85,7 +99,7 @@ public class DarkThemeSlice implements CustomSliceable {
|
|||||||
.setSubtitle(mContext.getText(R.string.dark_theme_slice_subtitle))
|
.setSubtitle(mContext.getText(R.string.dark_theme_slice_subtitle))
|
||||||
.setPrimaryAction(
|
.setPrimaryAction(
|
||||||
SliceAction.createToggle(toggleAction, null /* actionTitle */,
|
SliceAction.createToggle(toggleAction, null /* actionTitle */,
|
||||||
isChecked)))
|
isDarkThemeMode(mContext))))
|
||||||
.build();
|
.build();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -100,8 +114,7 @@ public class DarkThemeSlice implements CustomSliceable {
|
|||||||
false);
|
false);
|
||||||
// make toggle transition more smooth before dark theme takes effect
|
// make toggle transition more smooth before dark theme takes effect
|
||||||
new Handler(Looper.getMainLooper()).postDelayed(() -> {
|
new Handler(Looper.getMainLooper()).postDelayed(() -> {
|
||||||
mUiModeManager.setNightMode(
|
mUiModeManager.setNightModeActivated(isChecked);
|
||||||
isChecked ? UiModeManager.MODE_NIGHT_YES : UiModeManager.MODE_NIGHT_NO);
|
|
||||||
}, DELAY_TIME_EXECUTING_DARK_THEME);
|
}, DELAY_TIME_EXECUTING_DARK_THEME);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -110,10 +123,15 @@ public class DarkThemeSlice implements CustomSliceable {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Class getBackgroundWorkerClass() {
|
||||||
|
return DarkThemeWorker.class;
|
||||||
|
}
|
||||||
|
|
||||||
@VisibleForTesting
|
@VisibleForTesting
|
||||||
boolean isAvailable(Context context) {
|
boolean isAvailable(Context context) {
|
||||||
// checking dark theme mode.
|
// checking dark theme mode.
|
||||||
if (mUiModeManager.getNightMode() == UiModeManager.MODE_NIGHT_YES) {
|
if (isDarkThemeMode(context)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -121,7 +139,47 @@ public class DarkThemeSlice implements CustomSliceable {
|
|||||||
final BatteryManager batteryManager = context.getSystemService(BatteryManager.class);
|
final BatteryManager batteryManager = context.getSystemService(BatteryManager.class);
|
||||||
final int level = batteryManager.getIntProperty(BatteryManager.BATTERY_PROPERTY_CAPACITY);
|
final int level = batteryManager.getIntProperty(BatteryManager.BATTERY_PROPERTY_CAPACITY);
|
||||||
Log.d(TAG, "battery level=" + level);
|
Log.d(TAG, "battery level=" + level);
|
||||||
|
|
||||||
return level <= BATTERY_LEVEL_THRESHOLD;
|
return level <= BATTERY_LEVEL_THRESHOLD;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@VisibleForTesting
|
||||||
|
boolean isDarkThemeMode(Context context) {
|
||||||
|
final int currentNightMode =
|
||||||
|
context.getResources().getConfiguration().uiMode & Configuration.UI_MODE_NIGHT_MASK;
|
||||||
|
return currentNightMode == Configuration.UI_MODE_NIGHT_YES;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class DarkThemeWorker extends SliceBackgroundWorker<Void> {
|
||||||
|
private final Context mContext;
|
||||||
|
private final ContentObserver mContentObserver =
|
||||||
|
new ContentObserver(new Handler(Looper.getMainLooper())) {
|
||||||
|
@Override
|
||||||
|
public void onChange(boolean bChanged) {
|
||||||
|
if (mContext.getSystemService(PowerManager.class).isPowerSaveMode()) {
|
||||||
|
notifySliceChange();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
public DarkThemeWorker(Context context, Uri uri) {
|
||||||
|
super(context, uri);
|
||||||
|
mContext = context;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onSlicePinned() {
|
||||||
|
mContext.getContentResolver().registerContentObserver(
|
||||||
|
Settings.Global.getUriFor(LOW_POWER_MODE), false /* notifyForDescendants */,
|
||||||
|
mContentObserver);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onSliceUnpinned() {
|
||||||
|
mContext.getContentResolver().unregisterContentObserver(mContentObserver);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void close() throws IOException {
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@@ -22,10 +22,10 @@ import static org.mockito.Mockito.doReturn;
|
|||||||
import static org.mockito.Mockito.spy;
|
import static org.mockito.Mockito.spy;
|
||||||
import static org.mockito.Mockito.when;
|
import static org.mockito.Mockito.when;
|
||||||
|
|
||||||
import android.app.UiModeManager;
|
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.net.Uri;
|
import android.net.Uri;
|
||||||
import android.os.BatteryManager;
|
import android.os.BatteryManager;
|
||||||
|
import android.os.PowerManager;
|
||||||
|
|
||||||
import androidx.slice.Slice;
|
import androidx.slice.Slice;
|
||||||
import androidx.slice.SliceMetadata;
|
import androidx.slice.SliceMetadata;
|
||||||
@@ -47,10 +47,10 @@ import org.robolectric.RuntimeEnvironment;
|
|||||||
|
|
||||||
@RunWith(RobolectricTestRunner.class)
|
@RunWith(RobolectricTestRunner.class)
|
||||||
public class DarkThemeSliceTest {
|
public class DarkThemeSliceTest {
|
||||||
@Mock
|
|
||||||
private UiModeManager mUiModeManager;
|
|
||||||
@Mock
|
@Mock
|
||||||
private BatteryManager mBatteryManager;
|
private BatteryManager mBatteryManager;
|
||||||
|
@Mock
|
||||||
|
private PowerManager mPowerManager;
|
||||||
|
|
||||||
private Context mContext;
|
private Context mContext;
|
||||||
private DarkThemeSlice mDarkThemeSlice;
|
private DarkThemeSlice mDarkThemeSlice;
|
||||||
@@ -63,11 +63,12 @@ public class DarkThemeSliceTest {
|
|||||||
mFeatureFactory = FakeFeatureFactory.setupForTest();
|
mFeatureFactory = FakeFeatureFactory.setupForTest();
|
||||||
mFeatureFactory.slicesFeatureProvider = new SlicesFeatureProviderImpl();
|
mFeatureFactory.slicesFeatureProvider = new SlicesFeatureProviderImpl();
|
||||||
mFeatureFactory.slicesFeatureProvider.newUiSession();
|
mFeatureFactory.slicesFeatureProvider.newUiSession();
|
||||||
doReturn(mUiModeManager).when(mContext).getSystemService(UiModeManager.class);
|
doReturn(mPowerManager).when(mContext).getSystemService(PowerManager.class);
|
||||||
|
when(mPowerManager.isPowerSaveMode()).thenReturn(false);
|
||||||
|
|
||||||
// Set-up specs for SliceMetadata.
|
// Set-up specs for SliceMetadata.
|
||||||
SliceProvider.setSpecs(SliceLiveData.SUPPORTED_SPECS);
|
SliceProvider.setSpecs(SliceLiveData.SUPPORTED_SPECS);
|
||||||
mDarkThemeSlice = new DarkThemeSlice(mContext);
|
mDarkThemeSlice = spy(new DarkThemeSlice(mContext));
|
||||||
mDarkThemeSlice.sKeepSliceShow = false;
|
mDarkThemeSlice.sKeepSliceShow = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -80,7 +81,7 @@ public class DarkThemeSliceTest {
|
|||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void isAvailable_inDarkThemeMode_returnFalse() {
|
public void isAvailable_inDarkThemeMode_returnFalse() {
|
||||||
when(mUiModeManager.getNightMode()).thenReturn(UiModeManager.MODE_NIGHT_YES);
|
doReturn(true).when(mDarkThemeSlice).isDarkThemeMode(mContext);
|
||||||
|
|
||||||
assertThat(mDarkThemeSlice.isAvailable(mContext)).isFalse();
|
assertThat(mDarkThemeSlice.isAvailable(mContext)).isFalse();
|
||||||
}
|
}
|
||||||
@@ -100,23 +101,36 @@ public class DarkThemeSliceTest {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void getSlice_notAvailable_returnNull() {
|
public void getSlice_batterySaver_returnErrorSlice() {
|
||||||
when(mUiModeManager.getNightMode()).thenReturn(UiModeManager.MODE_NIGHT_YES);
|
when(mPowerManager.isPowerSaveMode()).thenReturn(true);
|
||||||
|
|
||||||
assertThat(mDarkThemeSlice.getSlice()).isNull();
|
final Slice mediaSlice = mDarkThemeSlice.getSlice();
|
||||||
|
final SliceMetadata metadata = SliceMetadata.from(mContext, mediaSlice);
|
||||||
|
assertThat(metadata.isErrorSlice()).isTrue();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void getSlice_newSession_notAvailable_returnNull() {
|
public void getSlice_notAvailable_returnErrorSlice() {
|
||||||
|
doReturn(true).when(mDarkThemeSlice).isDarkThemeMode(mContext);
|
||||||
|
|
||||||
|
final Slice mediaSlice = mDarkThemeSlice.getSlice();
|
||||||
|
final SliceMetadata metadata = SliceMetadata.from(mContext, mediaSlice);
|
||||||
|
assertThat(metadata.isErrorSlice()).isTrue();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void getSlice_newSession_notAvailable_returnErrorSlice() {
|
||||||
// previous displayed: yes
|
// previous displayed: yes
|
||||||
mDarkThemeSlice.sKeepSliceShow = true;
|
mDarkThemeSlice.sKeepSliceShow = true;
|
||||||
// Session: use original value + 1 to become a new session
|
// Session: use original value + 1 to become a new session
|
||||||
mDarkThemeSlice.sActiveUiSession =
|
mDarkThemeSlice.sActiveUiSession =
|
||||||
mFeatureFactory.slicesFeatureProvider.getUiSessionToken() + 1;
|
mFeatureFactory.slicesFeatureProvider.getUiSessionToken() + 1;
|
||||||
|
|
||||||
when(mUiModeManager.getNightMode()).thenReturn(UiModeManager.MODE_NIGHT_YES);
|
doReturn(true).when(mDarkThemeSlice).isDarkThemeMode(mContext);
|
||||||
|
|
||||||
assertThat(mDarkThemeSlice.getSlice()).isNull();
|
final Slice mediaSlice = mDarkThemeSlice.getSlice();
|
||||||
|
final SliceMetadata metadata = SliceMetadata.from(mContext, mediaSlice);
|
||||||
|
assertThat(metadata.isErrorSlice()).isTrue();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
@@ -149,7 +163,7 @@ public class DarkThemeSliceTest {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private void setBatteryCapacityLevel(int power_level) {
|
private void setBatteryCapacityLevel(int power_level) {
|
||||||
when(mUiModeManager.getNightMode()).thenReturn(UiModeManager.MODE_NIGHT_NO);
|
doReturn(false).when(mDarkThemeSlice).isDarkThemeMode(mContext);
|
||||||
doReturn(mBatteryManager).when(mContext).getSystemService(BatteryManager.class);
|
doReturn(mBatteryManager).when(mContext).getSystemService(BatteryManager.class);
|
||||||
when(mBatteryManager.getIntProperty(BatteryManager.BATTERY_PROPERTY_CAPACITY))
|
when(mBatteryManager.getIntProperty(BatteryManager.BATTERY_PROPERTY_CAPACITY))
|
||||||
.thenReturn(power_level);
|
.thenReturn(power_level);
|
||||||
|
Reference in New Issue
Block a user