私が欲しいもの
設定メニューに簡単なチェックボックスが必要です。これをチェックすると、アプリのデバイス管理が有効になり、アプリがアンインストールされなくなります。
チェックボックスをオフにすると、デバイス管理が無効になります。
私のアプリはセキュリティに関するものであり、アンインストールされないように保護する必要があります。これに対する簡単な解決策を得ることができますか?
PS - これに関するドキュメントを読みましたが、機能していないようです。
私が欲しいもの
設定メニューに簡単なチェックボックスが必要です。これをチェックすると、アプリのデバイス管理が有効になり、アプリがアンインストールされなくなります。
チェックボックスをオフにすると、デバイス管理が無効になります。
私のアプリはセキュリティに関するものであり、アンインストールされないように保護する必要があります。これに対する簡単な解決策を得ることができますか?
PS - これに関するドキュメントを読みましたが、機能していないようです。
これは不可能です。アプリをデバイス管理者にすることを自分で決定することはできません。ユーザーを設定アプリの適切な場所に誘導して、アプリACTION_ADD_DEVICE_ADMIN
をデバイス管理者にすることをユーザーに選択してもらうことは大歓迎です。
たとえば、このアクティビティは、( 経由で) 既にデバイス管理者であるかどうかを確認し、必要に応じてアクティビティisActiveAdmin()
を起動します。ACTION_ADD_DEVICE_ADMIN
/***
Copyright (c) 2012 CommonsWare, LLC
Licensed under the Apache License, Version 2.0 (the "License"); you may not
use this file except in compliance with the License. You may obtain a copy
of the License at http://www.apache.org/licenses/LICENSE-2.0. Unless required
by applicable law or agreed to in writing, software distributed under the
License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS
OF ANY KIND, either express or implied. See the License for the specific
language governing permissions and limitations under the License.
From _The Busy Coder's Guide to Android Development_
http://commonsware.com/Android
*/
package com.commonsware.android.lockme;
import android.app.Activity;
import android.app.admin.DevicePolicyManager;
import android.content.ComponentName;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;
public class LockMeNowActivity extends Activity {
private DevicePolicyManager mgr=null;
private ComponentName cn=null;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
cn=new ComponentName(this, AdminReceiver.class);
mgr=(DevicePolicyManager)getSystemService(DEVICE_POLICY_SERVICE);
}
public void lockMeNow(View v) {
if (mgr.isAdminActive(cn)) {
mgr.lockNow();
}
else {
Intent intent=
new Intent(DevicePolicyManager.ACTION_ADD_DEVICE_ADMIN);
intent.putExtra(DevicePolicyManager.EXTRA_DEVICE_ADMIN, cn);
intent.putExtra(DevicePolicyManager.EXTRA_ADD_EXPLANATION,
getString(R.string.device_admin_explanation));
startActivity(intent);
}
}
}
(このサンプルプロジェクトから)
2 つのエクストラ (EXTRA_DEVICE_ADMIN
およびEXTRA_ADD_EXPLANATION
) はオプションですが、良いアイデアです。最初のものは、サブクラスをComponentName
識別する必要があります。DeviceAdminReceiver
2 つ目は、ユーザーがアプリをデバイス管理者にする理由を説明する (文字列リソースからの) 文字列である必要があります。
私のアプリはセキュリティに関するものであり、アンインストールされないように保護する必要があります。
誰でもアクセスして、アプリをデバイス管理者にしないことを決定し (ここでも [設定] を介して)、アンインストールすることができるため、これはあまり防御にはなりません。
import android.app.admin.DeviceAdminReceiver;
import android.content.Context;
import android.content.Intent;
import android.util.Log;
import android.widget.Toast;
/**
* This is the component that is responsible for actual device administration.
* It becomes the receiver when a policy is applied. It is important that we
* subclass DeviceAdminReceiver class here and to implement its only required
* method onEnabled().
*/
public class DemoDeviceAdmin extends DeviceAdminReceiver {
static final String TAG = "DemoDeviceAdmin";
/** Called when this application is approved to be a device administrator. */
@Override
public void onEnabled(Context context, Intent intent) {
super.onEnabled(context, intent);
Toast.makeText(context, R.string.device_admin_enabled,
Toast.LENGTH_LONG).show();
Log.d(TAG, "onEnabled");
}
/** Called when this application is no longer the device administrator. */
@Override
public void onDisabled(Context context, Intent intent) {
super.onDisabled(context, intent);
Toast.makeText(context, R.string.device_admin_disabled,
Toast.LENGTH_LONG).show();
Log.d(TAG, "onDisabled");
}
@Override
public void onPasswordChanged(Context context, Intent intent) {
super.onPasswordChanged(context, intent);
Log.d(TAG, "onPasswordChanged");
}
@Override
public void onPasswordFailed(Context context, Intent intent) {
super.onPasswordFailed(context, intent);
Log.d(TAG, "onPasswordFailed");
}
@Override
public void onPasswordSucceeded(Context context, Intent intent) {
super.onPasswordSucceeded(context, intent);
Log.d(TAG, "onPasswordSucceeded");
}
}
MainActivity は次のようになります
devicePolicyManager = (DevicePolicyManager) getSystemService(Context.DEVICE_POLICY_SERVICE);
demoDeviceAdmin = new ComponentName(this, DemoDeviceAdmin.class);
Log.e("DeviceAdminActive==", "" + demoDeviceAdmin);
Intent intent = new Intent(DevicePolicyManager.ACTION_ADD_DEVICE_ADMIN);// adds new device administrator
intent.putExtra(DevicePolicyManager.EXTRA_DEVICE_ADMIN, demoDeviceAdmin);//ComponentName of the administrator component.
intent.putExtra(DevicePolicyManager.EXTRA_ADD_EXPLANATION,
"Disable app");//dditional explanation
startActivityForResult(intent, ACTIVATION_REQUEST);
マニフェストは次のようになります。
<!-- This is where we register our receiver -->
<receiver
android:name=".DemoDeviceAdmin"
android:permission="android.permission.BIND_DEVICE_ADMIN" >
<intent-filter>
<!-- This action is required -->
<action android:name="android.app.action.DEVICE_ADMIN_ENABLED" />
</intent-filter>
<!-- This is required this receiver to become device admin component. -->
<meta-data
android:name="android.app.device_admin"
android:resource="@xml/device_admin_sample" />
</receiver>