【轉】Android之自定義裝置管理
阿新 • • 發佈:2019-02-10
需求:控制手機的攝像頭和錄音全部禁用~
這篇文章使用android提供的裝置管理器類DeviceAdminReceiver ,來實現禁用手機攝像頭的功能,還能夠改變密碼,鎖屏,重啟等功能,具體功能請去查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; 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" android:id="@+id/activity_main" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" android:background="#edeaea"> <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.os.Bundle; import android.view.View; import android.view.ViewGroup; import android.view.WindowManager; import android.widget.Toast; public class MainActivity extends Activity { /** * 啟用元件的請求碼 */ 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在裝置管理器中的許可權。