WRITE_SETTINGS 권한을 얻을 수 없습니다.


85

Android M Preview 3에서 대상 API가 23 인 경우 Manifest.permission.WRITE_SETTTINGS 권한을 얻을 수없는 것 같습니다.

 requestPermissions(new String[]{Manifest.permission.WRITE_SETTINGS},
              101);

요청 권한은 예상 한 대화 상자를 표시하지 않지만이 권한없이 다음 호출을하면

 RingtoneManager.setActualDefaultRingtoneUri(activity, RingtoneManager.TYPE_RINGTONE, ringUri);

내가 허가를받지 않았기 때문에 전화를하겠습니다.

여기서 어디로 가야할지 모르겠습니다. 23을위한 새로운 벨소리 API가 있습니까? 아니면이 권한 변경으로 인해 시스템이 아닌 앱이 벨소리를 변경할 수 없게 되었습니까?

답변:


134

WRITE_SETTINGS문서에 따라 를 사용하려면 다음 을 수행하십시오.

  1. <uses-permission>매니페스트 의 요소를 정상적으로 유지합니다.

  2. Settings.System.canWrite()설정을 작성할 자격이 있는지 전화문의 하십시오.

  3. 경우 canWrite()반환 false, 시작 활동을 사용자가 실제로 설정에 기록하는 응용 프로그램을 허용하는이 동의 할 수 있습니다.ACTION_MANAGE_WRITE_SETTINGS

즉, 설정에 쓰기는 이제 기기 관리 API, 접근성 서비스 등과 유사한 이중 옵트 인 (설치에 동의하고 설정에서 별도로 동의)입니다.

또한 아직 사용해 보지 않았습니다. 이것은 어제 Android 6.0 변경 사항에 대해 수행 한 연구에 기반합니다 .


감사합니다 마크! 매력처럼 작동했습니다. developer.android.com/preview/features/runtime-permissions.html 은 권한을 요청하는 여러 가지 새로운 방법이있는 경우 약간의 업데이트가 필요합니다. (게시하기 전에 이미 귀하의 블로그를 읽었지만 필요할 때 해당 정보를 보유하지 않았습니다)
Justin

이것은 실제로 작동했습니다. 그러나 최종 사용자에게 이것은 나쁜 접근 방식입니다. Google이이 동작을 변경하는 징후가 있습니까?
Fhl

2
@Fhl : 나는 dangerous그들이 안드로이드 6.0에서 다른 것들과 함께 했던 정규 런타임 권한 접근 방식 대신 왜이 경로를 택했는지 모르겠습니다 . 조만간 이것이 바뀌면 놀랄 것입니다.
CommonsWare

2
이 같은 의도에서 응용 프로그램을 지정할 수 있습니다 :intent.setData(Uri.parse("package:" + Context.getPackageName()));
Olegas Gončarovas

8
주목해야 할 또 다른 사항은 Android에 버그가있는 것 같습니다.이 버그는 사용자가 위에서 설명한 대화 상자에서 쓰기 권한을 부여한 위치에 이전에 설치된 앱을 유발하는 것으로 보입니다. 여기서 토글 스위치는 활성화 된 위치에 있지만 canWrite는 false를 반환합니다. canWrite () 메서드가 true를 반환하려면 사용자는 스위치를 껐다가 다시 켜야합니다. 개발 중이지만 고객이 볼 수있는 것은 아닐 것입니다.
Matt Wolfe

45

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;
}

Fragment PopupwritePermission은 상황을 설명하는 창을 제공합니다. 확인 버튼을 클릭하면 권한을 부여 할 수있는 Android 시스템 메뉴가 열립니다.

private void openAndroidPermissionsMenu() {
    Intent intent = new Intent(Settings.ACTION_MANAGE_WRITE_SETTINGS);
    intent.setData(Uri.parse("package:" + getActivity().getPackageName()));
    startActivity(intent);
}

40

이전 답변은 훌륭하며 허가 요청에 대한 결과를 얻기 위해 약간의 추가 사항이 있습니다.

 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
        }
    }

코드를 입력하면 권한이 부여 되어도 제대로 작동하지만 여전히 사용자 정의 벨소리가 할당되지 않고 여전히 Write_Setting 권한이 거부되었습니다.
Zia Ur Rahman

4
ActivityCompat.requestPermissions (context, new String [] {Manifest.permission.WRITE_SETTINGS}, ....); 쓸 수 없다. 특별 허가입니다. 이 권한은 문서에 명시된 의도로만 요청할 수 있습니다. 또한 Marshmello 이전에는 설치시 영구적으로 권한이 부여됩니다
Anonymous

2
@yshahak 당신의 변수는 무엇 MainActivity.CODE_WRITE_SETTINGS_PERMISSION입니까?
Bruno Bieri

@BrunoBieri 네 맞아요, 생략했습니다. 자세한 내용이 표시되도록 답변을 편집하겠습니다.
yshahak

그래서 무엇 MainActivity.CODE_WRITE_SETTINGS_PERMISSION입니까?
대괄호

12

다음은 완전한 예입니다.

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);
    }
}

intent.setData (Uri.parse ( "package :"+ getActivity (). getPackageName ()));
Ole K

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();
        }

5

사용 권한은 android.permission.WRITE_SETTINGS그룹에 지금 signature|appop|pre23|preinstalled과 같은 android.permission.CHANGE_NETWORK_STATEandroid.permission.SYSTEM_ALERT_WINDOW

이것은 sdk 22 이하에서 얻을 수 있음을 의미합니다. 최신 버전에서는 앱 운영자 여야합니다.


4

나는 벨로우즈처럼 ..

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
        boolean retVal = true;
        retVal = Settings.System.canWrite(this);
        if (retVal == false) {
            if (!Settings.System.canWrite(getApplicationContext())) {

                Intent intent = new Intent(Settings.ACTION_MANAGE_WRITE_SETTINGS, Uri.parse("package:" + getPackageName()));
                Toast.makeText(getApplicationContext(), "Please, allow system settings for automatic logout ", Toast.LENGTH_LONG).show();
                startActivityForResult(intent, 200);
            }
        }else {
            Toast.makeText(getApplicationContext(), "You are not allowed to wright ", Toast.LENGTH_LONG).show();
        }
    }

매니페스트 권한

<uses-permission  android:name="android.permission.WRITE_SETTINGS" tools:ignore="ProtectedPermissions" />

2

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);
}

이 권한을 사용하십시오. <uses-permission android : name = "android.permission.WRITE_SETTINGS"/>
Sourabh Tejraj

1

Kotlin Version in Simple Steps

이 단계를 따르세요:

1.manifest.xml 일반적으로 권한의 사용 요소를 추가합니다 .

<uses-permission
    android:name="android.permission.WRITE_SETTINGS"
    tools:ignore="ProtectedPermissions" />

2. 설정을 변경하려는 위치에서 쓰기 액세스를 확인합니다.

if (context.canWriteSettings) {
    // change the settings here ...
} else {
    startManageWriteSettingsPermission()
}

3. 권한을 요청하는 경우에도 다음 코드 줄을 추가합니다.

private fun startManageWriteSettingsPermission() {
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
        Intent(
            Settings.ACTION_MANAGE_WRITE_SETTINGS,
            Uri.parse("package:${context.packageName}")
        ).let {
            startActivityForResult(it, REQUEST_CODE_WRITE_SETTINGS_PERMISSION)
        }
    }
}

override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
    super.onActivityResult(requestCode, resultCode, data)

    when (requestCode) {
        REQUEST_CODE_WRITE_SETTINGS_PERMISSION -> {
            if (context.canWriteSettings) {
                // change the settings here ...
            } else {
                Toast.makeText(context, "Write settings permission is not granted!", Toast.LENGTH_SHORT).show()
            }
        }
    }
}

val Context.canWriteSettings: Boolean
    get() = Build.VERSION.SDK_INT < Build.VERSION_CODES.M || Settings.System.canWrite(this)

companion object {
    private const val REQUEST_CODE_WRITE_SETTINGS_PERMISSION = 5
}
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.