广播的最佳实践——实现强制下线功能

本次项目实现一个简单登陆界面强制下线的功能(包含登陆界面、广播等内容)。

*要有最朴素的生活,与最遥远的梦想!*

新建一个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

这时用户将无法再对界面的任何元素进行操作,只能点击确定按钮,然后会重新回到登陆界面。这样,强制下线功能就实现啦!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值