Android 广播------实现强制下线功能

新建一个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这条广播

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值