首先,可以确定的一点是锁屏下弹出页面必须用Activity实现,用WindowManager和Dialog均不行,然后有个注意的地方就是某些手机默认是不允许应用在锁屏下弹出界面的,比如小米的MIUI和魅族的Flyme, 需要在设置中打开一项权限才可以:
并且这个权限不能像其他权限一样可以在代码中动态申请,只能用户手动选择允许。在我的手机上这个权限默认是禁止的,即便选择询问,有时也是不能弹出询问对话框的,必须选择允许才行。目测应该目前国内几大手机厂商出品的手机都会带这个功能。
因为在锁屏下可以弹出页面其实是一个危险的权限,需要得到用户的确认才行,否则你可以想象随便一个app就可以在锁屏下开启页面,那么它是不是可以抓取你的锁屏密码等用户隐私了。
下面是使用Activity在锁屏下弹出的方法步骤:
1. 设置要弹出的Activity页面
public class ScreenOnTestActivity extends Activity {
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_screen_on);
final Window win = getWindow();
//四个标志位分别是锁屏状态下显示,解锁,保持屏幕长亮,打开屏幕
//这样当Activity启动的时候,它会解锁并亮屏显示。
win.addFlags(WindowManager.LayoutParams.FLAG_SHOW_WHEN_LOCKED
| WindowManager.LayoutParams.FLAG_DISMISS_KEYGUARD
| WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON
| WindowManager.LayoutParams.FLAG_TURN_SCREEN_ON);
Log.e("AAA", "onCreate: ScreenOnTestActivity");
}
@Override
protected void onNewIntent(Intent intent) {
super.onNewIntent(intent);
//唤醒屏幕的代码, 再次亮起屏幕时,如果该Activity并未退出,会走onNewIntent(实际这个不加我的手机也能唤起)
PowerManager pm = (PowerManager) this.getSystemService(Context.POWER_SERVICE);
if (!pm.isScreenOn()) {
PowerManager.WakeLock wl = pm.newWakeLock(PowerManager.ACQUIRE_CAUSES_WAKEUP |
PowerManager.SCREEN_BRIGHT_WAKE_LOCK, "bright");
wl.acquire();
wl.release();
}
}
}
AndroidManifest.xml文件:
<activity android:name=".ScreenOnTestActivity"
android:launchMode="singleInstance"
android:excludeFromRecents="true"
android:taskAffinity=""
android:theme="@android:style/Theme.DeviceDefault.Light"/>
权限:
<uses-permission android:name="android.permission.DISABLE_KEYGUARD"/>
<uses-permission android:name="android.permission.WAKE_LOCK"/>
2.监听系统锁屏或者亮屏:
系统锁屏或者屏幕亮起,解锁的时候,系统会发送相应的广播,通过监听广播来判断
private void startScreenBroadcastReceiver() {
IntentFilter filter = new IntentFilter();
filter.addAction(Intent.ACTION_SCREEN_ON);
filter.addAction(Intent.ACTION_SCREEN_OFF);
filter.addAction(Intent.ACTION_USER_PRESENT);
registerReceiver(mScreenReceiver, filter);
}
private ScreenBroadcastReceiver mScreenReceiver = new ScreenBroadcastReceiver();
private class ScreenBroadcastReceiver extends BroadcastReceiver {
private String action = null;
@Override
public void onReceive(Context context, Intent intent) {
action = intent.getAction();
Log.e(TAG, "onReceive: " +action );
if (Intent.ACTION_SCREEN_ON.equals(action)) {
// 开屏
KeyguardManager km = (KeyguardManager) context.getSystemService(Context.KEYGUARD_SERVICE);
//判断是否为锁屏状态
if (km != null && km.inKeyguardRestrictedInputMode()) {
Intent alarmIntent = new Intent(context, ScreenOnTestActivity.class);
alarmIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(alarmIntent);
}
} else if (Intent.ACTION_SCREEN_OFF.equals(action)) {
// 锁屏
} else if (Intent.ACTION_USER_PRESENT.equals(action)) {
// 解锁
}
}
}
这个监听屏幕状态的广播接收器最好能放在常驻后台的Service服务中,但是常驻后台在最新的手机ROM上也是要设置手机权限的,从Android 7.0 8.0开始已经没有办法做到无感知的后台任务了,有两种方式,用户在手机上手动设置目标app允许后台运行的权限(类似于锁屏下弹出的权限设置),或者使用前台服务(需要绑定Notification)。
开头提到的权限选择,在我的手机上如果选择了“询问”,则亮屏时看不到界面,其实会在通知栏发一个通知告诉你:
这个提醒不太明显,用户可能感知不到,最好能在app内提示用户去手动选择。
上面文中都有提到通过WindowManager 悬浮窗的方式实现, 但经过尝试不可行,显示悬浮窗也需要在设置中开启应用的权限,悬浮窗可以在桌面显示,但是在锁屏下仍然无法看到悬浮窗,另外使用弹窗的方式也不可行,弹窗只能显示在应用内。