Не удается получить разрешение WRITE_SETTINGS
Когда у меня есть целевой API 23 на Android M Preview 3, я не могу получить разрешение Manifest.permission.WRITE_SETTTINGS.
requestPermissions(new String[]{Manifest.permission.WRITE_SETTINGS},
101);
Разрешение запроса не вызывает диалог, который я ожидал бы, но если я сделаю следующий вызов без этого разрешения,
RingtoneManager.setActualDefaultRingtoneUri(activity, RingtoneManager.TYPE_RINGTONE, ringUri);
Вызов будет за исключением того, что у меня нет разрешения.
Я не уверен, куда идти отсюда. Есть ли новый рингтон API для 23? Или изменилось ли это разрешение, чтобы не изменять какие-либо несистемные приложения для мелодии звонка?
Ответы
Ответ 1
Чтобы использовать WRITE_SETTINGS
, на основе документов:
-
Элемент <uses-permission>
в манифесте как обычно
-
Вызовите Settings.System.canWrite()
чтобы узнать, имеете ли вы право на запись настроек.
-
Если canWrite()
возвращает false
, запустите действие ACTION_MANAGE_WRITE_SETTINGS
чтобы пользователь мог согласиться с ним, чтобы позволить вашему приложению фактически выполнять запись в настройки.
Другими словами, запись в настройки теперь является двойной подпиской (согласиться на установку, согласиться отдельно в настройках, чтобы разрешить), сродни API администратора устройства, службами доступности и т.д.
Также обратите внимание, что я еще не пробовал использовать их - это основано на исследовании, которое я сделал вчера об изменениях в Android 6.0.
Ответ 2
В дополнение к ответу от CommonsWare и комментарию от Ogix, здесь есть некоторый фиктивный код:
private boolean checkSystemWritePermission() {
boolean retVal = true;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
retVal = Settings.System.canWrite(this);
Log.d(TAG, "Can Write Settings: " + retVal);
if(retVal){
Toast.makeText(this, "Write allowed :-)", Toast.LENGTH_LONG).show();
}else{
Toast.makeText(this, "Write not allowed :-(", Toast.LENGTH_LONG).show();
FragmentManager fm = getFragmentManager();
PopupWritePermission dialogFragment = new PopupWritePermission();
dialogFragment.show(fm, getString(R.string.popup_writesettings_title));
}
}
return retVal;
}
Фрагмент PopupwritePermission затем дает окно, где объясняется ситуация. Щелчок по кнопке ОК откроет системное меню Android, где можно предоставить разрешение:
private void openAndroidPermissionsMenu() {
Intent intent = new Intent(Settings.ACTION_MANAGE_WRITE_SETTINGS);
intent.setData(Uri.parse("package:" + getActivity().getPackageName()));
startActivity(intent);
}
Ответ 3
Предыдущие ответы великолепны, у меня есть только небольшое дополнение для получения результата для запроса разрешения.
public static void youDesirePermissionCode(Activity context){
boolean permission;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
permission = Settings.System.canWrite(context);
} else {
permission = ContextCompat.checkSelfPermission(context, Manifest.permission.WRITE_SETTINGS) == PackageManager.PERMISSION_GRANTED;
}
if (permission) {
//do your code
} else {
if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.M) {
Intent intent = new Intent(Settings.ACTION_MANAGE_WRITE_SETTINGS);
intent.setData(Uri.parse("package:" + context.getPackageName()));
context.startActivityForResult(intent, MainActivity.CODE_WRITE_SETTINGS_PERMISSION);
} else {
ActivityCompat.requestPermissions(context, new String[]{Manifest.permission.WRITE_SETTINGS}, MainActivity.CODE_WRITE_SETTINGS_PERMISSION);
}
}
}
А затем в Activity
:
@SuppressLint("NewApi")
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
if (requestCode == MainActivity.CODE_WRITE_SETTINGS_PERMISSION && Settings.System.canWrite(this)){
Log.d("TAG", "MainActivity.CODE_WRITE_SETTINGS_PERMISSION success");
//do your code
}
}
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
if (requestCode == MainActivity.CODE_WRITE_SETTINGS_PERMISSION && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
//do your code
}
}
Ответ 4
Это полный пример:
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
if (Settings.System.canWrite(context) {
// Do stuff here
}
else {
Intent intent = new Intent(android.provider.Settings.ACTION_MANAGE_WRITE_SETTINGS);
intent.setData(Uri.parse("package:" + getActivity().getPackageName()));
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(intent);
}
}
Ответ 5
Разрешение android.permission.WRITE_SETTINGS
теперь находится в группе signature|appop|pre23|preinstalled
, как android.permission.CHANGE_NETWORK_STATE
и android.permission.SYSTEM_ALERT_WINDOW
Это означает, что вы получаете его на sdk 22 и ниже. В новой версии вы должны быть оператором приложения.
Ответ 6
Упомяните ниже разрешение в AndroidManifest.xml
В разделе "Активность" используйте ниже, чтобы изменить настройки.
if(Settings.System.canWrite(this)){
// change setting here
}
else{
//Migrate to Setting write permission screen.
Intent intent = new Intent(Settings.ACTION_MANAGE_WRITE_SETTINGS);
intent.setData(Uri.parse("package:" + mContext.getPackageName()));
startActivity(intent);
}
Ответ 7
Начиная с Android Marshmellow, вы должны использовать разрешения времени выполнения, которые направлены на повышение безопасности, или использовать разрешения, когда здесь требуется документация
и для записи настроек документация здесь
В манифест добавить
<uses-permission android:name="android.permission.WRITE_SETTINGS" />
В вашем классе
private boolean checkSystemWritePermission() {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
if(Settings.System.canWrite(context))
return true;
else
openAndroidPermissionsMenu();
}
return false;
}
private void openAndroidPermissionsMenu() {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
Intent intent = new Intent(Settings.ACTION_MANAGE_WRITE_SETTINGS);
intent.setData(Uri.parse("package:" + context.getPackageName()));
context.startActivity(intent);
}
}
И используйте это так
try {
if (checkSystemWritePermission()) {
RingtoneManager.setActualDefaultRingtoneUri(context, RingtoneManager.TYPE_RINGTONE, newUri);
Toast.makeText(context, "Set as ringtoon successfully ", Toast.LENGTH_SHORT).show();
}else {
Toast.makeText(context, "Allow modify system settings ==> ON ", Toast.LENGTH_LONG).show();
}
} catch (Exception e) {
Log.i("ringtoon",e.toString());
Toast.makeText(context, "unable to set as Ringtoon ", Toast.LENGTH_SHORT).show();
}