| /* |
| * Copyright (C) 2008 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.systemui.statusbar.phone; |
| |
| import android.annotation.Nullable; |
| import android.app.ActivityTaskManager; |
| import android.app.AlarmManager; |
| import android.app.AlarmManager.AlarmClockInfo; |
| import android.app.IActivityManager; |
| import android.app.SynchronousUserSwitchObserver; |
| import android.content.BroadcastReceiver; |
| import android.content.Context; |
| import android.content.Intent; |
| import android.content.IntentFilter; |
| import android.content.SharedPreferences; |
| import android.content.res.Resources; |
| import android.media.AudioManager; |
| import android.os.Handler; |
| import android.os.RemoteException; |
| import android.os.UserHandle; |
| import android.os.UserManager; |
| import android.provider.Settings.Global; |
| import android.service.notification.ZenModeConfig; |
| import android.telecom.TelecomManager; |
| import android.text.format.DateFormat; |
| import android.util.Log; |
| import android.view.View; |
| |
| import androidx.lifecycle.Observer; |
| |
| import com.android.systemui.R; |
| import com.android.systemui.broadcast.BroadcastDispatcher; |
| import com.android.systemui.dagger.qualifiers.DisplayId; |
| import com.android.systemui.dagger.qualifiers.Main; |
| import com.android.systemui.dagger.qualifiers.UiBackground; |
| import com.android.systemui.qs.tiles.DndTile; |
| import com.android.systemui.qs.tiles.RotationLockTile; |
| import com.android.systemui.screenrecord.RecordingController; |
| import com.android.systemui.statusbar.CommandQueue; |
| import com.android.systemui.statusbar.policy.BluetoothController; |
| import com.android.systemui.statusbar.policy.CastController; |
| import com.android.systemui.statusbar.policy.CastController.CastDevice; |
| import com.android.systemui.statusbar.policy.DataSaverController; |
| import com.android.systemui.statusbar.policy.DataSaverController.Listener; |
| import com.android.systemui.statusbar.policy.DeviceProvisionedController; |
| import com.android.systemui.statusbar.policy.DeviceProvisionedController.DeviceProvisionedListener; |
| import com.android.systemui.statusbar.policy.HotspotController; |
| import com.android.systemui.statusbar.policy.KeyguardStateController; |
| import com.android.systemui.statusbar.policy.LocationController; |
| import com.android.systemui.statusbar.policy.NextAlarmController; |
| import com.android.systemui.statusbar.policy.RotationLockController; |
| import com.android.systemui.statusbar.policy.RotationLockController.RotationLockControllerCallback; |
| import com.android.systemui.statusbar.policy.SensorPrivacyController; |
| import com.android.systemui.statusbar.policy.UserInfoController; |
| import com.android.systemui.statusbar.policy.ZenModeController; |
| import com.android.systemui.util.RingerModeTracker; |
| import com.android.systemui.util.time.DateFormatUtil; |
| |
| import java.util.Locale; |
| import java.util.concurrent.Executor; |
| |
| import javax.inject.Inject; |
| |
| /** |
| * This class contains all of the policy about which icons are installed in the status bar at boot |
| * time. It goes through the normal API for icons, even though it probably strictly doesn't need to. |
| */ |
| public class PhoneStatusBarPolicy |
| implements BluetoothController.Callback, |
| CommandQueue.Callbacks, |
| RotationLockControllerCallback, |
| Listener, |
| ZenModeController.Callback, |
| DeviceProvisionedListener, |
| KeyguardStateController.Callback, |
| LocationController.LocationChangeCallback, |
| RecordingController.RecordingStateChangeCallback { |
| private static final String TAG = "PhoneStatusBarPolicy"; |
| private static final boolean DEBUG = Log.isLoggable(TAG, Log.DEBUG); |
| |
| static final int LOCATION_STATUS_ICON_ID = |
| com.android.internal.R.drawable.perm_group_location; |
| |
| private final String mSlotCast; |
| private final String mSlotHotspot; |
| private final String mSlotBluetooth; |
| private final String mSlotTty; |
| private final String mSlotZen; |
| private final String mSlotVolume; |
| private final String mSlotAlarmClock; |
| private final String mSlotManagedProfile; |
| private final String mSlotRotate; |
| private final String mSlotHeadset; |
| private final String mSlotDataSaver; |
| private final String mSlotLocation; |
| private final String mSlotSensorsOff; |
| private final String mSlotScreenRecord; |
| private final int mDisplayId; |
| private final SharedPreferences mSharedPreferences; |
| private final DateFormatUtil mDateFormatUtil; |
| private final TelecomManager mTelecomManager; |
| |
| private final Handler mHandler = new Handler(); |
| private final CastController mCast; |
| private final HotspotController mHotspot; |
| private final NextAlarmController mNextAlarmController; |
| private final AlarmManager mAlarmManager; |
| private final UserInfoController mUserInfoController; |
| private final IActivityManager mIActivityManager; |
| private final UserManager mUserManager; |
| private final StatusBarIconController mIconController; |
| private final CommandQueue mCommandQueue; |
| private final BroadcastDispatcher mBroadcastDispatcher; |
| private final Resources mResources; |
| private final RotationLockController mRotationLockController; |
| private final DataSaverController mDataSaver; |
| private final ZenModeController mZenController; |
| private final DeviceProvisionedController mProvisionedController; |
| private final KeyguardStateController mKeyguardStateController; |
| private final LocationController mLocationController; |
| private final Executor mUiBgExecutor; |
| private final SensorPrivacyController mSensorPrivacyController; |
| private final RecordingController mRecordingController; |
| private final RingerModeTracker mRingerModeTracker; |
| |
| private boolean mZenVisible; |
| private boolean mVolumeVisible; |
| private boolean mCurrentUserSetup; |
| |
| private boolean mManagedProfileIconVisible = false; |
| |
| private BluetoothController mBluetooth; |
| private AlarmManager.AlarmClockInfo mNextAlarm; |
| |
| @Inject |
| public PhoneStatusBarPolicy(StatusBarIconController iconController, |
| CommandQueue commandQueue, BroadcastDispatcher broadcastDispatcher, |
| @UiBackground Executor uiBgExecutor, @Main Resources resources, |
| CastController castController, HotspotController hotspotController, |
| BluetoothController bluetoothController, NextAlarmController nextAlarmController, |
| UserInfoController userInfoController, RotationLockController rotationLockController, |
| DataSaverController dataSaverController, ZenModeController zenModeController, |
| DeviceProvisionedController deviceProvisionedController, |
| KeyguardStateController keyguardStateController, |
| LocationController locationController, |
| SensorPrivacyController sensorPrivacyController, IActivityManager iActivityManager, |
| AlarmManager alarmManager, UserManager userManager, |
| RecordingController recordingController, |
| @Nullable TelecomManager telecomManager, @DisplayId int displayId, |
| @Main SharedPreferences sharedPreferences, DateFormatUtil dateFormatUtil, |
| RingerModeTracker ringerModeTracker) { |
| mIconController = iconController; |
| mCommandQueue = commandQueue; |
| mBroadcastDispatcher = broadcastDispatcher; |
| mResources = resources; |
| mCast = castController; |
| mHotspot = hotspotController; |
| mBluetooth = bluetoothController; |
| mNextAlarmController = nextAlarmController; |
| mAlarmManager = alarmManager; |
| mUserInfoController = userInfoController; |
| mIActivityManager = iActivityManager; |
| mUserManager = userManager; |
| mRotationLockController = rotationLockController; |
| mDataSaver = dataSaverController; |
| mZenController = zenModeController; |
| mProvisionedController = deviceProvisionedController; |
| mKeyguardStateController = keyguardStateController; |
| mLocationController = locationController; |
| mSensorPrivacyController = sensorPrivacyController; |
| mRecordingController = recordingController; |
| mUiBgExecutor = uiBgExecutor; |
| mTelecomManager = telecomManager; |
| mRingerModeTracker = ringerModeTracker; |
| |
| mSlotCast = resources.getString(com.android.internal.R.string.status_bar_cast); |
| mSlotHotspot = resources.getString(com.android.internal.R.string.status_bar_hotspot); |
| mSlotBluetooth = resources.getString(com.android.internal.R.string.status_bar_bluetooth); |
| mSlotTty = resources.getString(com.android.internal.R.string.status_bar_tty); |
| mSlotZen = resources.getString(com.android.internal.R.string.status_bar_zen); |
| mSlotVolume = resources.getString(com.android.internal.R.string.status_bar_volume); |
| mSlotAlarmClock = resources.getString(com.android.internal.R.string.status_bar_alarm_clock); |
| mSlotManagedProfile = resources.getString( |
| com.android.internal.R.string.status_bar_managed_profile); |
| mSlotRotate = resources.getString(com.android.internal.R.string.status_bar_rotate); |
| mSlotHeadset = resources.getString(com.android.internal.R.string.status_bar_headset); |
| mSlotDataSaver = resources.getString(com.android.internal.R.string.status_bar_data_saver); |
| mSlotLocation = resources.getString(com.android.internal.R.string.status_bar_location); |
| mSlotSensorsOff = resources.getString(com.android.internal.R.string.status_bar_sensors_off); |
| mSlotScreenRecord = resources.getString( |
| com.android.internal.R.string.status_bar_screen_record); |
| |
| mDisplayId = displayId; |
| mSharedPreferences = sharedPreferences; |
| mDateFormatUtil = dateFormatUtil; |
| } |
| |
| /** Initialize the object after construction. */ |
| public void init() { |
| // listen for broadcasts |
| IntentFilter filter = new IntentFilter(); |
| |
| filter.addAction(AudioManager.ACTION_HEADSET_PLUG); |
| filter.addAction(Intent.ACTION_SIM_STATE_CHANGED); |
| filter.addAction(TelecomManager.ACTION_CURRENT_TTY_MODE_CHANGED); |
| filter.addAction(Intent.ACTION_MANAGED_PROFILE_AVAILABLE); |
| filter.addAction(Intent.ACTION_MANAGED_PROFILE_UNAVAILABLE); |
| filter.addAction(Intent.ACTION_MANAGED_PROFILE_REMOVED); |
| mBroadcastDispatcher.registerReceiverWithHandler(mIntentReceiver, filter, mHandler); |
| Observer<Integer> observer = ringer -> mHandler.post(this::updateVolumeZen); |
| |
| mRingerModeTracker.getRingerMode().observeForever(observer); |
| mRingerModeTracker.getRingerModeInternal().observeForever(observer); |
| |
| // listen for user / profile change. |
| try { |
| mIActivityManager.registerUserSwitchObserver(mUserSwitchListener, TAG); |
| } catch (RemoteException e) { |
| // Ignore |
| } |
| |
| // TTY status |
| updateTTY(); |
| |
| // bluetooth status |
| updateBluetooth(); |
| |
| // Alarm clock |
| mIconController.setIcon(mSlotAlarmClock, R.drawable.stat_sys_alarm, null); |
| mIconController.setIconVisibility(mSlotAlarmClock, false); |
| |
| // zen |
| mIconController.setIcon(mSlotZen, R.drawable.stat_sys_dnd, null); |
| mIconController.setIconVisibility(mSlotZen, false); |
| |
| // volume |
| mIconController.setIcon(mSlotVolume, R.drawable.stat_sys_ringer_vibrate, null); |
| mIconController.setIconVisibility(mSlotVolume, false); |
| updateVolumeZen(); |
| |
| // cast |
| mIconController.setIcon(mSlotCast, R.drawable.stat_sys_cast, null); |
| mIconController.setIconVisibility(mSlotCast, false); |
| |
| // hotspot |
| mIconController.setIcon(mSlotHotspot, R.drawable.stat_sys_hotspot, |
| mResources.getString(R.string.accessibility_status_bar_hotspot)); |
| mIconController.setIconVisibility(mSlotHotspot, mHotspot.isHotspotEnabled()); |
| |
| // managed profile |
| mIconController.setIcon(mSlotManagedProfile, R.drawable.stat_sys_managed_profile_status, |
| mResources.getString(R.string.accessibility_managed_profile)); |
| mIconController.setIconVisibility(mSlotManagedProfile, mManagedProfileIconVisible); |
| |
| // data saver |
| mIconController.setIcon(mSlotDataSaver, R.drawable.stat_sys_data_saver, |
| mResources.getString(R.string.accessibility_data_saver_on)); |
| mIconController.setIconVisibility(mSlotDataSaver, false); |
| |
| mIconController.setIcon(mSlotLocation, LOCATION_STATUS_ICON_ID, |
| mResources.getString(R.string.accessibility_location_active)); |
| mIconController.setIconVisibility(mSlotLocation, false); |
| |
| // sensors off |
| mIconController.setIcon(mSlotSensorsOff, R.drawable.stat_sys_sensors_off, |
| mResources.getString(R.string.accessibility_sensors_off_active)); |
| mIconController.setIconVisibility(mSlotSensorsOff, |
| mSensorPrivacyController.isSensorPrivacyEnabled()); |
| |
| // screen record |
| mIconController.setIcon(mSlotScreenRecord, R.drawable.stat_sys_screen_record, null); |
| mIconController.setIconVisibility(mSlotScreenRecord, false); |
| |
| mRotationLockController.addCallback(this); |
| mBluetooth.addCallback(this); |
| mProvisionedController.addCallback(this); |
| mZenController.addCallback(this); |
| mCast.addCallback(mCastCallback); |
| mHotspot.addCallback(mHotspotCallback); |
| mNextAlarmController.addCallback(mNextAlarmCallback); |
| mDataSaver.addCallback(this); |
| mKeyguardStateController.addCallback(this); |
| mSensorPrivacyController.addCallback(mSensorPrivacyListener); |
| mLocationController.addCallback(this); |
| mRecordingController.addCallback(this); |
| |
| mCommandQueue.addCallback(this); |
| } |
| |
| @Override |
| public void onZenChanged(int zen) { |
| updateVolumeZen(); |
| } |
| |
| @Override |
| public void onConfigChanged(ZenModeConfig config) { |
| updateVolumeZen(); |
| } |
| |
| private void updateAlarm() { |
| final AlarmClockInfo alarm = mAlarmManager.getNextAlarmClock(UserHandle.USER_CURRENT); |
| final boolean hasAlarm = alarm != null && alarm.getTriggerTime() > 0; |
| int zen = mZenController.getZen(); |
| final boolean zenNone = zen == Global.ZEN_MODE_NO_INTERRUPTIONS; |
| mIconController.setIcon(mSlotAlarmClock, zenNone ? R.drawable.stat_sys_alarm_dim |
| : R.drawable.stat_sys_alarm, buildAlarmContentDescription()); |
| mIconController.setIconVisibility(mSlotAlarmClock, mCurrentUserSetup && hasAlarm); |
| } |
| |
| private String buildAlarmContentDescription() { |
| if (mNextAlarm == null) { |
| return mResources.getString(R.string.status_bar_alarm); |
| } |
| |
| String skeleton = mDateFormatUtil.is24HourFormat() ? "EHm" : "Ehma"; |
| String pattern = DateFormat.getBestDateTimePattern(Locale.getDefault(), skeleton); |
| String dateString = DateFormat.format(pattern, mNextAlarm.getTriggerTime()).toString(); |
| |
| return mResources.getString(R.string.accessibility_quick_settings_alarm, dateString); |
| } |
| |
| private final void updateVolumeZen() { |
| boolean zenVisible = false; |
| int zenIconId = 0; |
| String zenDescription = null; |
| |
| boolean volumeVisible = false; |
| int volumeIconId = 0; |
| String volumeDescription = null; |
| int zen = mZenController.getZen(); |
| |
| if (DndTile.isVisible(mSharedPreferences) || DndTile.isCombinedIcon(mSharedPreferences)) { |
| zenVisible = zen != Global.ZEN_MODE_OFF; |
| zenIconId = R.drawable.stat_sys_dnd; |
| zenDescription = mResources.getString(R.string.quick_settings_dnd_label); |
| } else if (zen == Global.ZEN_MODE_NO_INTERRUPTIONS) { |
| zenVisible = true; |
| zenIconId = R.drawable.stat_sys_dnd; |
| zenDescription = mResources.getString(R.string.interruption_level_none); |
| } else if (zen == Global.ZEN_MODE_IMPORTANT_INTERRUPTIONS) { |
| zenVisible = true; |
| zenIconId = R.drawable.stat_sys_dnd; |
| zenDescription = mResources.getString(R.string.interruption_level_priority); |
| } |
| |
| if (!ZenModeConfig.isZenOverridingRinger(zen, mZenController.getConsolidatedPolicy())) { |
| final Integer ringerModeInternal = |
| mRingerModeTracker.getRingerModeInternal().getValue(); |
| if (ringerModeInternal != null) { |
| if (ringerModeInternal == AudioManager.RINGER_MODE_VIBRATE) { |
| volumeVisible = true; |
| volumeIconId = R.drawable.stat_sys_ringer_vibrate; |
| volumeDescription = mResources.getString(R.string.accessibility_ringer_vibrate); |
| } else if (ringerModeInternal == AudioManager.RINGER_MODE_SILENT) { |
| volumeVisible = true; |
| volumeIconId = R.drawable.stat_sys_ringer_silent; |
| volumeDescription = mResources.getString(R.string.accessibility_ringer_silent); |
| } |
| } |
| } |
| |
| if (zenVisible) { |
| mIconController.setIcon(mSlotZen, zenIconId, zenDescription); |
| } |
| if (zenVisible != mZenVisible) { |
| mIconController.setIconVisibility(mSlotZen, zenVisible); |
| mZenVisible = zenVisible; |
| } |
| |
| if (volumeVisible) { |
| mIconController.setIcon(mSlotVolume, volumeIconId, volumeDescription); |
| } |
| if (volumeVisible != mVolumeVisible) { |
| mIconController.setIconVisibility(mSlotVolume, volumeVisible); |
| mVolumeVisible = volumeVisible; |
| } |
| updateAlarm(); |
| } |
| |
| @Override |
| public void onBluetoothDevicesChanged() { |
| updateBluetooth(); |
| } |
| |
| @Override |
| public void onBluetoothStateChange(boolean enabled) { |
| updateBluetooth(); |
| } |
| |
| private final void updateBluetooth() { |
| int iconId = R.drawable.stat_sys_data_bluetooth_connected; |
| String contentDescription = |
| mResources.getString(R.string.accessibility_quick_settings_bluetooth_on); |
| boolean bluetoothVisible = false; |
| if (mBluetooth != null) { |
| if (mBluetooth.isBluetoothConnected() |
| && (mBluetooth.isBluetoothAudioActive() |
| || !mBluetooth.isBluetoothAudioProfileOnly())) { |
| contentDescription = mResources.getString( |
| R.string.accessibility_bluetooth_connected); |
| bluetoothVisible = mBluetooth.isBluetoothEnabled(); |
| } |
| } |
| |
| mIconController.setIcon(mSlotBluetooth, iconId, contentDescription); |
| mIconController.setIconVisibility(mSlotBluetooth, bluetoothVisible); |
| } |
| |
| private final void updateTTY() { |
| if (mTelecomManager == null) { |
| updateTTY(TelecomManager.TTY_MODE_OFF); |
| } else { |
| updateTTY(mTelecomManager.getCurrentTtyMode()); |
| } |
| } |
| |
| private final void updateTTY(int currentTtyMode) { |
| boolean enabled = currentTtyMode != TelecomManager.TTY_MODE_OFF; |
| |
| if (DEBUG) Log.v(TAG, "updateTTY: enabled: " + enabled); |
| |
| if (enabled) { |
| // TTY is on |
| if (DEBUG) Log.v(TAG, "updateTTY: set TTY on"); |
| mIconController.setIcon(mSlotTty, R.drawable.stat_sys_tty_mode, |
| mResources.getString(R.string.accessibility_tty_enabled)); |
| mIconController.setIconVisibility(mSlotTty, true); |
| } else { |
| // TTY is off |
| if (DEBUG) Log.v(TAG, "updateTTY: set TTY off"); |
| mIconController.setIconVisibility(mSlotTty, false); |
| } |
| } |
| |
| private void updateCast() { |
| boolean isCasting = false; |
| for (CastDevice device : mCast.getCastDevices()) { |
| if (device.state == CastDevice.STATE_CONNECTING |
| || device.state == CastDevice.STATE_CONNECTED) { |
| isCasting = true; |
| break; |
| } |
| } |
| if (DEBUG) Log.v(TAG, "updateCast: isCasting: " + isCasting); |
| mHandler.removeCallbacks(mRemoveCastIconRunnable); |
| if (isCasting && !mRecordingController.isRecording()) { // screen record has its own icon |
| mIconController.setIcon(mSlotCast, R.drawable.stat_sys_cast, |
| mResources.getString(R.string.accessibility_casting)); |
| mIconController.setIconVisibility(mSlotCast, true); |
| } else { |
| // don't turn off the screen-record icon for a few seconds, just to make sure the user |
| // has seen it |
| if (DEBUG) Log.v(TAG, "updateCast: hiding icon in 3 sec..."); |
| mHandler.postDelayed(mRemoveCastIconRunnable, 3000); |
| } |
| } |
| |
| private void updateManagedProfile() { |
| // getLastResumedActivityUserId needds to acquire the AM lock, which may be contended in |
| // some cases. Since it doesn't really matter here whether it's updated in this frame |
| // or in the next one, we call this method from our UI offload thread. |
| mUiBgExecutor.execute(() -> { |
| final int userId; |
| try { |
| userId = ActivityTaskManager.getService().getLastResumedActivityUserId(); |
| boolean isManagedProfile = mUserManager.isManagedProfile(userId); |
| mHandler.post(() -> { |
| final boolean showIcon; |
| if (isManagedProfile && (!mKeyguardStateController.isShowing() |
| || mKeyguardStateController.isOccluded())) { |
| showIcon = true; |
| mIconController.setIcon(mSlotManagedProfile, |
| R.drawable.stat_sys_managed_profile_status, |
| mResources.getString(R.string.accessibility_managed_profile)); |
| } else { |
| showIcon = false; |
| } |
| if (mManagedProfileIconVisible != showIcon) { |
| mIconController.setIconVisibility(mSlotManagedProfile, showIcon); |
| mManagedProfileIconVisible = showIcon; |
| } |
| }); |
| } catch (RemoteException e) { |
| Log.w(TAG, "updateManagedProfile: ", e); |
| } |
| }); |
| } |
| |
| private final SynchronousUserSwitchObserver mUserSwitchListener = |
| new SynchronousUserSwitchObserver() { |
| @Override |
| public void onUserSwitching(int newUserId) throws RemoteException { |
| mHandler.post(() -> mUserInfoController.reloadUserInfo()); |
| } |
| |
| @Override |
| public void onUserSwitchComplete(int newUserId) throws RemoteException { |
| mHandler.post(() -> { |
| updateAlarm(); |
| updateManagedProfile(); |
| }); |
| } |
| }; |
| |
| private final HotspotController.Callback mHotspotCallback = new HotspotController.Callback() { |
| @Override |
| public void onHotspotChanged(boolean enabled, int numDevices) { |
| mIconController.setIconVisibility(mSlotHotspot, enabled); |
| } |
| }; |
| |
| private final CastController.Callback mCastCallback = new CastController.Callback() { |
| @Override |
| public void onCastDevicesChanged() { |
| updateCast(); |
| } |
| }; |
| |
| private final NextAlarmController.NextAlarmChangeCallback mNextAlarmCallback = |
| new NextAlarmController.NextAlarmChangeCallback() { |
| @Override |
| public void onNextAlarmChanged(AlarmManager.AlarmClockInfo nextAlarm) { |
| mNextAlarm = nextAlarm; |
| updateAlarm(); |
| } |
| }; |
| |
| private final SensorPrivacyController.OnSensorPrivacyChangedListener mSensorPrivacyListener = |
| new SensorPrivacyController.OnSensorPrivacyChangedListener() { |
| @Override |
| public void onSensorPrivacyChanged(boolean enabled) { |
| mHandler.post(() -> { |
| mIconController.setIconVisibility(mSlotSensorsOff, enabled); |
| }); |
| } |
| }; |
| |
| @Override |
| public void appTransitionStarting(int displayId, long startTime, long duration, |
| boolean forced) { |
| if (mDisplayId == displayId) { |
| updateManagedProfile(); |
| } |
| } |
| |
| @Override |
| public void onKeyguardShowingChanged() { |
| updateManagedProfile(); |
| } |
| |
| @Override |
| public void onUserSetupChanged() { |
| boolean userSetup = mProvisionedController.isUserSetup( |
| mProvisionedController.getCurrentUser()); |
| if (mCurrentUserSetup == userSetup) return; |
| mCurrentUserSetup = userSetup; |
| updateAlarm(); |
| } |
| |
| @Override |
| public void onRotationLockStateChanged(boolean rotationLocked, boolean affordanceVisible) { |
| boolean portrait = RotationLockTile.isCurrentOrientationLockPortrait( |
| mRotationLockController, mResources); |
| if (rotationLocked) { |
| if (portrait) { |
| mIconController.setIcon(mSlotRotate, R.drawable.stat_sys_rotate_portrait, |
| mResources.getString(R.string.accessibility_rotation_lock_on_portrait)); |
| } else { |
| mIconController.setIcon(mSlotRotate, R.drawable.stat_sys_rotate_landscape, |
| mResources.getString(R.string.accessibility_rotation_lock_on_landscape)); |
| } |
| mIconController.setIconVisibility(mSlotRotate, true); |
| } else { |
| mIconController.setIconVisibility(mSlotRotate, false); |
| } |
| } |
| |
| private void updateHeadsetPlug(Intent intent) { |
| boolean connected = intent.getIntExtra("state", 0) != 0; |
| boolean hasMic = intent.getIntExtra("microphone", 0) != 0; |
| if (connected) { |
| String contentDescription = mResources.getString(hasMic |
| ? R.string.accessibility_status_bar_headset |
| : R.string.accessibility_status_bar_headphones); |
| mIconController.setIcon(mSlotHeadset, hasMic ? R.drawable.stat_sys_headset_mic |
| : R.drawable.stat_sys_headset, contentDescription); |
| mIconController.setIconVisibility(mSlotHeadset, true); |
| } else { |
| mIconController.setIconVisibility(mSlotHeadset, false); |
| } |
| } |
| |
| @Override |
| public void onDataSaverChanged(boolean isDataSaving) { |
| mIconController.setIconVisibility(mSlotDataSaver, isDataSaving); |
| } |
| |
| @Override |
| public void onLocationActiveChanged(boolean active) { |
| updateLocation(); |
| } |
| |
| // Updates the status view based on the current state of location requests. |
| private void updateLocation() { |
| if (mLocationController.isLocationActive()) { |
| mIconController.setIconVisibility(mSlotLocation, true); |
| } else { |
| mIconController.setIconVisibility(mSlotLocation, false); |
| } |
| } |
| |
| private BroadcastReceiver mIntentReceiver = new BroadcastReceiver() { |
| @Override |
| public void onReceive(Context context, Intent intent) { |
| String action = intent.getAction(); |
| switch (action) { |
| case Intent.ACTION_SIM_STATE_CHANGED: |
| // Avoid rebroadcast because SysUI is direct boot aware. |
| if (intent.getBooleanExtra(Intent.EXTRA_REBROADCAST_ON_UNLOCK, false)) { |
| break; |
| } |
| break; |
| case TelecomManager.ACTION_CURRENT_TTY_MODE_CHANGED: |
| updateTTY(intent.getIntExtra(TelecomManager.EXTRA_CURRENT_TTY_MODE, |
| TelecomManager.TTY_MODE_OFF)); |
| break; |
| case Intent.ACTION_MANAGED_PROFILE_AVAILABLE: |
| case Intent.ACTION_MANAGED_PROFILE_UNAVAILABLE: |
| case Intent.ACTION_MANAGED_PROFILE_REMOVED: |
| updateManagedProfile(); |
| break; |
| case AudioManager.ACTION_HEADSET_PLUG: |
| updateHeadsetPlug(intent); |
| break; |
| } |
| } |
| }; |
| |
| private Runnable mRemoveCastIconRunnable = new Runnable() { |
| @Override |
| public void run() { |
| if (DEBUG) Log.v(TAG, "updateCast: hiding icon NOW"); |
| mIconController.setIconVisibility(mSlotCast, false); |
| } |
| }; |
| |
| // Screen Recording |
| @Override |
| public void onCountdown(long millisUntilFinished) { |
| if (DEBUG) Log.d(TAG, "screenrecord: countdown " + millisUntilFinished); |
| int countdown = (int) Math.floorDiv(millisUntilFinished + 500, 1000); |
| int resourceId = R.drawable.stat_sys_screen_record; |
| String description = Integer.toString(countdown); |
| switch (countdown) { |
| case 1: |
| resourceId = R.drawable.stat_sys_screen_record_1; |
| break; |
| case 2: |
| resourceId = R.drawable.stat_sys_screen_record_2; |
| break; |
| case 3: |
| resourceId = R.drawable.stat_sys_screen_record_3; |
| break; |
| } |
| mIconController.setIcon(mSlotScreenRecord, resourceId, description); |
| mIconController.setIconVisibility(mSlotScreenRecord, true); |
| // Set as assertive so talkback will announce the countdown |
| mIconController.setIconAccessibilityLiveRegion(mSlotScreenRecord, |
| View.ACCESSIBILITY_LIVE_REGION_ASSERTIVE); |
| } |
| |
| @Override |
| public void onCountdownEnd() { |
| if (DEBUG) Log.d(TAG, "screenrecord: hiding icon during countdown"); |
| mHandler.post(() -> mIconController.setIconVisibility(mSlotScreenRecord, false)); |
| // Reset talkback priority |
| mHandler.post(() -> mIconController.setIconAccessibilityLiveRegion(mSlotScreenRecord, |
| View.ACCESSIBILITY_LIVE_REGION_NONE)); |
| } |
| |
| @Override |
| public void onRecordingStart() { |
| if (DEBUG) Log.d(TAG, "screenrecord: showing icon"); |
| mIconController.setIcon(mSlotScreenRecord, |
| R.drawable.stat_sys_screen_record, |
| mResources.getString(R.string.screenrecord_ongoing_screen_only)); |
| mHandler.post(() -> mIconController.setIconVisibility(mSlotScreenRecord, true)); |
| } |
| |
| @Override |
| public void onRecordingEnd() { |
| // Ensure this is on the main thread |
| if (DEBUG) Log.d(TAG, "screenrecord: hiding icon"); |
| mHandler.post(() -> mIconController.setIconVisibility(mSlotScreenRecord, false)); |
| } |
| } |