/* * Copyright (C) 2021 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.display; import static android.hardware.SensorPrivacyManager.Sensors.CAMERA; import static android.provider.Settings.Secure.CAMERA_AUTOROTATE; import static androidx.lifecycle.Lifecycle.Event.ON_START; import static androidx.lifecycle.Lifecycle.Event.ON_STOP; import android.Manifest; import android.app.settings.SettingsEnums; import android.content.BroadcastReceiver; import android.content.Context; import android.content.Intent; import android.content.IntentFilter; import android.content.pm.PackageManager; import android.content.pm.ResolveInfo; import android.hardware.SensorPrivacyManager; import android.os.PowerManager; import android.provider.Settings; import android.service.rotationresolver.RotationResolverService; import android.text.TextUtils; import androidx.lifecycle.LifecycleObserver; import androidx.lifecycle.OnLifecycleEvent; import androidx.preference.Preference; import androidx.preference.PreferenceScreen; import com.android.internal.annotations.VisibleForTesting; import com.android.internal.view.RotationPolicy; import com.android.settings.R; import com.android.settings.core.TogglePreferenceController; import com.android.settings.overlay.FeatureFactory; import com.android.settingslib.core.instrumentation.MetricsFeatureProvider; import com.android.settingslib.devicestate.DeviceStateRotationLockSettingsManager; /** * SmartAutoRotateController controls whether auto rotation is enabled */ public class SmartAutoRotateController extends TogglePreferenceController implements Preference.OnPreferenceChangeListener, LifecycleObserver { protected Preference mPreference; private final MetricsFeatureProvider mMetricsFeatureProvider; private final SensorPrivacyManager mPrivacyManager; private final PowerManager mPowerManager; private final BroadcastReceiver mReceiver = new BroadcastReceiver() { @Override public void onReceive(Context context, Intent intent) { updateState(mPreference); } }; private final SensorPrivacyManager.OnSensorPrivacyChangedListener mPrivacyChangedListener = new SensorPrivacyManager.OnSensorPrivacyChangedListener() { @Override public void onSensorPrivacyChanged(int sensor, boolean enabled) { updateState(mPreference); } }; private final DeviceStateRotationLockSettingsManager mDeviceStateAutoRotateSettingsManager; private final DeviceStateRotationLockSettingsManager.DeviceStateRotationLockSettingsListener mDeviceStateRotationLockSettingsListener = () -> updateState(mPreference); private RotationPolicy.RotationPolicyListener mRotationPolicyListener; public SmartAutoRotateController(Context context, String preferenceKey) { super(context, preferenceKey); mMetricsFeatureProvider = FeatureFactory.getFeatureFactory().getMetricsFeatureProvider(); mPrivacyManager = SensorPrivacyManager.getInstance(context); mPowerManager = context.getSystemService(PowerManager.class); mDeviceStateAutoRotateSettingsManager = DeviceStateRotationLockSettingsManager.getInstance( context); } @Override public int getAvailabilityStatus() { if (!isRotationResolverServiceAvailable(mContext)) { return UNSUPPORTED_ON_DEVICE; } return !isRotationLocked() && hasSufficientPermission(mContext) && !isCameraLocked() && !isPowerSaveMode() ? AVAILABLE : DISABLED_DEPENDENT_SETTING; } protected boolean isRotationLocked() { if (DeviceStateAutoRotationHelper.isDeviceStateRotationEnabled(mContext)) { return mDeviceStateAutoRotateSettingsManager.isRotationLockedForAllStates(); } return RotationPolicy.isRotationLocked(mContext); } @Override public void updateState(Preference preference) { super.updateState(preference); if (preference != null) { preference.setEnabled(getAvailabilityStatus() == AVAILABLE); } } /** * Need this because all controller tests use RoboElectric. No easy way to mock this service, * so we mock the call we need */ @VisibleForTesting boolean isCameraLocked() { return mPrivacyManager.isSensorPrivacyEnabled(SensorPrivacyManager.Sensors.CAMERA); } @VisibleForTesting boolean isPowerSaveMode() { return mPowerManager.isPowerSaveMode(); } @OnLifecycleEvent(ON_START) public void onStart() { mContext.registerReceiver(mReceiver, new IntentFilter(PowerManager.ACTION_POWER_SAVE_MODE_CHANGED)); if (mRotationPolicyListener == null) { mRotationPolicyListener = new RotationPolicy.RotationPolicyListener() { @Override public void onChange() { updateState(mPreference); } }; } RotationPolicy.registerRotationPolicyListener(mContext, mRotationPolicyListener); mDeviceStateAutoRotateSettingsManager.registerListener( mDeviceStateRotationLockSettingsListener); mPrivacyManager.addSensorPrivacyListener(CAMERA, mPrivacyChangedListener); } @OnLifecycleEvent(ON_STOP) public void onStop() { mContext.unregisterReceiver(mReceiver); if (mRotationPolicyListener != null) { RotationPolicy.unregisterRotationPolicyListener(mContext, mRotationPolicyListener); mRotationPolicyListener = null; } mDeviceStateAutoRotateSettingsManager.unregisterListener( mDeviceStateRotationLockSettingsListener); mPrivacyManager.removeSensorPrivacyListener(CAMERA, mPrivacyChangedListener); } @Override public boolean isChecked() { return !isRotationLocked() && hasSufficientPermission(mContext) && !isCameraLocked() && !isPowerSaveMode() && Settings.Secure.getInt( mContext.getContentResolver(), CAMERA_AUTOROTATE, 0) == 1; } @Override public void displayPreference(PreferenceScreen screen) { super.displayPreference(screen); mPreference = screen.findPreference(getPreferenceKey()); } @Override public boolean setChecked(boolean isChecked) { mMetricsFeatureProvider.action(mContext, SettingsEnums.ACTION_CAMERA_ROTATE_TOGGLE, isChecked); Settings.Secure.putInt(mContext.getContentResolver(), CAMERA_AUTOROTATE, isChecked ? 1 : 0); return true; } @Override public int getSliceHighlightMenuRes() { return R.string.menu_key_display; } /** * Returns true if there is a {@link RotationResolverService} available */ public static boolean isRotationResolverServiceAvailable(Context context) { if (!context.getResources().getBoolean( R.bool.config_auto_rotate_face_detection_available)) { return false; } final PackageManager packageManager = context.getPackageManager(); final String resolvePackage = packageManager.getRotationResolverPackageName(); if (TextUtils.isEmpty(resolvePackage)) { return false; } final Intent intent = new Intent(RotationResolverService.SERVICE_INTERFACE).setPackage( resolvePackage); final ResolveInfo resolveInfo = packageManager.resolveService(intent, PackageManager.MATCH_SYSTEM_ONLY); return resolveInfo != null && resolveInfo.serviceInfo != null; } static boolean hasSufficientPermission(Context context) { final PackageManager packageManager = context.getPackageManager(); final String rotationPackage = packageManager.getRotationResolverPackageName(); return rotationPackage != null && packageManager.checkPermission( Manifest.permission.CAMERA, rotationPackage) == PackageManager.PERMISSION_GRANTED; } }