Snap for 10402374 from ab1b393412
to udc-qpr1-release
Change-Id: If9ffac88333a07415861a8389ac8eda62bf8737f
This commit is contained in:
@@ -81,6 +81,8 @@ public class ApprovalPreferenceController extends BasePreferenceController {
|
|||||||
final RestrictedSwitchPreference preference =
|
final RestrictedSwitchPreference preference =
|
||||||
(RestrictedSwitchPreference) pref;
|
(RestrictedSwitchPreference) pref;
|
||||||
final CharSequence label = mPkgInfo.applicationInfo.loadLabel(mPm);
|
final CharSequence label = mPkgInfo.applicationInfo.loadLabel(mPm);
|
||||||
|
final boolean isAllowedCn = mCn.flattenToShortString().length()
|
||||||
|
<= NotificationManager.MAX_SERVICE_COMPONENT_NAME_LENGTH;
|
||||||
final boolean isEnabled = isServiceEnabled(mCn);
|
final boolean isEnabled = isServiceEnabled(mCn);
|
||||||
preference.setChecked(isEnabled);
|
preference.setChecked(isEnabled);
|
||||||
preference.setOnPreferenceChangeListener((p, newValue) -> {
|
preference.setOnPreferenceChangeListener((p, newValue) -> {
|
||||||
@@ -105,7 +107,8 @@ public class ApprovalPreferenceController extends BasePreferenceController {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
preference.updateState(mCn.getPackageName(), mPkgInfo.applicationInfo.uid, isEnabled);
|
preference.updateState(
|
||||||
|
mCn.getPackageName(), mPkgInfo.applicationInfo.uid, isAllowedCn, isEnabled);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void disable(final ComponentName cn) {
|
public void disable(final ComponentName cn) {
|
||||||
|
@@ -164,15 +164,18 @@ public class LocaleDialogFragment extends InstrumentedDialogFragment {
|
|||||||
public void onClick(DialogInterface dialog, int which) {
|
public void onClick(DialogInterface dialog, int which) {
|
||||||
if (mDialogType == DIALOG_CONFIRM_SYSTEM_DEFAULT) {
|
if (mDialogType == DIALOG_CONFIRM_SYSTEM_DEFAULT) {
|
||||||
int result = Activity.RESULT_CANCELED;
|
int result = Activity.RESULT_CANCELED;
|
||||||
|
boolean changed = false;
|
||||||
if (which == DialogInterface.BUTTON_POSITIVE) {
|
if (which == DialogInterface.BUTTON_POSITIVE) {
|
||||||
result = Activity.RESULT_OK;
|
result = Activity.RESULT_OK;
|
||||||
|
changed = true;
|
||||||
}
|
}
|
||||||
Intent intent = new Intent();
|
Intent intent = new Intent();
|
||||||
Bundle bundle = new Bundle();
|
Bundle bundle = new Bundle();
|
||||||
bundle.putInt(ARG_DIALOG_TYPE, DIALOG_CONFIRM_SYSTEM_DEFAULT);
|
bundle.putInt(ARG_DIALOG_TYPE, DIALOG_CONFIRM_SYSTEM_DEFAULT);
|
||||||
intent.putExtras(bundle);
|
intent.putExtras(bundle);
|
||||||
mParent.onActivityResult(DIALOG_CONFIRM_SYSTEM_DEFAULT, result, intent);
|
mParent.onActivityResult(DIALOG_CONFIRM_SYSTEM_DEFAULT, result, intent);
|
||||||
mMetricsFeatureProvider.action(mContext, SettingsEnums.ACTION_CHANGE_LANGUAGE);
|
mMetricsFeatureProvider.action(mContext, SettingsEnums.ACTION_CHANGE_LANGUAGE,
|
||||||
|
changed);
|
||||||
}
|
}
|
||||||
mShouldKeepDialog = false;
|
mShouldKeepDialog = false;
|
||||||
}
|
}
|
||||||
|
@@ -16,6 +16,7 @@
|
|||||||
|
|
||||||
package com.android.settings.localepicker;
|
package com.android.settings.localepicker;
|
||||||
|
|
||||||
|
import android.app.settings.SettingsEnums;
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.graphics.Canvas;
|
import android.graphics.Canvas;
|
||||||
import android.os.Bundle;
|
import android.os.Bundle;
|
||||||
@@ -37,6 +38,7 @@ import androidx.recyclerview.widget.RecyclerView;
|
|||||||
import com.android.internal.app.LocalePicker;
|
import com.android.internal.app.LocalePicker;
|
||||||
import com.android.internal.app.LocaleStore;
|
import com.android.internal.app.LocaleStore;
|
||||||
import com.android.settings.R;
|
import com.android.settings.R;
|
||||||
|
import com.android.settings.overlay.FeatureFactory;
|
||||||
import com.android.settings.shortcut.ShortcutsUpdateTask;
|
import com.android.settings.shortcut.ShortcutsUpdateTask;
|
||||||
|
|
||||||
import java.text.NumberFormat;
|
import java.text.NumberFormat;
|
||||||
@@ -210,6 +212,13 @@ class LocaleDragAndDropAdapter
|
|||||||
Log.e(TAG, String.format(Locale.US,
|
Log.e(TAG, String.format(Locale.US,
|
||||||
"Negative position in onItemMove %d -> %d", fromPosition, toPosition));
|
"Negative position in onItemMove %d -> %d", fromPosition, toPosition));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (fromPosition != toPosition) {
|
||||||
|
FeatureFactory.getFactory(mContext).getMetricsFeatureProvider()
|
||||||
|
.action(mContext, SettingsEnums.ACTION_REORDER_LANGUAGE,
|
||||||
|
mDragLocale.getLocale().getDisplayName() + " move to " + toPosition);
|
||||||
|
}
|
||||||
|
|
||||||
notifyItemChanged(fromPosition); // to update the numbers
|
notifyItemChanged(fromPosition); // to update the numbers
|
||||||
notifyItemChanged(toPosition);
|
notifyItemChanged(toPosition);
|
||||||
notifyItemMoved(fromPosition, toPosition);
|
notifyItemMoved(fromPosition, toPosition);
|
||||||
@@ -244,8 +253,13 @@ class LocaleDragAndDropAdapter
|
|||||||
|
|
||||||
void removeChecked() {
|
void removeChecked() {
|
||||||
int itemCount = mFeedItemList.size();
|
int itemCount = mFeedItemList.size();
|
||||||
|
LocaleStore.LocaleInfo localeInfo;
|
||||||
for (int i = itemCount - 1; i >= 0; i--) {
|
for (int i = itemCount - 1; i >= 0; i--) {
|
||||||
if (mFeedItemList.get(i).getChecked()) {
|
localeInfo = mFeedItemList.get(i);
|
||||||
|
if (localeInfo.getChecked()) {
|
||||||
|
FeatureFactory.getFactory(mContext).getMetricsFeatureProvider()
|
||||||
|
.action(mContext, SettingsEnums.ACTION_REMOVE_LANGUAGE,
|
||||||
|
localeInfo.getLocale().getDisplayName());
|
||||||
mFeedItemList.remove(i);
|
mFeedItemList.remove(i);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -199,9 +199,11 @@ public class LocaleListEditor extends RestrictedSettingsFragment implements View
|
|||||||
localeInfo = (LocaleStore.LocaleInfo) data.getSerializableExtra(INTENT_LOCALE_KEY);
|
localeInfo = (LocaleStore.LocaleInfo) data.getSerializableExtra(INTENT_LOCALE_KEY);
|
||||||
String preferencesTags = Settings.System.getString(
|
String preferencesTags = Settings.System.getString(
|
||||||
getContext().getContentResolver(), Settings.System.LOCALE_PREFERENCES);
|
getContext().getContentResolver(), Settings.System.LOCALE_PREFERENCES);
|
||||||
|
localeInfo = mayAppendUnicodeTags(localeInfo, preferencesTags);
|
||||||
mAdapter.addLocale(mayAppendUnicodeTags(localeInfo, preferencesTags));
|
mAdapter.addLocale(localeInfo);
|
||||||
updateVisibilityOfRemoveMenu();
|
updateVisibilityOfRemoveMenu();
|
||||||
|
mMetricsFeatureProvider.action(getContext(), SettingsEnums.ACTION_ADD_LANGUAGE,
|
||||||
|
localeInfo.getLocale().getDisplayName());
|
||||||
} else if (requestCode == DIALOG_CONFIRM_SYSTEM_DEFAULT) {
|
} else if (requestCode == DIALOG_CONFIRM_SYSTEM_DEFAULT) {
|
||||||
localeInfo = mAdapter.getFeedItemList().get(0);
|
localeInfo = mAdapter.getFeedItemList().get(0);
|
||||||
if (resultCode == Activity.RESULT_OK) {
|
if (resultCode == Activity.RESULT_OK) {
|
||||||
@@ -214,6 +216,9 @@ public class LocaleListEditor extends RestrictedSettingsFragment implements View
|
|||||||
LocaleDialogFragment localeDialogFragment = LocaleDialogFragment.newInstance();
|
LocaleDialogFragment localeDialogFragment = LocaleDialogFragment.newInstance();
|
||||||
localeDialogFragment.setArguments(args);
|
localeDialogFragment.setArguments(args);
|
||||||
localeDialogFragment.show(mFragmentManager, TAG_DIALOG_NOT_AVAILABLE);
|
localeDialogFragment.show(mFragmentManager, TAG_DIALOG_NOT_AVAILABLE);
|
||||||
|
mMetricsFeatureProvider.action(getContext(),
|
||||||
|
SettingsEnums.ACTION_NOT_SUPPORTED_SYSTEM_LANGUAGE,
|
||||||
|
localeInfo.getLocale().getDisplayName());
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
mAdapter.notifyListChanged(localeInfo);
|
mAdapter.notifyListChanged(localeInfo);
|
||||||
|
@@ -67,7 +67,9 @@ public class NotificationAccessConfirmationActivity extends Activity
|
|||||||
mUserId = getIntent().getIntExtra(EXTRA_USER_ID, UserHandle.USER_NULL);
|
mUserId = getIntent().getIntExtra(EXTRA_USER_ID, UserHandle.USER_NULL);
|
||||||
CharSequence mAppLabel;
|
CharSequence mAppLabel;
|
||||||
|
|
||||||
if (mComponentName == null || mComponentName.getPackageName() == null) {
|
if (mComponentName == null || mComponentName.getPackageName() == null
|
||||||
|
|| mComponentName.flattenToString().length()
|
||||||
|
> NotificationManager.MAX_SERVICE_COMPONENT_NAME_LENGTH) {
|
||||||
finish();
|
finish();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@@ -66,7 +66,6 @@ public class NotificationAccessSettings extends EmptyTextSettings {
|
|||||||
private static final String TAG = "NotifAccessSettings";
|
private static final String TAG = "NotifAccessSettings";
|
||||||
static final String ALLOWED_KEY = "allowed";
|
static final String ALLOWED_KEY = "allowed";
|
||||||
static final String NOT_ALLOWED_KEY = "not_allowed";
|
static final String NOT_ALLOWED_KEY = "not_allowed";
|
||||||
private static final int MAX_CN_LENGTH = 500;
|
|
||||||
|
|
||||||
private static final ManagedServiceSettings.Config CONFIG =
|
private static final ManagedServiceSettings.Config CONFIG =
|
||||||
new ManagedServiceSettings.Config.Builder()
|
new ManagedServiceSettings.Config.Builder()
|
||||||
@@ -150,7 +149,8 @@ public class NotificationAccessSettings extends EmptyTextSettings {
|
|||||||
for (ServiceInfo service : services) {
|
for (ServiceInfo service : services) {
|
||||||
final ComponentName cn = new ComponentName(service.packageName, service.name);
|
final ComponentName cn = new ComponentName(service.packageName, service.name);
|
||||||
boolean isAllowed = mNm.isNotificationListenerAccessGranted(cn);
|
boolean isAllowed = mNm.isNotificationListenerAccessGranted(cn);
|
||||||
if (!isAllowed && cn.flattenToString().length() > MAX_CN_LENGTH) {
|
if (!isAllowed && cn.flattenToString().length()
|
||||||
|
> NotificationManager.MAX_SERVICE_COMPONENT_NAME_LENGTH) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -0,0 +1,104 @@
|
|||||||
|
/*
|
||||||
|
* 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.notification;
|
||||||
|
|
||||||
|
import static com.android.internal.notification.NotificationAccessConfirmationActivityContract.EXTRA_COMPONENT_NAME;
|
||||||
|
|
||||||
|
import static com.google.common.truth.Truth.assertThat;
|
||||||
|
|
||||||
|
import static org.robolectric.Shadows.shadowOf;
|
||||||
|
|
||||||
|
import android.annotation.Nullable;
|
||||||
|
import android.app.Activity;
|
||||||
|
import android.content.ComponentName;
|
||||||
|
import android.content.Intent;
|
||||||
|
import android.content.pm.ApplicationInfo;
|
||||||
|
import android.content.pm.PackageInfo;
|
||||||
|
import android.widget.TextView;
|
||||||
|
|
||||||
|
import com.android.settings.R;
|
||||||
|
|
||||||
|
import com.google.common.base.Strings;
|
||||||
|
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.runner.RunWith;
|
||||||
|
import org.robolectric.Robolectric;
|
||||||
|
import org.robolectric.RobolectricTestRunner;
|
||||||
|
import org.robolectric.RuntimeEnvironment;
|
||||||
|
|
||||||
|
@RunWith(RobolectricTestRunner.class)
|
||||||
|
public class NotificationAccessConfirmationActivityTest {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void start_showsDialog() {
|
||||||
|
ComponentName cn = new ComponentName("com.example", "com.example.SomeService");
|
||||||
|
installPackage(cn.getPackageName(), "X");
|
||||||
|
|
||||||
|
NotificationAccessConfirmationActivity activity = startActivityWithIntent(cn);
|
||||||
|
|
||||||
|
assertThat(activity.isFinishing()).isFalse();
|
||||||
|
assertThat(getDialogText(activity)).isEqualTo(
|
||||||
|
activity.getString(R.string.notification_listener_security_warning_summary, "X"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void start_withMissingPackage_finishes() {
|
||||||
|
ComponentName cn = new ComponentName("com.example", "com.example.SomeService");
|
||||||
|
|
||||||
|
NotificationAccessConfirmationActivity activity = startActivityWithIntent(cn);
|
||||||
|
|
||||||
|
assertThat(getDialogText(activity)).isNull();
|
||||||
|
assertThat(activity.isFinishing()).isTrue();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void start_componentNameTooLong_finishes() {
|
||||||
|
ComponentName longCn = new ComponentName("com.example", Strings.repeat("Blah", 150));
|
||||||
|
installPackage(longCn.getPackageName(), "<Unused>");
|
||||||
|
|
||||||
|
NotificationAccessConfirmationActivity activity = startActivityWithIntent(longCn);
|
||||||
|
|
||||||
|
assertThat(getDialogText(activity)).isNull();
|
||||||
|
assertThat(activity.isFinishing()).isTrue();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static NotificationAccessConfirmationActivity startActivityWithIntent(
|
||||||
|
ComponentName cn) {
|
||||||
|
return Robolectric.buildActivity(
|
||||||
|
NotificationAccessConfirmationActivity.class,
|
||||||
|
new Intent().putExtra(EXTRA_COMPONENT_NAME, cn))
|
||||||
|
.setup()
|
||||||
|
.get();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void installPackage(String packageName, String appName) {
|
||||||
|
PackageInfo pi = new PackageInfo();
|
||||||
|
pi.packageName = packageName;
|
||||||
|
pi.applicationInfo = new ApplicationInfo();
|
||||||
|
pi.applicationInfo.packageName = packageName;
|
||||||
|
pi.applicationInfo.name = appName;
|
||||||
|
shadowOf(RuntimeEnvironment.application.getPackageManager()).installPackage(pi);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nullable
|
||||||
|
private static String getDialogText(Activity activity) {
|
||||||
|
TextView tv = activity.getWindow().findViewById(android.R.id.message);
|
||||||
|
CharSequence text = (tv != null ? tv.getText() : null);
|
||||||
|
return text != null ? text.toString() : null;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@@ -83,6 +83,36 @@ public class ApprovalPreferenceControllerTest {
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void updateState_enabled() {
|
||||||
|
when(mAppOpsManager.noteOpNoThrow(anyInt(), anyInt(), anyString())).thenReturn(
|
||||||
|
AppOpsManager.MODE_ALLOWED);
|
||||||
|
when(mNm.isNotificationListenerAccessGranted(mCn)).thenReturn(true);
|
||||||
|
RestrictedSwitchPreference pref = new RestrictedSwitchPreference(
|
||||||
|
mContext);
|
||||||
|
pref.setAppOps(mAppOpsManager);
|
||||||
|
|
||||||
|
mController.updateState(pref);
|
||||||
|
|
||||||
|
assertThat(pref.isEnabled()).isTrue();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void updateState_invalidCn_disabled() {
|
||||||
|
ComponentName longCn = new ComponentName("com.example.package",
|
||||||
|
com.google.common.base.Strings.repeat("Blah", 150));
|
||||||
|
mController.setCn(longCn);
|
||||||
|
when(mAppOpsManager.noteOpNoThrow(anyInt(), anyInt(), anyString())).thenReturn(
|
||||||
|
AppOpsManager.MODE_ALLOWED);
|
||||||
|
RestrictedSwitchPreference pref = new RestrictedSwitchPreference(
|
||||||
|
mContext);
|
||||||
|
pref.setAppOps(mAppOpsManager);
|
||||||
|
|
||||||
|
mController.updateState(pref);
|
||||||
|
|
||||||
|
assertThat(pref.isEnabled()).isFalse();
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void updateState_checked() {
|
public void updateState_checked() {
|
||||||
when(mAppOpsManager.noteOpNoThrow(anyInt(), anyInt(), anyString())).thenReturn(
|
when(mAppOpsManager.noteOpNoThrow(anyInt(), anyInt(), anyString())).thenReturn(
|
||||||
|
Reference in New Issue
Block a user