文章目錄
- 功能簡介
- 關閉所有活動
- 登陸界面
- 發送強制下線的廣播
- 廣播接收器
- 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,返回登陸界面: