35
votes

I have an app that plays intermittent sounds while its activity is open, and the user is always expecting it to make these noises, but unfortunately it is constrained by the music stream volume. The options I have discovered are thus:

  1. Adjust the music stream's volume, possibly deafening the user if they happen to be playing music at the time.
  2. Call MediaPlayer.setVolume() which is ineffective if the music stream's volume is 0.
  3. Handle volume button presses myself which presents two issues: volume button presses adjust the ringer volume unless my audio is playing, and I have been as of yet unable to catch the onKey event for a volume button press using my OnKeyListener. If there were some way to force the adjustment to apply to the music stream while my activity has focus, or to just catch the button presses myself this would work.
  4. Play a silent looping piece of audio in the background to keep the music stream in context for volume adjustments.

I think the 3rd option is probably best since it cedes control of the volume to the user, but if there were some way to just override the system volume in a single mediaplayer instance that would work too.

5

5 Answers

75
votes

Got another suggestion via Google Groups that is the platform integrated solution I was looking for and works fine:

Please don't handle the volume keys yourself - it is almost impossible to guarantee that you won't break the behavior of the volume keys.

Call this API in your onCreate():

setVolumeControlStream(AudioManager.STREAM_MUSIC);

This tells the AudioManager that when your application has focus, the volume keys should adjust music volume.

2
votes

It is usefull of this API: setVolumeControlStream(AudioManager.STREAM_MUSIC);

Unless this I will spend more time to fix that volume bug.

1
votes

If you are using the 'OnKeyDown' method, you can get around this issue by doing the following:

public boolean onKeyDown(int keyCode, KeyEvent event) {    
if (keyCode == KeyEvent.KEYCODE_VOLUME_UP || keyCode == KeyEvent.KEYCODE_VOLUME_DOWN) {
                setVolumeControlStream(AudioManager.STREAM_MUSIC);}}
0
votes

This won't help you if you have override onKeyDown(int keycode,KeyEvent event) method of an activity.

0
votes
   var audioManager:AudioManager?=null
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        registerReceiver(mVolumeReceiver, IntentFilter("android.media.VOLUME_CHANGED_ACTION"))
        audioManager = getSystemService(Context.AUDIO_SERVICE) as AudioManager
    }

    private var mVolumeReceiver: BroadcastReceiver = object : BroadcastReceiver() {
        override fun onReceive(context: Context, intent: Intent) {

            if (intent.action == "android.media.VOLUME_CHANGED_ACTION") {
                var currentVolume = audioManager?.getStreamVolume(AudioManager.STREAM_MUSIC)
            }
        }
    }