强制下线的应用场景很多, 比如我们好多账号都有抢登的显现, 会被挤下线.
详细实现:
其实实现强制下线功能的思路也比较简单, 只需要在界面弹出一个对话框, 让用户无法进行其他操作, 必须点击对话框中的确定按钮, 然后回到登录界面即可.
但是又有一个问题, 我们通知用户强制下线, 用户可能正处于任何一个界面, 难道需要在每个界面上都编写一个弹出对话框的逻辑? 当然不是! 我们可以借助本章中所学的广播知识来轻松实现这一功能.
public class ActivityConllector { public static List<Activity> activities = new ArrayList<>(); public static void addActivity(Activity activity){ activities.add(activity); } public static void removeActivity(Activity activity){ activities.remove(activity); } public static void finishAll(){ for (Activity activity : activities){ if(!activity.isFinishing()){ activity.finish(); } } }
创建BaseActivity类作为所有活动的父类, 代码如下:
public class BaseActivity extends AppCompatActivity{ @Override protected void onCreate(@Nullable Bundle savedInstanceState) { super.onCreate(savedInstanceState); ActivityConllector.addActivity(this); } @Override protected void onDestroy() { super.onDestroy(); ActivityConllector.removeActivity(this); } }
首先编写布局文件activity_login文件:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical"> <LinearLayout android:layout_width="match_parent" android:layout_height="60dp" android:orientation="horizontal"> <TextView android:layout_width="90dp" android:layout_height="wrap_content" android:text="User Name" android:textSize="18sp" android:layout_gravity="center_vertical"/> <EditText android:id="@+id/account" android:layout_width="0dp" android:layout_height="wrap_content" android:layout_weight="1" android:layout_gravity="center_vertical" /> </LinearLayout> <LinearLayout android:layout_width="match_parent" android:layout_height="60dp" android:orientation="horizontal"> <TextView android:layout_width="90dp" android:layout_height="wrap_content" android:text="Password" android:textSize="18sp" android:layout_gravity="center_vertical"/> <EditText android:id="@+id/password" android:layout_width="0dp" android:layout_height="wrap_content" android:layout_weight="1" android:layout_gravity="center_vertical" android:inputType="textPassword"/> </LinearLayout> <Button android:id="@+id/login" android:layout_width="match_parent" android:layout_height="60dp" android:text="Login"/> </LinearLayout>
LoginActivity.java中代码:
public class LoginActivity extends BaseActivity implements View.OnClickListener{ private EditText userName; private EditText passWord; private Button login; @Override protected void onCreate(@Nullable Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_login); userName = (EditText)findViewById(R.id.account); passWord = (EditText)findViewById(R.id.password); login = (Button) findViewById(R.id.login); login.setOnClickListener(this); } @Override public void onClick(View v) { switch (v.getId()){ case R.id.login: String username = userName.getText().toString(); String password = passWord.getText().toString(); if(username.equals("admin") && password.equals("123")){ Intent intent = new Intent(LoginActivity.this, MainActivity.class); startActivity(intent); finish(); } else { Toast.makeText(this, "用户名或者密码错误!", Toast.LENGTH_SHORT).show(); } } } }
我们这时可以理解成, 登录界面之后, 进入的activity_main.xml就是主界面. 主界面没有其他功能, 只有一个用于显示的TextView和一个用作下线功能的按钮
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:id="@+id/activity_main" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical"> <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="这是主页面" android:layout_gravity="center_horizontal"/> <Button android:id="@+id/force_offline" android:layout_width="match_parent" android:layout_height="wrap_content" android:text="强制下线"/> </LinearLayout>
MainActivity.java中代码:
public class MainActivity extends BaseActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); Button forceOffline = (Button) findViewById(R.id.force_offline); forceOffline.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { Intent intent = new Intent("com.junzaivip.broadcastbestpractice.FORCE_OFFLINE"); sendBroadcast(intent); } }); } }
我们按钮的点击事件里面发送了一条广播, 广播的值为com.junzaivip.broadcastbestpractice.FORCE_OFFLINE, 这条广播就是用于通知程序强制用户下线的.
也就是说强制用户下线的逻辑并不是写在MainActivity里的, 而是应该写在接收这条广播的接收器里面. 这样的好处是, 强制下线的功能就不会依附于任何的界面, 不管是在程序的任何地方, 只需要发出一条这样的广播,就可以完成强制下线的操作了.
那么, 需要创建一个广播接收器, 用于接收这条强制下线的广播, 唯一的问题就是, 应该在哪里创建呢? 由于广播接收器里面需要弹出一个对话框来阻塞用户的正常操作. 但如果创建的是一个静态注册的广播接收器, 是没有办法在onReceive()方法里弹出对话框这样的UI控件的, 而我们显然不能也不可能在每个活动中注册一个动态的广播接收器.
那么到底应该怎么办呢? 其实很明显, 只需要在BaseActivity中动态注册一个广播接收器就可以了. 因为所有的活动都继承自BaseActivity的.
BaseActivity中的代码:
public class BaseActivity extends AppCompatActivity{ @Override protected void onCreate(@Nullable Bundle savedInstanceState) { super.onCreate(savedInstanceState); ActivityConllector.addActivity(this); } @Override protected void onDestroy() { super.onDestroy(); ActivityConllector.removeActivity(this); } class ForceOffLineReceiver extends BroadcastReceiver{ @Override public void onReceive(final Context context, final Intent intent) { AlertDialog.Builder builder = new AlertDialog.Builder(context); builder.setTitle("warning"); builder.setMessage("您被强制下线, 请您重新登录!"); builder.setCancelable(false); // 将对话框设置为不可取消 // 给按钮添加注册监听 builder.setPositiveButton("OK", new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { // 点击按钮所调用的方法 ActivityConllector.finishAll();//销毁所有的活动 Intent intent1 = new Intent(context, LoginActivity.class); context.startActivity(intent); } }); builder.show(); } } }
我们需要设置主活动为LoginActivity, 而不再是MainActivity, 模拟访问一个程序首先在登录页面.
修改AndroidManifest.xml文件:
<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"> </activity> <activity android:name=".LoginActivity"> <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> </activity> </application>
运行效果如下:
源码: BroadcastBestPractice_jb51.rar
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。