新建一个ActivityCollector类,用于管理所有的活动
public class ActivityCollector {
public static List<Activity> activities = new ArrayList<Activity>();
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 Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
ActivityCollector.addActivity(this);
}
@Override
protected void onDestroy() {
super.onDestroy();
ActivityCollector.removeActivity(this);
}
}
接着创建一个登录界面的布局
新建布局文件login.xml
<pre name="code" class="html"><?xml version="1.0" encoding="utf-8"?>
<TableLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:stretchColumns="1" >
<TableRow>
<TextView
android:layout_height="wrap_content"
android:text="Account : " />
<EditText
android:id="@+id/account"
android:layout_height="wrap_content"
android:hint="Input your account" />
</TableRow>
<TableRow>
<TextView
android:layout_height="wrap_content"
android:text="Password : " />
<EditText
android:id="@+id/password"
android:layout_height="wrap_content"
android:inputType="textPassword" />
</TableRow>
<TableRow>
<Button
android:id="@+id/login"
android:layout_height="wrap_content"
android:layout_span="2"
android:text="Login" />
</TableRow>
</TableLayout>
TableLayout允许我们使用表格的方式来排列控件
在TableLayout中每加入一个TableRow就表示在表格中添加了一行
在TableRow中每加入一个控件,就表示在该行中加入了一列
TableRow中的控件是不能指定宽度的
第一行有一个TextView和一个用于输入账号的EditText
第二行也有一个TextView和一个用于输入密码的EditText
通过android:inputType="textPassword"将EditText变成密码输入框
android:stretchColumns="1",表示如果表格不能完全占满屏幕宽度,就将第二列进行拉伸
android:layout_span="2",让登录按钮占据两列的空间(对单元格进行合并)
接下来是编写登录界面的活动
新建LoginActivity继承自BaseActivity
public class LoginActivity extends BaseActivity {
private EditText accountEdit;
private EditText passwordEdit;
private Button login;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.login);
accountEdit = (EditText) findViewById(R.id.account);
passwordEdit = (EditText) findViewById(R.id.password);
login = (Button) findViewById(R.id.login);
login.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
String account = accountEdit.getText().toString();
String password = passwordEdit.getText().toString();
// 如果账号是admin且密码是123456,就认为登录成功
if (account.equals("admin") && password.equals("123456")) {
Intent intent = new Intent(LoginActivity.this, MainActivity.class);
startActivity(intent);
finish();
} else {
Toast.makeText(LoginActivity.this, "account or password is invalid", Toast.LENGTH_SHORT).show();
}
}
});
}
}
setContenView()将login布局加载进来
在登录按钮的点击事件里对输入的账号和密码进行判断
如果账号是admin并且密码是123456,则认为登录成功并跳转到MainActivity,否则提示用户账号或密码错误
可以将MainActivity理解为登陆成功后进入的程序主界面
修改activity_main.xml
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:paddingBottom="@dimen/activity_vertical_margin"
android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
tools:context="com.example.broadcastbestpractice.MainActivity" >
<Button
android:id="@+id/force_offline"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Send force offline broadcast" />
</LinearLayout>
然后修改MainActivity
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 OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent("com.example.broadcastbestpractice.FORCE_OFFLINE");
sendBroadcast(intent);
}
});
}
}
广播的值为com.example.broadcastbestpractice.FORCE_OFFLINE
这条广播是用于通知程序强制用户下线的
强制用户下线的逻辑并不是写在MainActivity里,而是应该写在接收这条广播的广播接收器里,这样强制下线的功能就不会依附于任何的界面了。
不管是在程序的任何地方,只需要发出这样一条广播,就可以完成强制下线的操作了。
接下来创建一个广播接收器
新建ForceOfflineReceiver继承自BroadcastReceiver
public class ForceOfflineReceiver extends BroadcastReceiver {
@Override
public void onReceive(final Context context, Intent intent) {
AlertDialog.Builder dialogBuilder = new AlertDialog.Builder(context);
dialogBuilder.setMessage("You are force to be offline. Please try to login again.");
dialogBuilder.setCancelable(false);
dialogBuilder.setPositiveButton("OK", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
ActivityCollector.finishAll(); // 销毁所有活动
Intent intent = new Intent(context, LoginActivity.class);
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(intent); // 重新启动LoginActivity
}
});
AlertDialog alertDialog = dialogBuilder.create();
// 需要设置AlertDialog的类型,保证在广播接收器中可以正常弹出
alertDialog.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);
alertDialog.show();
}
}
使用AlertDialog.Builder来构建一个对话框
注意:一定要调用setCancelable()方法将对话框设置为不可取消,否则用户点击一下Back键就可以关闭对话框继续使用程序了。
使用setPositiveButton()方法给对话框注册确定按钮
如果点击了确定按钮,就调用ActivityCollector的finishAll()方法来销毁所有活动,并重新启动LoginActivity活动
另外,由于我们是在广播接收器里启动活动的,因此一定要给Intent加入FLAG_ACTIVITY_NEW_TASK这个标志
最后,还需要把对话框的类型设为TYPE_SYSTEM_ALERT,否则他将无法在广播接收器里弹出
接下来对AndroidManifest.xml文件进行配置
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.broadcastbestpractice"
android:versionCode="1"
android:versionName="1.0" >
<uses-sdk
android:minSdkVersion="14"
android:targetSdkVersion="21" />
<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />
<application
android:allowBackup="true"
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:theme="@style/AppTheme" >
<activity
android:name=".LoginActivity"
android:label="@string/app_name" >
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<activity android:name=".MainActivity" >
</activity>
<receiver android:name=".ForceOfflineReceiver" >
<intent-filter>
<action android:name="com.example.broadcastbestpractice.FORCE_OFFLINE" />
</intent-filter>
</receiver>
</application>
</manifest>
由于在ForceOfflineReceiver里弹出了系统级别的对话框,因此需要声明android.permission.SYSTEM_ALERT_WINDOW权限
然后对LoginActivity进行注册,并将它设置为主活动,因为肯定不能让用户启动程序就直接进入Mainactivity
最后再对ForceOfflineReceiver进行注册,并指定它接收com.example.broadcastbestpractice.FORCE_OFFLINE这条广播