Merge "Add a filter to take out unnecessary homepage data."

This commit is contained in:
Fan Zhang
2018-10-09 16:48:04 +00:00
committed by Android (Google) Code Review
3 changed files with 179 additions and 104 deletions

View File

@@ -16,13 +16,21 @@
package com.android.settings.homepage; package com.android.settings.homepage;
import static android.app.slice.Slice.HINT_ERROR;
import static androidx.slice.widget.SliceLiveData.SUPPORTED_SPECS;
import android.content.ContentProviderClient;
import android.content.ContentResolver;
import android.content.Context; import android.content.Context;
import android.content.pm.PackageManager; import android.content.pm.PackageManager;
import android.database.Cursor; import android.database.Cursor;
import android.net.Uri;
import android.util.Log; import android.util.Log;
import androidx.annotation.NonNull; import androidx.annotation.NonNull;
import androidx.annotation.VisibleForTesting; import androidx.annotation.VisibleForTesting;
import androidx.slice.Slice;
import com.android.settings.homepage.deviceinfo.DataUsageSlice; import com.android.settings.homepage.deviceinfo.DataUsageSlice;
import com.android.settings.homepage.deviceinfo.DeviceInfoSlice; import com.android.settings.homepage.deviceinfo.DeviceInfoSlice;
@@ -31,6 +39,7 @@ import com.android.settingslib.utils.AsyncLoaderCompat;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.List; import java.util.List;
import java.util.stream.Collectors;
public class CardContentLoader extends AsyncLoaderCompat<List<ContextualCard>> { public class CardContentLoader extends AsyncLoaderCompat<List<ContextualCard>> {
private static final String TAG = "CardContentLoader"; private static final String TAG = "CardContentLoader";
@@ -59,8 +68,7 @@ public class CardContentLoader extends AsyncLoaderCompat<List<ContextualCard>> {
try (Cursor cursor = getContextualCardsFromProvider()) { try (Cursor cursor = getContextualCardsFromProvider()) {
if (cursor.getCount() == 0) { if (cursor.getCount() == 0) {
result.addAll(createStaticCards()); result.addAll(createStaticCards());
return result; } else {
}
for (cursor.moveToFirst(); !cursor.isAfterLast(); cursor.moveToNext()) { for (cursor.moveToFirst(); !cursor.isAfterLast(); cursor.moveToNext()) {
final ContextualCard card = new ContextualCard(cursor); final ContextualCard card = new ContextualCard(cursor);
if (card.isCustomCard()) { if (card.isCustomCard()) {
@@ -70,7 +78,8 @@ public class CardContentLoader extends AsyncLoaderCompat<List<ContextualCard>> {
} }
} }
} }
return result; }
return filter(result);
} }
@VisibleForTesting @VisibleForTesting
@@ -95,15 +104,15 @@ public class CardContentLoader extends AsyncLoaderCompat<List<ContextualCard>> {
.build()); .build());
//TODO(b/115971399): Will change following values of SliceUri and Name //TODO(b/115971399): Will change following values of SliceUri and Name
// after landing these slice cards. // after landing these slice cards.
add(new ContextualCard.Builder() // add(new ContextualCard.Builder()
.setSliceUri("content://com.android.settings.slices/intent/battery_card") // .setSliceUri("content://com.android.settings.slices/battery_card")
.setName(packageName + "/" + "battery_card") // .setName(packageName + "/" + "battery_card")
.setPackageName(packageName) // .setPackageName(packageName)
.setRankingScore(rankingScore) // .setRankingScore(rankingScore)
.setAppVersion(appVersionCode) // .setAppVersion(appVersionCode)
.setCardType(ContextualCard.CardType.SLICE) // .setCardType(ContextualCard.CardType.SLICE)
.setIsHalfWidth(true) // .setIsHalfWidth(true)
.build()); // .build());
add(new ContextualCard.Builder() add(new ContextualCard.Builder()
.setSliceUri(DeviceInfoSlice.DEVICE_INFO_CARD_URI.toString()) .setSliceUri(DeviceInfoSlice.DEVICE_INFO_CARD_URI.toString())
.setName(packageName + "/" + DeviceInfoSlice.PATH_DEVICE_INFO_CARD) .setName(packageName + "/" + DeviceInfoSlice.PATH_DEVICE_INFO_CARD)
@@ -126,6 +135,41 @@ public class CardContentLoader extends AsyncLoaderCompat<List<ContextualCard>> {
return result; return result;
} }
@VisibleForTesting
List<ContextualCard> filter(List<ContextualCard> candidates) {
return candidates.stream().filter(card -> isCardEligibleToDisplay(card)).collect(
Collectors.toList());
}
@VisibleForTesting
boolean isCardEligibleToDisplay(ContextualCard card) {
if (card.isCustomCard()) {
return true;
}
final Uri uri = card.getSliceUri();
if (!ContentResolver.SCHEME_CONTENT.equals(uri.getScheme())) {
return false;
}
//check if the uri has a provider associated with.
final ContentProviderClient provider =
mContext.getContentResolver().acquireContentProviderClient(uri);
if (provider == null) {
return false;
}
//release contentProviderClient to prevent from memory leak.
provider.release();
final Slice slice = Slice.bindSlice(mContext, uri, SUPPORTED_SPECS);
if (slice == null || slice.hasHint(HINT_ERROR)) {
return false;
}
return true;
}
private long getAppVersionCode() { private long getAppVersionCode() {
try { try {
return mContext.getPackageManager().getPackageInfo(mContext.getPackageName(), return mContext.getPackageManager().getPackageInfo(mContext.getPackageName(),

View File

@@ -18,128 +18,92 @@ package com.android.settings.homepage;
import static com.google.common.truth.Truth.assertThat; import static com.google.common.truth.Truth.assertThat;
import static org.mockito.Mockito.doReturn;
import static org.mockito.Mockito.spy;
import android.content.Context; import android.content.Context;
import android.database.Cursor; import android.net.Uri;
import android.database.MatrixCursor;
import com.android.settings.homepage.deviceinfo.DataUsageSlice;
import com.android.settings.homepage.deviceinfo.DeviceInfoSlice;
import com.android.settings.slices.SettingsSliceProvider;
import com.android.settings.testutils.SettingsRobolectricTestRunner; import com.android.settings.testutils.SettingsRobolectricTestRunner;
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.robolectric.RuntimeEnvironment; import org.robolectric.RuntimeEnvironment;
import org.robolectric.shadows.ShadowContentResolver;
import java.util.Arrays;
import java.util.List; import java.util.List;
import java.util.stream.Collectors;
@RunWith(SettingsRobolectricTestRunner.class) @RunWith(SettingsRobolectricTestRunner.class)
public class CardContentLoaderTest { public class CardContentLoaderTest {
private static final String[] QUERY_PROJECTION = {
CardDatabaseHelper.CardColumns.NAME,
CardDatabaseHelper.CardColumns.TYPE,
CardDatabaseHelper.CardColumns.SCORE,
CardDatabaseHelper.CardColumns.SLICE_URI,
CardDatabaseHelper.CardColumns.CATEGORY,
CardDatabaseHelper.CardColumns.LOCALIZED_TO_LOCALE,
CardDatabaseHelper.CardColumns.PACKAGE_NAME,
CardDatabaseHelper.CardColumns.APP_VERSION,
CardDatabaseHelper.CardColumns.TITLE_RES_NAME,
CardDatabaseHelper.CardColumns.TITLE_TEXT,
CardDatabaseHelper.CardColumns.SUMMARY_RES_NAME,
CardDatabaseHelper.CardColumns.SUMMARY_TEXT,
CardDatabaseHelper.CardColumns.ICON_RES_NAME,
CardDatabaseHelper.CardColumns.ICON_RES_ID,
CardDatabaseHelper.CardColumns.CARD_ACTION,
CardDatabaseHelper.CardColumns.EXPIRE_TIME_MS,
CardDatabaseHelper.CardColumns.SUPPORT_HALF_WIDTH
};
private Context mContext; private Context mContext;
private CardContentLoader mCardContentLoader; private CardContentLoader mCardContentLoader;
private SettingsSliceProvider mProvider;
@Before @Before
public void setUp() { public void setUp() {
mContext = RuntimeEnvironment.application; mContext = RuntimeEnvironment.application;
mCardContentLoader = spy(new CardContentLoader(mContext)); mCardContentLoader = new CardContentLoader(mContext);
mProvider = new SettingsSliceProvider();
ShadowContentResolver.registerProviderInternal(SettingsSliceProvider.SLICE_AUTHORITY,
mProvider);
} }
@Test @Test
public void loadInBackground_hasDataInDb_shouldReturnData() { public void createStaticCards_shouldReturnTwoCards() {
final Cursor cursor = generateTwoRowContextualCards(); final List<ContextualCard> defaultData = mCardContentLoader.createStaticCards();
doReturn(cursor).when(mCardContentLoader).getContextualCardsFromProvider();
final List<ContextualCard> contextualCards = mCardContentLoader.loadInBackground(); assertThat(defaultData).hasSize(2);
assertThat(contextualCards.size()).isEqualTo(cursor.getCount());
} }
@Test @Test
public void loadInBackground_hasNoData_shouldReturnThreeDefaultData() { public void createStaticCards_shouldContainDataUsageAndDeviceInfo() {
final Cursor cursor = generateEmptyContextualCards(); final Uri dataUsage = DataUsageSlice.DATA_USAGE_CARD_URI;
doReturn(cursor).when(mCardContentLoader).getContextualCardsFromProvider(); final Uri deviceInfo = DeviceInfoSlice.DEVICE_INFO_CARD_URI;
final List<Uri> expectedUris = Arrays.asList(dataUsage, deviceInfo);
final List<ContextualCard> contextualCards = mCardContentLoader.loadInBackground(); final List<Uri> actualCardUris = mCardContentLoader.createStaticCards().stream().map(
ContextualCard::getSliceUri).collect(Collectors.toList());
assertThat(contextualCards.size()).isEqualTo(mCardContentLoader.createStaticCards().size()); assertThat(actualCardUris).containsExactlyElementsIn(expectedUris);
} }
private MatrixCursor generateEmptyContextualCards() { @Test
final MatrixCursor result = new MatrixCursor(QUERY_PROJECTION); public void isCardEligibleToDisplay_customCard_returnTrue() {
return result; final ContextualCard customCard = new ContextualCard.Builder()
.setName("custom_card")
.setCardType(ContextualCard.CardType.DEFAULT)
.setTitleText("custom_title")
.setSummaryText("custom_summary")
.build();
assertThat(mCardContentLoader.isCardEligibleToDisplay(customCard)).isTrue();
} }
private MatrixCursor generateTwoRowContextualCards() { @Test
final MatrixCursor result = generateEmptyContextualCards(); public void isCardEligibleToDisplay_invalidScheme_returnFalse() {
result.addRow(generateFirstFakeData()); final String sliceUri = "contet://com.android.settings.slices/action/flashlight";
result.addRow(generateSecondFakeData());
return result; assertThat(
mCardContentLoader.isCardEligibleToDisplay(getContextualCard(sliceUri))).isFalse();
} }
private Object[] generateFirstFakeData() { @Test
final Object[] ref = new Object[]{ public void isCardEligibleToDisplay_noProvider_returnFalse() {
"auto_rotate", /* NAME */ final String sliceUri = "content://com.android.settings.test.slices/action/flashlight";
ContextualCard.CardType.SLICE, /* TYPE */
0.5, /* SCORE */ assertThat(
"content://com.android.settings.slices/action/auto_rotate", /* SLICE_URI */ mCardContentLoader.isCardEligibleToDisplay(getContextualCard(sliceUri))).isFalse();
2, /* CATEGORY */
"", /* LOCALIZED_TO_LOCALE */
"com.android.settings", /* PACKAGE_NAME */
1l, /* APP_VERSION */
"", /* TITLE_RES_NAME */
"", /* TITLE_TEXT */
"", /* SUMMARY_RES_NAME */
"", /* SUMMARY_TEXT */
"", /* ICON_RES_NAME */
0, /* ICON_RES_ID */
0, /* CARD_ACTION */
-1, /* EXPIRE_TIME_MS */
0 /* SUPPORT_HALF_WIDTH */
};
return ref;
} }
private Object[] generateSecondFakeData() { private ContextualCard getContextualCard(String sliceUri) {
final Object[] ref = new Object[]{ return new ContextualCard.Builder()
"toggle_airplane", /* NAME */ .setName("test_card")
ContextualCard.CardType.SLICE, /* TYPE */ .setCardType(ContextualCard.CardType.SLICE)
0.5, /* SCORE */ .setSliceUri(sliceUri)
"content://com.android.settings.slices/action/toggle_airplane", /* SLICE_URI */ .build();
2, /* CATEGORY */
"", /* LOCALIZED_TO_LOCALE */
"com.android.settings", /* PACKAGE_NAME */
1l, /* APP_VERSION */
"", /* TITLE_RES_NAME */
"", /* TITLE_TEXT */
"", /* SUMMARY_RES_NAME */
"", /* SUMMARY_TEXT */
"", /* ICON_RES_NAME */
0, /* ICON_RES_ID */
0, /* CARD_ACTION */
-1, /* EXPIRE_TIME_MS */
0 /* SUPPORT_HALF_WIDTH */
};
return ref;
} }
} }

View File

@@ -0,0 +1,67 @@
/*
* Copyright (C) 2018 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 com.android.settings.homepage;
import static com.google.common.truth.Truth.assertThat;
import android.content.Context;
import android.support.test.InstrumentationRegistry;
import android.support.test.runner.AndroidJUnit4;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
import java.util.ArrayList;
import java.util.List;
@RunWith(AndroidJUnit4.class)
public class CardContentLoaderTest {
private Context mContext;
private CardContentLoader mCardContentLoader;
@Before
public void setUp() {
mContext = InstrumentationRegistry.getTargetContext();
mCardContentLoader = new CardContentLoader(mContext);
}
@Test
public void filter_twoInvalidCards_shouldReturnOneCard() {
final String sliceUri1 = "content://com.android.settings.slices/action/flashlight"; //valid
final String sliceUri2 = "content://com.android.settings.test.slices/action/flashlight";
final String sliceUri3 = "cotent://com.android.settings.slices/action/flashlight";
final List<ContextualCard> cards = new ArrayList<>();
cards.add(getContextualCard(sliceUri1));
cards.add(getContextualCard(sliceUri2));
cards.add(getContextualCard(sliceUri3));
final List<ContextualCard> result = mCardContentLoader.filter(cards);
assertThat(result).hasSize(1);
}
private ContextualCard getContextualCard(String sliceUri) {
return new ContextualCard.Builder()
.setName("test_card")
.setCardType(ContextualCard.CardType.SLICE)
.setSliceUri(sliceUri)
.build();
}
}