美文网首页
android使用广播实现强制下线功能

android使用广播实现强制下线功能

作者: 别看后面有人 | 来源:发表于2021-06-29 11:45 被阅读0次

    强制下线是我们常见的功能,比如QQ在别处登录了,就会将你强制挤下线。思路是在界面上添加一个对话框,让用户无法进行其他操作,而且对话框只有“确定”按钮,然后回到登录界面即可。强制下线的功能需要关闭所有的activity,我们可以用之前创建的基类BaseActivity和ActivityColletor来协助实现该功能
    一、创建LoginActivity作为登录界面,布局文件如下:

    <?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_height="match_parent">
    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="60dp">
        <TextView
            android:layout_width="90dp"
            android:layout_height="match_parent"
            android:gravity="center"
            android:textSize="18sp"
            android:text="用户名"/>
        <EditText
            android:id="@+id/accountEdit"
            android:layout_width="0dp"
            android:layout_height="match_parent"
            android:layout_weight="1"
            android:hint="请输入用户名"
            android:layout_gravity="center_horizontal"/>
    </LinearLayout>
        <LinearLayout
            android:layout_width="match_parent"
            android:layout_height="60dp">
            <TextView
                android:layout_width="90dp"
                android:layout_height="match_parent"
                android:gravity="center"
                android:textSize="18sp"
                android:text="密码"/>
            <EditText
                android:id="@+id/pwdEdit"
                android:layout_width="0dp"
                android:layout_height="match_parent"
                android:layout_weight="1"
                android:hint="请输入密码"
                android:layout_gravity="center_horizontal"/>
        </LinearLayout>
        <Button
            android:id="@+id/login"
            android:text="登录"
            android:layout_width="200dp"
            android:layout_height="60dp"
            android:layout_gravity="center_horizontal"/>
    </LinearLayout>
    

    LoginActivity中的代码如下:

    class LoginAcitivity:BaseActivity() {
        override fun onCreate(savedInstanceState: Bundle?) {
            super.onCreate(savedInstanceState)
    
            setContentView(R.layout.activity_login)
    
            login.setOnClickListener {
                val account=accountEdit.text.toString()
                val pwd=pwdEdit.text.toString()
    
                //如果账号为admin,密码为123456 就认为登录成功
                if (account=="admin"&&pwd=="123456"){
                    val initent=Intent(this,MainActivity::class.java)
                    startActivity(initent)
                    finish()
                }else{
                    Toast.makeText(this,"账号或者密码错误",Toast.LENGTH_LONG).show()
                }
            }
        }
    }
    

    二、创建MainActivity,该页面是登录成功之后的主界面
    xml代码如下:

    <LinearLayout
        xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent"
        android:layout_height="match_parent">
    <Button
        android:id="@+id/forceoffline"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="强制下线"/>
    </LinearLayout>
    

    MainActivity中的代码如下:

    class MainActivity:BaseActivity() {
        override fun onCreate(savedInstanceState: Bundle?) {
            super.onCreate(savedInstanceState)
            setContentView(R.layout.activity_main)
             forceoffline.setOnClickListener {
                 val intent=Intent("com.app.activitytest.broadcast.FORCE_OFFLINE")
                 sendBroadcast(intent)
             }
        }
    }
    

    这里我们广播的值”com.app.activitytest.broadcast.FORCE_OFFLINE“,这条广播就是通知用户强制下线的,强制用户下线的逻辑并不在这里,而是应该在接收广播的BroadcastReceiver里面,这样强制下线的功能就不会依附在程序的任何界面了,只有发出这一条广播,就可以完成用户强制下线了。
    我们创建的BroadcastReceiver需要弹出一个对话框来阻塞用户的正常操作,但是创建一个静态注册的BroadcastReceiver,是没有办法在onReceive方法里弹出对话框这样的UI控件的,而显然我们也不能在每一个activity中都注册一个动态的BroadcastReceiver。那么我们需要怎么办呢?其实只需要在BaseActivity中动态注册一个BroadcastReceiver就可以了,因为所有的Activity都继承自BaseActivity
    修改BaseActivity代码:

    open class BaseActivity :AppCompatActivity() {
        val TAG="BaseActivity"
        lateinit var receiver:ForceOfflineReceiver
    
        /**
         * 在onCreat()方法中调用ActivityCollector的addActivity()方法,
         * 表明当前正在创建的activity添加到集合里,
         */
        override fun onCreate(savedInstanceState: Bundle?) {
            super.onCreate(savedInstanceState)
            Log.d(TAG, "onCreate: "+javaClass.simpleName)
            ActivityCollector.addAcivity(this)
        }
    
        override fun onResume() {
            super.onResume()
    
            val intentFilter=IntentFilter()
            intentFilter.addAction("com.app.activitytest.broadcast.FORCE_OFFLINE")
            receiver=ForceOfflineReceiver()
            registerReceiver(receiver,intentFilter)
        }
    
        override fun onPause() {
            super.onPause()
            unregisterReceiver(receiver)
        }
        /**
         * 在onDestroy()方法中使用ActivityCollector.removeActivity()方法,
         * 表明从集合里移除一个马上要销毁的activity
         */
        override fun onDestroy() {
            super.onDestroy()
            ActivityCollector.removeActivity(this)
    
        }
    
        inner class ForceOfflineReceiver:BroadcastReceiver(){
            override fun onReceive(context: Context?, intent: Intent?) {
                if (context != null) {
                    AlertDialog.Builder(context).apply {
                        setTitle("警告")
                        setMessage("你被强制退出,请尝试重写登录")
                        setCancelable(false)
                        setPositiveButton("确定"){_,_->
                            ActivityCollector.finishAll()
                            val intent=Intent(context,LoginAcitivity::class.java)
                            context.startActivity(intent)
    
                        }
                        show()
                    }
                }
            }
    
        }
    }
    

    我们来看一下forceOfflineReceiver中的代码,这次代码中的使用了AlertDialog.Builder构建的对话框,注意这里一定要设置setCancelable()方法为不可取消,否则用户按一下Back键就可以关闭对话框继续使用程序了。然后使用setPositiveButton()方法来给对话框注册确定按钮,当用户点击了确定之后,就调用了ActivityCollector的finishAll()方法,销毁了所以的activity,并且重新启动LoginActivity。
    forceOfflineReceiver这个BroadcastReceiver是怎么注册的,这里重写了onResumne()方法和onPause()方法,然后分别在这俩个方法中注册和取消。原因是我们始终需要处于栈顶的Activity才能接收到这条强制下线的广播,非栈顶的activity没有必要接受这条广播。当一个activity失去栈顶的位置的时候就会自动取消BroadcastReceiver的注册

    三、mainfest文件中的注册:

        <activity
                android:name=".LoginAcitivity"
                android:label="First app">
                <intent-filter>
                    <action android:name="android.intent.action.MAIN" />
    
                    <category android:name="android.intent.category.LAUNCHER" />
                </intent-filter>
            </activity>
    
            <activity android:name=".MainActivity"/>
    

    相关文章

      网友评论

          本文标题:android使用广播实现强制下线功能

          本文链接:https://www.haomeiwen.com/subject/mmbhultx.html