1. 程式人生 > >你理解Android裝置管理的妙用麼?

你理解Android裝置管理的妙用麼?

    前段時間公司有個需求,需要控制手機的攝像頭和錄音全部禁用,以達到使用者公司那不可告人的小祕密~ 嘿嘿嘿~我把這個需求告訴了我的大哥..於是乎大哥會心一笑對我說..留個後門..看看是不是A.V公司的工作人員要配置的內部手機..開過玩笑~ 評審哥哥不要當真~

    這篇文章我會使用android提供的裝置管理器類,來實現禁用手機攝像頭的功能,還能夠改變密碼,鎖屏,重啟等功能,具體功能情去查API,如果我沒記錯的話有5千多行... ... 但是這篇文章主要就介紹常用的幾個功能需求,後面還會介紹如何讓使用者點選啟用後,禁止使用者手動點取消啟用的流氓目的... = = 下面看程式碼:


1.首先你需要先建立xml檔案如上圖所示。

2.在device_admin.xml檔案中,寫入如下內容:(下面的內容就好比申請許可權)

<device-admin xmlns:android="http://schemas.android.com/apk/res/android">
    <uses-policies>

        <!--停用相機-->
        <disable-camera/>
        <!--鎖屏時禁用某些功能-->
        <disable-keyguard-features/>
        <!--設定儲存裝置加密-->
        <encrypted-storage/>
        <!--設定鎖定螢幕密碼的有效期-->
        <expire-password/>
        <!--鎖定螢幕-->
        <force-lock/>
        <!--設定密碼規則-->
        <limit-password/>
        <!--更改螢幕解鎖密碼-->
        <reset-password/>
        <!--設定裝置全域性代理-->
        <set-global-proxy/>
        <!--監控螢幕解鎖嘗試次數-->
        <watch-login/>
        <!--恢復出廠設定-->
        <wipe-data/>

    </uses-policies>

</device-admin>

3.你需要建立一個DeviceReceiver(你可以理解這是一個裝置管理的監聽器)
import android.app.admin.DeviceAdminReceiver;
import android.content.Context;
import android.content.Intent;
import android.widget.Toast;

/**
 * Created by Administrator on 2017/8/16 0016.
 */

public class DeviceReceiver extends DeviceAdminReceiver {

    @Override
    public void onEnabled(Context context, Intent intent) {
        super.onEnabled(context, intent);
        Toast.makeText(context, "裝置管理器:已啟用", Toast.LENGTH_SHORT).show();
    }

    @Override
    public void onDisabled(Context context, Intent intent) {
        super.onDisabled(context, intent);
        Toast.makeText(context, "裝置管理器:未啟用", Toast.LENGTH_SHORT).show();
    }

    @Override
    public CharSequence onDisableRequested(Context context, Intent intent) {
        Intent intent1 = context.getPackageManager().getLaunchIntentForPackage("com.android.settings");
        intent1.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
        context.startActivity(intent1);
        try {
            Thread.sleep(7000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        return "This is a onDisableRequested response message";
    }

    @Override
    public void onPasswordChanged(Context context, Intent intent) {
        super.onPasswordChanged(context, intent);
        Toast.makeText(context, "裝置管理;密碼已經改變", Toast.LENGTH_SHORT).show();
    }

    @Override
    public void onPasswordFailed(Context context, Intent intent) {
        super.onPasswordFailed(context, intent);
        Toast.makeText(context, "裝置管理:改變密碼失敗", Toast.LENGTH_SHORT).show();
    }

    @Override
    public void onPasswordSucceeded(Context context, Intent intent) {
        super.onPasswordSucceeded(context, intent);
        Toast.makeText(context, "裝置管理;改變密碼成功", Toast.LENGTH_SHORT).show();
    }

}

4.接下來就是很重要的清單檔案中註冊receive了:
 <!--裝置管理-->
        <receiver android:name=".DeviceReceiver"
            android:permission="android.permission.BIND_DEVICE_ADMIN">
            <meta-data
                android:name="android.app.device_admin"
                android:resource="@xml/device_admin"/>
            <intent-filter>
                <action android:name="android.app.action.DEVICE_ADMIN_ENABLED"/>
                <!--<category android:name="android.intent.action.BOOT_COMPLETED"/>-->
                <!--<category android:name="android.intent.category.HOME"/>-->
            </intent-filter>
        </receiver>

5.接下來是佈局檔案,僅僅是設定幾個按鈕:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/activity_main"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    android:background="#edeaea"
    tools:context="com.sq.devicedemo.MainActivity">

    <Button
        android:id="@+id/btn_action"
        android:text="啟用裝置管理器"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_margin="5dp"/>

    <Button
        android:id="@+id/btn_cancel_active"
        android:text="取消啟用裝置管理器"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_margin="5dp"/>

    <Button
        android:id="@+id/btn_change_password"
        android:text="修改密碼為1234"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" />

    <Button
        android:id="@+id/btn_cancel_password"
        android:text="取消密碼"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" />

    <Button
        android:id="@+id/btn_lock"
        android:text="鎖屏"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_margin="5dp"/>

    <Button
        android:id="@+id/btn_setCameraDisabled"
        android:text="禁用 相機"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_margin="5dp"/>

    <Button
        android:id="@+id/btn_setCameraDisabled1"
        android:text="啟動 相機"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_margin="5dp"/>

</LinearLayout>

6.MainActivity中程式碼部分,初始化和判斷裝置管理器狀態,以及使用者是否同意點選的回撥方法:
import android.app.Activity;
import android.app.admin.DevicePolicyManager;
import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.os.Build;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.view.ViewGroup;
import android.view.WindowManager;
import android.widget.Toast;

public class MainActivity extends AppCompatActivity {

    /**
     * 啟用元件的請求碼
     */
    private static final int REQUEST_CODE_ACTIVE_COMPONENT = 1;

    /**
     * 裝置安全管理服務,2.2之前需要通過反射技術獲取
     */
    private DevicePolicyManager devicePolicyManager = null;
    /**
     * 對應自定義DeviceAdminReceiver的元件
     */
    private ComponentName componentName = null;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        setTranslucent(this);

        devicePolicyManager = (DevicePolicyManager) getSystemService(Context.DEVICE_POLICY_SERVICE);
        componentName = new ComponentName(this,DeviceReceiver.class);

        /**
         * 啟用裝置管理器
         */
        findViewById(R.id.btn_action).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                if (isAdminActive()){
                    Toast.makeText(MainActivity.this, "裝置管理器已啟用", Toast.LENGTH_SHORT).show();
                }else {
                    // 開啟管理器的啟用視窗
                    Intent intent = new Intent(DevicePolicyManager.ACTION_ADD_DEVICE_ADMIN);
                    // 指定需要啟用的元件
                    intent.putExtra(DevicePolicyManager.EXTRA_DEVICE_ADMIN,componentName);
                    intent.putExtra(DevicePolicyManager.EXTRA_ADD_EXPLANATION, "(啟用視窗中的描述資訊)");
                    startActivityForResult(intent, REQUEST_CODE_ACTIVE_COMPONENT);
                }
            }
        });

        /**
         * 取消啟用
         */
        findViewById(R.id.btn_cancel_active).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                if (isAdminActive()){
                    devicePolicyManager.removeActiveAdmin(componentName);
                    Toast.makeText(MainActivity.this, "將觸發deviceAdminReceiver.onDisabled", Toast.LENGTH_SHORT).show();
                }else {
                    Toast.makeText(MainActivity.this, "裝置管理未啟用", Toast.LENGTH_SHORT).show();
                }
            }
        });

        /**
         * 鎖屏
         */
        findViewById(R.id.btn_lock).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                if (isAdminActive()){
                    devicePolicyManager.lockNow();
                }else {
                    Toast.makeText(MainActivity.this, "裝置管理未啟用", Toast.LENGTH_SHORT).show();
                }
            }
        });

        /**
         * 禁止使用攝像頭
         */
        findViewById(R.id.btn_setCameraDisabled).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                if (isAdminActive()){
                    devicePolicyManager.setCameraDisabled(componentName,true);
                }
            }
        });

        /**
         * 啟動攝像頭
         */
        findViewById(R.id.btn_setCameraDisabled1).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                if (isAdminActive()){
                    devicePolicyManager.setCameraDisabled(componentName,false);
                }
            }
        });

        /**
         * 設定密碼
         */
        findViewById(R.id.btn_change_password).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                if (isAdminActive()){
                    devicePolicyManager.resetPassword("123456",1);
                }
            }
        });

        /**
         * 取消密碼
         */
        findViewById(R.id.btn_cancel_password).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                if (isAdminActive()){
                    devicePolicyManager.resetPassword("",0);
                }
            }
        });

    }

    /**
     * 判斷該組建是否有系統管理員的許可權(系統安全-裝置管理器 中是否啟用)
     * @return
     */
    private boolean isAdminActive(){
        return devicePolicyManager.isAdminActive(componentName);
    }

    /**
     * 使用者是否點選啟用或取消的回撥
     * @param requestCode
     * @param resultCode
     * @param data
     */
    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        if (requestCode == REQUEST_CODE_ACTIVE_COMPONENT) {
            // 啟用元件的響應
            if (resultCode == Activity.RESULT_CANCELED) {
                Toast.makeText(this, "使用者手動取消啟用", Toast.LENGTH_SHORT).show();
            } else {
                Toast.makeText(this, "已觸發DeviceAdminReceiver.onEnabled", Toast.LENGTH_SHORT).show();
            }
        }

    }

    public static void setTranslucent(Activity activity) {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
            // 設定狀態列透明
            activity.getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
            // 設定根佈局的引數
            ViewGroup rootView = (ViewGroup) ((ViewGroup) activity.findViewById(android.R.id.content)).getChildAt(0);
            rootView.setFitsSystemWindows(true);
            rootView.setClipToPadding(true);
        }
    }

}

上個攝像頭被禁用的效果圖(微信呼叫掃一掃介面):


    不知道你們有沒有注意,在第三步中有一處添加了其他的程式碼部分,目的是使用者手動點選取消啟用時,開啟執行緒阻塞,當睡眠超過5s後,詢問是否真的取消裝置管理器許可權的彈框會失效(通過檢視android原始碼可以知道)為了穩妥點設定成7s,當然也有不同的方法啦,譬如迴圈不停的鎖屏,甚至可以寫重啟手機...只是鎖屏後不同手機有差異,手速快一點一樣可以在螢幕亮暗的瞬間點確定..當然你手機設定密碼的話就不會有這種問題~ 還有一種老方法已經不能在比較新的android系統使用了,就是在裝置管理器的介面彈出遮擋佈局,來變相禁止使用者擅自取消許可權... android系統已經知道了這個bug,禁止一切彈窗或是佈局活動頁在此介面彈出,會報沒有許可權異常且自動取消本app在裝置管理器中的許可權。