In Android 7 (API level 24) my app is not allowed to mute phone (set ringer mode to silent)

AndroidAndroid AudiomanagerAndroid 7.0-Nougat

Android Problem Overview


I have an app that mutes the phone by using AudioManager and setting ringer mode to silent with this code:

AudioManager audioManager = 
    (AudioManager) context.getSystemService(Context.AUDIO_SERVICE);
try {
	audioManager.setRingerMode(AudioManager.RINGER_MODE_SILENT)
} catch (Exception e) {
 	e.printStackTrace();
}

This works with Android 6, but now with Android 7, I get the following error:

System.err: java.lang.SecurityException: Not allowed to change Do Not Disturb state
System.err: at android.os.Parcel.readException(Parcel.java:1683)
System.err: at android.os.Parcel.readException(Parcel.java:1636)
System.err: at android.media.IAudioService$Stub$Proxy.setRingerModeExternal(IAudioService.java:962)
System.err: at android.media.AudioManager.setRingerMode(AudioManager.java:1022)
System.err: at controllers.SearchResultController.mutePhone(SearchResultController.java:185)

Are there any new permissions I need to ask for to make this work?

I looked through the Android permissions list, but couldn't find any that seemed relevant.

Android Solutions


Solution 1 - Android

Thanks for your answers, here is a little more detail.

To be able to set ringer mode to silent, you must ask permission to access notification policy (like @ucsunil said).

<uses-permission android:name="android.permission.ACCESS_NOTIFICATION_POLICY" />

Then, check if you have this permission. If you do not, open the settings for "Do Not Disturb access" for your app:

NotificationManager notificationManager = 
    (NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE);

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M
    && !notificationManager.isNotificationPolicyAccessGranted()) {

    Intent intent = new Intent(
                        android.provider.Settings
                        .ACTION_NOTIFICATION_POLICY_ACCESS_SETTINGS);

    startActivity(intent);
}

When you run startActivity(), Android opens the Do Not Disturb access settings for your app.

What confused me, was that the way to ask for this permission is completely different from other permissions.

Just for reference, here is the way to ask for permission READ_CONTACTS:

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M
    && ActivityCompat.checkSelfPermission(activity,
        Manifest.permission.READ_CONTACTS)
        == PackageManager.PERMISSION_DENIED) {

    ActivityCompat.requestPermissions(activity,
        new String[]{ Manifest.permission.READ_CONTACTS },
            REQUEST_CODE_READ_CONTACTS);
}

Solution 2 - Android

The access you are looking for is:

<uses-permission android:name="android.permission.ACCESS_NOTIFICATION_POLICY" />

You need to request the user to provide you with this specific permission.

Solution 3 - Android

ACCESS_NOTIFICATION_POLICY is considered a "normal" permission which means you do NOT have to request permission at runtime. You only have to add the permission to AndroidManifest.xml

    <uses-permission android:name="android.permission.ACCESS_NOTIFICATION_POLICY" />

Then, to mute all notifications you can set the Interruption filter to INTERRUPTION_FILTER_NONE

NotificationManager notificationManager = (NotificationManager)getSystemService(Context.NOTIFICATION_SERVICE);

        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
            previous_notification_interrupt_setting = notificationManager.getCurrentInterruptionFilter();
            notificationManager.setInterruptionFilter(NotificationManager.INTERRUPTION_FILTER_NONE);
        }

INTERRUPTION_FILTER_ALARMS - Alarms only
INTERRUPTION_FILTER_ALL - Normal filter
INTERRUPTION_FILTER_NONE - No interruptions (MUTE)
INTERRUPTION_FILTER_PRIORITY - Priority interruptions

I use

 int previous_notification_interrupt_setting = notificationManager.getCurrentInterruptionFilter();

to reset the filter later when onPause() or onStop()

Solution 4 - Android

I've resolved this issue by opening Setting menu and asking user to grant permission for it if api > 23.

Just call requestMutePermissions() from anywhere and it will work!

private void requestMutePermissions() {
	try {
		if (Build.VERSION.SDK_INT < 23) {
			AudioManager audioManager = (AudioManager)getApplicationContext().getSystemService(Context.AUDIO_SERVICE);
            audioManager.setRingerMode(AudioManager.RINGER_MODE_SILENT);
		} else if( Build.VERSION.SDK_INT >= 23 ) {
			this.requestForDoNotDisturbPermissionOrSetDoNotDisturbForApi23AndUp();
		}
	} catch ( SecurityException e ) {

	}
}

private void requestForDoNotDisturbPermissionOrSetDoNotDisturbForApi23AndUp() {

	NotificationManager notificationManager = (NotificationManager) getApplicationContext().getSystemService(Context.NOTIFICATION_SERVICE);
    // if user granted access else ask for permission
	if ( notificationManager.isNotificationPolicyAccessGranted()) {
		AudioManager audioManager = (AudioManager) getApplicationContext().getSystemService(Context.AUDIO_SERVICE);
        audioManager.setRingerMode(AudioManager.RINGER_MODE_SILENT);
	} else{
		// Open Setting screen to ask for permisssion
		Intent intent = new Intent(android.provider.Settings.ACTION_NOTIFICATION_POLICY_ACCESS_SETTINGS);
		startActivityForResult( intent, ON_DO_NOT_DISTURB_CALLBACK_CODE );
	}
}

@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
	// Check which request we're responding to
	if (requestCode == ON_DO_NOT_DISTURB_CALLBACK_CODE ) {
		this.requestForDoNotDisturbPermissionOrSetDoNotDisturbForApi23AndUp();
	}
}

Solution 5 - Android

Based on Android documentation, you have to grant access to Do Not Disturb.

> From N onward, ringer mode adjustments that would toggle Do Not Disturb are not allowed unless the app has been granted Do Not Disturb Access. See isNotificationPolicyAccessGranted().

Please check the link

Hope it helps!

Solution 6 - Android

Hy, If anyone needs this permission in Kotlin, here you are

var notificationManager = this.getSystemService(Context.NOTIFICATION_SERVICE) as NotificationManager
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M
        && !notificationManager.isNotificationPolicyAccessGranted
    ) {
        val intent = Intent(
            Settings.ACTION_NOTIFICATION_POLICY_ACCESS_SETTINGS
        )
        startActivity(intent)
    }

I really appreciate your advices that was life saving! :D

Solution 7 - Android

I faced this issue only in an Emulator. When i run the app on my mobile it never comes and it works as expected.

Solution 8 - Android

In an Android device you have to grant DoNotDisturb access to your app

Attributions

All content for this solution is sourced from the original question on Stackoverflow.

The content on this page is licensed under the Attribution-ShareAlike 4.0 International (CC BY-SA 4.0) license.

Content TypeOriginal AuthorOriginal Content on Stackoverflow
QuestionBjarte Aune OlsenView Question on Stackoverflow
Solution 1 - AndroidBjarte Aune OlsenView Answer on Stackoverflow
Solution 2 - AndroiducsunilView Answer on Stackoverflow
Solution 3 - AndroidWinston BrownView Answer on Stackoverflow
Solution 4 - AndroidWaqar UlHaqView Answer on Stackoverflow
Solution 5 - AndroidLucas SantosView Answer on Stackoverflow
Solution 6 - AndroidBotond NémethView Answer on Stackoverflow
Solution 7 - AndroidUzumakiLView Answer on Stackoverflow
Solution 8 - AndroidSHUBH TECHView Answer on Stackoverflow