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 {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);ActivityCollector.addActivity(this);}@Overrideprotected 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;@Overrideprotected 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() {@Overridepublic 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 {@Overridepublic 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() {@Overridepublic 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这条广播
更多相关文章
- android弹出选择对话框-仿某团购网android客户端栏目选择
- 利用半透明对话框实现android运行时的提示界面
- 如何查看android的wifi密码?究竟保存在哪里
- Android 将Activity以对话框(Dialog)形式显示
- Android自定义dialog对话框悬浮在界面上
- 【原创】Android之修改AlertDialog对话框及使用系统Holo风格
- Android 在Service开启对话框的方法