Android学习笔记(十五)——实战:强制下线

  //此系列博文是《第一行Android代码》的学习笔记,如有错漏,欢迎指正!

  实现强制下线功能的思路也比较简单,只需要在界面上弹出一个对话框, 让用户无法进行任何其他操作, 必须要点击对话框中的确定按钮,然后回到登录界面即可。我们被通知需要强制下线时可能正处于任何一个界面, 但我们不需要在每个界面上都编写一个弹出对话框的逻辑,可以使用广播来轻松地实现这一功能。

一、创建一个 ActivityCollector 类用于管理所有的活动:

 1 public class ActivityCollector {
 2     public static List<Activity> activities = new ArrayList<Activity>();
 3     public static void addActivity(Activity activity) {
 4         activities.add(activity);
 5     }
 6     public static void removeActivity(Activity activity) {
 7         activities.remove(activity);
 8     }
 9     public static void finishAll() {
10         for (Activity activity : activities) {
11             if (!activity.isFinishing()) {
12                 activity.finish();
13             }
14         }
15     }
16 }
View Code

  ActivityCollector类可以方便我们管理返回栈里的活动。

二、创建 BaseActivity类作为所有活动的父类:

 1 public class BaseActivity extends Activity {
 2     @Override
 3     protected void onCreate(Bundle savedInstanceState) {
 4         super.onCreate(savedInstanceState);
 5         ActivityCollector.addActivity(this);
 6     }
 7     @Override
 8     protected void onDestroy() {
 9         super.onDestroy();
10         ActivityCollector.removeActivity(this);
11     }
12 }
View Code

  BaseActivity类与ActivityCollector类结合,每生成一个活动就加到ActivityCollector的List里,每销毁一个活动就在List将其减去。

三、创建登录界面的布局:
  使用TableView创建一个登陆界面:

 1 <TableLayout xmlns:android="http://schemas.android.com/apk/res/android"
 2     android:layout_width="match_parent"
 3     android:layout_height="match_parent"
 4     android:stretchColumns="1" >
 5     <TableRow>
 6         <TextView
 7             android:layout_height="wrap_content"
 8             android:text="Account:" />
 9         <EditText
10             android:id="@+id/account"
11             android:layout_height="wrap_content"
12             android:hint="Input your account" />
13     </TableRow>
14     <TableRow>
15         <TextView
16             android:layout_height="wrap_content"
17             android:text="Password:" />
18         <EditText
19             android:id="@+id/password"
20             android:layout_height="wrap_content"
21             android:inputType="textPassword" />
22     </TableRow>
23     <TableRow>
24         <Button
25             android:id="@+id/login"
26             android:layout_height="wrap_content"
27             android:layout_span="2"
28             android:text="Login" />
29     </TableRow>
30 </TableLayout>
View Code

四、创建 LoginActivity继承自 BaseActivity:

  登陆活动的代码:

 1 public class LoginActivity extends BaseActivity {
 2     private EditText accountEdit;
 3     private EditText passwordEdit;
 4     private Button login;
 5     @Override
 6     protected void onCreate(Bundle savedInstanceState) {
 7         super.onCreate(savedInstanceState);
 8         setContentView(R.layout.login);
 9         accountEdit = (EditText) findViewById(R.id.account);
10         passwordEdit = (EditText) findViewById(R.id.password);
11         login = (Button) findViewById(R.id.login);
12         login.setOnClickListener(new View.OnClickListener() {
13             @Override
14             public void onClick(View v) {
15                 String account = accountEdit.getText().toString();
16                 String password = passwordEdit.getText().toString();
17 // 如果账号是Vincent且密码是233333,就认为登录成功
18                 if (account.equals("Vincent") && password.equals("233333")) {
19                     Intent intent = new Intent(LoginActivity.this, MainActivity.class);
20                     startActivity(intent);
21                     finish();
22                 } else {
23                     Toast.makeText(LoginActivity.this, "account or password is invalid",
24                             Toast.LENGTH_SHORT).show();
25                 }
26             }
27         });
28     }
29 }
View Code

  这里我们模拟了一个非常简单的登录功能。首先使用 setContentView()方法将 login布局加载进来, 并调用 findViewById()方法分别获取到账号输入框、 密码输入框以及登录按钮的实例。然后在登录按钮的点击事件里面对输入的账号和密码进行判断,如果账号和密码正确,就认为登录成功并跳转到 MainActivity,否则就提示用户账号或密码错误。

五、在设置主活动布局并在主活动中添加代码:

  主活动相当于登陆之后的界面,我们在里面添加一个button,按下它之后就弹出提醒框:

 1 <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
 2     xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent"
 3     android:layout_height="match_parent" android:paddingLeft="@dimen/activity_horizontal_margin"
 4     android:paddingRight="@dimen/activity_horizontal_margin"
 5     android:paddingTop="@dimen/activity_vertical_margin"
 6     android:paddingBottom="@dimen/activity_vertical_margin" tools:context=".MainActivity">
 7 
 8     <Button
 9         android:id="@+id/force_offline"
10         android:layout_width="match_parent"
11         android:layout_height="wrap_content"
12         android:text="你好哇" />
13 </RelativeLayout>
View Code
 1 public class MainActivity extends BaseActivity {
 2 
 3     @Override
 4     protected void onCreate(Bundle savedInstanceState) {
 5         super.onCreate(savedInstanceState);
 6         setContentView(R.layout.activity_main);
 7         Button forceOffline = (Button) findViewById(R.id.force_offline);
 8         forceOffline.setOnClickListener(new View.OnClickListener() {
 9             @Override
10             public void onClick(View v) {
11                 Intent intent = new Intent("com.example.broadcastbestpractice.FORCE_OFFLINE");
12                 sendBroadcast(intent);
13             }
14         });
15     }
16 }
View Code

   我们在按钮的点击事件里面发送了一条广播,广播的值为 com.example.broadcastbestpractice.FORCE_OFFLINE,这条广播就是用于通知程序强制用户下线的。也就是说强制用户下线的逻辑并不是写在 MainActivity里的,而是应该写在接收这条广播的广播接收器里面,这样强制下线的功能就不会依附于任何的界面,不管是在程序的任何地方,只需要发出这样一条广播,就可以完成强制下线的操作了。

六、创建一个广播接收器:
  此广播接收器用于在任何收到“下线”的通知时强迫用户下线:

 1 public class ForceOfflineReceiver extends BroadcastReceiver {
 2     @Override
 3     public void onReceive(final Context context, Intent intent) {
 4         AlertDialog.Builder dialogBuilder = new AlertDialog.Builder(context);
 5         dialogBuilder.setTitle("Warning");
 6         dialogBuilder.setMessage("您已经掉线了");
 7         dialogBuilder.setCancelable(false);
 8         dialogBuilder.setPositiveButton("OK", new DialogInterface.OnClickListener() {
 9                     @Override
10                     public void onClick(DialogInterface dialog, int which) {
11                         ActivityCollector.finishAll(); // 销毁所有活动
12                         Intent intent = new Intent(context, LoginActivity.class);
13                         intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
14                         context.startActivity(intent); // 重新启动LoginActivity
15                     }
16                 });
17         AlertDialog alertDialog = dialogBuilder.create();
18 // 需要设置AlertDialog的类型,保证在广播接收器中可以正常弹出
19         alertDialog.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);
20         alertDialog.show();
21     }
22 }
View Code

   首先肯定是使用 AlertDialog.Builder 来构建一个对话框,注意这里一定要调用 setCancelable()方法将对话框设为不可取消, 否则用户按一下 Back键就可以关闭对话框继续使用程序了。然后使用 setPositiveButton()方法来给对话框注册确定按钮,当用户点击了确定按钮时,就调用 ActivityCollector 的 finishAll()方法来销毁掉所有活动,并重新启动 LoginActivity 这个活动。另外,由于我们是在广播接收器里启动活动的,因此一定要给Intent 加入 FLAG_ACTIVITY_NEW_TASK 这个标志。最后,还需要把对话框的类型设为TYPE_SYSTEM_ALERT,不然它将无法在广播接收器里弹出。

七、在manifest文件中注册活动与声明权限:

 1 <manifest xmlns:android="http://schemas.android.com/apk/res/android"
 2     package="com.mycompany.practice" >
 3 
 4     <uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />
 5     <application
 6         android:allowBackup="true"
 7         android:icon="@mipmap/ic_launcher"
 8         android:label="@string/app_name"
 9         android:theme="@style/AppTheme" >
10     <activity
11         android:name=".LoginActivity"
12         android:label="@string/app_name" >
13         <intent-filter>
14             <action android:name="android.intent.action.MAIN" />
15             <category android:name="android.intent.category.LAUNCHER" />
16         </intent-filter>
17     </activity>
18 
19     <activity android:name=".MainActivity" >
20     </activity>
21     <receiver android:name=".ForceOfflineReceiver" >
22         <intent-filter>
23             <action android:name="com.example.broadcastbestpractice.FORCE_OFFLINE" />
24         </intent-filter>
25     </receiver>
26     </application>
27 
28 </manifest>
View Code

   这里有几点内容需要注意, 首先由于我们在 ForceOfflineReceiver 里弹出了一个系统级别的对话框,因此必须要声明 android.permission.SYSTEM_ALERT_WINDOW 权限。然后对LoginActivity 进行注册,并把它设置为主活动,否则程序一启动就直接进入MainActivity 吧。最后再对 ForceOfflineReceiver 进行注册,并指定它接收 com.example.broadcastbestpractice.FORCE_OFFLINE 这条广播。

  最后,我们运行程序,效果如下:

  

 

  //End.

posted @ 2016-04-19 19:41  Vincent_Bryan  阅读(377)  评论(0编辑  收藏  举报