一、效果图
![](https://img-blog.csdnimg.cn/d7ec6713f68c40988e6a59728acf28b8.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBARHVja2Rhbg==,size_14,color_FFFFFF,t_70,g_se,x_16)
![](https://img-blog.csdnimg.cn/14ae7eb10cd54daa9bbcbb1870ef7b9e.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBARHVja2Rhbg==,size_14,color_FFFFFF,t_70,g_se,x_16)
二、源代码
静态广播弹出的弹窗是由悬浮弹窗所实现的package com.study.forcelogoutdemo
import android.content.Intent
import androidx.appcompat.app.AppCompatActivity
import android.os.Bundle
import android.view.View
class MainActivity : BaseActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
}
/**
* 发送动态注册的广播
*/
fun sendDynamic(view: View) {
with(Intent()) {
action = FORCE_LOGOUT_ACTION
sendBroadcast(this)
}
}
/**
* 发送静态注册的广播
*/
fun sendStatic(view: View) {
with(Intent()) {
action = STATIC_FORCE_LOGOUT_ACTION
/**
* Android8.0之后要想使静态注册的广播接收者接收到广播,那么必须指明具体接收者。
* 如果不指明具体接收者,那么则表示改广播是隐式广播,静态注册的广播接收者将无法接收该广播。
* 不过目前还有部分系统的隐式广播仍可以被静态注册的广播接收者接收
*/
setPackage(packageName)
sendBroadcast(this)
}
}
}
package com.study.forcelogoutdemo
import android.app.AlertDialog
import android.content.BroadcastReceiver
import android.content.Context
import android.content.Intent
import android.content.Intent.FLAG_ACTIVITY_NEW_TASK
import android.net.Uri
import android.os.Build
import android.provider.Settings
import android.util.Log
import android.view.WindowManager
import android.widget.Toast
/**
* androidx.appcompat.app.AlertDialog在显示对话框时会报如下错误
* You need to use a Theme.AppCompat theme (or descendant) with this activity.
* 意思是:使用这个对话框必须是在使用Theme.AppCompat或其子类主题的activity才可以。
* 所以在静态注册的广播中是无法使用这个包名下面的AlertDialog类
*
* 不过可以使用 这个android.app.AlertDialog这个类提供的AlertDialog的组件
*
*/
class StaticForceReceiver : BroadcastReceiver() {
/**
* android.app.ReceiverRestrictedContext@8cd96ed
*/
override fun onReceive(context: Context, intent: Intent) {
Log.e(javaClass.simpleName, context.toString())
if (Build.VERSION.SDK_INT >= 23) {
if (Settings.canDrawOverlays(context)) {
//Android6.0以上
showDialogTip(context)
} else {
//去设置页面打开该权限
Intent(
Settings.ACTION_MANAGE_OVERLAY_PERMISSION,
Uri.parse("package:" + context.packageName)
).apply {
addFlags(FLAG_ACTIVITY_NEW_TASK)
context.startActivity(this)
}
Toast.makeText(context, "请去设置中打开应用悬浮窗的权限", Toast.LENGTH_SHORT).show()
}
} else {
//Android6.0以上
showDialogTip(context)
}
}
private fun showDialogTip(context: Context) {
val dialog = AlertDialog.Builder(context).run {
setMessage("静态注册的广播检测到您的登录状态异常,请重新登录")
setCancelable(false)
setPositiveButton("退出登录") { dialog, _ ->
dialog.dismiss()
ActivityCollector.clearAllActivity()
Intent(context, LoginActivity::class.java).run {
addFlags(FLAG_ACTIVITY_NEW_TASK)
context.startActivity(this)
}
}
create()
}
if (Build.VERSION.SDK_INT >= 26) {
dialog.window!!.setType(WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY)
} else { //兼容低版本
dialog.window!!.setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT)
}
dialog.show()
}
}
报错示意图:
![](https://img-blog.csdnimg.cn/ab35b03799b04043bc49f5e073b3a683.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBARHVja2Rhbg==,size_20,color_FFFFFF,t_70,g_se,x_16#pic_center)