/* * Copyright (C) 2010 The Android Open Source Project * * 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. */ package com.android.settings; import android.app.AppOpsManager; import org.xmlpull.v1.XmlPullParserException; import android.app.Activity; import android.app.ActivityManagerNative; import android.app.AlertDialog; import android.app.Dialog; import android.app.admin.DeviceAdminInfo; import android.app.admin.DeviceAdminReceiver; import android.app.admin.DevicePolicyManager; import android.content.ComponentName; import android.content.Context; import android.content.DialogInterface; import android.content.Intent; import android.content.pm.ActivityInfo; import android.content.pm.ApplicationInfo; import android.content.pm.PackageInfo; import android.content.pm.PackageManager; import android.content.pm.PackageManager.NameNotFoundException; import android.content.pm.ResolveInfo; import android.content.res.Resources; import android.os.Bundle; import android.os.Handler; import android.os.RemoteCallback; import android.os.RemoteException; import android.os.UserHandle; import android.text.TextUtils.TruncateAt; import android.util.EventLog; import android.util.Log; import android.view.Display; import android.view.View; import android.view.ViewGroup; import android.view.WindowManager; import android.widget.AppSecurityPermissions; import android.widget.Button; import android.widget.ImageView; import android.widget.TextView; import java.io.IOException; import java.util.ArrayList; import java.util.List; public class DeviceAdminAdd extends Activity { static final String TAG = "DeviceAdminAdd"; static final int DIALOG_WARNING = 1; private static final int MAX_ADD_MSG_LINES_PORTRAIT = 5; private static final int MAX_ADD_MSG_LINES_LANDSCAPE = 2; private static final int MAX_ADD_MSG_LINES = 15; Handler mHandler; DevicePolicyManager mDPM; AppOpsManager mAppOps; DeviceAdminInfo mDeviceAdmin; CharSequence mAddMsgText; String mProfileOwnerName; ImageView mAdminIcon; TextView mAdminName; TextView mAdminDescription; TextView mAddMsg; TextView mProfileOwnerWarning; ImageView mAddMsgExpander; boolean mAddMsgEllipsized = true; TextView mAdminWarning; ViewGroup mAdminPolicies; Button mActionButton; Button mCancelButton; final ArrayList mAddingPolicies = new ArrayList(); final ArrayList mActivePolicies = new ArrayList(); boolean mAdding; boolean mRefreshing; boolean mWaitingForRemoveMsg; boolean mAddingProfileOwner; int mCurSysAppOpMode; int mCurToastAppOpMode; @Override protected void onCreate(Bundle icicle) { super.onCreate(icicle); mHandler = new Handler(getMainLooper()); mDPM = (DevicePolicyManager)getSystemService(Context.DEVICE_POLICY_SERVICE); mAppOps = (AppOpsManager)getSystemService(Context.APP_OPS_SERVICE); PackageManager packageManager = getPackageManager(); if ((getIntent().getFlags()&Intent.FLAG_ACTIVITY_NEW_TASK) != 0) { Log.w(TAG, "Cannot start ADD_DEVICE_ADMIN as a new task"); finish(); return; } String action = getIntent().getAction(); ComponentName who = (ComponentName)getIntent().getParcelableExtra( DevicePolicyManager.EXTRA_DEVICE_ADMIN); if (who == null) { Log.w(TAG, "No component specified in " + action); finish(); return; } if (action != null && action.equals(DevicePolicyManager.ACTION_SET_PROFILE_OWNER)) { setResult(RESULT_CANCELED); setFinishOnTouchOutside(true); mAddingProfileOwner = true; mProfileOwnerName = getIntent().getStringExtra(DevicePolicyManager.EXTRA_PROFILE_OWNER_NAME); String callingPackage = getCallingPackage(); if (callingPackage == null || !callingPackage.equals(who.getPackageName())) { Log.e(TAG, "Unknown or incorrect caller"); finish(); return; } try { PackageInfo packageInfo = packageManager.getPackageInfo(callingPackage, 0); if ((packageInfo.applicationInfo.flags & ApplicationInfo.FLAG_SYSTEM) == 0) { Log.e(TAG, "Cannot set a non-system app as a profile owner"); finish(); return; } } catch (NameNotFoundException nnfe) { Log.e(TAG, "Cannot find the package " + callingPackage); finish(); return; } } ActivityInfo ai; try { ai = packageManager.getReceiverInfo(who, PackageManager.GET_META_DATA); } catch (PackageManager.NameNotFoundException e) { Log.w(TAG, "Unable to retrieve device policy " + who, e); finish(); return; } // When activating, make sure the given component name is actually a valid device admin. // No need to check this when deactivating, because it is safe to deactivate an active // invalid device admin. if (!mDPM.isAdminActive(who)) { List avail = packageManager.queryBroadcastReceivers( new Intent(DeviceAdminReceiver.ACTION_DEVICE_ADMIN_ENABLED), PackageManager.GET_DISABLED_UNTIL_USED_COMPONENTS); int count = avail == null ? 0 : avail.size(); boolean found = false; for (int i=0; i newPolicies = mDeviceAdmin.getUsedPolicies(); for (int i = 0; i < newPolicies.size(); i++) { DeviceAdminInfo.PolicyInfo pi = newPolicies.get(i); if (!mDPM.hasGrantedPolicy(who, pi.ident)) { mRefreshing = true; break; } } if (!mRefreshing) { // Nothing changed (or policies were removed) - return immediately setResult(Activity.RESULT_OK); finish(); return; } } } // If we're trying to add a profile owner and user setup hasn't completed yet, no // need to prompt for permission. Just add and finish. if (mAddingProfileOwner && !mDPM.hasUserSetupCompleted()) { addAndFinish(); return; } mAddMsgText = getIntent().getCharSequenceExtra(DevicePolicyManager.EXTRA_ADD_EXPLANATION); setContentView(R.layout.device_admin_add); mAdminIcon = (ImageView)findViewById(R.id.admin_icon); mAdminName = (TextView)findViewById(R.id.admin_name); mAdminDescription = (TextView)findViewById(R.id.admin_description); mProfileOwnerWarning = (TextView) findViewById(R.id.profile_owner_warning); mAddMsg = (TextView)findViewById(R.id.add_msg); mAddMsgExpander = (ImageView) findViewById(R.id.add_msg_expander); mAddMsg.setOnClickListener(new View.OnClickListener() { public void onClick(View v) { toggleMessageEllipsis(v); } }); // toggleMessageEllipsis also handles initial layout: toggleMessageEllipsis(mAddMsg); mAdminWarning = (TextView) findViewById(R.id.admin_warning); mAdminPolicies = (ViewGroup) findViewById(R.id.admin_policies); mCancelButton = (Button) findViewById(R.id.cancel_button); mCancelButton.setOnClickListener(new View.OnClickListener() { public void onClick(View v) { EventLog.writeEvent(EventLogTags.EXP_DET_DEVICE_ADMIN_DECLINED_BY_USER, mDeviceAdmin.getActivityInfo().applicationInfo.uid); finish(); } }); mActionButton = (Button) findViewById(R.id.action_button); mActionButton.setOnClickListener(new View.OnClickListener() { public void onClick(View v) { if (mAdding) { addAndFinish(); } else if (!mWaitingForRemoveMsg) { try { // Don't allow the admin to put a dialog up in front // of us while we interact with the user. ActivityManagerNative.getDefault().stopAppSwitches(); } catch (RemoteException e) { } mWaitingForRemoveMsg = true; mDPM.getRemoveWarning(mDeviceAdmin.getComponent(), new RemoteCallback(mHandler) { @Override protected void onResult(Bundle bundle) { CharSequence msg = bundle != null ? bundle.getCharSequence( DeviceAdminReceiver.EXTRA_DISABLE_WARNING) : null; continueRemoveAction(msg); } }); // Don't want to wait too long. getWindow().getDecorView().getHandler().postDelayed(new Runnable() { @Override public void run() { continueRemoveAction(null); } }, 2*1000); } } }); } void addAndFinish() { try { mDPM.setActiveAdmin(mDeviceAdmin.getComponent(), mRefreshing); EventLog.writeEvent(EventLogTags.EXP_DET_DEVICE_ADMIN_ACTIVATED_BY_USER, mDeviceAdmin.getActivityInfo().applicationInfo.uid); setResult(Activity.RESULT_OK); } catch (RuntimeException e) { // Something bad happened... could be that it was // already set, though. Log.w(TAG, "Exception trying to activate admin " + mDeviceAdmin.getComponent(), e); if (mDPM.isAdminActive(mDeviceAdmin.getComponent())) { setResult(Activity.RESULT_OK); } } if (mAddingProfileOwner) { try { mDPM.setProfileOwner(mDeviceAdmin.getComponent(), mProfileOwnerName, UserHandle.myUserId()); } catch (RuntimeException re) { setResult(Activity.RESULT_CANCELED); } } finish(); } void continueRemoveAction(CharSequence msg) { if (!mWaitingForRemoveMsg) { return; } mWaitingForRemoveMsg = false; if (msg == null) { try { ActivityManagerNative.getDefault().resumeAppSwitches(); } catch (RemoteException e) { } mDPM.removeActiveAdmin(mDeviceAdmin.getComponent()); finish(); } else { try { // Continue preventing anything from coming in front. ActivityManagerNative.getDefault().stopAppSwitches(); } catch (RemoteException e) { } Bundle args = new Bundle(); args.putCharSequence( DeviceAdminReceiver.EXTRA_DISABLE_WARNING, msg); showDialog(DIALOG_WARNING, args); } } @Override protected void onResume() { super.onResume(); updateInterface(); // As long as we are running, don't let this admin overlay stuff on top of the screen. final int uid = mDeviceAdmin.getActivityInfo().applicationInfo.uid; final String pkg = mDeviceAdmin.getActivityInfo().applicationInfo.packageName; mCurSysAppOpMode = mAppOps.checkOp(AppOpsManager.OP_SYSTEM_ALERT_WINDOW, uid, pkg); mCurToastAppOpMode = mAppOps.checkOp(AppOpsManager.OP_TOAST_WINDOW, uid, pkg); mAppOps.setMode(AppOpsManager.OP_SYSTEM_ALERT_WINDOW, uid, pkg, AppOpsManager.MODE_IGNORED); mAppOps.setMode(AppOpsManager.OP_TOAST_WINDOW, uid, pkg, AppOpsManager.MODE_IGNORED); } @Override protected void onPause() { super.onPause(); // As long as we are running, don't let this admin overlay stuff on top of the screen. final int uid = mDeviceAdmin.getActivityInfo().applicationInfo.uid; final String pkg = mDeviceAdmin.getActivityInfo().applicationInfo.packageName; mAppOps.setMode(AppOpsManager.OP_SYSTEM_ALERT_WINDOW, uid, pkg, mCurSysAppOpMode); mAppOps.setMode(AppOpsManager.OP_TOAST_WINDOW, uid, pkg, mCurToastAppOpMode); try { ActivityManagerNative.getDefault().resumeAppSwitches(); } catch (RemoteException e) { } } @Override protected Dialog onCreateDialog(int id, Bundle args) { switch (id) { case DIALOG_WARNING: { CharSequence msg = args.getCharSequence(DeviceAdminReceiver.EXTRA_DISABLE_WARNING); AlertDialog.Builder builder = new AlertDialog.Builder( DeviceAdminAdd.this); builder.setMessage(msg); builder.setPositiveButton(R.string.dlg_ok, new DialogInterface.OnClickListener() { public void onClick(DialogInterface dialog, int which) { try { ActivityManagerNative.getDefault().resumeAppSwitches(); } catch (RemoteException e) { } mDPM.removeActiveAdmin(mDeviceAdmin.getComponent()); finish(); } }); builder.setNegativeButton(R.string.dlg_cancel, null); return builder.create(); } default: return super.onCreateDialog(id, args); } } static void setViewVisibility(ArrayList views, int visibility) { final int N = views.size(); for (int i=0; i policies = mDeviceAdmin.getUsedPolicies(); for (int i=0; i policies = mDeviceAdmin.getUsedPolicies(); for (int i=0; i d.getWidth() ? MAX_ADD_MSG_LINES_PORTRAIT : MAX_ADD_MSG_LINES_LANDSCAPE; } }