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

更多相关文章

  1. android弹出选择对话框-仿某团购网android客户端栏目选择
  2. 利用半透明对话框实现android运行时的提示界面
  3. 如何查看android的wifi密码?究竟保存在哪里
  4. Android 将Activity以对话框(Dialog)形式显示
  5. Android自定义dialog对话框悬浮在界面上
  6. 【原创】Android之修改AlertDialog对话框及使用系统Holo风格
  7. Android 在Service开启对话框的方法

随机推荐

  1. 关于Android工程的构建工具-不定时更新
  2. Android拍照、录像、录音代码范例
  3. android 让 webView 中的超链接失效
  4. android Bitmap getByteCount和getRowByt
  5. Android视频播放项目总结之 使用Android
  6. android实现 服务器功能
  7. android的PowerManager和PowerManager.Wa
  8. android MAT使用
  9. Android(安卓)3.0 http网络请求
  10. 关于用eclipse开发android经常出现R异常