Replace the build-time boolean resource with a run time device config flag that
signals whether the ring notification slider should be displayed or not
Bug: b/38477228
Test: make DEBUG_ROBOLECTRIC=1 ROBOTEST_FILTER=NotificationVolumePreferenceControllerTest RunSettingsRoboTests -j40
make DEBUG_ROBOLECTRIC=1 ROBOTEST_FILTER=RingVolumePreferenceControllerTest RunSettingsRoboTests -j40
Change-Id: I8b9a2cbd5af7fa1bba56ff9ba62771d677d4a932
328 lines
11 KiB
Java
328 lines
11 KiB
Java
/*
|
|
* Copyright (C) 2016 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.notification;
|
|
|
|
import android.app.ActivityThread;
|
|
import android.app.INotificationManager;
|
|
import android.app.NotificationManager;
|
|
import android.content.BroadcastReceiver;
|
|
import android.content.ComponentName;
|
|
import android.content.Context;
|
|
import android.content.Intent;
|
|
import android.content.IntentFilter;
|
|
import android.media.AudioManager;
|
|
import android.os.Handler;
|
|
import android.os.Looper;
|
|
import android.os.Message;
|
|
import android.os.ServiceManager;
|
|
import android.os.Vibrator;
|
|
import android.provider.DeviceConfig;
|
|
import android.service.notification.NotificationListenerService;
|
|
import android.text.TextUtils;
|
|
import android.util.Log;
|
|
|
|
import androidx.lifecycle.OnLifecycleEvent;
|
|
|
|
import com.android.internal.annotations.VisibleForTesting;
|
|
import com.android.internal.config.sysui.SystemUiDeviceConfigFlags;
|
|
import com.android.settings.R;
|
|
import com.android.settings.Utils;
|
|
import com.android.settingslib.core.lifecycle.Lifecycle;
|
|
|
|
import java.util.Objects;
|
|
import java.util.Set;
|
|
|
|
/**
|
|
* This slider can represent both ring and notification, if the corresponding streams are aliased,
|
|
* and only ring if the streams are not aliased.
|
|
*/
|
|
public class RingVolumePreferenceController extends VolumeSeekBarPreferenceController {
|
|
|
|
private static final String TAG = "RingVolumePreferenceController";
|
|
private static final String KEY_RING_VOLUME = "ring_volume";
|
|
|
|
private Vibrator mVibrator;
|
|
private int mRingerMode = AudioManager.RINGER_MODE_NORMAL;
|
|
private ComponentName mSuppressor;
|
|
private final RingReceiver mReceiver = new RingReceiver();
|
|
private final H mHandler = new H();
|
|
|
|
private int mMuteIcon;
|
|
|
|
private int mNormalIconId;
|
|
@VisibleForTesting
|
|
int mVibrateIconId;
|
|
@VisibleForTesting
|
|
int mSilentIconId;
|
|
|
|
@VisibleForTesting
|
|
int mTitleId;
|
|
|
|
private boolean mSeparateNotification;
|
|
|
|
private INotificationManager mNoMan;
|
|
|
|
private static final boolean CONFIG_DEFAULT_VAL = false;
|
|
|
|
public RingVolumePreferenceController(Context context) {
|
|
this(context, KEY_RING_VOLUME);
|
|
}
|
|
|
|
public RingVolumePreferenceController(Context context, String key) {
|
|
super(context, key);
|
|
mVibrator = (Vibrator) mContext.getSystemService(Context.VIBRATOR_SERVICE);
|
|
if (mVibrator != null && !mVibrator.hasVibrator()) {
|
|
mVibrator = null;
|
|
}
|
|
mSeparateNotification = DeviceConfig.getBoolean(DeviceConfig.NAMESPACE_SYSTEMUI,
|
|
SystemUiDeviceConfigFlags.VOLUME_SEPARATE_NOTIFICATION, CONFIG_DEFAULT_VAL);
|
|
loadPreferenceIconResources(mSeparateNotification);
|
|
updateRingerMode();
|
|
}
|
|
|
|
private void loadPreferenceIconResources(boolean separateNotification) {
|
|
if (separateNotification) {
|
|
mTitleId = R.string.separate_ring_volume_option_title;
|
|
mNormalIconId = R.drawable.ic_ring_volume;
|
|
mSilentIconId = R.drawable.ic_ring_volume_off;
|
|
} else {
|
|
mTitleId = R.string.ring_volume_option_title;
|
|
mNormalIconId = R.drawable.ic_notifications;
|
|
mSilentIconId = R.drawable.ic_notifications_off_24dp;
|
|
}
|
|
// todo: set a distinct vibrate icon for ring vs notification
|
|
mVibrateIconId = R.drawable.ic_volume_ringer_vibrate;
|
|
}
|
|
|
|
/**
|
|
* As the responsibility of this slider changes, so should its title & icon
|
|
*/
|
|
public void onDeviceConfigChange(DeviceConfig.Properties properties) {
|
|
Set<String> changeSet = properties.getKeyset();
|
|
if (changeSet.contains(SystemUiDeviceConfigFlags.VOLUME_SEPARATE_NOTIFICATION)) {
|
|
boolean valueUpdated = readSeparateNotificationVolumeConfig();
|
|
if (valueUpdated) {
|
|
updateEffectsSuppressor();
|
|
selectPreferenceIconState();
|
|
setPreferenceTitle();
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* side effect: updates the cached value of the config, and also the icon
|
|
* @return has the config changed?
|
|
*/
|
|
private boolean readSeparateNotificationVolumeConfig() {
|
|
boolean newVal = DeviceConfig.getBoolean(DeviceConfig.NAMESPACE_SYSTEMUI,
|
|
SystemUiDeviceConfigFlags.VOLUME_SEPARATE_NOTIFICATION, CONFIG_DEFAULT_VAL);
|
|
|
|
boolean valueUpdated = newVal != mSeparateNotification;
|
|
if (valueUpdated) {
|
|
mSeparateNotification = newVal;
|
|
loadPreferenceIconResources(newVal);
|
|
}
|
|
|
|
return valueUpdated;
|
|
}
|
|
|
|
@OnLifecycleEvent(Lifecycle.Event.ON_RESUME)
|
|
@Override
|
|
public void onResume() {
|
|
super.onResume();
|
|
mReceiver.register(true);
|
|
readSeparateNotificationVolumeConfig();
|
|
DeviceConfig.addOnPropertiesChangedListener(DeviceConfig.NAMESPACE_SYSTEMUI,
|
|
ActivityThread.currentApplication().getMainExecutor(), this::onDeviceConfigChange);
|
|
updateEffectsSuppressor();
|
|
selectPreferenceIconState();
|
|
setPreferenceTitle();
|
|
}
|
|
|
|
@OnLifecycleEvent(Lifecycle.Event.ON_PAUSE)
|
|
@Override
|
|
public void onPause() {
|
|
super.onPause();
|
|
mReceiver.register(false);
|
|
DeviceConfig.removeOnPropertiesChangedListener(this::onDeviceConfigChange);
|
|
}
|
|
|
|
@Override
|
|
public String getPreferenceKey() {
|
|
return KEY_RING_VOLUME;
|
|
}
|
|
|
|
@Override
|
|
public int getAvailabilityStatus() {
|
|
return Utils.isVoiceCapable(mContext) && !mHelper.isSingleVolume()
|
|
? AVAILABLE : UNSUPPORTED_ON_DEVICE;
|
|
}
|
|
|
|
@Override
|
|
public boolean isSliceable() {
|
|
return TextUtils.equals(getPreferenceKey(), KEY_RING_VOLUME);
|
|
}
|
|
|
|
@Override
|
|
public boolean isPublicSlice() {
|
|
return true;
|
|
}
|
|
|
|
@Override
|
|
public boolean useDynamicSliceSummary() {
|
|
return true;
|
|
}
|
|
|
|
@Override
|
|
public int getAudioStream() {
|
|
return AudioManager.STREAM_RING;
|
|
}
|
|
|
|
@Override
|
|
public int getMuteIcon() {
|
|
return mMuteIcon;
|
|
}
|
|
|
|
@VisibleForTesting
|
|
void updateRingerMode() {
|
|
final int ringerMode = mHelper.getRingerModeInternal();
|
|
if (mRingerMode == ringerMode) return;
|
|
mRingerMode = ringerMode;
|
|
selectPreferenceIconState();
|
|
}
|
|
|
|
private void updateEffectsSuppressor() {
|
|
final ComponentName suppressor = NotificationManager.from(mContext).getEffectsSuppressor();
|
|
if (Objects.equals(suppressor, mSuppressor)) return;
|
|
|
|
if (mNoMan == null) {
|
|
mNoMan = INotificationManager.Stub.asInterface(
|
|
ServiceManager.getService(Context.NOTIFICATION_SERVICE));
|
|
}
|
|
|
|
final int hints;
|
|
try {
|
|
hints = mNoMan.getHintsFromListenerNoToken();
|
|
} catch (android.os.RemoteException ex) {
|
|
Log.w(TAG, "updateEffectsSuppressor: " + ex.getMessage());
|
|
return;
|
|
}
|
|
|
|
if (hintsMatch(hints, DeviceConfig.getBoolean(DeviceConfig.NAMESPACE_SYSTEMUI,
|
|
SystemUiDeviceConfigFlags.VOLUME_SEPARATE_NOTIFICATION, false))) {
|
|
mSuppressor = suppressor;
|
|
if (mPreference != null) {
|
|
final String text = SuppressorHelper.getSuppressionText(mContext, suppressor);
|
|
mPreference.setSuppressionText(text);
|
|
}
|
|
}
|
|
}
|
|
|
|
@VisibleForTesting
|
|
boolean hintsMatch(int hints, boolean notificationSeparated) {
|
|
return (hints & NotificationListenerService.HINT_HOST_DISABLE_CALL_EFFECTS) != 0
|
|
|| (hints & NotificationListenerService.HINT_HOST_DISABLE_EFFECTS) != 0
|
|
|| ((hints & NotificationListenerService.HINT_HOST_DISABLE_NOTIFICATION_EFFECTS)
|
|
!= 0 && !notificationSeparated);
|
|
}
|
|
|
|
@VisibleForTesting
|
|
void setPreference(VolumeSeekBarPreference volumeSeekBarPreference) {
|
|
mPreference = volumeSeekBarPreference;
|
|
}
|
|
|
|
@VisibleForTesting
|
|
void setVibrator(Vibrator vibrator) {
|
|
mVibrator = vibrator;
|
|
}
|
|
|
|
private void selectPreferenceIconState() {
|
|
if (mPreference != null) {
|
|
if (mRingerMode == AudioManager.RINGER_MODE_NORMAL) {
|
|
mPreference.showIcon(mNormalIconId);
|
|
} else {
|
|
if (mRingerMode == AudioManager.RINGER_MODE_VIBRATE && mVibrator != null) {
|
|
mMuteIcon = mVibrateIconId;
|
|
} else {
|
|
mMuteIcon = mSilentIconId;
|
|
}
|
|
mPreference.showIcon(mMuteIcon);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* This slider can represent both ring and notification, or only ring.
|
|
* Note: This cannot be used in the constructor, as the reference to preference object would
|
|
* still be null.
|
|
*/
|
|
private void setPreferenceTitle() {
|
|
if (mPreference != null) {
|
|
mPreference.setTitle(mTitleId);
|
|
}
|
|
}
|
|
|
|
private final class H extends Handler {
|
|
private static final int UPDATE_EFFECTS_SUPPRESSOR = 1;
|
|
private static final int UPDATE_RINGER_MODE = 2;
|
|
|
|
private H() {
|
|
super(Looper.getMainLooper());
|
|
}
|
|
|
|
@Override
|
|
public void handleMessage(Message msg) {
|
|
switch (msg.what) {
|
|
case UPDATE_EFFECTS_SUPPRESSOR:
|
|
updateEffectsSuppressor();
|
|
break;
|
|
case UPDATE_RINGER_MODE:
|
|
updateRingerMode();
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
private class RingReceiver extends BroadcastReceiver {
|
|
private boolean mRegistered;
|
|
|
|
public void register(boolean register) {
|
|
if (mRegistered == register) return;
|
|
if (register) {
|
|
final IntentFilter filter = new IntentFilter();
|
|
filter.addAction(NotificationManager.ACTION_EFFECTS_SUPPRESSOR_CHANGED);
|
|
filter.addAction(AudioManager.INTERNAL_RINGER_MODE_CHANGED_ACTION);
|
|
mContext.registerReceiver(this, filter);
|
|
} else {
|
|
mContext.unregisterReceiver(this);
|
|
}
|
|
mRegistered = register;
|
|
}
|
|
|
|
@Override
|
|
public void onReceive(Context context, Intent intent) {
|
|
final String action = intent.getAction();
|
|
if (NotificationManager.ACTION_EFFECTS_SUPPRESSOR_CHANGED.equals(action)) {
|
|
mHandler.sendEmptyMessage(H.UPDATE_EFFECTS_SUPPRESSOR);
|
|
} else if (AudioManager.INTERNAL_RINGER_MODE_CHANGED_ACTION.equals(action)) {
|
|
mHandler.sendEmptyMessage(H.UPDATE_RINGER_MODE);
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|