Android 在 Android Phone 锁屏上显示 Activity
Android Showing Activity on Android Phone LockScreen
当用户打开他的 phone
时,我正在尝试显示 activity
我做了这些
1.Foreground 带有广播接收器的服务在 LockScreen
上显示 activity
public class OverlayService extends Service {
NotificationManager manager;
@Override
public IBinder onBind(Intent intent) {
throw new UnsupportedOperationException("Not yet implemented");
}
@Override
public void onCreate() {
super.onCreate();
if (Build.VERSION.SDK_INT > Build.VERSION_CODES.O)
startMyOwnForeground();
else {
startForeground(1, new Notification());
Toast.makeText(this, " Service Is Running", Toast.LENGTH_LONG).show();
}
}
@RequiresApi(Build.VERSION_CODES.O)
private void startMyOwnForeground() {
String NOTIFICATION_CHANNEL_ID = ".....";
String channelName = "Foreground Service";
NotificationChannel chan = new NotificationChannel(NOTIFICATION_CHANNEL_ID, channelName, NotificationManager.IMPORTANCE_DEFAULT);
chan.setLightColor(Color.BLUE);
chan.setLockscreenVisibility(Notification.VISIBILITY_PRIVATE);
chan.setShowBadge(false);
manager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
assert manager != null;
manager.createNotificationChannel(chan);
NotificationCompat.Builder notificationBuilder = new NotificationCompat.Builder(this, NOTIFICATION_CHANNEL_ID);
Notification notification = notificationBuilder.setOngoing(true)
.setContentTitle("text_my")
.setContentText("my_text")
.setPriority(NotificationManager.IMPORTANCE_DEFAULT)
.setCategory(Notification.CATEGORY_EVENT)
.build();
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
startForeground(2, notification, ServiceInfo.FOREGROUND_SERVICE_TYPE_DATA_SYNC);
}
startForeground(2, notification);
Toast.makeText(this, " ...Service Is Running", Toast.LENGTH_LONG).show();
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
registerOverlayReceiver();
return START_STICKY;
}
@Override
public void onDestroy() {
super.onDestroy();
unregisterOverlayReceiver();
}
private void registerOverlayReceiver() {
IntentFilter filter = new IntentFilter();
filter.addAction(Intent.ACTION_SCREEN_ON);
filter.addAction(ACTION_DEBUG);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
filter.addAction(Intent.ACTION_USER_UNLOCKED);
}
registerReceiver(overlayReceiver, filter);
}
private void unregisterOverlayReceiver() {
unregisterReceiver(overlayReceiver);
}
private static final String ACTION_DEBUG = "....text";
private final BroadcastReceiver overlayReceiver = new BroadcastReceiver() {
@Override
public void onReceive(Context context, Intent intent) {
String action = intent.getAction();
assert action != null;
if (action.equals(Intent.ACTION_SCREEN_ON)) {
showOverlayActivity(context);
} else if (action.equals(ACTION_DEBUG)) {
showOverlayActivity(context);
}
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N && action.equals(Intent.ACTION_USER_UNLOCKED)) {
showOverlayActivity(context);
}
}
};
private void showOverlayActivity(Context context) {
Intent intent = new Intent(context, com.rad.mls.OverlayActivity.class);
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(intent);
}
}
- 在 Overlay 中Activity 在 LockScreen 上显示
final Window window = getWindow();
window.addFlags(WindowManager.LayoutParams.FLAG_SHOW_WHEN_LOCKED
| WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON | WindowManager.LayoutParams.FLAG_DISMISS_KEYGUARD);
- Android 清单中的权限
<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />
<uses-permission android:name="android.permission.FOREGROUND_SERVICE" />
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.VIBRATE" />
<uses-permission android:name="android.permission.WAKE_LOCK" />
<uses-permission android:name="android.permission.USE_FULL_SCREEN_INTENT" />
<uses-permission android:name="android.permission.DISABLE_KEYGUARD" />
- Activity 清单
<activity
android:name=".OverlayActivity"
android:excludeFromRecents="true"
android:screenOrientation="portrait"
android:showOnLockScreen="true"
android:theme="@style/Theme.Lockscreen" />
- 我启动前台服务的方式
Intent i = new Intent(this, OverlayService.class);
i.setAction("C.ACTION_START_SERVICE");
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) startForegroundService(i);
else
startService(i);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
Toast.makeText(this, "Give app permission to DISPLAY(DRAW) OVER TOP OF OTHER APPS", Toast.LENGTH_LONG).show();
Intent intent;
if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.M) {
intent = new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION);
} else {
intent = new Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
}
Uri uri = Uri.fromParts("package", getPackageName(), null);
intent.setData(uri);
startActivity(intent);
finish();
} else {
finish();
}
我也问过用户如果他们在 MIUI 上去权限并允许应用程序在锁屏上显示但仍然
MAIN PROBLEM : Although it works on most devices but still in many
devices activity didn't show over Pattern/Password lock also in many
devices the service get automatically killed after some time and it
stops showing activity
Is there a proper way you can show a layout with event listeners on lockscreen
Or what's the ways which is most near to perfect in acheiving it?
实际上在 8+ 上你需要一些其他代码并且不推荐使用的标志将不起作用
if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.O_MR1){
setShowWhenLocked(true)
setTurnScreenOn(true)
}
if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.O){
(getSystemService(Context.KEYGUARD_SERVICE) as? KeyguardManager)?.requesDismissKeyguard()
}
也不要忘记能够在 Android 10+ 中启动活动的后门(例如系统警报 window 权限)
MIUI 也添加了很多垃圾项目 android 我找到的最全面的代码在 QuickLyric 的源代码中
当用户打开他的 phone
时,我正在尝试显示 activity我做了这些
1.Foreground 带有广播接收器的服务在 LockScreen
上显示 activitypublic class OverlayService extends Service {
NotificationManager manager;
@Override
public IBinder onBind(Intent intent) {
throw new UnsupportedOperationException("Not yet implemented");
}
@Override
public void onCreate() {
super.onCreate();
if (Build.VERSION.SDK_INT > Build.VERSION_CODES.O)
startMyOwnForeground();
else {
startForeground(1, new Notification());
Toast.makeText(this, " Service Is Running", Toast.LENGTH_LONG).show();
}
}
@RequiresApi(Build.VERSION_CODES.O)
private void startMyOwnForeground() {
String NOTIFICATION_CHANNEL_ID = ".....";
String channelName = "Foreground Service";
NotificationChannel chan = new NotificationChannel(NOTIFICATION_CHANNEL_ID, channelName, NotificationManager.IMPORTANCE_DEFAULT);
chan.setLightColor(Color.BLUE);
chan.setLockscreenVisibility(Notification.VISIBILITY_PRIVATE);
chan.setShowBadge(false);
manager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
assert manager != null;
manager.createNotificationChannel(chan);
NotificationCompat.Builder notificationBuilder = new NotificationCompat.Builder(this, NOTIFICATION_CHANNEL_ID);
Notification notification = notificationBuilder.setOngoing(true)
.setContentTitle("text_my")
.setContentText("my_text")
.setPriority(NotificationManager.IMPORTANCE_DEFAULT)
.setCategory(Notification.CATEGORY_EVENT)
.build();
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
startForeground(2, notification, ServiceInfo.FOREGROUND_SERVICE_TYPE_DATA_SYNC);
}
startForeground(2, notification);
Toast.makeText(this, " ...Service Is Running", Toast.LENGTH_LONG).show();
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
registerOverlayReceiver();
return START_STICKY;
}
@Override
public void onDestroy() {
super.onDestroy();
unregisterOverlayReceiver();
}
private void registerOverlayReceiver() {
IntentFilter filter = new IntentFilter();
filter.addAction(Intent.ACTION_SCREEN_ON);
filter.addAction(ACTION_DEBUG);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
filter.addAction(Intent.ACTION_USER_UNLOCKED);
}
registerReceiver(overlayReceiver, filter);
}
private void unregisterOverlayReceiver() {
unregisterReceiver(overlayReceiver);
}
private static final String ACTION_DEBUG = "....text";
private final BroadcastReceiver overlayReceiver = new BroadcastReceiver() {
@Override
public void onReceive(Context context, Intent intent) {
String action = intent.getAction();
assert action != null;
if (action.equals(Intent.ACTION_SCREEN_ON)) {
showOverlayActivity(context);
} else if (action.equals(ACTION_DEBUG)) {
showOverlayActivity(context);
}
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N && action.equals(Intent.ACTION_USER_UNLOCKED)) {
showOverlayActivity(context);
}
}
};
private void showOverlayActivity(Context context) {
Intent intent = new Intent(context, com.rad.mls.OverlayActivity.class);
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(intent);
}
}
- 在 Overlay 中Activity 在 LockScreen 上显示
final Window window = getWindow();
window.addFlags(WindowManager.LayoutParams.FLAG_SHOW_WHEN_LOCKED
| WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON | WindowManager.LayoutParams.FLAG_DISMISS_KEYGUARD);
- Android 清单中的权限
<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />
<uses-permission android:name="android.permission.FOREGROUND_SERVICE" />
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.VIBRATE" />
<uses-permission android:name="android.permission.WAKE_LOCK" />
<uses-permission android:name="android.permission.USE_FULL_SCREEN_INTENT" />
<uses-permission android:name="android.permission.DISABLE_KEYGUARD" />
- Activity 清单
<activity
android:name=".OverlayActivity"
android:excludeFromRecents="true"
android:screenOrientation="portrait"
android:showOnLockScreen="true"
android:theme="@style/Theme.Lockscreen" />
- 我启动前台服务的方式
Intent i = new Intent(this, OverlayService.class);
i.setAction("C.ACTION_START_SERVICE");
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) startForegroundService(i);
else
startService(i);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
Toast.makeText(this, "Give app permission to DISPLAY(DRAW) OVER TOP OF OTHER APPS", Toast.LENGTH_LONG).show();
Intent intent;
if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.M) {
intent = new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION);
} else {
intent = new Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
}
Uri uri = Uri.fromParts("package", getPackageName(), null);
intent.setData(uri);
startActivity(intent);
finish();
} else {
finish();
}
我也问过用户如果他们在 MIUI 上去权限并允许应用程序在锁屏上显示但仍然
MAIN PROBLEM : Although it works on most devices but still in many devices activity didn't show over Pattern/Password lock also in many devices the service get automatically killed after some time and it stops showing activity
Is there a proper way you can show a layout with event listeners on lockscreen Or what's the ways which is most near to perfect in acheiving it?
实际上在 8+ 上你需要一些其他代码并且不推荐使用的标志将不起作用
if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.O_MR1){
setShowWhenLocked(true)
setTurnScreenOn(true)
}
if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.O){
(getSystemService(Context.KEYGUARD_SERVICE) as? KeyguardManager)?.requesDismissKeyguard()
}
也不要忘记能够在 Android 10+ 中启动活动的后门(例如系统警报 window 权限) MIUI 也添加了很多垃圾项目 android 我找到的最全面的代码在 QuickLyric 的源代码中