如果把生活比喻为创作的意境,那么阅读就像阳光。
本讲内容:使用广播实现强制下线功能
强制下线功能比较常见,譬如:你的QQ在别处登录了,就会将你强制挤下线。
实现强制下线功能思路:在界面上弹出一个对话框,让用户无法进行任何其它操作,必须点击对话框中的确定按钮,然后回到登录界面即可,可是这样就存在一个问题,因为我们被通知需要强制下线时可能正处于任何一个界面,难道需要在每个界面都编写一个弹出对话框?解决这一问题,我们可以使用广播。
注意:强制下线功能需要先关闭掉所有的活动然后回到登录界面。
示例一:
下面是res/layout/login.xml 布局文件:
<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:hint="Input your password" />
</TableRow>
<TableRow>
<Button
android:id="@+id/login"
android:layout_height="wrap_content"
android:layout_span="2"
android:text="Login" />
</TableRow>
</TableLayout>
下面是res/layout/activity_main.xml 布局文件:
<RelativeLayout 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" >
<Button
android:id="@+id/force_offline"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Send force offline bradcast" />
</RelativeLayout>
下面是ActivityCollector.java文件:
//管理所有活动
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 .java文件:
//创建BaseActivity类作为所有类的父类
public class BaseActivity extends Activity{
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
ActivityCollector.addActivity(this);
}
protected void onDestroy() {
super.onDestroy();
ActivityCollector.removeActivity(this);
}
}
下面是 LoginActivity .java文件:
public class LoginActivity extends BaseActivity{
private EditText accountEdit;
private EditText passwordEdit;
private Button login;
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.login);
initView();
}
private void initView() {
accountEdit=(EditText) findViewById(R.id.account);
passwordEdit=(EditText) findViewById(R.id.password);
login=(Button) findViewById(R.id.login);
login.setOnClickListener(new OnClickListener() {
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, "acount or password is invalid!", Toast.LENGTH_LONG).show();
}
}
});
}
}
下面是 MainActivity .java文件:
public class MainActivity extends BaseActivity {
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() {
public void onClick(View v) {
Intent intent=new Intent("com.example.broadcast.FORCE_OFFLINE");
sendBroadcast(intent);
}
});
}
}
下面是ForceOfflineReceiver.java文件:
public class ForceOfflineReceiver extends BroadcastReceiver{
public void onReceive(final Context context, Intent intent) {//注意写final
AlertDialog.Builder dialogBuilder=new AlertDialog.Builder(context);
dialogBuilder.setTitle("Warning");
dialogBuilder.setMessage("You are forced to be offline. Please try to login again");
dialogBuilder.setCancelable(false);//设置对话框为不可取消,防止按Back键关闭
dialogBuilder.setPositiveButton("OK", new DialogInterface.OnClickListener() {
public void onClick(DialogInterface dialog, int which) {
ActivityCollector.finishAll();//销毁所有对象
Intent intent=new Intent(context,LoginActivity.class);
//由于我们是在广播接收器里启动活动的,因此一定要给Intent加入FLAG_ACTIVITY_NEW_TASK这个标志
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(intent);
}
});
AlertDialog alertDialog=dialogBuilder.create();
//需要反对话框的类型设置为TYPE_SYSTEM_ALERT(系统级别的对话框),不然它将无法在广播接收器里弹出
alertDialog.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);
alertDialog.show();
}
}
在AndroidManifest.xml文件
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.text03"
android:versionCode="1"
android:versionName="1.0" >
<uses-sdk
android:minSdkVersion="8"
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.broadcast.FORCE_OFFLINE"/>
</intent-filter>
</receiver>
</application>
</manifest>
Take your time and enjoy it 要原码的、路过的、学习过的请留个言,顶个呗~~