您的位置:首页 > 移动开发 > Android开发

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

2016-04-11 21:23 567 查看
新建一个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这条广播
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: