ไม่สามารถรับสิทธิ์ WRITE_SETTINGS


85

เมื่อฉันมี 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);

การโทรจะยกเว้นเพราะฉันไม่ได้รับอนุญาต

ฉันไม่แน่ใจว่าจะไปที่ไหนจากที่นี่ มี Ringtone API ใหม่สำหรับ 23 หรือไม่? หรือการอนุญาตนี้เปลี่ยนไปทำให้แอปที่ไม่ใช่ระบบเปลี่ยนเสียงเรียกเข้าไม่ได้?

คำตอบ:


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จำเป็นต้องมีการอัปเดตหากเราจะมีวิธีการใหม่ ๆ ในการขอสิทธิ์ (ฉันอ่านบล็อกของคุณแล้วก่อนโพสต์ แต่เห็นได้ชัดว่าไม่ได้เก็บข้อมูลส่วนนั้นไว้เมื่อฉันต้องการ)
จัสติน

สิ่งนี้ได้ผลแน่นอน แต่สำหรับผู้ใช้นี่เป็นแนวทางที่ไม่ดี มีสัญญาณว่า Google เปลี่ยนแปลงพฤติกรรมนี้หรือไม่
Fhl

2
@Fhl: ฉันไม่รู้ว่าทำไมพวกเขาถึงไปเส้นทางนี้แทนที่จะใช้dangerousวิธีการอนุญาตรันไทม์ปกติที่พวกเขาไปกับสิ่งอื่น ๆ ใน Android 6.0 ฉันจะแปลกใจถ้าสิ่งนี้เปลี่ยนแปลงเร็ว ๆ นี้
CommonsWare

2
คุณสามารถระบุแอปของคุณในเจตนาดังนี้intent.setData(Uri.parse("package:" + Context.getPackageName()));
Olegas Gončarovas

8
สิ่งที่ควรทราบอีกประการหนึ่งคือดูเหมือนว่าจะมีข้อบกพร่องใน Android ที่ทำให้แอปที่ติดตั้งไว้ก่อนหน้านี้ซึ่งผู้ใช้ให้สิทธิ์การเขียนในกล่องโต้ตอบที่อธิบายไว้ข้างต้นโดยที่สวิตช์สลับจะถูกวางไว้ในตำแหน่งที่เปิดใช้งาน แต่ canWrite จะส่งกลับเท็จ .. ในการรับ canWrite () วิธีการคืนค่าจริงผู้ใช้ต้องสลับการปิดและเปิดใหม่ ... ฉันเห็นสิ่งนี้อยู่ระหว่างการพัฒนา แต่หวังว่าจะไม่เป็นอย่างที่ลูกค้าเห็น
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 (บริบทสตริงใหม่ [] {Manifest.permission.WRITE_SETTINGS}, .... ); ไม่สามารถใช้งานได้ เป็นการอนุญาตพิเศษ เราสามารถขออนุญาตนี้ได้โดยมีเจตนาตามที่ระบุไว้ในเอกสารเท่านั้น ก่อนที่ Marshmello จะได้รับอนุญาตตลอดเวลาในการติดตั้ง
ไม่ระบุชื่อ

2
@yshahak ตัวแปรของคุณคือMainActivity.CODE_WRITE_SETTINGS_PERMISSIONอะไร?
Bruno Bieri

@BrunoBieri ใช่คุณใช่ฉันละเว้นสิ่งนั้น ฉันจะแก้ไขคำตอบของฉันเพื่อให้มันถูกต้อง
yshahak

มีอะไรMainActivity.CODE_WRITE_SETTINGS_PERMISSIONเหรอ?
วงเล็บ

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

ในฐานะของหุ่นยนต์ Marshmellow คุณจำเป็นต้องใช้สิทธิ์รันไทม์ซึ่งมีวัตถุประสงค์เพื่อการรักษาความปลอดภัยมากขึ้นหรือการใช้งานได้รับอนุญาตเมื่อต้องที่นี่เป็นdocumenatation

และสำหรับเอกสารการตั้งค่าการเขียนอยู่ที่นี่

ในรายการเพิ่ม

<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

ซึ่งหมายความว่าคุณได้รับใน 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);
}

โปรดใช้การอนุญาตนี้ <
use

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.