本次项目实现一个简单登陆界面强制下线的功能(包含登陆界面、广播等内容)。
*要有最朴素的生活,与最遥远的梦想!*
新建一个BroadcastBestPractice项目。
1、强制下线功能需要先关闭掉所有的活动,然后回到登陆界面。因此先创建一个ActivityCollector类用于管理所有的活动。
代码如下:
package com.example.broadcastbestpractice;
import java.util.ArrayList;
import java.util.List;
import android.app.Activity;
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(); //关闭所有活动
}
}
}
}
2、然后创建BaseActivity类作为所有活动的父类,这样所有继承BaseActivity的活动
均会被添加到ActivityCollector类中,也可以随时调用父类的方法,关闭所有活动。
代码如下:
package com.example.broadcastbestpractice;
import android.app.Activity;
import android.os.Bundle;
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);
}
}
3、需要创建登陆界面的布局,新建布局文件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> //第一行,包含2个控件
<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 > //第二行,包含2个控件
<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>
4、既然登陆界面的布局已经完成,接下来就编写登陆界面的活动。
新建LoginActivity继承自BaseActivity 。
代码如下:
package com.example.broadcastbestpractice;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.EditText;
import android.widget.Toast;
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); //加载布局login.xml的布局
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(); //将输入的账号给account
String password=passwordEdit.getText().toString(); //将输入的密码给password
if(account.equals("admin")&&password.equals("123456")) //如果账号是admin,并且密码是123456,则登陆成功
{ Intent intent =new Intent(LoginActivity.this,MainActivity.class);
startActivity(intent); //跳转到MainActivity活动的界面
finish();
}
else{
Toast.makeText(LoginActivity.this, "account or password is invaild",Toast.LENGTH_SHORT).show(); //如果账号密码不匹配,提示账号或密码错误
}
});
}
}
5、MainActivity便是登陆成功后进入的程序主界面了,在这里加入强制下线功能。修改activity_main.xml的布局,代码如下:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
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 broadcast " />
</LinearLayout>
修改MainActivity中的代码,代码如下:
package com.example.broadcastbestpractice;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
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,这条广播就是用于通知程序强制用户下线的。
}
});
}
}
6、创建一个广播接收器,毋庸置疑,强制用户下线的逻辑应该写在接收广播接收器里面,这样强制下线的功能就不会依附于任何的界面了。新建ForceOfflineReceiver 类继承自BroadcastReceiver ,代码如下:
package com.example.broadcastbestpractice;
import android.app.AlertDialog;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.DialogInterface;
import android.content.Intent;
import android.view.WindowManager;
public class ForceOfflineReceiver extends BroadcastReceiver {
@Override
public void onReceive(final Context context,Intent intent){
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); //false,将对话框设为不可取消,否则按一下Back键就可以关闭对话框继续使用程序了
dialogBuilder.setPositiveButton("OK", new DialogInterface.OnClickListener(){
@Override //给对话框注册确定按钮,当点击确定按钮时,就调用ActivityCollector的finishAll()方法来销毁当前所有活动,并重启LoginActivity这个活动
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);
}
});
AlertDialog alertDialog=dialogBuilder.create();
alertDialog.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT); //把对话框类型设为TYPE_SYSTEM_ALERT,不然对话框无法在广播接收器里弹出
alertDialog.show(); //显示对话框
}
}
7、最后对AndroidManifest.xml文件进行配置,代码如下:
<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> //将LoginActivity设置为主活动
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity> //注册MainActivity活动
<activity android:name=".MainActivity">
</activity>
<receiver android:name=".ForceOfflineReceiver" > //注册接收器
<intent-filter>
<action android:name="com.example.broadcastbestpractice.FORCE_OFFLINE"/> //让接收器接收com.example.broadcastbestpractice.FORCE_OFFLINE这条广播
</intent-filter>
</receiver>
</application>
</manifest>
8、最后运行程序,上图!!!
首先进入登陆界面,并可以在这里输入账号和密码,如图1所示
图1
如果输入的账号是admin ,密码是123456,点击按钮就会进入到程序的主界面,如图2所示
图2
这时点击一下发送广播的按钮,就会发出一条强制下线的广播,ForceOfflineReceiver接收器收到这条广播后就会弹出一个对话框提示用户被强制下线,如图3所示
图3
这时用户将无法再对界面的任何元素进行操作,只能点击确定按钮,然后会重新回到登陆界面。这样,强制下线功能就实现啦!