天天看點

Android開發中強制下線功能實作

強制下線的應用場景很多, 比如我們好多賬号都有搶登的顯現, 會被擠下線.

Android開發中強制下線功能實作

實作效果:

Android開發中強制下線功能實作

源碼:

https://github.com/junzaivip/BroadcastBestPractice

詳細實作:

其實實作強制下線功能的思路也比較簡單, 隻需要在界面彈出一個對話框, 讓使用者無法進行其他操作, 必須點選對話框中的确定按鈕, 然後回到登入界面即可.

但是又有一個問題, 我們通知使用者強制下線, 使用者可能正處于任何一個界面, 難道需要在每個界面上都編寫一個彈出對話框的邏輯? 當然不是! 我們可以借助本章中所學的廣播知識來輕松實作這一功能.

public class ActivityConllector {
    public static List<Activity> activities = new ArrayList<>();

    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 AppCompatActivity{
    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        ActivityConllector.addActivity(this);
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        ActivityConllector.removeActivity(this);
    }
}
           

首先編寫布局檔案activity_login檔案:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical">
    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="60dp"
        android:orientation="horizontal">
        <TextView
            android:layout_width="90dp"
            android:layout_height="wrap_content"
            android:text="User Name"
            android:textSize="18sp"
            android:layout_gravity="center_vertical"/>
        <EditText
            android:id="@+id/account"
            android:layout_width="0dp"
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:layout_gravity="center_vertical"
            />
    </LinearLayout>
    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="60dp"
        android:orientation="horizontal">
        <TextView
            android:layout_width="90dp"
            android:layout_height="wrap_content"
            android:text="Password"
            android:textSize="18sp"
            android:layout_gravity="center_vertical"/>
        <EditText
            android:id="@+id/password"
            android:layout_width="0dp"
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:layout_gravity="center_vertical"
            android:inputType="textPassword"/>
    </LinearLayout>
    <Button
        android:id="@+id/login"
        android:layout_width="match_parent"
        android:layout_height="60dp"
        android:text="Login"/>
</LinearLayout>
           

LoginActivity.java中代碼:

public class LoginActivity extends BaseActivity implements View.OnClickListener{
    private EditText userName;
    private EditText passWord;
    private Button login;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_login);
        userName = (EditText)findViewById(R.id.account);
        passWord = (EditText)findViewById(R.id.password);
        login = (Button) findViewById(R.id.login);
        login.setOnClickListener(this);

    }

    @Override
    public void onClick(View v) {
        switch (v.getId()){
            case R.id.login:
                String username = userName.getText().toString();
                String password = passWord.getText().toString();
                if(username.equals("admin") && password.equals("123")){
                    Intent intent = new Intent(LoginActivity.this, MainActivity.class);
                    startActivity(intent);
                    finish();
                } else {
                    Toast.makeText(this, "使用者名或者密碼錯誤!", Toast.LENGTH_SHORT).show();
                }

        }

    }
}
           

我們

這時

可以了解成, 登入界面之後, 進入的activity_main.xml就是主界面. 主界面沒有其他功能, 隻有一個用于顯示的TextView和一個用作下線功能的按鈕

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/activity_main"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical">

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="這是首頁面"
        android:layout_gravity="center_horizontal"/>

    <Button
        android:id="@+id/force_offline"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="強制下線"/>
</LinearLayout>
           

MainActivity.java中代碼:

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 View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Intent intent = new Intent("com.junzaivip.broadcastbestpractice.FORCE_OFFLINE");
                sendBroadcast(intent);
            }
        });
    }
}
           

我們按鈕的點選事件裡面發送了一條廣播, 廣播的值為com.junzaivip.broadcastbestpractice.FORCE_OFFLINE, 這條廣播就是用于通知程式強制使用者下線的.

也就是說強制使用者下線的邏輯并不是寫在MainActivity裡的, 而是應該寫在接收這條廣播的接收器裡面. 這樣的好處是, 強制下線的功能就不會依附于任何的界面, 不管是在程式的任何地方, 隻需要發出一條這樣的廣播,就可以完成強制下線的操作了.

那麼, 需要建立一個廣播接收器, 用于接收這條強制下線的廣播, 唯一的問題就是, 應該在哪裡建立呢? 由于廣播接收器裡面需要彈出一個對話框來阻塞使用者的正常操作. 但如果建立的是一個靜态注冊的廣播接收器, 是沒有辦法在onReceive()方法裡彈出對話框這樣的UI控件的, 而我們顯然不能也不可能在每個活動中注冊一個動态的廣播接收器.

那麼到底應該怎麼辦呢? 其實很明顯, 隻需要在BaseActivity中動态注冊一個廣播接收器就可以了. 因為所有的活動都繼承自BaseActivity的.

BaseActivity中的代碼:

public class BaseActivity  extends AppCompatActivity{
    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        ActivityConllector.addActivity(this);
    }
    @Override
    protected void onDestroy() {
        super.onDestroy();
        ActivityConllector.removeActivity(this);
    }
    class ForceOffLineReceiver extends BroadcastReceiver{
        @Override
        public void onReceive(final Context context, final Intent intent) {
            AlertDialog.Builder builder = new AlertDialog.Builder(context);
            builder.setTitle("warning");
            builder.setMessage("您被強制下線, 請您重新登入!");
            builder.setCancelable(false); // 将對話框設定為不可取消
            // 給按鈕添加注冊監聽
            builder.setPositiveButton("OK", new DialogInterface.OnClickListener() {
                @Override
                public void onClick(DialogInterface dialog, int which) {
                    // 點選按鈕所調用的方法
                    ActivityConllector.finishAll();//銷毀所有的活動
                    Intent intent1 = new Intent(context, LoginActivity.class);
                    context.startActivity(intent);
                }
            });
            builder.show();
        }
    }

}
           

我們需要設定主活動為LoginActivity, 而不再是MainActivity, 模拟通路一個程式首先在登入頁面.

修改AndroidManifest.xml檔案:

<application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:supportsRtl="true"
        android:theme="@style/AppTheme">
        <activity android:name=".MainActivity">
        </activity>
        <activity android:name=".LoginActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>
           

運作效果如下:

Android開發中強制下線功能實作

參考: 郭霖: <<第一行代碼>>

繼續閱讀