文章目录
- 功能简介
- 关闭所有活动
- 登陆界面
- 发送强制下线的广播
- 广播接收器
- AndroidManifest.xml
- 运行结果
功能简介
强制下线功能只需要弹出一个对话框,让用户只能点击确定按钮,回到登录界面。
如果在每一个活动中添加一个对话框的话太过繁琐,用广播实现是一个好办法。下面逐步进行实现:
关闭所有活动
强制下线功能需要先关闭所有的活动,我们只需要用 AcitivityCollector
类来管理所有的活动,然后用 BaseActivity
类作为所有活动的父类,如此一来即可通过 ActivityCollector.finishAll()
退出所有程序。
AcitivityCollector.java
和 BaseActivity.java 我们在该篇博客中实现过(生产环境中关于 Activity 的小技巧),这里直接拿过来用:
public class ActivityCollector {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();}}}
}
public class BaseActivity extends AppCompatActivity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);ActivityCollector.addActivity(this);}@Overrideprotected void onDestroy() {super.onDestroy();ActivityCollector.removeActivity(this);}
}
登陆界面
首先我们创建一个 LoginActivity
作为登录界面。
布局文件 login_layout.xml
:
<LinearLayoutxmlns:android="http://schemas.android.com/apk/res/android"android:orientation="vertical"android:layout_width="match_parent"android:layout_height="match_parent"><LinearLayoutandroid:orientation="horizontal"android:layout_width="match_parent"android:layout_height="60dp"><TextViewandroid:layout_width="90dp"android:layout_height="wrap_content"android:layout_gravity="center_vertical"android:textSize="18sp"android:text="Account:"/><EditTextandroid:id="@+id/account"android:layout_width="0dp"android:layout_height="wrap_content"android:layout_weight="1"android:hint="Please input a account"android:layout_gravity="center_vertical"/></LinearLayout><LinearLayoutandroid:orientation="horizontal"android:layout_width="match_parent"android:layout_height="60dp"><TextViewandroid:layout_width="90dp"android:layout_height="wrap_content"android:layout_gravity="center_vertical"android:textSize="18sp"android:text="Password:"/><EditTextandroid:id="@+id/password"android:layout_width="0dp"android:layout_height="wrap_content"android:layout_weight="1"android:layout_gravity="center_vertical"android:hint="Please input a password"android:inputType="textPassword" /></LinearLayout><Buttonandroid:id="@+id/button_login"android:layout_width="match_parent"android:layout_height="60dp"android:text="login"/></LinearLayout>
- android:inputType :让输入的内容变成我们熟知的小圆点形式(见下图
运行结果:
活动文件 LoginActivity.java
:
// 继承自自定义的父类BaseActivity
public class LoginActivity extends BaseActivity {private EditText accountEdit;private EditText passwordEdit;private Button login;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.login_layout);accountEdit = findViewById(R.id.account);passwordEdit = findViewById(R.id.password);login = findViewById(R.id.button_login);login.setOnClickListener((View v)->{String account = accountEdit.getText().toString();String password = passwordEdit.getText().toString();if(account.equals("cmy") && password.equals("201314")){Intent intent = new Intent(LoginActivity.this, ForceQuitActivity.class);startActivity(intent);finish();}else{if (account.isEmpty() || password.isEmpty()){Toast.makeText(this, "账号或密码不不能为空", Toast.LENGTH_LONG).show();}Toast.makeText(this, "账号或密码不正确", Toast.LENGTH_LONG).show();}});}
}
发送强制下线的广播
登陆界面账号密码正确后,跳转到 ForceQuitActivity.java 活动,在 onCreate
方法中初始化发送广播的按钮,并实现发送广播的过程:
public class ForceQuitActivity extends BaseActivity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.force_quit_layout);Button forceOffline = findViewById(R.id.force_offline);forceOffline.setOnClickListener((View v)->{// 将要发送的广播植入IntentIntent intent = new Intent("com.example.activitytest.Activity.FORCE_OFFLINE");sendBroadcast(intent);});}
}
布局文件 force_quit_layout.xml
:
<LinearLayoutxmlns:android="http://schemas.android.com/apk/res/android"android:orientation="vertical"android:layout_width="match_parent"android:layout_height="match_parent"><Buttonandroid:id="@+id/force_offline"android:layout_width="match_parent"android:layout_height="wrap_content"android:text="send force offline broadcast"/>
</LinearLayout>
发送广播的按钮:
广播接收器
接下来就应该创建一个广播接收器来接收强制下线广播了,问题是应该创建在哪里?
- 静态注册的广播接收器无法在
onReceive()
方法中弹出对话框这种UI
控件。 - 也不可能在每个活动中都注册一个动态的广播接收器。
因此可以在 BaseActivity 中动态注册一个广播接收器,供其他子类调用:
public class BaseActivity extends AppCompatActivity {private ForceOfflineReceiver receiver;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);ActivityCollector.addActivity(this);}// 活动位于栈顶且准备好与用户交互@Overrideprotected void onResume() {super.onResume();IntentFilter intentFilter = new IntentFilter();intentFilter.addAction("com.example.activitytest.Activity.FORCE_OFFLINE");receiver = new ForceOfflineReceiver();// 动态注册registerReceiver(receiver, intentFilter);}// 在系统准备去启动或者恢复另一个活动时调用,通常会释放一些占用 CPU 的资源,保存一些关键数据;@Overrideprotected void onPause() {super.onPause();// 动态注册要在结束时取消注册// 本程序中点击提示框中的OK按钮后会跳转到登陆界面,此时就是强制下线广播职责结束的时候if(receiver != null){unregisterReceiver(receiver);receiver = null;}}@Overrideprotected void onDestroy() {super.onDestroy();ActivityCollector.removeActivity(this);}class ForceOfflineReceiver extends BroadcastReceiver{@Overridepublic void onReceive(final Context context, Intent intent) {AlertDialog.Builder dialog = new AlertDialog.Builder(context);dialog.setTitle("Warning");dialog.setMessage("You are forced to be offline. Please try to login again.");dialog.setCancelable(false);dialog.setPositiveButton("OK", (DialogInterface dialogInterface, int which)->{ActivityCollector.finishAll(); // 销毁所有活动// 重新启动LoginActivityIntent intent1 = new Intent(context, LoginActivity.class);context.startActivity(intent1);});dialog.show();}}
}
以往注册和取消注册广播接收器都是在 onCreate()
和 onDestroy()
里进行的,而这里我们在 onResume()
和 onPause()
中注册/取消注册,这是因为我们需要保证只有处于栈顶的活动才能收到这条广播,当活动失去栈顶位置时应该取消广播接收器的注册。
AndroidManifest.xml
将登录界面 LoginActivity 设为程序主界面。
运行结果
启动程序,展示登陆界面:
账号密码正确,跳转到发送广播界面:
点击按钮发送广播,弹出强制下线对话框:
点击OK,返回登陆界面: