Подтвердить что ты не робот

Не удается получить разрешение 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? Или изменилось ли это разрешение, чтобы не изменять какие-либо несистемные приложения для мелодии звонка?

4b9b3361

Ответ 1

Чтобы использовать WRITE_SETTINGS, на основе документов:

  1. Элемент <uses-permission> в манифесте как обычно

  2. Вызовите Settings.System.canWrite() чтобы узнать, имеете ли вы право на запись настроек.

  3. Если 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();
        }