How to hook into the Power button in Android?
Asked Answered
P

9

35

On an Android device, where the only buttons are the volume buttons and a power button, I want to make the app react to presses on the power button (long and short). How is this done?

Petrie answered 13/9, 2010 at 17:58 Comment(1)
Here is the fully working answer to this questionSimony
P
8

Solution:

@Override
public boolean dispatchKeyEvent(KeyEvent event) {
    if (event.getKeyCode() == KeyEvent.KEYCODE_POWER) {
        Intent i = new Intent(this, ActivitySetupMenu.class);
        startActivity(i);
        return true;
    }

    return super.dispatchKeyEvent(event);
}
Petrie answered 15/9, 2010 at 7:29 Comment(2)
Yes, on single press this is not invoked. when you long press this reacts.Abstractionism
Would this work when the phone is off (er...screen locked)? I want a long-press of the power button to launch the voice command input so i don't have to take two seconds to unlock the phone, then invoke the voice command.Rodent
G
43

The existing answers don't completely answer the question and leave out enough details that they won't work without more investigation. I'll share what I've learned solving this.

First you need to add the following permission to your manifest file:

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

To handle short and long presses add the following overrides to your activity class:

@Override
public boolean onKeyDown(int keyCode, KeyEvent event) {
    if (keyCode == KeyEvent.KEYCODE_POWER) {
        // Do something here...
        event.startTracking(); // Needed to track long presses
        return true;
    }
    return super.onKeyDown(keyCode, event);
}

@Override
public boolean onKeyLongPress(int keyCode, KeyEvent event) {
    if (keyCode == KeyEvent.KEYCODE_POWER) {
        // Do something here...
        return true;
    }
    return super.onKeyLongPress(keyCode, event);
}

Note: It is worth noting that onKeyDown() will fire multiple times before onKeyLongPress does so you may want to trigger on onKeyUp() instead or other logic to prevent acting upon a series of onKeyDown() calls when the user is really holding it down.

I think this next part is for Cyanogenmod only. If the PREVENT_POWER_KEY constant is undefined then you should not need it.

To start intercepting the power key you need to set the following flag from your activity:

getWindow().addFlags(WindowManager.LayoutParams.PREVENT_POWER_KEY);

To stop intercepting the power key (allowing standard functionality):

getWindow().clearFlags(WindowManager.LayoutParams.PREVENT_POWER_KEY);

You can switch back and forth between the two modes repeatedly in your program if you wish.

Glucose answered 28/4, 2012 at 15:57 Comment(10)
does this work with all API versions? I tries APIs 8 and 15 and there is no PREVENT_POWER_KEY param within LayoutParams. Could you maybe here value of this param?Bacitracin
The value for WindowManager.LayoutParams.PREVENT_POWER_KEY is 0x80000000 But what it looks like is that this is either a hidden value or specific only to Cyanogenmod (which would be disturbing if true). The project I did this for was API 10 but I had modified the sdk to expose hidden APIs and I was running it on Cyanogenmod 7.Glucose
onKeyDown only works when a long press event occurs the actually onKeyLongPress never gets fired by the Power button, at least on a Motorola Droid X2.Conveyancer
@JPM, @JT.: Exactly the same problem here (Galaxy Note, ICS): onKeyDown only works when a long press event occurs, and the actually onKeyLongPress never gets fired. Moreover, Power button is not prevented by the uses-permission in the manifest.Chaunce
Its not working on Samsung S(Api lable 2.2). but working on ICS. How to solve this problem?Allaround
NOT WORKING an Samsung Galaxy Tab 10 (ICS API 15)Wrongful
I can't see permission you have mentioned above, its not listed here developer.android.com/reference/android/…Attorney
The PREVENT_POWER_KEY was required for cyanogenmod only. Apparently stock Android has removed the ability to capture the power key sometime in the last couple of years. This approach may suit the needs of anybody who needs to do something when the power key is pressed: #7682516Glucose
@Glucose I am on CyanogenMod 12 and I could not get it to work. Also I can detect other keys like volume up/down, back, menu buttons etc (saw in Log) but I could NOT detect power button using the onKeyDown. The screen just turn off.Gyrate
how can I achieve the same using Service as a base class ?Kelliekellina
P
8

Solution:

@Override
public boolean dispatchKeyEvent(KeyEvent event) {
    if (event.getKeyCode() == KeyEvent.KEYCODE_POWER) {
        Intent i = new Intent(this, ActivitySetupMenu.class);
        startActivity(i);
        return true;
    }

    return super.dispatchKeyEvent(event);
}
Petrie answered 15/9, 2010 at 7:29 Comment(2)
Yes, on single press this is not invoked. when you long press this reacts.Abstractionism
Would this work when the phone is off (er...screen locked)? I want a long-press of the power button to launch the voice command input so i don't have to take two seconds to unlock the phone, then invoke the voice command.Rodent
H
5

On you activity add:

public boolean onKeyDown(int keyCode, KeyEvent event) {
    if (event.getKeyCode() == KeyEvent.KEYCODE_POWER) {
        // do what you want with the power button
        return true;
    }
    return super.onKeyDown(keyCode, event);
}

Though... this kind of keys are somehow special... not sure if it can give problems to you.

Hohenzollern answered 13/9, 2010 at 18:8 Comment(0)
W
4

As mentioned here https://mcmap.net/q/82477/-how-can-i-catch-the-power-button-long-press

@Override
public void onWindowFocusChanged(boolean hasFocus) {
    super.onWindowFocusChanged(hasFocus);
    if(!hasFocus) {
       Intent closeDialog = new Intent(Intent.ACTION_CLOSE_SYSTEM_DIALOGS);
        sendBroadcast(closeDialog);
    }
}
Williamson answered 5/4, 2013 at 7:54 Comment(1)
There is a problem with this solution, you will need to override this method for activities, menu, dialogs, fragments ... if you only do it in your activity, the user will be able to access the power menu when there is a dialog showing.Vito
L
4

Sharing a method to listen for Power button long press. Works with API 23+ permissions:

  1. Asking for system permission to draw overlay (This is not a normal or vulnerable permission). This is not a user permission, so You should really know, what you are doing, by asking for it.

    public class MainActivity extends AppCompatActivity {
    
        public final static int REQUEST_CODE = 10101;
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_main);
            if (checkDrawOverlayPermission()) {
                startService(new Intent(this, PowerButtonService.class));
            }
        }
    
        public boolean checkDrawOverlayPermission() {
            if (Build.VERSION.SDK_INT < Build.VERSION_CODES.M) {
                return true;
            }
            if (!Settings.canDrawOverlays(this)) {
                /** if not construct intent to request permission */
                Intent intent = new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION,
                    Uri.parse("package:" + getPackageName()));
            /** request permission via start activity for result */
                startActivityForResult(intent, REQUEST_CODE);
                return false;
            } else {
                return true;
            }
        }
    
        @Override
        @TargetApi(Build.VERSION_CODES.M)
        protected void onActivityResult(int requestCode, int resultCode, Intent data) {
            if (requestCode == REQUEST_CODE) {
                if (Settings.canDrawOverlays(this)) {
                    startService(new Intent(this, PowerButtonService.class));
                }
            }
        }
    }
    
  2. Starting a service and adds a special view to WindowManager

  3. Waiting for an action inside View's onCloseSystemDialogs method.

    public class PowerButtonService extends Service {
    
        public PowerButtonService() {
    
        }
    
        @Override
        public void onCreate() {
            super.onCreate();
            LinearLayout mLinear = new LinearLayout(getApplicationContext()) {
    
                //home or recent button
                public void onCloseSystemDialogs(String reason) {
                    if ("globalactions".equals(reason)) {
                        Log.i("Key", "Long press on power button");
                    } else if ("homekey".equals(reason)) {
                        //home key pressed
                    } else if ("recentapss".equals(reason)) {
                        // recent apps button clicked
                    }
                }
    
                @Override
                public boolean dispatchKeyEvent(KeyEvent event) {
                    if (event.getKeyCode() == KeyEvent.KEYCODE_BACK
                        || event.getKeyCode() == KeyEvent.KEYCODE_VOLUME_UP
                        || event.getKeyCode() == KeyEvent.KEYCODE_VOLUME_DOWN
                        || event.getKeyCode() == KeyEvent.KEYCODE_CAMERA
                        || event.getKeyCode() == KeyEvent.KEYCODE_POWER) {
                        Log.i("Key", "keycode " + event.getKeyCode());
                    }
                    return super.dispatchKeyEvent(event);
                }
            };
    
            mLinear.setFocusable(true);
    
            View mView = LayoutInflater.from(this).inflate(R.layout.service_layout, mLinear);
            WindowManager wm = (WindowManager) getSystemService(WINDOW_SERVICE);
    
            //params
            WindowManager.LayoutParams params = new WindowManager.LayoutParams(
                100,
                100,
                WindowManager.LayoutParams.TYPE_SYSTEM_ALERT,
                WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL
                        | WindowManager.LayoutParams.FLAG_FULLSCREEN
                        | WindowManager.LayoutParams.FLAG_LAYOUT_IN_SCREEN
                        | WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON,
                PixelFormat.TRANSLUCENT);
            params.gravity = Gravity.LEFT | Gravity.CENTER_VERTICAL;
            wm.addView(mView, params);
        }
    
        @Override
        public IBinder onBind(Intent intent) {
            return null;
        }
    }
    

Manifest:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
          package="powerbuttonpress">

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

    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:supportsRtl="true"
        android:theme="@style/AppTheme">
        <activity android:name=".MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN"/>
                <category android:name="android.intent.category.LAUNCHER"/>
            </intent-filter>
        </activity>

        <service
            android:name=".PowerButtonService"
            android:enabled="true"
            android:exported="true">
        </service>

    </application>

</manifest>

service_layout:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical">

</LinearLayout>
Leucine answered 29/8, 2016 at 11:30 Comment(7)
Brilliant! Thanks. One problem here is that soft keyboard cant be opened :(Effieeffigy
@R. Zagórski Hi, but this does not print anythingPanek
If anyone is having issues with the soft keyboard not showing, you can add WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE to your windowmanager params to fix itKalle
Not sure, what is calling the onCloseSystemDialogs() method?Maximomaximum
this method is not working for android 9, does anyone try it on android 9.?Muscovado
This worked for me for a long time but seems to have been plugged in recent security patch of Android 9. "globalactions" is no longer reported although other reasons like "home" areColtish
Its working perfectly for oreo and older versions. Not working in Pie. Anyone got a solution for making this work in pie?Fishbowl
N
2

Use Broadcast receiver for power button (Screen On/Off)event

here is solution: create broadcast receiver class

public class CallBroadCastReciever extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {

        if (intent.getAction().equals(Intent.ACTION_SCREEN_OFF)) {
            //The screen off position
            Toast.makeText(context,"Screen is off",Toast.LENGTH_LONG).show();
        }
        else if (intent.getAction().equals(Intent.ACTION_SCREEN_ON)) {
            //The screen on position
     Toast.makeText(context,"Screen is on",Toast.LENGTH_LONG).show();
        }
    }}

then follow below steps: 1. Initialize receiver in activity

CallBroadCastReciever broadcastReceiver = new CallBroadCastReciever();

2.Declare receiver in manifest file in tag

 <receiver android:name="com.android.CallBroadCastReciever">
<intent-filter>
    <action android:name="android.intent.action.SCREEN_OFF"/>
    <action android:name="android.intent.action.SCREEN_ON"/>
</intent-filter>
</receiver>

3.For send action and data from activity to receiver put below code in onCreate() or onStrat() method

IntentFilter intentFilter = new IntentFilter(); intentFilter.addAction(Intent.ACTION_SCREEN_OFF);

if (broadcastReceiver != null) {
    registerReceiver(broadcastReceiver, intentFilter);
}}

4.Don't forget to unregister receiver in onDestory() method

 protected void onDestroy() {
        super.onDestroy();
 if (broadcastReceiver != null) {
                unregisterReceiver(broadcastReceiver);
             } }
Nita answered 13/12, 2019 at 11:51 Comment(2)
but this won't work when the app is not running ?Underrate
I am using the same approach for flashlight light blinking to stop on power button but its not working.Haywood
O
1

You could overwrite the public boolean onKeyDown(int keyCode, KeyEvent event) and public boolean onKeyUp(int keyCode, KeyEvent event) functions in your Activity class and test if keyCode is equal to KeyEvent.KEYCODE_POWER.

I haven't tested this, but I would assume the system treats this like it does the Home key in that you cannot stop the system from receiving the key event, you can only observe that it occurs. To test this, try returning True from the above functions and see if this catches the key event.

Oliana answered 13/9, 2010 at 18:8 Comment(0)
B
0

For all android versions use this code.

I tried R. Zagórski's answer, but I am not able to run this code on Pie. However, I have updated their code in my answer.

PowerButtonService:

public class PowerButtonService extends Service {

    public PowerButtonService() {

    }

    @Override
    public void onCreate() {
        super.onCreate();
        LinearLayout mLinear = new LinearLayout(getApplicationContext()) {

            //home or recent button
            public void onCloseSystemDialogs(String reason) {
                if ("globalactions".equals(reason)) {
                    Log.i("Key", "Long press on power button");
                    Intent intent = new Intent(Intent.ACTION_CALL);
                    intent.setData(Uri.parse("tel:" + "0000000000"));
                    intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
                    intent.addFlags(Intent.FLAG_FROM_BACKGROUND);
                    startActivity(intent);
                } else if ("homekey".equals(reason)) {
                    //home key pressed
                } else if ("recentapss".equals(reason)) {
                    // recent apps button clicked
                }
            }

            @Override
            public boolean dispatchKeyEvent(KeyEvent event) {
                if (event.getKeyCode() == KeyEvent.KEYCODE_BACK
                        || event.getKeyCode() == KeyEvent.KEYCODE_VOLUME_UP
                        || event.getKeyCode() == KeyEvent.KEYCODE_VOLUME_DOWN
                        || event.getKeyCode() == KeyEvent.KEYCODE_CAMERA
                        || event.getKeyCode() == KeyEvent.KEYCODE_POWER) {
                    Log.i("Key", "keycode " + event.getKeyCode());
                }
                return super.dispatchKeyEvent(event);
            }
        };

        mLinear.setFocusable(true);

      **// here I done with EDIT** 

        int LAYOUT_FLAG;
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.P) {
            LAYOUT_FLAG = WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY;
        } else {
            LAYOUT_FLAG = WindowManager.LayoutParams.TYPE_PHONE;
        }
        View mView = LayoutInflater.from(this).inflate(R.layout.service_layout, mLinear);
        WindowManager wm = (WindowManager) getSystemService(WINDOW_SERVICE);

        //params
        WindowManager.LayoutParams params = new WindowManager.LayoutParams(
                WindowManager.LayoutParams.WRAP_CONTENT,
                WindowManager.LayoutParams.WRAP_CONTENT,
                LAYOUT_FLAG,
                WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE | WindowManager.LayoutParams.FLAG_LAYOUT_IN_SCREEN | WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE,
                PixelFormat.TRANSLUCENT);
        params.gravity = Gravity.LEFT | Gravity.CENTER_VERTICAL;
        wm.addView(mView, params);
    }

    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }
}

and service_layout:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical" android:layout_width="match_parent"
    android:layout_gravity="center"
    android:gravity="center"
    android:layout_height="match_parent">


</LinearLayout>
Break answered 22/4, 2019 at 12:33 Comment(2)
I wouldn't use this code, it's going to make a phone call in the background to some mystery number.Drusy
@JeffSchmitz You have to edit phone number here - intent.setData(Uri.parse("tel:" + "0000000000"));Break
P
-1

you have to use this:

BroadcastReceiver screenoff = new BroadcastReceiver() {

public static final String Screenoff = "android.intent.action.SCREEN_OFF";

@Override
public void onReceive(Context context, Intent intent) {
        if (!intent.getAction().equals(Screenoff)) return;
        //put code to handle power press here
        return;

}};
Pontine answered 31/7, 2011 at 6:18 Comment(2)
Does this work? I'm pretty sure the screen would still go off.Korwin
That intent filter didn't work, I'm receiving nothing on Android 6.0.1Estevez

© 2022 - 2024 — McMap. All rights reserved.