Update theme setting to not require reboot

Add calls into the OverlayManagerService to update the overlays
directly and remove reboot dialog.

Test: make RunSettingsRobotTests
Change-Id: I92841e643eab70b4b194f60cce95ce3cedb75972
Fixes: 34701451
This commit is contained in:
Jason Monk
2017-03-07 15:59:36 -05:00
parent 69ef3f4e69
commit 2859b9a302
5 changed files with 199 additions and 84 deletions

View File

@@ -15,15 +15,19 @@ package com.android.settings.display;
import static com.android.internal.logging.nano.MetricsProto.MetricsEvent.ACTION_THEME; import static com.android.internal.logging.nano.MetricsProto.MetricsEvent.ACTION_THEME;
import android.app.AlertDialog;
import android.app.UiModeManager;
import android.content.Context; import android.content.Context;
import android.content.DialogInterface.OnClickListener; import android.content.om.IOverlayManager;
import android.content.om.OverlayInfo;
import android.content.pm.PackageManager;
import android.content.pm.PackageManager.NameNotFoundException;
import android.os.RemoteException;
import android.os.ServiceManager;
import android.os.UserHandle;
import android.support.annotation.VisibleForTesting; import android.support.annotation.VisibleForTesting;
import android.support.v7.preference.ListPreference; import android.support.v7.preference.ListPreference;
import android.support.v7.preference.Preference; import android.support.v7.preference.Preference;
import android.text.TextUtils; import android.text.TextUtils;
import com.android.settings.R; import com.android.settings.R;
import com.android.settings.core.PreferenceController; import com.android.settings.core.PreferenceController;
import com.android.settings.core.instrumentation.MetricsFeatureProvider; import com.android.settings.core.instrumentation.MetricsFeatureProvider;
@@ -31,17 +35,27 @@ import com.android.settings.overlay.FeatureFactory;
import libcore.util.Objects; import libcore.util.Objects;
import java.util.List;
public class ThemePreferenceController extends PreferenceController implements public class ThemePreferenceController extends PreferenceController implements
Preference.OnPreferenceChangeListener { Preference.OnPreferenceChangeListener {
private static final String KEY_THEME = "theme"; private static final String KEY_THEME = "theme";
private final UiModeManager mUiModeManager;
private final MetricsFeatureProvider mMetricsFeatureProvider; private final MetricsFeatureProvider mMetricsFeatureProvider;
private final OverlayManager mOverlayService;
private final PackageManager mPackageManager;
public ThemePreferenceController(Context context) { public ThemePreferenceController(Context context) {
this(context, ServiceManager.getService(Context.OVERLAY_SERVICE) != null
? new OverlayManager() : null);
}
@VisibleForTesting
ThemePreferenceController(Context context, OverlayManager overlayManager) {
super(context); super(context);
mUiModeManager = context.getSystemService(UiModeManager.class); mOverlayService = overlayManager;
mPackageManager = context.getPackageManager();
mMetricsFeatureProvider = FeatureFactory.getFactory(context).getMetricsFeatureProvider(); mMetricsFeatureProvider = FeatureFactory.getFactory(context).getMetricsFeatureProvider();
} }
@@ -61,12 +75,18 @@ public class ThemePreferenceController extends PreferenceController implements
@Override @Override
public void updateState(Preference preference) { public void updateState(Preference preference) {
ListPreference pref = (ListPreference) preference; ListPreference pref = (ListPreference) preference;
String[] options = getAvailableThemes(); String[] pkgs = getAvailableThemes();
for (int i = 0; i < options.length; i++) { CharSequence[] labels = new CharSequence[pkgs.length];
options[i] = nullToDefault(options[i]); for (int i = 0; i < pkgs.length; i++) {
try {
labels[i] = mPackageManager.getApplicationInfo(pkgs[i], 0)
.loadLabel(mPackageManager);
} catch (NameNotFoundException e) {
labels[i] = pkgs[i];
}
} }
pref.setEntries(options); pref.setEntries(labels);
pref.setEntryValues(options); pref.setEntryValues(pkgs);
String theme = getCurrentTheme(); String theme = getCurrentTheme();
if (TextUtils.isEmpty(theme)) { if (TextUtils.isEmpty(theme)) {
theme = mContext.getString(R.string.default_theme); theme = mContext.getString(R.string.default_theme);
@@ -77,49 +97,76 @@ public class ThemePreferenceController extends PreferenceController implements
@Override @Override
public boolean onPreferenceChange(Preference preference, Object newValue) { public boolean onPreferenceChange(Preference preference, Object newValue) {
if (Objects.equal(newValue, mUiModeManager.getTheme())) { String current = getTheme();
if (Objects.equal(newValue, current)) {
return true; return true;
} }
// TODO: STOPSHIP Don't require reboot and remove this prompt. try {
OnClickListener onConfirm = (d, i) -> { mOverlayService.setEnabledExclusive((String) newValue, true, UserHandle.myUserId());
mUiModeManager.setTheme(defaultToNull((String) newValue)); } catch (RemoteException e) {
((ListPreference) preference).setValue((String) newValue); return false;
}; }
new AlertDialog.Builder(mContext) return true;
.setTitle(R.string.change_theme_reboot) }
.setPositiveButton(com.android.internal.R.string.global_action_restart, onConfirm)
.setNegativeButton(android.R.string.cancel, null) private String getTheme() {
.show(); try {
return false; List<OverlayInfo> infos = mOverlayService.getOverlayInfosForTarget("android",
UserHandle.myUserId());
for (int i = 0, size = infos.size(); i < size; i++) {
if (infos.get(i).isEnabled()) {
return infos.get(i).packageName;
}
}
} catch (RemoteException e) {
}
return null;
} }
@Override @Override
public boolean isAvailable() { public boolean isAvailable() {
if (mOverlayService == null) return false;
String[] themes = getAvailableThemes(); String[] themes = getAvailableThemes();
return themes != null && themes.length > 1; return themes != null && themes.length > 1;
} }
@VisibleForTesting @VisibleForTesting
String getCurrentTheme() { String getCurrentTheme() {
return mUiModeManager.getTheme(); return getTheme();
} }
@VisibleForTesting @VisibleForTesting
String[] getAvailableThemes() { String[] getAvailableThemes() {
return mUiModeManager.getAvailableThemes(); try {
List<OverlayInfo> infos = mOverlayService.getOverlayInfosForTarget("android",
UserHandle.myUserId());
String[] pkgs = new String[infos.size()];
for (int i = 0, size = infos.size(); i < size; i++) {
pkgs[i] = infos.get(i).packageName;
}
return pkgs;
} catch (RemoteException e) {
}
return new String[0];
} }
private String nullToDefault(String input) { public static class OverlayManager {
if (TextUtils.isEmpty(input)) { private final IOverlayManager mService;
return mContext.getString(R.string.default_theme);
}
return input;
}
private String defaultToNull(String input) { public OverlayManager() {
if (mContext.getString(R.string.default_theme).equals(input)) { mService = IOverlayManager.Stub.asInterface(
return null; ServiceManager.getService(Context.OVERLAY_SERVICE));
}
public void setEnabledExclusive(String pkg, boolean enabled, int userId)
throws RemoteException {
mService.setEnabledExclusive(pkg, enabled, userId);
}
public List<OverlayInfo> getOverlayInfosForTarget(String target, int userId)
throws RemoteException {
return mService.getOverlayInfosForTarget(target, userId);
} }
return input;
} }
} }

View File

@@ -0,0 +1,26 @@
/*
* Copyright (C) 2017 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file
* except in compliance with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software distributed under the
* License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the specific language governing
* permissions and limitations under the License.
*/
package android.content.om;
import android.os.IBinder;
public class IOverlayManager {
public static class Stub {
public static IOverlayManager asInterface(IBinder b) {
return null;
}
}
}

View File

@@ -0,0 +1,18 @@
/*
* Copyright (C) 2017 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file
* except in compliance with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software distributed under the
* License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the specific language governing
* permissions and limitations under the License.
*/
package android.content.om;
public class OverlayInfo {
}

View File

@@ -17,10 +17,15 @@
package com.android.settings.display; package com.android.settings.display;
import android.content.Context; import android.content.Context;
import android.content.pm.ApplicationInfo;
import android.content.pm.PackageManager;
import android.content.pm.PackageManager.NameNotFoundException;
import android.support.v7.preference.ListPreference; import android.support.v7.preference.ListPreference;
import com.android.settings.R; import com.android.settings.R;
import com.android.settings.SettingsRobolectricTestRunner; import com.android.settings.SettingsRobolectricTestRunner;
import com.android.settings.TestConfig; import com.android.settings.TestConfig;
import com.android.settings.display.ThemePreferenceController.OverlayManager;
import org.junit.Before; import org.junit.Before;
import org.junit.Test; import org.junit.Test;
import org.junit.runner.RunWith; import org.junit.runner.RunWith;
@@ -28,9 +33,13 @@ import org.mockito.Mock;
import org.mockito.MockitoAnnotations; import org.mockito.MockitoAnnotations;
import org.robolectric.annotation.Config; import org.robolectric.annotation.Config;
import static org.mockito.Matchers.any;
import static org.mockito.Matchers.anyInt;
import static org.mockito.Mockito.doReturn; import static org.mockito.Mockito.doReturn;
import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.spy; import static org.mockito.Mockito.spy;
import static org.mockito.Mockito.verify; import static org.mockito.Mockito.verify;
import static org.mockito.Mockito.when;
@RunWith(SettingsRobolectricTestRunner.class) @RunWith(SettingsRobolectricTestRunner.class)
@Config(manifest = TestConfig.MANIFEST_PATH, sdk = TestConfig.SDK_VERSION) @Config(manifest = TestConfig.MANIFEST_PATH, sdk = TestConfig.SDK_VERSION)
@@ -40,24 +49,19 @@ public class ThemePreferenceControllerTest {
private ListPreference mPreference; private ListPreference mPreference;
@Mock @Mock
private Context mContext; private Context mContext;
@Mock
private PackageManager mPackageManager;
@Mock
private ApplicationInfo mApplicationInfo;
private ThemePreferenceController mController; private ThemePreferenceController mController;
@Before @Before
public void setUp() { public void setUp() throws NameNotFoundException {
MockitoAnnotations.initMocks(this); MockitoAnnotations.initMocks(this);
mController = spy(new ThemePreferenceController(mContext)); when(mPackageManager.getApplicationInfo(any(), anyInt())).thenReturn(mApplicationInfo);
} when(mContext.getPackageManager()).thenReturn(mPackageManager);
mController = spy(new ThemePreferenceController(mContext, mock(OverlayManager.class)));
@Test
public void updateState_nullTheme_shouldSetSummaryToDefault() {
final String[] themes = {"Theme1", "Theme2"};
doReturn(null).when(mController).getCurrentTheme();
doReturn(themes).when(mController).getAvailableThemes();
mController.updateState(mPreference);
verify(mPreference).setSummary(mContext.getString(R.string.default_theme));
} }
@Test @Test

View File

@@ -14,95 +14,115 @@
* limitations under the License. * limitations under the License.
*/ */
package com.android.settings.core; package com.android.settings.display;
import static junit.framework.TestCase.assertNotNull;
import static org.junit.Assert.assertEquals; import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertFalse; import static org.junit.Assert.assertFalse;
import static org.junit.Assert.assertTrue; import static org.junit.Assert.assertTrue;
import static org.mockito.Matchers.any;
import static org.mockito.Matchers.anyInt;
import static org.mockito.Matchers.eq; import static org.mockito.Matchers.eq;
import static org.mockito.Mockito.mock; import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.verify; import static org.mockito.Mockito.verify;
import static org.mockito.Mockito.when; import static org.mockito.Mockito.when;
import android.app.UiModeManager;
import android.content.Context;
import android.content.ContextWrapper; import android.content.ContextWrapper;
import android.content.om.OverlayInfo;
import android.content.pm.ApplicationInfo;
import android.content.pm.PackageManager;
import android.support.test.InstrumentationRegistry; import android.support.test.InstrumentationRegistry;
import android.support.test.filters.SmallTest; import android.support.test.filters.SmallTest;
import android.support.test.runner.AndroidJUnit4; import android.support.test.runner.AndroidJUnit4;
import android.support.v7.preference.ListPreference; import android.support.v7.preference.ListPreference;
import com.android.settings.R;
import com.android.settings.display.ThemePreferenceController; import com.android.settings.display.ThemePreferenceController;
import com.android.settings.display.ThemePreferenceController.OverlayManager;
import org.junit.Before; import org.junit.Before;
import org.junit.Test; import org.junit.Test;
import org.junit.runner.RunWith; import org.junit.runner.RunWith;
import org.mockito.ArgumentCaptor; import org.mockito.ArgumentCaptor;
import java.util.ArrayList;
@SmallTest @SmallTest
@RunWith(AndroidJUnit4.class) @RunWith(AndroidJUnit4.class)
public class ThemePreferenceControllerTest { public class ThemePreferenceControllerTest {
private UiModeManager mMockUiModeManager; private OverlayManager mMockOverlayManager;
private ContextWrapper mContext; private ContextWrapper mContext;
private ThemePreferenceController mPreferenceController; private ThemePreferenceController mPreferenceController;
private PackageManager mMockPackageManager;
@Before @Before
public void setup() { public void setup() {
mMockUiModeManager = mock(UiModeManager.class); mMockOverlayManager = mock(OverlayManager.class);
mContext = new ContextWrapper(InstrumentationRegistry.getTargetContext()) { mMockPackageManager = mock(PackageManager.class);
mContext = new ContextWrapper(InstrumentationRegistry.getContext()) {
@Override @Override
public Object getSystemService(String name) { public PackageManager getPackageManager() {
if (Context.UI_MODE_SERVICE.equals(name)) { return mMockPackageManager;
return mMockUiModeManager;
}
return super.getSystemService(name);
} }
}; };
mPreferenceController = new ThemePreferenceController(mContext); mPreferenceController = new ThemePreferenceController(mContext, mMockOverlayManager);
} }
@Test @Test
public void testUpdateState() { public void testUpdateState() throws Exception {
when(mMockUiModeManager.getAvailableThemes()).thenReturn(new String[] { OverlayInfo info1 = new OverlayInfo("com.android.Theme1", "android",
null, "", OverlayInfo.STATE_ENABLED, 0);
"Theme1", OverlayInfo info2 = new OverlayInfo("com.android.Theme2", "android",
"Theme2", "", 0, 0);
when(mMockPackageManager.getApplicationInfo(any(), anyInt())).thenAnswer(inv -> {
ApplicationInfo info = mock(ApplicationInfo.class);
if ("com.android.Theme1".equals(inv.getArguments()[0])) {
when(info.loadLabel(any())).thenReturn("Theme1");
} else {
when(info.loadLabel(any())).thenReturn("Theme2");
}
return info;
}); });
when(mMockUiModeManager.getTheme()).thenReturn("Theme1"); when(mMockOverlayManager.getOverlayInfosForTarget(any(), anyInt())).thenReturn(
list(info1, info2));
ListPreference pref = mock(ListPreference.class); ListPreference pref = mock(ListPreference.class);
mPreferenceController.updateState(pref); mPreferenceController.updateState(pref);
ArgumentCaptor<String[]> arg = ArgumentCaptor.forClass(String[].class); ArgumentCaptor<String[]> arg = ArgumentCaptor.forClass(String[].class);
verify(pref).setEntries(arg.capture()); verify(pref).setEntries(arg.capture());
String[] entries = arg.getValue();
assertEquals(3, entries.length); CharSequence[] entries = arg.getValue();
assertNotNull(entries[0]); assertEquals(2, entries.length);
assertEquals("Theme1", entries[1]); assertEquals("Theme1", entries[0]);
assertEquals("Theme2", entries[2]); assertEquals("Theme2", entries[1]);
verify(pref).setEntryValues(arg.capture()); verify(pref).setEntryValues(arg.capture());
String[] entryValues = arg.getValue(); CharSequence[] entryValues = arg.getValue();
assertEquals(3, entryValues.length); assertEquals("com.android.Theme1", entryValues[0]);
assertNotNull(entryValues[0]); assertEquals("com.android.Theme2", entryValues[1]);
assertEquals("Theme1", entryValues[1]);
assertEquals("Theme2", entryValues[2]);
verify(pref).setValue(eq("Theme1")); verify(pref).setValue(eq("com.android.Theme1"));
} }
@Test @Test
public void testAvailable_false() { public void testAvailable_false() throws Exception {
when(mMockUiModeManager.getAvailableThemes()).thenReturn(new String[1]); when(mMockOverlayManager.getOverlayInfosForTarget(any(), anyInt()))
.thenReturn(list(new OverlayInfo("", "", "", 0, 0)));
assertFalse(mPreferenceController.isAvailable()); assertFalse(mPreferenceController.isAvailable());
} }
@Test @Test
public void testAvailable_true() { public void testAvailable_true() throws Exception {
when(mMockUiModeManager.getAvailableThemes()).thenReturn(new String[2]); when(mMockOverlayManager.getOverlayInfosForTarget(any(), anyInt()))
.thenReturn(list(new OverlayInfo("", "", "", 0, 0),
new OverlayInfo("", "", "", 0, 0)));
assertTrue(mPreferenceController.isAvailable()); assertTrue(mPreferenceController.isAvailable());
} }
private ArrayList<OverlayInfo> list(OverlayInfo... infos) {
ArrayList<OverlayInfo> list = new ArrayList<>();
for (int i = 0; i < infos.length; i++) {
list.add(infos[i]);
}
return list;
}
} }