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、ユーザー補助サービスなどと同様に、ダブルオプトイン(インストールに同意する、設定で個別に同意する)になりました。

また、これらをまだ使用していないことにも注意してください。これは、昨日Android6.0の変更で行っ調査に基づいています


ありがとうマーク!チャームのように働いた。権限をリクエストするための新しい方法が複数ある場合は、developer.android.com / preview / features /runtime-permissions.htmlを更新する必要があります。(私は投稿する前にあなたのブログをすでに読んでいましたが、明らかに必要なときにその情報を保持していませんでした)
Justin

これは確かに機能しました。しかし、エンドユーザーにとってこれは悪いアプローチです。Googleがこの行動を変える兆候はありますか?
fhl 2015

2
@Fhl:dangerousAndroid 6.0で他のものを使用した通常のランタイム権限アプローチではなく、なぜこのルートを使用したのかわかりません。これがすぐに変わると私は驚きます。
CommonsWare 2015

2
次のような意図でアプリを指定できます。intent.setData(Uri.parse("package:" + Context.getPackageName()));
OlegasGončarovas 2015年

8
もう1つの注意点は、Androidにバグがあり、以前にインストールされたアプリで、ユーザーが上記のダイアログで書き込み権限を付与した場合、トグルスイッチは有効な位置に配置されますが、canWriteはfalseを返します。 canWrite()メソッドをtrueに戻すには、ユーザーはスイッチをオフにしてからオンに戻す必要があります...これは開発中ですが、顧客に表示されるものではないことを願っています。
Matt Wolfe 2016年

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

次に、フラグメントPopupwritePermissionは、状況が説明されるウィンドウを提供します。[OK]ボタンをクリックすると、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許可が拒否された問題を持っています。
ジアウルラフマン

4
ActivityCompat.requestPermissions(context、new String [] {Manifest.permission.WRITE_SETTINGS}、....); 使用できません。特別な許可です。この許可を要求できるのは、ドキュメントに記載されている意図がある場合のみです。また、Marshmelloの前は、インストール時に永続的に許可が付与されます
匿名

2
@yshahakあなたの変数は何MainActivity.CODE_WRITE_SETTINGS_PERMISSIONですか?
BrunoBieri18年

@BrunoBieriはい、そうです、私はそれを省略しました。答えが冗長になるように編集します。
yshahak

それで、何MainActivity.CODE_WRITE_SETTINGS_PERMISSIONですか?
ブラケット2018

13

これは完全な例です:

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

8

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_STATEありますandroid.permission.SYSTEM_ALERT_WINDOW

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


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 2018年

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.