Updating CreateShortcut activity to use the new ShortcutManager APIs

for creating shortcuts

Test:  adb shell am instrument -e class com.android.settings.CreateShortcutTest \
 -w com.android.settings.tests/android.support.test.runner.AndroidJUnitRunner

Change-Id: I09fb73ae2a0687a3269f3714bdb2c5d92c99ea87
This commit is contained in:
Sunny Goyal
2017-01-19 12:06:48 -08:00
parent 5782b390dd
commit 9a6b675096
3 changed files with 194 additions and 27 deletions

View File

@@ -17,15 +17,21 @@
package com.android.settings; package com.android.settings;
import android.app.LauncherActivity; import android.app.LauncherActivity;
import android.content.ComponentName;
import android.content.Context; import android.content.Context;
import android.content.Intent; import android.content.Intent;
import android.content.pm.ActivityInfo; import android.content.pm.ActivityInfo;
import android.content.pm.PackageManager; import android.content.pm.PackageManager;
import android.content.pm.ResolveInfo; import android.content.pm.ResolveInfo;
import android.content.pm.ShortcutInfo;
import android.content.pm.ShortcutManager;
import android.graphics.Bitmap; import android.graphics.Bitmap;
import android.graphics.Bitmap.Config; import android.graphics.Bitmap.Config;
import android.graphics.Canvas; import android.graphics.Canvas;
import android.graphics.drawable.Icon;
import android.net.ConnectivityManager; import android.net.ConnectivityManager;
import android.os.AsyncTask;
import android.support.annotation.VisibleForTesting;
import android.view.ContextThemeWrapper; import android.view.ContextThemeWrapper;
import android.view.LayoutInflater; import android.view.LayoutInflater;
import android.view.View; import android.view.View;
@@ -35,34 +41,52 @@ import android.widget.ListView;
import com.android.settings.Settings.TetherSettingsActivity; import com.android.settings.Settings.TetherSettingsActivity;
import java.util.ArrayList;
import java.util.List; import java.util.List;
public class CreateShortcut extends LauncherActivity { public class CreateShortcut extends LauncherActivity {
@VisibleForTesting
static final String SHORTCUT_ID_PREFIX = "component-shortcut-";
@Override @Override
protected Intent getTargetIntent() { protected Intent getTargetIntent() {
Intent targetIntent = new Intent(Intent.ACTION_MAIN, null); return getBaseIntent().addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
targetIntent.addCategory("com.android.settings.SHORTCUT");
targetIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
return targetIntent;
} }
@Override @Override
protected void onListItemClick(ListView l, View v, int position, long id) { protected void onListItemClick(ListView l, View v, int position, long id) {
Intent shortcutIntent = intentForPosition(position); ListItem item = itemForPosition(position);
setResult(RESULT_OK, createResultIntent(intentForPosition(position),
item.resolveInfo, item.label));
finish();
}
protected Intent createResultIntent(Intent shortcutIntent, ResolveInfo resolveInfo,
CharSequence label) {
shortcutIntent.setFlags(Intent.FLAG_ACTIVITY_RESET_TASK_IF_NEEDED); shortcutIntent.setFlags(Intent.FLAG_ACTIVITY_RESET_TASK_IF_NEEDED);
Intent intent = new Intent();
ActivityInfo activityInfo = resolveInfo.activityInfo;
Bitmap icon = activityInfo.icon != 0 ? createIcon(activityInfo.icon) : null;
String shortcutId = SHORTCUT_ID_PREFIX +
shortcutIntent.getComponent().flattenToShortString();
ShortcutInfo info = new ShortcutInfo.Builder(this, shortcutId)
.setShortLabel(label)
.setIntent(shortcutIntent)
.setIcon(icon != null ? Icon.createWithBitmap(icon) :
Icon.createWithResource(this, R.mipmap.ic_launcher_settings))
.build();
Intent intent = getSystemService(ShortcutManager.class).createShortcutResultIntent(info);
if (intent == null) {
intent = new Intent();
}
intent.putExtra(Intent.EXTRA_SHORTCUT_ICON_RESOURCE, intent.putExtra(Intent.EXTRA_SHORTCUT_ICON_RESOURCE,
Intent.ShortcutIconResource.fromContext(this, R.mipmap.ic_launcher_settings)); Intent.ShortcutIconResource.fromContext(this, R.mipmap.ic_launcher_settings));
intent.putExtra(Intent.EXTRA_SHORTCUT_INTENT, shortcutIntent); intent.putExtra(Intent.EXTRA_SHORTCUT_INTENT, shortcutIntent);
intent.putExtra(Intent.EXTRA_SHORTCUT_NAME, itemForPosition(position).label); intent.putExtra(Intent.EXTRA_SHORTCUT_NAME, label);
ResolveInfo resolveInfo = itemForPosition(position).resolveInfo; intent.putExtra(Intent.EXTRA_SHORTCUT_ICON, icon);
ActivityInfo activityInfo = resolveInfo.activityInfo; return intent;
if (activityInfo.icon != 0) {
intent.putExtra(Intent.EXTRA_SHORTCUT_ICON, createIcon(activityInfo.icon));
}
setResult(RESULT_OK, intent);
finish();
} }
private Bitmap createIcon(int resource) { private Bitmap createIcon(int resource) {
@@ -110,4 +134,43 @@ public class CreateShortcut extends LauncherActivity {
} }
return activities; return activities;
} }
@VisibleForTesting
static Intent getBaseIntent() {
return new Intent(Intent.ACTION_MAIN).addCategory("com.android.settings.SHORTCUT");
}
public static class ShortcutsUpdateTask extends AsyncTask<Void, Void, Void> {
private final Context mContext;
public ShortcutsUpdateTask(Context context) {
mContext = context;
}
@Override
public Void doInBackground(Void... params) {
ShortcutManager sm = mContext.getSystemService(ShortcutManager.class);
PackageManager pm = mContext.getPackageManager();
List<ShortcutInfo> updates = new ArrayList<>();
for (ShortcutInfo info : sm.getPinnedShortcuts()) {
if (!info.getId().startsWith(SHORTCUT_ID_PREFIX)) {
continue;
}
ComponentName cn = ComponentName.unflattenFromString(
info.getId().substring(SHORTCUT_ID_PREFIX.length()));
ResolveInfo ri = pm.resolveActivity(getBaseIntent().setComponent(cn), 0);
if (ri == null) {
continue;
}
updates.add(new ShortcutInfo.Builder(mContext, info.getId())
.setShortLabel(ri.loadLabel(pm)).build());
}
if (!updates.isEmpty()) {
sm.updateShortcuts(updates);
}
return null;
}
}
} }

View File

@@ -34,6 +34,7 @@ import android.widget.CompoundButton;
import com.android.internal.app.LocalePicker; import com.android.internal.app.LocalePicker;
import com.android.internal.app.LocaleStore; import com.android.internal.app.LocaleStore;
import com.android.settings.CreateShortcut;
import com.android.settings.R; import com.android.settings.R;
import java.text.NumberFormat; import java.text.NumberFormat;
@@ -299,6 +300,8 @@ class LocaleDragAndDropAdapter
LocalePicker.updateLocales(mLocalesToSetNext); LocalePicker.updateLocales(mLocalesToSetNext);
mLocalesSetLast = mLocalesToSetNext; mLocalesSetLast = mLocalesToSetNext;
new CreateShortcut.ShortcutsUpdateTask(mContext).execute();
mLocalesToSetNext = null; mLocalesToSetNext = null;
mNumberFormatter = NumberFormat.getNumberInstance(Locale.getDefault()); mNumberFormatter = NumberFormat.getNumberInstance(Locale.getDefault());

View File

@@ -16,29 +16,130 @@
package com.android.settings; package com.android.settings;
import android.app.Instrumentation;
import android.content.Intent;
import android.support.test.InstrumentationRegistry;
import android.support.test.filters.SmallTest;
import android.support.test.runner.AndroidJUnit4;
import org.junit.Test;
import org.junit.runner.RunWith;
import static android.support.test.espresso.Espresso.onView; import static android.support.test.espresso.Espresso.onView;
import static android.support.test.espresso.assertion.ViewAssertions.doesNotExist; import static android.support.test.espresso.assertion.ViewAssertions.doesNotExist;
import static android.support.test.espresso.matcher.ViewMatchers.withText; import static android.support.test.espresso.matcher.ViewMatchers.withText;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertNotNull;
import static org.mockito.Mockito.any;
import static org.mockito.Mockito.doReturn;
import static org.mockito.Mockito.eq;
import static org.mockito.Mockito.spy;
import static org.mockito.Mockito.times;
import static org.mockito.Mockito.verify;
import static org.mockito.Mockito.when;
import android.app.Instrumentation;
import android.content.ComponentName;
import android.content.Context;
import android.content.ContextWrapper;
import android.content.Intent;
import android.content.pm.ResolveInfo;
import android.content.pm.ShortcutInfo;
import android.content.pm.ShortcutManager;
import android.support.test.InstrumentationRegistry;
import android.support.test.filters.SmallTest;
import android.support.test.runner.AndroidJUnit4;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.mockito.ArgumentCaptor;
import org.mockito.Captor;
import org.mockito.Mock;
import org.mockito.MockitoAnnotations;
import java.util.Arrays;
import java.util.List;
/**
* Tests for {@link CreateShortcutTest}
*
m SettingsTests &&
adb install \
-r -g ${ANDROID_PRODUCT_OUT}/data/app/SettingsTests/SettingsTests.apk &&
adb shell am instrument -e class com.android.settings.CreateShortcutTest \
-w com.android.settings.tests/android.support.test.runner.AndroidJUnitRunner
*/
@RunWith(AndroidJUnit4.class) @RunWith(AndroidJUnit4.class)
@SmallTest @SmallTest
public class CreateShortcutTest { public class CreateShortcutTest {
private static final String SHORTCUT_ID_PREFIX = CreateShortcut.SHORTCUT_ID_PREFIX;
private Instrumentation mInstrumentation;
private Context mContext;
@Mock ShortcutManager mShortcutManager;
@Captor ArgumentCaptor<List<ShortcutInfo>> mListCaptor;
@Before
public void setup() {
MockitoAnnotations.initMocks(this);
mInstrumentation = InstrumentationRegistry.getInstrumentation();
mContext = mInstrumentation.getTargetContext();
}
@Test @Test
public void test_layoutDoesNotHaveCancelButton() { public void test_layoutDoesNotHaveCancelButton() {
Instrumentation instrumentation = InstrumentationRegistry.getInstrumentation(); mInstrumentation.startActivitySync(new Intent(Intent.ACTION_CREATE_SHORTCUT)
instrumentation.startActivitySync(new Intent(Intent.ACTION_CREATE_SHORTCUT) .setClassName(mContext, CreateShortcut.class.getName()));
.setClassName(instrumentation.getTargetContext(),
CreateShortcut.class.getName()));
onView(withText(R.string.cancel)).check(doesNotExist()); onView(withText(R.string.cancel)).check(doesNotExist());
} }
@Test
public void createResultIntent() {
CreateShortcut orgActivity = (CreateShortcut) mInstrumentation.startActivitySync(
new Intent(Intent.ACTION_CREATE_SHORTCUT)
.setClassName(mContext, CreateShortcut.class.getName()));
CreateShortcut activity = spy(orgActivity);
doReturn(mShortcutManager).when(activity).getSystemService(eq(Context.SHORTCUT_SERVICE));
when(mShortcutManager.createShortcutResultIntent(any(ShortcutInfo.class)))
.thenReturn(new Intent().putExtra("d1", "d2"));
Intent intent = CreateShortcut.getBaseIntent()
.setClass(activity, Settings.ManageApplicationsActivity.class);
ResolveInfo ri = activity.getPackageManager().resolveActivity(intent, 0);
Intent result = activity.createResultIntent(intent, ri, "dummy");
assertEquals("d2", result.getStringExtra("d1"));
assertNotNull(result.getParcelableExtra(Intent.EXTRA_SHORTCUT_INTENT));
ArgumentCaptor<ShortcutInfo> infoCaptor = ArgumentCaptor.forClass(ShortcutInfo.class);
verify(mShortcutManager, times(1))
.createShortcutResultIntent(infoCaptor.capture());
String expectedId = SHORTCUT_ID_PREFIX + intent.getComponent().flattenToShortString();
assertEquals(expectedId, infoCaptor.getValue().getId());
}
@Test
public void shortcutsUpdateTask() {
mContext = spy(new ContextWrapper(mInstrumentation.getTargetContext()));
doReturn(mShortcutManager).when(mContext).getSystemService(eq(Context.SHORTCUT_SERVICE));
List<ShortcutInfo> pinnedShortcuts = Arrays.asList(
makeShortcut("d1"), makeShortcut("d2"),
makeShortcut(Settings.ManageApplicationsActivity.class),
makeShortcut("d3"),
makeShortcut(Settings.SoundSettingsActivity.class));
when(mShortcutManager.getPinnedShortcuts()).thenReturn(pinnedShortcuts);
new CreateShortcut.ShortcutsUpdateTask(mContext).doInBackground();
verify(mShortcutManager, times(1)).updateShortcuts(mListCaptor.capture());
List<ShortcutInfo> updates = mListCaptor.getValue();
assertEquals(2, updates.size());
assertEquals(pinnedShortcuts.get(2).getId(), updates.get(0).getId());
assertEquals(pinnedShortcuts.get(4).getId(), updates.get(1).getId());
}
private ShortcutInfo makeShortcut(Class<?> className) {
ComponentName cn = new ComponentName(mContext, className);
return makeShortcut(SHORTCUT_ID_PREFIX + cn.flattenToShortString());
}
private ShortcutInfo makeShortcut(String id) {
return new ShortcutInfo.Builder(mContext, id).build();
}
} }