web-dev-qa-db-ja.com

WRITE_SETTINGS権限を取得できません

Android M Preview 3に23のターゲットAPIがある場合、Manifest.permission.WRITE_SETTTINGS権限を取得できないようです。

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

許可を要求してもダイアログは表示されませんが、この許可なしで次の呼び出しを行うと、

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

私が許可を持っていないので、電話は例外です。

ここからどこに行くかわかりません。 23用の新しい着信音APIはありますか?または、この許可の変更により、システム以外のアプリが着信音を変更できなくなっただけですか?

73
Justin

ドキュメントに基づいてWRITE_SETTINGSを使用するには:

  1. 通常どおり、マニフェストに<uses-permission>要素を含めます。

  2. Settings.System.canWrite()を呼び出す 設定を書き出す資格があるかどうかを確認します。

  3. canWrite()falseを返す場合、 ACTION_MANAGE_WRITE_SETTINGSアクティビティ を起動して、ユーザーがそこに同意して、アプリが実際に設定に書き込むことを許可できるようにします。

つまり、設定への書き込みは、デバイス管理API、アクセシビリティサービスなどに似たダブルオプトイン(インストールに同意し、設定で個別に許可)になりました。

また、これらをまだ使用していないことに注意してください。これは、 昨日Android 6.0の変更について行った研究 に基づいています。

112
CommonsWare

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は、状況を説明するウィンドウを提供します。 [OK]ボタンをクリックすると、Androidシステムメニューが開き、アクセス許可を付与できます。

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

以前の回答は素晴らしいです。許可を求めるための結果を得るための追加はほんの少しです。

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

これは完全な例です。

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);
    }
}
10
Daniel Raouf

Android Marshmellowの時点で、セキュリティを強化するランタイムパーミッションを使用するか、ここで必要な場合にパーミッションを使用する必要があります documenatation

書き込み設定のドキュメントは here です

マニフェストに追加

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

許可Android.permission.WRITE_SETTINGSは、signature|appop|pre23|preinstalledAndroid.permission.CHANGE_NETWORK_STATEのようなグループAndroid.permission.SYSTEM_ALERT_WINDOWになりました

これは、SDK 22以下で入手できることを意味します。新しいバージョンでは、アプリのオペレーターである必要があります。

4
passsy

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);
}
2
Sourabh Tejraj