Bug: 374968702 Test: UserAspectRatioDetailsTest Test: UserAspectRatioManagerTest Test: manual, verify which option is selected by default after app installation Flag: com.android.window.flags.universal_resizable_by_default Change-Id: I04063e4a130d42a4bfd7e548940ca5e654a6d012
384 lines
18 KiB
Java
384 lines
18 KiB
Java
/*
|
|
* Copyright (C) 2023 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.applications.appcompat;
|
|
|
|
import static android.content.pm.ActivityInfo.OVERRIDE_ANY_ORIENTATION_TO_USER;
|
|
import static android.content.pm.PackageManager.USER_MIN_ASPECT_RATIO_16_9;
|
|
import static android.content.pm.PackageManager.USER_MIN_ASPECT_RATIO_3_2;
|
|
import static android.content.pm.PackageManager.USER_MIN_ASPECT_RATIO_4_3;
|
|
import static android.content.pm.PackageManager.USER_MIN_ASPECT_RATIO_APP_DEFAULT;
|
|
import static android.content.pm.PackageManager.USER_MIN_ASPECT_RATIO_DISPLAY_SIZE;
|
|
import static android.content.pm.PackageManager.USER_MIN_ASPECT_RATIO_FULLSCREEN;
|
|
import static android.content.pm.PackageManager.USER_MIN_ASPECT_RATIO_SPLIT_SCREEN;
|
|
import static android.content.pm.PackageManager.USER_MIN_ASPECT_RATIO_UNSET;
|
|
import static android.os.UserHandle.getUserHandleForUid;
|
|
import static android.os.UserHandle.getUserId;
|
|
import static android.view.WindowManager.PROPERTY_COMPAT_ALLOW_ORIENTATION_OVERRIDE;
|
|
import static android.view.WindowManager.PROPERTY_COMPAT_ALLOW_USER_ASPECT_RATIO_FULLSCREEN_OVERRIDE;
|
|
import static android.view.WindowManager.PROPERTY_COMPAT_ALLOW_USER_ASPECT_RATIO_OVERRIDE;
|
|
|
|
import static java.lang.Boolean.FALSE;
|
|
|
|
import android.app.AppGlobals;
|
|
import android.app.compat.CompatChanges;
|
|
import android.content.Context;
|
|
import android.content.pm.ActivityInfo;
|
|
import android.content.pm.ApplicationInfo;
|
|
import android.content.pm.IPackageManager;
|
|
import android.content.pm.LauncherApps;
|
|
import android.content.pm.PackageManager;
|
|
import android.os.RemoteException;
|
|
import android.os.UserHandle;
|
|
import android.provider.DeviceConfig;
|
|
import android.util.ArrayMap;
|
|
import android.util.Log;
|
|
import android.util.SparseIntArray;
|
|
|
|
import androidx.annotation.NonNull;
|
|
import androidx.annotation.Nullable;
|
|
|
|
import com.android.settings.R;
|
|
import com.android.settings.Utils;
|
|
import com.android.window.flags.Flags;
|
|
|
|
import com.google.common.annotations.VisibleForTesting;
|
|
|
|
import java.util.Map;
|
|
|
|
/**
|
|
* Helper class for handling app aspect ratio override
|
|
* {@link PackageManager.UserMinAspectRatio} set by user
|
|
*/
|
|
public class UserAspectRatioManager {
|
|
private static final boolean DEFAULT_VALUE_ENABLE_USER_ASPECT_RATIO_SETTINGS = true;
|
|
@VisibleForTesting
|
|
static final String KEY_ENABLE_USER_ASPECT_RATIO_SETTINGS =
|
|
"enable_app_compat_aspect_ratio_user_settings";
|
|
static final String KEY_ENABLE_USER_ASPECT_RATIO_FULLSCREEN =
|
|
"enable_app_compat_user_aspect_ratio_fullscreen";
|
|
private static final boolean DEFAULT_VALUE_ENABLE_USER_ASPECT_RATIO_FULLSCREEN = true;
|
|
|
|
final boolean mIsUserMinAspectRatioAppDefaultFlagEnabled = Flags.userMinAspectRatioAppDefault();
|
|
private final boolean mIgnoreActivityOrientationRequest;
|
|
|
|
private final Context mContext;
|
|
private final IPackageManager mIPm;
|
|
/** Apps that have launcher entry defined in manifest */
|
|
private final Map<Integer, String> mUserAspectRatioMap;
|
|
private final Map<Integer, CharSequence> mUserAspectRatioA11yMap;
|
|
private final SparseIntArray mUserAspectRatioOrder;
|
|
|
|
public UserAspectRatioManager(@NonNull Context context) {
|
|
this(context, AppGlobals.getPackageManager());
|
|
}
|
|
|
|
@VisibleForTesting
|
|
UserAspectRatioManager(@NonNull Context context, @NonNull IPackageManager pm) {
|
|
mContext = context;
|
|
mIPm = pm;
|
|
mUserAspectRatioA11yMap = new ArrayMap<>();
|
|
mUserAspectRatioOrder = new SparseIntArray();
|
|
mUserAspectRatioMap = getUserMinAspectRatioMapping();
|
|
mIgnoreActivityOrientationRequest = getValueFromDeviceConfig(
|
|
"ignore_activity_orientation_request", false);
|
|
}
|
|
|
|
/**
|
|
* Whether user aspect ratio settings is enabled for device.
|
|
*/
|
|
public static boolean isFeatureEnabled(Context context) {
|
|
final boolean isBuildTimeFlagEnabled = context.getResources().getBoolean(
|
|
com.android.internal.R.bool.config_appCompatUserAppAspectRatioSettingsIsEnabled);
|
|
return getValueFromDeviceConfig(KEY_ENABLE_USER_ASPECT_RATIO_SETTINGS,
|
|
DEFAULT_VALUE_ENABLE_USER_ASPECT_RATIO_SETTINGS) && isBuildTimeFlagEnabled;
|
|
}
|
|
|
|
/**
|
|
* @return user-specific {@link PackageManager.UserMinAspectRatio} override for an app
|
|
*/
|
|
@PackageManager.UserMinAspectRatio
|
|
public int getUserMinAspectRatioValue(@NonNull String packageName, int uid)
|
|
throws RemoteException {
|
|
final int aspectRatio = mIPm.getUserMinAspectRatio(packageName, uid);
|
|
return hasAspectRatioOption(aspectRatio, packageName)
|
|
? aspectRatio : USER_MIN_ASPECT_RATIO_UNSET;
|
|
}
|
|
|
|
// TODO b/374903057 reuse method from ActivityRecord
|
|
boolean isUniversalResizeable(@NonNull String packageName, int userId) {
|
|
try {
|
|
final ApplicationInfo info = mIPm.getApplicationInfo(
|
|
packageName, 0 /* flags */, userId);
|
|
if (info == null || info.category == ApplicationInfo.CATEGORY_GAME) {
|
|
return false;
|
|
}
|
|
final boolean compatEnabled = Flags.universalResizableByDefault()
|
|
&& info.isChangeEnabled(ActivityInfo.UNIVERSAL_RESIZABLE_BY_DEFAULT);
|
|
return compatEnabled || mIgnoreActivityOrientationRequest;
|
|
} catch (RemoteException e) {
|
|
Log.e("UserAspectRatioManager", "Could not access application info for "
|
|
+ packageName + ":\n" + e);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @return corresponding string for {@link PackageManager.UserMinAspectRatio} value
|
|
*/
|
|
@NonNull
|
|
public String getUserMinAspectRatioEntry(@PackageManager.UserMinAspectRatio int aspectRatio,
|
|
@NonNull String packageName, int userId) {
|
|
final String appDefault = getAspectRatioStringOrDefault(
|
|
mUserAspectRatioMap.get(USER_MIN_ASPECT_RATIO_UNSET),
|
|
USER_MIN_ASPECT_RATIO_UNSET);
|
|
|
|
if (!hasAspectRatioOption(aspectRatio, packageName)) {
|
|
return appDefault;
|
|
}
|
|
|
|
return isUnsetAndRequiresFullscreenOverride(packageName, userId, aspectRatio)
|
|
? getUserMinAspectRatioEntry(USER_MIN_ASPECT_RATIO_FULLSCREEN, packageName, userId)
|
|
: mUserAspectRatioMap.getOrDefault(aspectRatio, appDefault);
|
|
}
|
|
|
|
/**
|
|
* @return corresponding accessible string for {@link PackageManager.UserMinAspectRatio} value
|
|
*/
|
|
@NonNull
|
|
public CharSequence getAccessibleEntry(@PackageManager.UserMinAspectRatio int aspectRatio,
|
|
@NonNull String packageName) {
|
|
final int userId = mContext.getUserId();
|
|
return isUnsetAndRequiresFullscreenOverride(packageName, userId, aspectRatio)
|
|
? getAccessibleEntry(USER_MIN_ASPECT_RATIO_FULLSCREEN, packageName)
|
|
: mUserAspectRatioA11yMap.getOrDefault(aspectRatio,
|
|
getUserMinAspectRatioEntry(aspectRatio, packageName, userId));
|
|
}
|
|
|
|
/**
|
|
* @return corresponding aspect ratio string for package name and user
|
|
*/
|
|
@NonNull
|
|
public String getUserMinAspectRatioEntry(@NonNull String packageName, int userId)
|
|
throws RemoteException {
|
|
final int aspectRatio = getUserMinAspectRatioValue(packageName, userId);
|
|
return getUserMinAspectRatioEntry(aspectRatio, packageName, userId);
|
|
}
|
|
|
|
/**
|
|
* @return the order of the aspect ratio option corresponding to
|
|
* config_userAspectRatioOverrideValues
|
|
*/
|
|
int getUserMinAspectRatioOrder(@PackageManager.UserMinAspectRatio int option) {
|
|
return mUserAspectRatioOrder.get(option);
|
|
}
|
|
|
|
/**
|
|
* Whether user aspect ratio option is specified in
|
|
* {@link R.array.config_userAspectRatioOverrideValues}
|
|
* and is enabled by device config
|
|
*/
|
|
public boolean hasAspectRatioOption(@PackageManager.UserMinAspectRatio int option,
|
|
String packageName) {
|
|
if (option == USER_MIN_ASPECT_RATIO_FULLSCREEN && !isFullscreenOptionEnabled(packageName)) {
|
|
return false;
|
|
}
|
|
return mUserAspectRatioMap.containsKey(option);
|
|
}
|
|
|
|
/**
|
|
* Sets user-specified {@link PackageManager.UserMinAspectRatio} override for an app
|
|
*/
|
|
public void setUserMinAspectRatio(@NonNull String packageName, int uid,
|
|
@PackageManager.UserMinAspectRatio int aspectRatio) throws RemoteException {
|
|
mIPm.setUserMinAspectRatio(packageName, uid, aspectRatio);
|
|
}
|
|
|
|
/**
|
|
* Whether an app's aspect ratio can be overridden by user. Only apps with launcher entry
|
|
* will be overridable.
|
|
*/
|
|
public boolean canDisplayAspectRatioUi(@NonNull ApplicationInfo app) {
|
|
Boolean appAllowsUserAspectRatioOverride = readComponentProperty(
|
|
mContext.getPackageManager(), app.packageName,
|
|
PROPERTY_COMPAT_ALLOW_USER_ASPECT_RATIO_OVERRIDE);
|
|
return !FALSE.equals(appAllowsUserAspectRatioOverride) && hasLauncherEntry(app);
|
|
}
|
|
|
|
/**
|
|
* Whether the app has been overridden to fullscreen by device manufacturer or
|
|
* whether the app's aspect ratio has been overridden by the user.
|
|
*/
|
|
public boolean isAppOverridden(@NonNull ApplicationInfo app,
|
|
@PackageManager.UserMinAspectRatio int userOverride) {
|
|
return (userOverride != USER_MIN_ASPECT_RATIO_UNSET
|
|
&& userOverride != USER_MIN_ASPECT_RATIO_APP_DEFAULT)
|
|
|| isUnsetAndRequiresFullscreenOverride(app.packageName, getUserId(app.uid),
|
|
userOverride);
|
|
}
|
|
|
|
/**
|
|
* Whether fullscreen option in per-app user aspect ratio settings is enabled
|
|
*/
|
|
@VisibleForTesting
|
|
boolean isFullscreenOptionEnabled(String packageName) {
|
|
Boolean appAllowsFullscreenOption = readComponentProperty(mContext.getPackageManager(),
|
|
packageName, PROPERTY_COMPAT_ALLOW_USER_ASPECT_RATIO_FULLSCREEN_OVERRIDE);
|
|
final boolean isBuildTimeFlagEnabled = mContext.getResources().getBoolean(
|
|
com.android.internal.R.bool.config_appCompatUserAppAspectRatioFullscreenIsEnabled);
|
|
return !FALSE.equals(appAllowsFullscreenOption) && isBuildTimeFlagEnabled
|
|
&& getValueFromDeviceConfig(KEY_ENABLE_USER_ASPECT_RATIO_FULLSCREEN,
|
|
DEFAULT_VALUE_ENABLE_USER_ASPECT_RATIO_FULLSCREEN);
|
|
}
|
|
|
|
/**
|
|
* Whether the device manufacturer has overridden app's orientation to
|
|
* {@link android.content.pm.ActivityInfo#SCREEN_ORIENTATION_USER} to force app to fullscreen
|
|
* or app is universal resizeable, and app has not opted-out from the treatment
|
|
*/
|
|
boolean isOverrideToFullscreenEnabled(String pkgName, int userId) {
|
|
Boolean appAllowsOrientationOverride = readComponentProperty(mContext.getPackageManager(),
|
|
pkgName, PROPERTY_COMPAT_ALLOW_ORIENTATION_OVERRIDE);
|
|
return mIsUserMinAspectRatioAppDefaultFlagEnabled
|
|
&& hasAspectRatioOption(USER_MIN_ASPECT_RATIO_FULLSCREEN, pkgName)
|
|
&& !FALSE.equals(appAllowsOrientationOverride)
|
|
&& (isFullscreenCompatChangeEnabled(pkgName, userId)
|
|
|| isUniversalResizeable(pkgName, userId));
|
|
}
|
|
|
|
boolean isFullscreenCompatChangeEnabled(String pkgName, int userId) {
|
|
return CompatChanges.isChangeEnabled(
|
|
OVERRIDE_ANY_ORIENTATION_TO_USER, pkgName, UserHandle.of(userId));
|
|
}
|
|
|
|
/**
|
|
* Whether the aspect ratio is unset and we desire to interpret it as fullscreen rather than
|
|
* app default because of manufacturer override or because the app is universal resizeable
|
|
*/
|
|
private boolean isUnsetAndRequiresFullscreenOverride(String pkgName, int userId,
|
|
@PackageManager.UserMinAspectRatio int aspectRatio) {
|
|
return aspectRatio == USER_MIN_ASPECT_RATIO_UNSET
|
|
&& isOverrideToFullscreenEnabled(pkgName, userId);
|
|
}
|
|
|
|
private boolean hasLauncherEntry(@NonNull ApplicationInfo app) {
|
|
return !mContext.getSystemService(LauncherApps.class)
|
|
.getActivityList(app.packageName, getUserHandleForUid(app.uid))
|
|
.isEmpty();
|
|
}
|
|
|
|
private static boolean getValueFromDeviceConfig(String name, boolean defaultValue) {
|
|
return DeviceConfig.getBoolean(DeviceConfig.NAMESPACE_WINDOW_MANAGER, name, defaultValue);
|
|
}
|
|
|
|
@NonNull
|
|
private Map<Integer, String> getUserMinAspectRatioMapping() {
|
|
final String[] userMinAspectRatioStrings = mContext.getResources().getStringArray(
|
|
R.array.config_userAspectRatioOverrideEntries);
|
|
final int[] userMinAspectRatioValues = mContext.getResources().getIntArray(
|
|
R.array.config_userAspectRatioOverrideValues);
|
|
if (userMinAspectRatioStrings.length != userMinAspectRatioValues.length) {
|
|
throw new RuntimeException(
|
|
"config_userAspectRatioOverride options cannot be different length");
|
|
}
|
|
|
|
final Map<Integer, String> userMinAspectRatioMap = new ArrayMap<>();
|
|
for (int i = 0; i < userMinAspectRatioValues.length; i++) {
|
|
final int aspectRatioVal = userMinAspectRatioValues[i];
|
|
final String aspectRatioString = getAspectRatioStringOrDefault(
|
|
userMinAspectRatioStrings[i], aspectRatioVal);
|
|
boolean containsColon = aspectRatioString.contains(":");
|
|
switch (aspectRatioVal) {
|
|
// Only map known values of UserMinAspectRatio and ignore unknown entries
|
|
case USER_MIN_ASPECT_RATIO_FULLSCREEN:
|
|
case USER_MIN_ASPECT_RATIO_UNSET:
|
|
case USER_MIN_ASPECT_RATIO_SPLIT_SCREEN:
|
|
case USER_MIN_ASPECT_RATIO_DISPLAY_SIZE:
|
|
case USER_MIN_ASPECT_RATIO_4_3:
|
|
case USER_MIN_ASPECT_RATIO_16_9:
|
|
case USER_MIN_ASPECT_RATIO_3_2:
|
|
if (containsColon) {
|
|
String[] aspectRatioDigits = aspectRatioString.split(":");
|
|
String accessibleString = getAccessibleOption(aspectRatioDigits[0],
|
|
aspectRatioDigits[1]);
|
|
final CharSequence accessibleSequence = Utils.createAccessibleSequence(
|
|
aspectRatioString, accessibleString);
|
|
mUserAspectRatioA11yMap.put(aspectRatioVal, accessibleSequence);
|
|
}
|
|
userMinAspectRatioMap.put(aspectRatioVal, aspectRatioString);
|
|
mUserAspectRatioOrder.put(aspectRatioVal, i);
|
|
}
|
|
}
|
|
if (!userMinAspectRatioMap.containsKey(USER_MIN_ASPECT_RATIO_UNSET)) {
|
|
throw new RuntimeException("config_userAspectRatioOverrideValues options must have"
|
|
+ " USER_MIN_ASPECT_RATIO_UNSET value");
|
|
}
|
|
if (mIsUserMinAspectRatioAppDefaultFlagEnabled) {
|
|
userMinAspectRatioMap.put(USER_MIN_ASPECT_RATIO_APP_DEFAULT,
|
|
userMinAspectRatioMap.get(USER_MIN_ASPECT_RATIO_UNSET));
|
|
mUserAspectRatioOrder.put(USER_MIN_ASPECT_RATIO_APP_DEFAULT,
|
|
mUserAspectRatioOrder.get(USER_MIN_ASPECT_RATIO_UNSET));
|
|
if (mUserAspectRatioA11yMap.containsKey(USER_MIN_ASPECT_RATIO_UNSET)) {
|
|
mUserAspectRatioA11yMap.put(USER_MIN_ASPECT_RATIO_APP_DEFAULT,
|
|
mUserAspectRatioA11yMap.get(USER_MIN_ASPECT_RATIO_UNSET));
|
|
}
|
|
}
|
|
return userMinAspectRatioMap;
|
|
}
|
|
|
|
@NonNull
|
|
private String getAccessibleOption(String numerator, String denominator) {
|
|
return mContext.getString(R.string.user_aspect_ratio_option_a11y,
|
|
numerator, denominator);
|
|
}
|
|
|
|
@NonNull
|
|
private String getAspectRatioStringOrDefault(@Nullable String aspectRatioString,
|
|
@PackageManager.UserMinAspectRatio int aspectRatioVal) {
|
|
if (aspectRatioString != null) {
|
|
return aspectRatioString;
|
|
}
|
|
// Options are customized per device and if strings are set to @null, use default
|
|
switch (aspectRatioVal) {
|
|
case USER_MIN_ASPECT_RATIO_FULLSCREEN:
|
|
return mContext.getString(R.string.user_aspect_ratio_fullscreen);
|
|
case USER_MIN_ASPECT_RATIO_SPLIT_SCREEN:
|
|
return mContext.getString(R.string.user_aspect_ratio_half_screen);
|
|
case USER_MIN_ASPECT_RATIO_DISPLAY_SIZE:
|
|
return mContext.getString(R.string.user_aspect_ratio_device_size);
|
|
case USER_MIN_ASPECT_RATIO_4_3:
|
|
return mContext.getString(R.string.user_aspect_ratio_4_3);
|
|
case USER_MIN_ASPECT_RATIO_16_9:
|
|
return mContext.getString(R.string.user_aspect_ratio_16_9);
|
|
case USER_MIN_ASPECT_RATIO_3_2:
|
|
return mContext.getString(R.string.user_aspect_ratio_3_2);
|
|
default:
|
|
return mContext.getString(R.string.user_aspect_ratio_app_default);
|
|
}
|
|
}
|
|
|
|
@Nullable
|
|
private static Boolean readComponentProperty(PackageManager pm, String packageName,
|
|
String propertyName) {
|
|
try {
|
|
return pm.getProperty(propertyName, packageName).getBoolean();
|
|
} catch (PackageManager.NameNotFoundException e) {
|
|
// No such property name
|
|
}
|
|
return null;
|
|
}
|
|
}
|