Merge "[Panlingual] Improve performance of displaying app list in Settings."

This commit is contained in:
TreeHugger Robot
2022-11-12 00:16:12 +00:00
committed by Android (Google) Code Review
7 changed files with 97 additions and 82 deletions

View File

@@ -17,48 +17,55 @@
package com.android.settings.applications; package com.android.settings.applications;
import android.annotation.NonNull; import android.annotation.NonNull;
import android.app.ActivityManager;
import android.app.LocaleConfig; import android.app.LocaleConfig;
import android.content.Context; import android.content.Context;
import android.content.Intent; import android.content.Intent;
import android.content.pm.PackageInfo; import android.content.pm.ApplicationInfo;
import android.content.pm.PackageManager; import android.content.pm.PackageManager;
import android.content.pm.ResolveInfo; import android.content.pm.ResolveInfo;
import android.os.LocaleList; import android.os.LocaleList;
import android.text.TextUtils;
import android.util.FeatureFlagUtils; import android.util.FeatureFlagUtils;
import android.util.Log; import android.util.Log;
import androidx.annotation.VisibleForTesting;
import com.android.settings.R; import com.android.settings.R;
import java.util.List; import java.util.List;
/** This class provides methods that help dealing with per app locale. */ /**
* This class provides methods that help dealing with per app locale.
*/
public class AppLocaleUtil { public class AppLocaleUtil {
private static final String TAG = AppLocaleUtil.class.getSimpleName(); private static final String TAG = AppLocaleUtil.class.getSimpleName();
public static final Intent LAUNCHER_ENTRY_INTENT = public static final Intent LAUNCHER_ENTRY_INTENT =
new Intent(Intent.ACTION_MAIN).addCategory(Intent.CATEGORY_LAUNCHER); new Intent(Intent.ACTION_MAIN).addCategory(Intent.CATEGORY_LAUNCHER);
@VisibleForTesting
static LocaleConfig sLocaleConfig;
/** /**
* Decides the UI display of per app locale. * Decides the UI display of per app locale.
*/ */
public static boolean canDisplayLocaleUi( public static boolean canDisplayLocaleUi(
@NonNull Context context, @NonNull Context context,
@NonNull String packageName, @NonNull ApplicationInfo app,
@NonNull List<ResolveInfo> infos) { @NonNull List<ResolveInfo> infos) {
boolean isDisallowedPackage = isDisallowedPackage(context, packageName); boolean isDisallowedPackage = isDisallowedPackage(context, app.packageName);
boolean hasLauncherEntry = hasLauncherEntry(packageName, infos); boolean hasLauncherEntry = hasLauncherEntry(app.packageName, infos);
boolean isSignedWithPlatformKey = isSignedWithPlatformKey(context, packageName); boolean isSignedWithPlatformKey = app.isSignedWithPlatformKey();
boolean canDisplay = !isDisallowedPackage boolean canDisplay = !isDisallowedPackage
&& !isSignedWithPlatformKey && !isSignedWithPlatformKey
&& hasLauncherEntry && hasLauncherEntry
&& isAppLocaleSupported(context, packageName); && isAppLocaleSupported(context, app.packageName);
Log.i(TAG, "Can display preference - [" + packageName + "] :" Log.i(TAG, "Can display preference - [" + app.packageName + "] :"
+ " isDisallowedPackage : " + isDisallowedPackage + " isDisallowedPackage : " + isDisallowedPackage
+ " / isSignedWithPlatformKey : " + isSignedWithPlatformKey + " / isSignedWithPlatformKey : " + isSignedWithPlatformKey
+ " / hasLauncherEntry : " + hasLauncherEntry + " / hasLauncherEntry : " + hasLauncherEntry
+ " / canDisplay : " + canDisplay); + " / canDisplay : " + canDisplay + " / 1.1");
return canDisplay; return canDisplay;
} }
@@ -66,30 +73,13 @@ public class AppLocaleUtil {
final String[] disallowedPackages = context.getResources().getStringArray( final String[] disallowedPackages = context.getResources().getStringArray(
R.array.config_disallowed_app_localeChange_packages); R.array.config_disallowed_app_localeChange_packages);
for (String disallowedPackage : disallowedPackages) { for (String disallowedPackage : disallowedPackages) {
if (packageName.equals(disallowedPackage)) { if (TextUtils.equals(packageName, disallowedPackage)) {
return true; return true;
} }
} }
return false; return false;
} }
private static boolean isSignedWithPlatformKey(Context context, String packageName) {
PackageInfo packageInfo = null;
PackageManager packageManager = context.getPackageManager();
ActivityManager activityManager = context.getSystemService(ActivityManager.class);
try {
packageInfo = packageManager.getPackageInfoAsUser(
packageName, /* flags= */ 0,
activityManager.getCurrentUser());
} catch (PackageManager.NameNotFoundException ex) {
Log.e(TAG, "package not found: " + packageName);
}
if (packageInfo == null) {
return false;
}
return packageInfo.applicationInfo.isSignedWithPlatformKey();
}
private static boolean hasLauncherEntry(String packageName, List<ResolveInfo> infos) { private static boolean hasLauncherEntry(String packageName, List<ResolveInfo> infos) {
return infos.stream() return infos.stream()
.anyMatch(info -> info.activityInfo.packageName.equals(packageName)); .anyMatch(info -> info.activityInfo.packageName.equals(packageName));
@@ -99,7 +89,13 @@ public class AppLocaleUtil {
* Check the function of per app language is supported by current application. * Check the function of per app language is supported by current application.
*/ */
public static boolean isAppLocaleSupported(Context context, String packageName) { public static boolean isAppLocaleSupported(Context context, String packageName) {
LocaleList localeList = getPackageLocales(context, packageName); LocaleList localeList;
if (sLocaleConfig != null) {
localeList = getPackageLocales(sLocaleConfig);
} else {
localeList = getPackageLocales(context, packageName);
}
if (localeList != null) { if (localeList != null) {
return localeList.size() > 0; return localeList.size() > 0;
} }
@@ -118,9 +114,8 @@ public class AppLocaleUtil {
public static String[] getAssetLocales(Context context, String packageName) { public static String[] getAssetLocales(Context context, String packageName) {
try { try {
PackageManager packageManager = context.getPackageManager(); PackageManager packageManager = context.getPackageManager();
String[] locales = packageManager.getResourcesForApplication( String[] locales = packageManager.getResourcesForApplication(packageName)
packageManager.getPackageInfo(packageName, PackageManager.MATCH_ALL) .getAssets().getNonSystemLocales();
.applicationInfo).getAssets().getNonSystemLocales();
if (locales == null) { if (locales == null) {
Log.i(TAG, "[" + packageName + "] locales are null."); Log.i(TAG, "[" + packageName + "] locales are null.");
} }
@@ -137,6 +132,14 @@ public class AppLocaleUtil {
return new String[0]; return new String[0];
} }
@VisibleForTesting
static LocaleList getPackageLocales(LocaleConfig localeConfig) {
if (localeConfig.getStatus() == LocaleConfig.STATUS_SUCCESS) {
return localeConfig.getSupportedLocales();
}
return null;
}
/** /**
* Get locales from LocaleConfig. * Get locales from LocaleConfig.
*/ */
@@ -144,9 +147,7 @@ public class AppLocaleUtil {
try { try {
LocaleConfig localeConfig = LocaleConfig localeConfig =
new LocaleConfig(context.createPackageContext(packageName, 0)); new LocaleConfig(context.createPackageContext(packageName, 0));
if (localeConfig.getStatus() == LocaleConfig.STATUS_SUCCESS) { return getPackageLocales(localeConfig);
return localeConfig.getSupportedLocales();
}
} catch (PackageManager.NameNotFoundException e) { } catch (PackageManager.NameNotFoundException e) {
Log.w(TAG, "Can not found the package name : " + packageName + " / " + e); Log.w(TAG, "Can not found the package name : " + packageName + " / " + e);
} }

View File

@@ -55,8 +55,7 @@ public class AppStateLocaleBridge extends AppStateBaseBridge {
AppInfoByProfiles appInfoByProfiles = getAppInfo(UserHandle.getUserId(uid)); AppInfoByProfiles appInfoByProfiles = getAppInfo(UserHandle.getUserId(uid));
app.extraInfo = AppLocaleUtil.canDisplayLocaleUi(appInfoByProfiles.mContextAsUser, app.extraInfo = AppLocaleUtil.canDisplayLocaleUi(appInfoByProfiles.mContextAsUser,
app.info.packageName, app.info, appInfoByProfiles.mListInfos) ? Boolean.TRUE : Boolean.FALSE;
appInfoByProfiles.mListInfos) ? Boolean.TRUE : Boolean.FALSE;
} }
@Override @Override
@@ -67,8 +66,7 @@ public class AppStateLocaleBridge extends AppStateBaseBridge {
AppInfoByProfiles appInfoByProfiles = getAppInfo(UserHandle.getUserId(app.info.uid)); AppInfoByProfiles appInfoByProfiles = getAppInfo(UserHandle.getUserId(app.info.uid));
app.extraInfo = AppLocaleUtil.canDisplayLocaleUi(appInfoByProfiles.mContextAsUser, app.extraInfo = AppLocaleUtil.canDisplayLocaleUi(appInfoByProfiles.mContextAsUser,
app.info.packageName, app.info, appInfoByProfiles.mListInfos) ? Boolean.TRUE : Boolean.FALSE;
appInfoByProfiles.mListInfos) ? Boolean.TRUE : Boolean.FALSE;
} }
} }

View File

@@ -84,6 +84,6 @@ public class AppLocalePreferenceController extends AppInfoPreferenceControllerBa
@VisibleForTesting @VisibleForTesting
boolean canDisplayLocaleUi() { boolean canDisplayLocaleUi() {
return AppLocaleUtil return AppLocaleUtil
.canDisplayLocaleUi(mContext, mParent.getAppEntry().info.packageName, mListInfos); .canDisplayLocaleUi(mContext, mParent.getAppEntry().info, mListInfos);
} }
} }

View File

@@ -69,7 +69,7 @@ private class AppLocalePresenter(
ResolveInfoFlags.of(PackageManager.GET_META_DATA.toLong()), ResolveInfoFlags.of(PackageManager.GET_META_DATA.toLong()),
app.userId, app.userId,
) )
AppLocaleUtil.canDisplayLocaleUi(context, app.packageName, resolveInfos) AppLocaleUtil.canDisplayLocaleUi(context, app, resolveInfos)
} }
val summaryFlow = flow { emit(getSummary()) } val summaryFlow = flow { emit(getSummary()) }

View File

@@ -58,7 +58,7 @@ class AppLanguagesListModel(private val context: Context) : AppListModel<AppLang
appList.map { app -> appList.map { app ->
AppLanguagesRecord(app, AppLanguagesRecord(app,
AppLocaleUtil.canDisplayLocaleUi(context, AppLocaleUtil.canDisplayLocaleUi(context,
app.packageName, resolveInfos)) app, resolveInfos))
} }
} }

View File

@@ -77,7 +77,8 @@ class AppLocalePreferenceTest {
.strictness(Strictness.LENIENT) .strictness(Strictness.LENIENT)
.startMocking() .startMocking()
whenever(context.packageManager).thenReturn(packageManager) whenever(context.packageManager).thenReturn(packageManager)
whenever(AppLocaleUtil.canDisplayLocaleUi(any(), eq(PACKAGE_NAME), any())).thenReturn(true) whenever(AppLocaleUtil.canDisplayLocaleUi(any(), ArgumentMatchers.eq(APP), any()))
.thenReturn(true)
whenever(AppLocaleDetails.getSummary(any(), ArgumentMatchers.eq(APP))).thenReturn(SUMMARY) whenever(AppLocaleDetails.getSummary(any(), ArgumentMatchers.eq(APP))).thenReturn(SUMMARY)
} }
@@ -88,7 +89,8 @@ class AppLocalePreferenceTest {
@Test @Test
fun whenCanNotDisplayLocalUi_notDisplayed() { fun whenCanNotDisplayLocalUi_notDisplayed() {
whenever(AppLocaleUtil.canDisplayLocaleUi(any(), eq(PACKAGE_NAME), any())).thenReturn(false) whenever(AppLocaleUtil.canDisplayLocaleUi(any(), ArgumentMatchers.eq(APP), any()))
.thenReturn(false)
setContent() setContent()

View File

@@ -17,27 +17,29 @@
package com.android.settings.applications; package com.android.settings.applications;
import static org.junit.Assert.assertFalse; import static org.junit.Assert.assertFalse;
import static org.junit.Assert.assertNull;
import static org.junit.Assert.assertTrue; import static org.junit.Assert.assertTrue;
import static org.mockito.ArgumentMatchers.anyInt; import static org.mockito.ArgumentMatchers.anyInt;
import static org.mockito.ArgumentMatchers.anyString;
import static org.mockito.Mockito.mock; import static org.mockito.Mockito.mock;
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.ActivityManager; import android.app.ActivityManager;
import android.app.LocaleConfig;
import android.content.Context; import android.content.Context;
import android.content.pm.ActivityInfo; import android.content.pm.ActivityInfo;
import android.content.pm.ApplicationInfo; import android.content.pm.ApplicationInfo;
import android.content.pm.PackageInfo;
import android.content.pm.PackageManager; import android.content.pm.PackageManager;
import android.content.pm.ResolveInfo; import android.content.pm.ResolveInfo;
import android.content.res.Resources; import android.content.res.Resources;
import android.os.LocaleList;
import android.util.FeatureFlagUtils;
import androidx.test.core.app.ApplicationProvider; import androidx.test.core.app.ApplicationProvider;
import androidx.test.ext.junit.runners.AndroidJUnit4; import androidx.test.ext.junit.runners.AndroidJUnit4;
import org.junit.After;
import org.junit.Before; import org.junit.Before;
import org.junit.Ignore;
import org.junit.Test; import org.junit.Test;
import org.junit.runner.RunWith; import org.junit.runner.RunWith;
import org.mockito.Mock; import org.mockito.Mock;
@@ -53,14 +55,15 @@ public class AppLocaleUtilTest {
@Mock @Mock
private ActivityManager mActivityManager; private ActivityManager mActivityManager;
@Mock @Mock
private ApplicationInfo mApplicationInfo;
@Mock
private Resources mResources; private Resources mResources;
@Mock
private LocaleConfig mLocaleConfig;
private Context mContext; private Context mContext;
private String mDisallowedPackage = "com.disallowed.package"; private String mDisallowedPackage = "com.disallowed.package";
private String mAllowedPackage = "com.allowed.package"; private String mAllowedPackage = "com.allowed.package";
private List<ResolveInfo> mListResolveInfo; private List<ResolveInfo> mListResolveInfo;
private ApplicationInfo mAppInfo;
@Before @Before
public void setUp() { public void setUp() {
@@ -68,47 +71,72 @@ public class AppLocaleUtilTest {
mContext = spy(ApplicationProvider.getApplicationContext()); mContext = spy(ApplicationProvider.getApplicationContext());
when(mContext.getPackageManager()).thenReturn(mPackageManager); when(mContext.getPackageManager()).thenReturn(mPackageManager);
when(mContext.getSystemService(ActivityManager.class)).thenReturn(mActivityManager); when(mContext.getSystemService(ActivityManager.class)).thenReturn(mActivityManager);
FeatureFlagUtils.setEnabled(mContext, FeatureFlagUtils.SETTINGS_APP_LOCALE_OPT_IN_ENABLED,
true);
setDisallowedPackageName(mDisallowedPackage); setDisallowedPackageName(mDisallowedPackage);
mAppInfo = new ApplicationInfo();
mLocaleConfig = mock(LocaleConfig.class);
}
@After
public void tearDown() {
AppLocaleUtil.sLocaleConfig = null;
} }
@Test @Test
@Ignore("b/231904717") public void canDisplayLocaleUi_showUI() {
public void canDisplayLocaleUi_showUI() throws PackageManager.NameNotFoundException { when(mLocaleConfig.getStatus()).thenReturn(LocaleConfig.STATUS_SUCCESS);
setApplicationInfo(/*no platform key*/ false); when(mLocaleConfig.getSupportedLocales()).thenReturn(LocaleList.forLanguageTags("en-US"));
AppLocaleUtil.sLocaleConfig = mLocaleConfig;
setActivityInfo(mAllowedPackage); setActivityInfo(mAllowedPackage);
mAppInfo.packageName = mAllowedPackage;
assertTrue(AppLocaleUtil.canDisplayLocaleUi(mContext, mAllowedPackage, mListResolveInfo)); assertTrue(AppLocaleUtil.canDisplayLocaleUi(mContext, mAppInfo, mListResolveInfo));
} }
@Test @Test
@Ignore("b/231904717") public void canDisplayLocaleUi_notShowUI_hasPlatformKey() {
public void canDisplayLocaleUi_notShowUI_hasPlatformKey()
throws PackageManager.NameNotFoundException {
setApplicationInfo(/*has platform key*/ true);
setActivityInfo(mAllowedPackage); setActivityInfo(mAllowedPackage);
mAppInfo.privateFlags |= ApplicationInfo.PRIVATE_FLAG_SIGNED_WITH_PLATFORM_KEY;
mAppInfo.packageName = mAllowedPackage;
assertFalse(AppLocaleUtil.canDisplayLocaleUi(mContext, mAllowedPackage, mListResolveInfo)); assertFalse(AppLocaleUtil.canDisplayLocaleUi(mContext, mAppInfo, mListResolveInfo));
} }
@Test @Test
@Ignore("b/231904717") public void canDisplayLocaleUi_notShowUI_noLauncherEntry() {
public void canDisplayLocaleUi_notShowUI_noLauncherEntry()
throws PackageManager.NameNotFoundException {
setApplicationInfo(/*no platform key*/false);
setActivityInfo(""); setActivityInfo("");
assertFalse(AppLocaleUtil.canDisplayLocaleUi(mContext, mAllowedPackage, mListResolveInfo)); assertFalse(AppLocaleUtil.canDisplayLocaleUi(mContext, mAppInfo, mListResolveInfo));
} }
@Test @Test
@Ignore("b/231904717") public void canDisplayLocaleUi_notShowUI_matchDisallowedPackageList() {
public void canDisplayLocaleUi_notShowUI_matchDisallowedPackageList()
throws PackageManager.NameNotFoundException {
setApplicationInfo(/*no platform key*/false);
setActivityInfo(""); setActivityInfo("");
mAppInfo.packageName = mDisallowedPackage;
assertFalse(AppLocaleUtil assertFalse(AppLocaleUtil
.canDisplayLocaleUi(mContext, mDisallowedPackage, mListResolveInfo)); .canDisplayLocaleUi(mContext, mAppInfo, mListResolveInfo));
}
@Test
public void getPackageLocales_getLocales_success() {
when(mLocaleConfig.getStatus()).thenReturn(LocaleConfig.STATUS_SUCCESS);
when(mLocaleConfig.getSupportedLocales()).thenReturn(LocaleList.forLanguageTags("en-US"));
LocaleList result = AppLocaleUtil.getPackageLocales(mLocaleConfig);
assertFalse(result.isEmpty());
}
@Test
public void getPackageLocales_getLocales_failed() {
when(mLocaleConfig.getStatus()).thenReturn(LocaleConfig.STATUS_PARSING_FAILED);
LocaleList result = AppLocaleUtil.getPackageLocales(mLocaleConfig);
assertNull(result);
} }
private void setDisallowedPackageName(String packageName) { private void setDisallowedPackageName(String packageName) {
@@ -116,20 +144,6 @@ public class AppLocaleUtilTest {
when(mResources.getStringArray(anyInt())).thenReturn(new String[]{packageName}); when(mResources.getStringArray(anyInt())).thenReturn(new String[]{packageName});
} }
private void setApplicationInfo(boolean signedWithPlatformKey)
throws PackageManager.NameNotFoundException {
ApplicationInfo applicationInfo = new ApplicationInfo();
if (signedWithPlatformKey) {
applicationInfo.privateFlags = applicationInfo.privateFlags
| ApplicationInfo.PRIVATE_FLAG_SIGNED_WITH_PLATFORM_KEY;
}
PackageInfo packageInfo = new PackageInfo();
packageInfo.applicationInfo = applicationInfo;
when(mPackageManager.getPackageInfoAsUser(anyString(), anyInt(), anyInt())).thenReturn(
packageInfo);
}
private void setActivityInfo(String packageName) { private void setActivityInfo(String packageName) {
ResolveInfo resolveInfo = mock(ResolveInfo.class); ResolveInfo resolveInfo = mock(ResolveInfo.class);
ActivityInfo activityInfo = mock(ActivityInfo.class); ActivityInfo activityInfo = mock(ActivityInfo.class);