Loading res/values/strings.xml +12 −0 Original line number Diff line number Diff line Loading @@ -7681,6 +7681,18 @@ <!-- Configure Notifications: Work profile section header [CHAR LIMIT=30] --> <string name="profile_section_header">Work notifications</string> <!-- Configure Notifications: setting title [CHAR LIMIT=80] --> <string name="asst_capability_prioritizer_title">Automatic Notification Prioritizer</string> <!-- Configure Notifications: setting summary [CHAR LIMIT=200] --> <string name="asst_capability_prioritizer_summary">Automatically silence and demote less important notifications</string> <!-- Configure Notifications: setting title [CHAR LIMIT=80] --> <string name="asst_capabilities_actions_replies_title">Smart actions and replies</string> <!-- Configure Notifications: setting summary [CHAR LIMIT=200] --> <string name="asst_capabilities_actions_replies_summary">Automatically add contextual notification actions and quick replies to notifications</string> <!-- Configure Notifications: setting title [CHAR LIMIT=80] --> <string name="hide_silent_icons_title">Hide silent notification status icons</string> res/xml/configure_notification_settings.xml +12 −0 Original line number Diff line number Diff line Loading @@ -26,6 +26,18 @@ settings:fragment="com.android.settings.notification.NotificationAssistantPicker" settings:controller="com.android.settings.notification.NotificationAssistantPreferenceController"/> <SwitchPreference android:key="asst_capability_prioritizer" android:title="@string/asst_capability_prioritizer_title" android:summary="@string/asst_capability_prioritizer_summary" settings:controller="com.android.settings.notification.AssistantCapabilityPreferenceController" /> <SwitchPreference android:key="asst_capabilities_actions_replies" android:title="@string/asst_capabilities_actions_replies_title" android:summary="@string/asst_capabilities_actions_replies_summary" settings:controller="com.android.settings.notification.AssistantCapabilityPreferenceController" /> <SwitchPreference android:key="hide_silent_icons" android:title="@string/hide_silent_icons_title" Loading src/com/android/settings/notification/AssistantCapabilityPreferenceController.java 0 → 100644 +75 −0 Original line number Diff line number Diff line /* * Copyright (C) 2019 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.content.Context; import android.os.UserHandle; import android.service.notification.Adjustment; import com.android.settings.core.TogglePreferenceController; import com.google.common.annotations.VisibleForTesting; import java.util.List; public class AssistantCapabilityPreferenceController extends TogglePreferenceController { static final String PRIORITIZER_KEY = "asst_capability_prioritizer"; static final String SMART_KEY = "asst_capabilities_actions_replies"; private NotificationBackend mBackend; public AssistantCapabilityPreferenceController(Context context, String key) { super(context, key); mBackend = new NotificationBackend(); } @VisibleForTesting void setBackend(NotificationBackend backend) { mBackend = backend; } @Override public boolean isChecked() { List<String> capabilities = mBackend.getAssistantCapabilities(mContext.getPackageName()); if (PRIORITIZER_KEY.equals(getPreferenceKey())) { return capabilities.contains(Adjustment.KEY_IMPORTANCE); } else if (SMART_KEY.equals(getPreferenceKey())) { return capabilities.contains(Adjustment.KEY_CONTEXTUAL_ACTIONS) && capabilities.contains(Adjustment.KEY_TEXT_REPLIES); } return false; } @Override public boolean setChecked(boolean isChecked) { if (PRIORITIZER_KEY.equals(getPreferenceKey())) { mBackend.allowAssistantCapability(Adjustment.KEY_IMPORTANCE, isChecked); } else if (SMART_KEY.equals(getPreferenceKey())) { mBackend.allowAssistantCapability(Adjustment.KEY_CONTEXTUAL_ACTIONS, isChecked); mBackend.allowAssistantCapability(Adjustment.KEY_TEXT_REPLIES, isChecked); } return true; } @Override public int getAvailabilityStatus() { return mBackend.getAllowedNotificationAssistant() != null ? AVAILABLE : DISABLED_DEPENDENT_SETTING; } } src/com/android/settings/notification/NotificationBackend.java +21 −0 Original line number Diff line number Diff line Loading @@ -339,6 +339,27 @@ public class NotificationBackend { } } public void allowAssistantCapability(String capability, boolean allowed) { try { if (allowed) { sINM.allowAssistantCapability(capability); } else { sINM.disallowAssistantCapability(capability); } } catch (Exception e) { Log.w(TAG, "Error calling NoMan", e); } } public List<String> getAssistantCapabilities(String pkg) { try { return sINM.getAllowedAssistantCapabilities(pkg); } catch (Exception e) { Log.w(TAG, "Error calling NoMan", e); } return new ArrayList<>(); } protected void recordAggregatedUsageEvents(Context context, AppRow appRow) { long now = System.currentTimeMillis(); long startTime = now - (DateUtils.DAY_IN_MILLIS * DAYS_TO_CHECK); Loading tests/robotests/src/com/android/settings/notification/AssistantCapabilityPreferenceControllerTest.java 0 → 100644 +169 −0 Original line number Diff line number Diff line /* * Copyright (C) 2019 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 static com.android.settings.core.BasePreferenceController.AVAILABLE; import static com.android.settings.core.BasePreferenceController.DISABLED_DEPENDENT_SETTING; import static com.android.settings.notification.AssistantCapabilityPreferenceController.PRIORITIZER_KEY; import static com.android.settings.notification.AssistantCapabilityPreferenceController.SMART_KEY; import static com.google.common.truth.Truth.assertThat; import static org.mockito.ArgumentMatchers.anyString; import static org.mockito.Mockito.mock; import static org.mockito.Mockito.verify; import static org.mockito.Mockito.when; import android.content.ComponentName; import android.content.Context; import android.service.notification.Adjustment; import org.junit.Before; import org.junit.Test; import org.junit.runner.RunWith; import org.mockito.Mock; import org.mockito.MockitoAnnotations; import org.robolectric.RobolectricTestRunner; import org.robolectric.RuntimeEnvironment; import java.util.ArrayList; import java.util.List; import androidx.preference.Preference; import androidx.preference.PreferenceScreen; @RunWith(RobolectricTestRunner.class) public class AssistantCapabilityPreferenceControllerTest { @Mock private NotificationBackend mBackend; @Mock private PreferenceScreen mScreen; private Context mContext; private AssistantCapabilityPreferenceController mPrioritizerController; private AssistantCapabilityPreferenceController mChipController; private Preference mPrioritizerPreference; private Preference mChipPreference; @Before public void setUp() { MockitoAnnotations.initMocks(this); mContext = RuntimeEnvironment.application; mPrioritizerController = new AssistantCapabilityPreferenceController( mContext, PRIORITIZER_KEY); mPrioritizerController.setBackend(mBackend); mPrioritizerPreference = new Preference(mContext); mPrioritizerPreference.setKey(mPrioritizerController.getPreferenceKey()); when(mScreen.findPreference( mPrioritizerController.getPreferenceKey())).thenReturn(mPrioritizerPreference); mChipController = new AssistantCapabilityPreferenceController(mContext, SMART_KEY); mChipController.setBackend(mBackend); mChipPreference = new Preference(mContext); mChipPreference.setKey(mChipController.getPreferenceKey()); when(mScreen.findPreference( mChipController.getPreferenceKey())).thenReturn(mChipPreference); } @Test public void getAvailabilityStatus_NAS() { when(mBackend.getAllowedNotificationAssistant()).thenReturn(mock(ComponentName.class)); assertThat(mPrioritizerController.getAvailabilityStatus()) .isEqualTo(AVAILABLE); assertThat(mChipController.getAvailabilityStatus()) .isEqualTo(AVAILABLE); } @Test public void getAvailabilityStatus_noNAS() { when(mBackend.getAllowedNotificationAssistant()).thenReturn(null); assertThat(mPrioritizerController.getAvailabilityStatus()) .isEqualTo(DISABLED_DEPENDENT_SETTING); assertThat(mChipController.getAvailabilityStatus()) .isEqualTo(DISABLED_DEPENDENT_SETTING); } @Test public void isChecked_prioritizerSettingIsOff_false() { List<String> capabilities = new ArrayList<>(); capabilities.add(Adjustment.KEY_USER_SENTIMENT); when(mBackend.getAssistantCapabilities(anyString())).thenReturn(capabilities); assertThat(mPrioritizerController.isChecked()).isFalse(); } @Test public void isChecked_prioritizerSettingIsOn_true() { List<String> capabilities = new ArrayList<>(); capabilities.add(Adjustment.KEY_IMPORTANCE); when(mBackend.getAssistantCapabilities(anyString())).thenReturn(capabilities); assertThat(mPrioritizerController.isChecked()).isTrue(); } @Test public void isChecked_chipSettingIsOff_false() { List<String> capabilities = new ArrayList<>(); capabilities.add(Adjustment.KEY_IMPORTANCE); when(mBackend.getAssistantCapabilities(anyString())).thenReturn(capabilities); assertThat(mChipController.isChecked()).isFalse(); capabilities = new ArrayList<>(); capabilities.add(Adjustment.KEY_CONTEXTUAL_ACTIONS); when(mBackend.getAssistantCapabilities(anyString())).thenReturn(capabilities); assertThat(mChipController.isChecked()).isFalse(); capabilities = new ArrayList<>(); capabilities.add(Adjustment.KEY_TEXT_REPLIES); when(mBackend.getAssistantCapabilities(anyString())).thenReturn(capabilities); assertThat(mChipController.isChecked()).isFalse(); } @Test public void isChecked_chipSettingIsOn_true() { List<String> capabilities = new ArrayList<>(); capabilities.add(Adjustment.KEY_TEXT_REPLIES); capabilities.add(Adjustment.KEY_CONTEXTUAL_ACTIONS); when(mBackend.getAssistantCapabilities(anyString())).thenReturn(capabilities); assertThat(mChipController.isChecked()).isTrue(); } @Test public void onPreferenceChange_prioritizerOn() { mPrioritizerController.onPreferenceChange(mPrioritizerPreference, true); verify(mBackend).allowAssistantCapability(Adjustment.KEY_IMPORTANCE, true); } @Test public void onPreferenceChange_prioritizerOff() { mPrioritizerController.onPreferenceChange(mPrioritizerPreference, false); verify(mBackend).allowAssistantCapability(Adjustment.KEY_IMPORTANCE, false); } @Test public void onPreferenceChange_chipsOn() { mChipController.onPreferenceChange(mChipPreference, true); verify(mBackend).allowAssistantCapability(Adjustment.KEY_CONTEXTUAL_ACTIONS, true); verify(mBackend).allowAssistantCapability(Adjustment.KEY_TEXT_REPLIES, true); } @Test public void onPreferenceChange_chipsOff() { mChipController.onPreferenceChange(mChipPreference, false); verify(mBackend).allowAssistantCapability(Adjustment.KEY_CONTEXTUAL_ACTIONS, false); verify(mBackend).allowAssistantCapability(Adjustment.KEY_TEXT_REPLIES, false); } } Loading
res/values/strings.xml +12 −0 Original line number Diff line number Diff line Loading @@ -7681,6 +7681,18 @@ <!-- Configure Notifications: Work profile section header [CHAR LIMIT=30] --> <string name="profile_section_header">Work notifications</string> <!-- Configure Notifications: setting title [CHAR LIMIT=80] --> <string name="asst_capability_prioritizer_title">Automatic Notification Prioritizer</string> <!-- Configure Notifications: setting summary [CHAR LIMIT=200] --> <string name="asst_capability_prioritizer_summary">Automatically silence and demote less important notifications</string> <!-- Configure Notifications: setting title [CHAR LIMIT=80] --> <string name="asst_capabilities_actions_replies_title">Smart actions and replies</string> <!-- Configure Notifications: setting summary [CHAR LIMIT=200] --> <string name="asst_capabilities_actions_replies_summary">Automatically add contextual notification actions and quick replies to notifications</string> <!-- Configure Notifications: setting title [CHAR LIMIT=80] --> <string name="hide_silent_icons_title">Hide silent notification status icons</string>
res/xml/configure_notification_settings.xml +12 −0 Original line number Diff line number Diff line Loading @@ -26,6 +26,18 @@ settings:fragment="com.android.settings.notification.NotificationAssistantPicker" settings:controller="com.android.settings.notification.NotificationAssistantPreferenceController"/> <SwitchPreference android:key="asst_capability_prioritizer" android:title="@string/asst_capability_prioritizer_title" android:summary="@string/asst_capability_prioritizer_summary" settings:controller="com.android.settings.notification.AssistantCapabilityPreferenceController" /> <SwitchPreference android:key="asst_capabilities_actions_replies" android:title="@string/asst_capabilities_actions_replies_title" android:summary="@string/asst_capabilities_actions_replies_summary" settings:controller="com.android.settings.notification.AssistantCapabilityPreferenceController" /> <SwitchPreference android:key="hide_silent_icons" android:title="@string/hide_silent_icons_title" Loading
src/com/android/settings/notification/AssistantCapabilityPreferenceController.java 0 → 100644 +75 −0 Original line number Diff line number Diff line /* * Copyright (C) 2019 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.content.Context; import android.os.UserHandle; import android.service.notification.Adjustment; import com.android.settings.core.TogglePreferenceController; import com.google.common.annotations.VisibleForTesting; import java.util.List; public class AssistantCapabilityPreferenceController extends TogglePreferenceController { static final String PRIORITIZER_KEY = "asst_capability_prioritizer"; static final String SMART_KEY = "asst_capabilities_actions_replies"; private NotificationBackend mBackend; public AssistantCapabilityPreferenceController(Context context, String key) { super(context, key); mBackend = new NotificationBackend(); } @VisibleForTesting void setBackend(NotificationBackend backend) { mBackend = backend; } @Override public boolean isChecked() { List<String> capabilities = mBackend.getAssistantCapabilities(mContext.getPackageName()); if (PRIORITIZER_KEY.equals(getPreferenceKey())) { return capabilities.contains(Adjustment.KEY_IMPORTANCE); } else if (SMART_KEY.equals(getPreferenceKey())) { return capabilities.contains(Adjustment.KEY_CONTEXTUAL_ACTIONS) && capabilities.contains(Adjustment.KEY_TEXT_REPLIES); } return false; } @Override public boolean setChecked(boolean isChecked) { if (PRIORITIZER_KEY.equals(getPreferenceKey())) { mBackend.allowAssistantCapability(Adjustment.KEY_IMPORTANCE, isChecked); } else if (SMART_KEY.equals(getPreferenceKey())) { mBackend.allowAssistantCapability(Adjustment.KEY_CONTEXTUAL_ACTIONS, isChecked); mBackend.allowAssistantCapability(Adjustment.KEY_TEXT_REPLIES, isChecked); } return true; } @Override public int getAvailabilityStatus() { return mBackend.getAllowedNotificationAssistant() != null ? AVAILABLE : DISABLED_DEPENDENT_SETTING; } }
src/com/android/settings/notification/NotificationBackend.java +21 −0 Original line number Diff line number Diff line Loading @@ -339,6 +339,27 @@ public class NotificationBackend { } } public void allowAssistantCapability(String capability, boolean allowed) { try { if (allowed) { sINM.allowAssistantCapability(capability); } else { sINM.disallowAssistantCapability(capability); } } catch (Exception e) { Log.w(TAG, "Error calling NoMan", e); } } public List<String> getAssistantCapabilities(String pkg) { try { return sINM.getAllowedAssistantCapabilities(pkg); } catch (Exception e) { Log.w(TAG, "Error calling NoMan", e); } return new ArrayList<>(); } protected void recordAggregatedUsageEvents(Context context, AppRow appRow) { long now = System.currentTimeMillis(); long startTime = now - (DateUtils.DAY_IN_MILLIS * DAYS_TO_CHECK); Loading
tests/robotests/src/com/android/settings/notification/AssistantCapabilityPreferenceControllerTest.java 0 → 100644 +169 −0 Original line number Diff line number Diff line /* * Copyright (C) 2019 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 static com.android.settings.core.BasePreferenceController.AVAILABLE; import static com.android.settings.core.BasePreferenceController.DISABLED_DEPENDENT_SETTING; import static com.android.settings.notification.AssistantCapabilityPreferenceController.PRIORITIZER_KEY; import static com.android.settings.notification.AssistantCapabilityPreferenceController.SMART_KEY; import static com.google.common.truth.Truth.assertThat; import static org.mockito.ArgumentMatchers.anyString; import static org.mockito.Mockito.mock; import static org.mockito.Mockito.verify; import static org.mockito.Mockito.when; import android.content.ComponentName; import android.content.Context; import android.service.notification.Adjustment; import org.junit.Before; import org.junit.Test; import org.junit.runner.RunWith; import org.mockito.Mock; import org.mockito.MockitoAnnotations; import org.robolectric.RobolectricTestRunner; import org.robolectric.RuntimeEnvironment; import java.util.ArrayList; import java.util.List; import androidx.preference.Preference; import androidx.preference.PreferenceScreen; @RunWith(RobolectricTestRunner.class) public class AssistantCapabilityPreferenceControllerTest { @Mock private NotificationBackend mBackend; @Mock private PreferenceScreen mScreen; private Context mContext; private AssistantCapabilityPreferenceController mPrioritizerController; private AssistantCapabilityPreferenceController mChipController; private Preference mPrioritizerPreference; private Preference mChipPreference; @Before public void setUp() { MockitoAnnotations.initMocks(this); mContext = RuntimeEnvironment.application; mPrioritizerController = new AssistantCapabilityPreferenceController( mContext, PRIORITIZER_KEY); mPrioritizerController.setBackend(mBackend); mPrioritizerPreference = new Preference(mContext); mPrioritizerPreference.setKey(mPrioritizerController.getPreferenceKey()); when(mScreen.findPreference( mPrioritizerController.getPreferenceKey())).thenReturn(mPrioritizerPreference); mChipController = new AssistantCapabilityPreferenceController(mContext, SMART_KEY); mChipController.setBackend(mBackend); mChipPreference = new Preference(mContext); mChipPreference.setKey(mChipController.getPreferenceKey()); when(mScreen.findPreference( mChipController.getPreferenceKey())).thenReturn(mChipPreference); } @Test public void getAvailabilityStatus_NAS() { when(mBackend.getAllowedNotificationAssistant()).thenReturn(mock(ComponentName.class)); assertThat(mPrioritizerController.getAvailabilityStatus()) .isEqualTo(AVAILABLE); assertThat(mChipController.getAvailabilityStatus()) .isEqualTo(AVAILABLE); } @Test public void getAvailabilityStatus_noNAS() { when(mBackend.getAllowedNotificationAssistant()).thenReturn(null); assertThat(mPrioritizerController.getAvailabilityStatus()) .isEqualTo(DISABLED_DEPENDENT_SETTING); assertThat(mChipController.getAvailabilityStatus()) .isEqualTo(DISABLED_DEPENDENT_SETTING); } @Test public void isChecked_prioritizerSettingIsOff_false() { List<String> capabilities = new ArrayList<>(); capabilities.add(Adjustment.KEY_USER_SENTIMENT); when(mBackend.getAssistantCapabilities(anyString())).thenReturn(capabilities); assertThat(mPrioritizerController.isChecked()).isFalse(); } @Test public void isChecked_prioritizerSettingIsOn_true() { List<String> capabilities = new ArrayList<>(); capabilities.add(Adjustment.KEY_IMPORTANCE); when(mBackend.getAssistantCapabilities(anyString())).thenReturn(capabilities); assertThat(mPrioritizerController.isChecked()).isTrue(); } @Test public void isChecked_chipSettingIsOff_false() { List<String> capabilities = new ArrayList<>(); capabilities.add(Adjustment.KEY_IMPORTANCE); when(mBackend.getAssistantCapabilities(anyString())).thenReturn(capabilities); assertThat(mChipController.isChecked()).isFalse(); capabilities = new ArrayList<>(); capabilities.add(Adjustment.KEY_CONTEXTUAL_ACTIONS); when(mBackend.getAssistantCapabilities(anyString())).thenReturn(capabilities); assertThat(mChipController.isChecked()).isFalse(); capabilities = new ArrayList<>(); capabilities.add(Adjustment.KEY_TEXT_REPLIES); when(mBackend.getAssistantCapabilities(anyString())).thenReturn(capabilities); assertThat(mChipController.isChecked()).isFalse(); } @Test public void isChecked_chipSettingIsOn_true() { List<String> capabilities = new ArrayList<>(); capabilities.add(Adjustment.KEY_TEXT_REPLIES); capabilities.add(Adjustment.KEY_CONTEXTUAL_ACTIONS); when(mBackend.getAssistantCapabilities(anyString())).thenReturn(capabilities); assertThat(mChipController.isChecked()).isTrue(); } @Test public void onPreferenceChange_prioritizerOn() { mPrioritizerController.onPreferenceChange(mPrioritizerPreference, true); verify(mBackend).allowAssistantCapability(Adjustment.KEY_IMPORTANCE, true); } @Test public void onPreferenceChange_prioritizerOff() { mPrioritizerController.onPreferenceChange(mPrioritizerPreference, false); verify(mBackend).allowAssistantCapability(Adjustment.KEY_IMPORTANCE, false); } @Test public void onPreferenceChange_chipsOn() { mChipController.onPreferenceChange(mChipPreference, true); verify(mBackend).allowAssistantCapability(Adjustment.KEY_CONTEXTUAL_ACTIONS, true); verify(mBackend).allowAssistantCapability(Adjustment.KEY_TEXT_REPLIES, true); } @Test public void onPreferenceChange_chipsOff() { mChipController.onPreferenceChange(mChipPreference, false); verify(mBackend).allowAssistantCapability(Adjustment.KEY_CONTEXTUAL_ACTIONS, false); verify(mBackend).allowAssistantCapability(Adjustment.KEY_TEXT_REPLIES, false); } }