Home | History | Annotate | Download | only in settings
      1 /*
      2  * Copyright (C) 2010 The Android Open Source Project
      3  *
      4  * Licensed under the Apache License, Version 2.0 (the "License");
      5  * you may not use this file except in compliance with the License.
      6  * You may obtain a copy of the License at
      7  *
      8  *      http://www.apache.org/licenses/LICENSE-2.0
      9  *
     10  * Unless required by applicable law or agreed to in writing, software
     11  * distributed under the License is distributed on an "AS IS" BASIS,
     12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     13  * See the License for the specific language governing permissions and
     14  * limitations under the License.
     15  */
     16 
     17 package com.android.settings;
     18 
     19 import android.app.Activity;
     20 import android.app.ActivityManager;
     21 import android.app.AlertDialog;
     22 import android.app.AppOpsManager;
     23 import android.app.Dialog;
     24 import android.app.admin.DeviceAdminInfo;
     25 import android.app.admin.DeviceAdminReceiver;
     26 import android.app.admin.DevicePolicyManager;
     27 import android.content.ComponentName;
     28 import android.content.Context;
     29 import android.content.DialogInterface;
     30 import android.content.Intent;
     31 import android.content.pm.ActivityInfo;
     32 import android.content.pm.ApplicationInfo;
     33 import android.content.pm.PackageInfo;
     34 import android.content.pm.PackageManager;
     35 import android.content.pm.PackageManager.NameNotFoundException;
     36 import android.content.pm.ResolveInfo;
     37 import android.content.pm.UserInfo;
     38 import android.content.res.Resources;
     39 import android.os.Binder;
     40 import android.os.Bundle;
     41 import android.os.Handler;
     42 import android.os.IBinder;
     43 import android.os.RemoteCallback;
     44 import android.os.RemoteException;
     45 import android.os.UserHandle;
     46 import android.os.UserManager;
     47 import android.text.TextUtils;
     48 import android.text.TextUtils.TruncateAt;
     49 import android.util.EventLog;
     50 import android.util.Log;
     51 import android.view.Display;
     52 import android.view.View;
     53 import android.view.ViewGroup;
     54 import android.view.ViewTreeObserver;
     55 import android.view.WindowManager;
     56 import android.widget.AppSecurityPermissions;
     57 import android.widget.Button;
     58 import android.widget.ImageView;
     59 import android.widget.TextView;
     60 
     61 import com.android.internal.logging.nano.MetricsProto;
     62 import com.android.settings.fuelgauge.BatteryUtils;
     63 import com.android.settings.overlay.FeatureFactory;
     64 import com.android.settings.users.UserDialogs;
     65 import com.android.settingslib.RestrictedLockUtils;
     66 import com.android.settingslib.RestrictedLockUtils.EnforcedAdmin;
     67 
     68 import org.xmlpull.v1.XmlPullParserException;
     69 
     70 import java.io.IOException;
     71 import java.util.ArrayList;
     72 import java.util.List;
     73 import java.util.Optional;
     74 
     75 public class DeviceAdminAdd extends Activity {
     76     static final String TAG = "DeviceAdminAdd";
     77 
     78     static final int DIALOG_WARNING = 1;
     79 
     80     private static final int MAX_ADD_MSG_LINES_PORTRAIT = 5;
     81     private static final int MAX_ADD_MSG_LINES_LANDSCAPE = 2;
     82     private static final int MAX_ADD_MSG_LINES = 15;
     83 
     84     /**
     85      * Optional key to map to the package name of the Device Admin.
     86      * Currently only used when uninstalling an active device admin.
     87      */
     88     public static final String EXTRA_DEVICE_ADMIN_PACKAGE_NAME =
     89             "android.app.extra.DEVICE_ADMIN_PACKAGE_NAME";
     90 
     91     public static final String EXTRA_CALLED_FROM_SUPPORT_DIALOG =
     92             "android.app.extra.CALLED_FROM_SUPPORT_DIALOG";
     93 
     94     private final IBinder mToken = new Binder();
     95     Handler mHandler;
     96 
     97     DevicePolicyManager mDPM;
     98     AppOpsManager mAppOps;
     99     DeviceAdminInfo mDeviceAdmin;
    100     CharSequence mAddMsgText;
    101     String mProfileOwnerName;
    102 
    103     ImageView mAdminIcon;
    104     TextView mAdminName;
    105     TextView mAdminDescription;
    106     TextView mAddMsg;
    107     TextView mProfileOwnerWarning;
    108     ImageView mAddMsgExpander;
    109     boolean mAddMsgEllipsized = true;
    110     TextView mAdminWarning;
    111     TextView mSupportMessage;
    112     ViewGroup mAdminPolicies;
    113     Button mActionButton;
    114     Button mUninstallButton;
    115     Button mCancelButton;
    116 
    117     boolean mUninstalling = false;
    118     boolean mAdding;
    119     boolean mRefreshing;
    120     boolean mWaitingForRemoveMsg;
    121     boolean mAddingProfileOwner;
    122     boolean mAdminPoliciesInitialized;
    123 
    124     boolean mIsCalledFromSupportDialog = false;
    125 
    126     @Override
    127     protected void onCreate(Bundle icicle) {
    128         super.onCreate(icicle);
    129 
    130         mHandler = new Handler(getMainLooper());
    131 
    132         mDPM = (DevicePolicyManager)getSystemService(Context.DEVICE_POLICY_SERVICE);
    133         mAppOps = (AppOpsManager)getSystemService(Context.APP_OPS_SERVICE);
    134         PackageManager packageManager = getPackageManager();
    135 
    136         if ((getIntent().getFlags()&Intent.FLAG_ACTIVITY_NEW_TASK) != 0) {
    137             Log.w(TAG, "Cannot start ADD_DEVICE_ADMIN as a new task");
    138             finish();
    139             return;
    140         }
    141 
    142         mIsCalledFromSupportDialog = getIntent().getBooleanExtra(
    143                 EXTRA_CALLED_FROM_SUPPORT_DIALOG, false);
    144 
    145         String action = getIntent().getAction();
    146         ComponentName who = (ComponentName)getIntent().getParcelableExtra(
    147                 DevicePolicyManager.EXTRA_DEVICE_ADMIN);
    148         if (who == null) {
    149             String packageName = getIntent().getStringExtra(EXTRA_DEVICE_ADMIN_PACKAGE_NAME);
    150             Optional<ComponentName> installedAdmin = findAdminWithPackageName(packageName);
    151             if (!installedAdmin.isPresent()) {
    152                 Log.w(TAG, "No component specified in " + action);
    153                 finish();
    154                 return;
    155             }
    156             who = installedAdmin.get();
    157             mUninstalling = true;
    158         }
    159 
    160         if (action != null && action.equals(DevicePolicyManager.ACTION_SET_PROFILE_OWNER)) {
    161             setResult(RESULT_CANCELED);
    162             setFinishOnTouchOutside(true);
    163             mAddingProfileOwner = true;
    164             mProfileOwnerName =
    165                     getIntent().getStringExtra(DevicePolicyManager.EXTRA_PROFILE_OWNER_NAME);
    166             String callingPackage = getCallingPackage();
    167             if (callingPackage == null || !callingPackage.equals(who.getPackageName())) {
    168                 Log.e(TAG, "Unknown or incorrect caller");
    169                 finish();
    170                 return;
    171             }
    172             try {
    173                 PackageInfo packageInfo = packageManager.getPackageInfo(callingPackage, 0);
    174                 if ((packageInfo.applicationInfo.flags & ApplicationInfo.FLAG_SYSTEM) == 0) {
    175                     Log.e(TAG, "Cannot set a non-system app as a profile owner");
    176                     finish();
    177                     return;
    178                 }
    179             } catch (NameNotFoundException nnfe) {
    180                 Log.e(TAG, "Cannot find the package " + callingPackage);
    181                 finish();
    182                 return;
    183             }
    184         }
    185 
    186         ActivityInfo ai;
    187         try {
    188             ai = packageManager.getReceiverInfo(who, PackageManager.GET_META_DATA);
    189         } catch (PackageManager.NameNotFoundException e) {
    190             Log.w(TAG, "Unable to retrieve device policy " + who, e);
    191             finish();
    192             return;
    193         }
    194 
    195         // When activating, make sure the given component name is actually a valid device admin.
    196         // No need to check this when deactivating, because it is safe to deactivate an active
    197         // invalid device admin.
    198         if (!mDPM.isAdminActive(who)) {
    199             List<ResolveInfo> avail = packageManager.queryBroadcastReceivers(
    200                     new Intent(DeviceAdminReceiver.ACTION_DEVICE_ADMIN_ENABLED),
    201                     PackageManager.GET_DISABLED_UNTIL_USED_COMPONENTS);
    202             int count = avail == null ? 0 : avail.size();
    203             boolean found = false;
    204             for (int i=0; i<count; i++) {
    205                 ResolveInfo ri = avail.get(i);
    206                 if (ai.packageName.equals(ri.activityInfo.packageName)
    207                         && ai.name.equals(ri.activityInfo.name)) {
    208                     try {
    209                         // We didn't retrieve the meta data for all possible matches, so
    210                         // need to use the activity info of this specific one that was retrieved.
    211                         ri.activityInfo = ai;
    212                         DeviceAdminInfo dpi = new DeviceAdminInfo(this, ri);
    213                         found = true;
    214                     } catch (XmlPullParserException e) {
    215                         Log.w(TAG, "Bad " + ri.activityInfo, e);
    216                     } catch (IOException e) {
    217                         Log.w(TAG, "Bad " + ri.activityInfo, e);
    218                     }
    219                     break;
    220                 }
    221             }
    222             if (!found) {
    223                 Log.w(TAG, "Request to add invalid device admin: " + who);
    224                 finish();
    225                 return;
    226             }
    227         }
    228 
    229         ResolveInfo ri = new ResolveInfo();
    230         ri.activityInfo = ai;
    231         try {
    232             mDeviceAdmin = new DeviceAdminInfo(this, ri);
    233         } catch (XmlPullParserException e) {
    234             Log.w(TAG, "Unable to retrieve device policy " + who, e);
    235             finish();
    236             return;
    237         } catch (IOException e) {
    238             Log.w(TAG, "Unable to retrieve device policy " + who, e);
    239             finish();
    240             return;
    241         }
    242 
    243         // This admin already exists, an we have two options at this point.  If new policy
    244         // bits are set, show the user the new list.  If nothing has changed, simply return
    245         // "OK" immediately.
    246         if (DevicePolicyManager.ACTION_ADD_DEVICE_ADMIN.equals(getIntent().getAction())) {
    247             mRefreshing = false;
    248             if (mDPM.isAdminActive(who)) {
    249                 if (mDPM.isRemovingAdmin(who, android.os.Process.myUserHandle().getIdentifier())) {
    250                     Log.w(TAG, "Requested admin is already being removed: " + who);
    251                     finish();
    252                     return;
    253                 }
    254 
    255                 ArrayList<DeviceAdminInfo.PolicyInfo> newPolicies = mDeviceAdmin.getUsedPolicies();
    256                 for (int i = 0; i < newPolicies.size(); i++) {
    257                     DeviceAdminInfo.PolicyInfo pi = newPolicies.get(i);
    258                     if (!mDPM.hasGrantedPolicy(who, pi.ident)) {
    259                         mRefreshing = true;
    260                         break;
    261                     }
    262                 }
    263                 if (!mRefreshing) {
    264                     // Nothing changed (or policies were removed) - return immediately
    265                     setResult(Activity.RESULT_OK);
    266                     finish();
    267                     return;
    268                 }
    269             }
    270         }
    271 
    272         // If we're trying to add a profile owner and user setup hasn't completed yet, no
    273         // need to prompt for permission. Just add and finish.
    274         if (mAddingProfileOwner && !mDPM.hasUserSetupCompleted()) {
    275             addAndFinish();
    276             return;
    277         }
    278 
    279         mAddMsgText = getIntent().getCharSequenceExtra(DevicePolicyManager.EXTRA_ADD_EXPLANATION);
    280 
    281         setContentView(R.layout.device_admin_add);
    282 
    283         mAdminIcon = (ImageView)findViewById(R.id.admin_icon);
    284         mAdminName = (TextView)findViewById(R.id.admin_name);
    285         mAdminDescription = (TextView)findViewById(R.id.admin_description);
    286         mProfileOwnerWarning = (TextView) findViewById(R.id.profile_owner_warning);
    287 
    288         mAddMsg = (TextView)findViewById(R.id.add_msg);
    289         mAddMsgExpander = (ImageView) findViewById(R.id.add_msg_expander);
    290         final View.OnClickListener onClickListener = new View.OnClickListener() {
    291             @Override
    292             public void onClick(View v) {
    293                 toggleMessageEllipsis(mAddMsg);
    294             }
    295         };
    296         mAddMsgExpander.setOnClickListener(onClickListener);
    297         mAddMsg.setOnClickListener(onClickListener);
    298 
    299         // Determine whether the message can be collapsed - getLineCount() gives the correct
    300         // number of lines only after a layout pass.
    301         mAddMsg.getViewTreeObserver().addOnGlobalLayoutListener(
    302                 new ViewTreeObserver.OnGlobalLayoutListener() {
    303                     @Override
    304                     public void onGlobalLayout() {
    305                         final int maxLines = getEllipsizedLines();
    306                         // hide the icon if number of visible lines does not exceed maxLines
    307                         boolean hideMsgExpander = mAddMsg.getLineCount() <= maxLines;
    308                         mAddMsgExpander.setVisibility(hideMsgExpander ? View.GONE : View.VISIBLE);
    309                         if (hideMsgExpander) {
    310                             mAddMsg.setOnClickListener(null);
    311                             ((View)mAddMsgExpander.getParent()).invalidate();
    312                         }
    313                         mAddMsg.getViewTreeObserver().removeOnGlobalLayoutListener(this);
    314                     }
    315                 });
    316 
    317         // toggleMessageEllipsis also handles initial layout:
    318         toggleMessageEllipsis(mAddMsg);
    319 
    320         mAdminWarning = (TextView) findViewById(R.id.admin_warning);
    321         mAdminPolicies = (ViewGroup) findViewById(R.id.admin_policies);
    322         mSupportMessage = (TextView) findViewById(R.id.admin_support_message);
    323 
    324         mCancelButton = (Button) findViewById(R.id.cancel_button);
    325         mCancelButton.setFilterTouchesWhenObscured(true);
    326         mCancelButton.setOnClickListener(new View.OnClickListener() {
    327             public void onClick(View v) {
    328                 EventLog.writeEvent(EventLogTags.EXP_DET_DEVICE_ADMIN_DECLINED_BY_USER,
    329                     mDeviceAdmin.getActivityInfo().applicationInfo.uid);
    330                 finish();
    331             }
    332         });
    333 
    334         mUninstallButton = (Button) findViewById(R.id.uninstall_button);
    335         mUninstallButton.setFilterTouchesWhenObscured(true);
    336         mUninstallButton.setOnClickListener(new View.OnClickListener() {
    337             public void onClick(View v) {
    338                 EventLog.writeEvent(EventLogTags.EXP_DET_DEVICE_ADMIN_UNINSTALLED_BY_USER,
    339                         mDeviceAdmin.getActivityInfo().applicationInfo.uid);
    340                 mDPM.uninstallPackageWithActiveAdmins(mDeviceAdmin.getPackageName());
    341                 finish();
    342             }
    343         });
    344 
    345         mActionButton = (Button) findViewById(R.id.action_button);
    346 
    347         final View restrictedAction = findViewById(R.id.restricted_action);
    348         restrictedAction.setFilterTouchesWhenObscured(true);
    349         restrictedAction.setOnClickListener(new View.OnClickListener() {
    350             public void onClick(View v) {
    351                 if (!mActionButton.isEnabled()) {
    352                     showPolicyTransparencyDialogIfRequired();
    353                     return;
    354                 }
    355                 if (mAdding) {
    356                     addAndFinish();
    357                 } else if (isManagedProfile(mDeviceAdmin)
    358                         && mDeviceAdmin.getComponent().equals(mDPM.getProfileOwner())) {
    359                     final int userId = UserHandle.myUserId();
    360                     UserDialogs.createRemoveDialog(DeviceAdminAdd.this, userId,
    361                             new DialogInterface.OnClickListener() {
    362                                 @Override
    363                                 public void onClick(DialogInterface dialog, int which) {
    364                                     UserManager um = UserManager.get(DeviceAdminAdd.this);
    365                                     um.removeUser(userId);
    366                                     finish();
    367                                 }
    368                             }
    369                     ).show();
    370                 } else if (mUninstalling) {
    371                     mDPM.uninstallPackageWithActiveAdmins(mDeviceAdmin.getPackageName());
    372                     finish();
    373                 } else if (!mWaitingForRemoveMsg) {
    374                     try {
    375                         // Don't allow the admin to put a dialog up in front
    376                         // of us while we interact with the user.
    377                         ActivityManager.getService().stopAppSwitches();
    378                     } catch (RemoteException e) {
    379                     }
    380                     mWaitingForRemoveMsg = true;
    381                     mDPM.getRemoveWarning(mDeviceAdmin.getComponent(),
    382                             new RemoteCallback(new RemoteCallback.OnResultListener() {
    383                                 @Override
    384                                 public void onResult(Bundle result) {
    385                                     CharSequence msg = result != null
    386                                             ? result.getCharSequence(
    387                                             DeviceAdminReceiver.EXTRA_DISABLE_WARNING)
    388                                             : null;
    389                                     continueRemoveAction(msg);
    390                                 }
    391                             }, mHandler));
    392                     // Don't want to wait too long.
    393                     getWindow().getDecorView().getHandler().postDelayed(new Runnable() {
    394                         @Override public void run() {
    395                             continueRemoveAction(null);
    396                         }
    397                     }, 2*1000);
    398                 }
    399             }
    400         });
    401     }
    402 
    403     /**
    404      * Shows a dialog to explain why the button is disabled if required.
    405      */
    406     private void showPolicyTransparencyDialogIfRequired() {
    407         if (isManagedProfile(mDeviceAdmin)
    408                 && mDeviceAdmin.getComponent().equals(mDPM.getProfileOwner())) {
    409             if (hasBaseCantRemoveProfileRestriction()) {
    410                 // If DISALLOW_REMOVE_MANAGED_PROFILE is set by the system, there's no
    411                 // point showing a dialog saying it's disabled by an admin.
    412                 return;
    413             }
    414             EnforcedAdmin enforcedAdmin = getAdminEnforcingCantRemoveProfile();
    415             if (enforcedAdmin != null) {
    416                 RestrictedLockUtils.sendShowAdminSupportDetailsIntent(
    417                         DeviceAdminAdd.this,
    418                         enforcedAdmin);
    419             }
    420         }
    421     }
    422 
    423     void addAndFinish() {
    424         try {
    425             logSpecialPermissionChange(true, mDeviceAdmin.getComponent().getPackageName());
    426             mDPM.setActiveAdmin(mDeviceAdmin.getComponent(), mRefreshing);
    427             EventLog.writeEvent(EventLogTags.EXP_DET_DEVICE_ADMIN_ACTIVATED_BY_USER,
    428                 mDeviceAdmin.getActivityInfo().applicationInfo.uid);
    429 
    430             unrestrictAppIfPossible(BatteryUtils.getInstance(this));
    431 
    432             setResult(Activity.RESULT_OK);
    433         } catch (RuntimeException e) {
    434             // Something bad happened...  could be that it was
    435             // already set, though.
    436             Log.w(TAG, "Exception trying to activate admin "
    437                     + mDeviceAdmin.getComponent(), e);
    438             if (mDPM.isAdminActive(mDeviceAdmin.getComponent())) {
    439                 setResult(Activity.RESULT_OK);
    440             }
    441         }
    442         if (mAddingProfileOwner) {
    443             try {
    444                 mDPM.setProfileOwner(mDeviceAdmin.getComponent(),
    445                         mProfileOwnerName, UserHandle.myUserId());
    446             } catch (RuntimeException re) {
    447                 setResult(Activity.RESULT_CANCELED);
    448             }
    449         }
    450         finish();
    451     }
    452 
    453     void unrestrictAppIfPossible(BatteryUtils batteryUtils) {
    454         // Unrestrict admin app if it is already been restricted
    455         final String packageName = mDeviceAdmin.getComponent().getPackageName();
    456         final int uid = batteryUtils.getPackageUid(packageName);
    457         if (batteryUtils.isForceAppStandbyEnabled(uid, packageName)) {
    458             batteryUtils.setForceAppStandby(uid, packageName, AppOpsManager.MODE_ALLOWED);
    459         }
    460     }
    461 
    462     void continueRemoveAction(CharSequence msg) {
    463         if (!mWaitingForRemoveMsg) {
    464             return;
    465         }
    466         mWaitingForRemoveMsg = false;
    467         if (msg == null) {
    468             try {
    469                 ActivityManager.getService().resumeAppSwitches();
    470             } catch (RemoteException e) {
    471             }
    472             logSpecialPermissionChange(false, mDeviceAdmin.getComponent().getPackageName());
    473             mDPM.removeActiveAdmin(mDeviceAdmin.getComponent());
    474             finish();
    475         } else {
    476             try {
    477                 // Continue preventing anything from coming in front.
    478                 ActivityManager.getService().stopAppSwitches();
    479             } catch (RemoteException e) {
    480             }
    481             Bundle args = new Bundle();
    482             args.putCharSequence(
    483                     DeviceAdminReceiver.EXTRA_DISABLE_WARNING, msg);
    484             showDialog(DIALOG_WARNING, args);
    485         }
    486     }
    487 
    488     void logSpecialPermissionChange(boolean allow, String packageName) {
    489         int logCategory = allow ? MetricsProto.MetricsEvent.APP_SPECIAL_PERMISSION_ADMIN_ALLOW :
    490                 MetricsProto.MetricsEvent.APP_SPECIAL_PERMISSION_ADMIN_DENY;
    491         FeatureFactory.getFactory(this).getMetricsFeatureProvider().action(this, logCategory, packageName);
    492     }
    493 
    494     @Override
    495     protected void onResume() {
    496         super.onResume();
    497         mActionButton.setEnabled(true);
    498         updateInterface();
    499         // As long as we are running, don't let anyone overlay stuff on top of the screen.
    500         mAppOps.setUserRestriction(AppOpsManager.OP_SYSTEM_ALERT_WINDOW, true, mToken);
    501         mAppOps.setUserRestriction(AppOpsManager.OP_TOAST_WINDOW, true, mToken);
    502 
    503     }
    504 
    505     @Override
    506     protected void onPause() {
    507         super.onPause();
    508         // This just greys out the button. The actual listener is attached to R.id.restricted_action
    509         mActionButton.setEnabled(false);
    510         mAppOps.setUserRestriction(AppOpsManager.OP_SYSTEM_ALERT_WINDOW, false, mToken);
    511         mAppOps.setUserRestriction(AppOpsManager.OP_TOAST_WINDOW, false, mToken);
    512         try {
    513             ActivityManager.getService().resumeAppSwitches();
    514         } catch (RemoteException e) {
    515         }
    516     }
    517 
    518     @Override
    519     protected void onUserLeaveHint() {
    520         super.onUserLeaveHint();
    521         // In case this is triggered from support dialog, finish this activity once the user leaves
    522         // so that this won't appear as a background next time support dialog is triggered. This
    523         // is because the support dialog activity and this belong to the same task and we can't
    524         // start this in new activity since we need to know the calling package in this activity.
    525         if (mIsCalledFromSupportDialog) {
    526             finish();
    527         }
    528     }
    529 
    530     @Override
    531     protected Dialog onCreateDialog(int id, Bundle args) {
    532         switch (id) {
    533             case DIALOG_WARNING: {
    534                 CharSequence msg = args.getCharSequence(DeviceAdminReceiver.EXTRA_DISABLE_WARNING);
    535                 AlertDialog.Builder builder = new AlertDialog.Builder(this);
    536                 builder.setMessage(msg);
    537                 builder.setPositiveButton(R.string.dlg_ok,
    538                         new DialogInterface.OnClickListener() {
    539                     public void onClick(DialogInterface dialog, int which) {
    540                         try {
    541                             ActivityManager.getService().resumeAppSwitches();
    542                         } catch (RemoteException e) {
    543                         }
    544                         mDPM.removeActiveAdmin(mDeviceAdmin.getComponent());
    545                         finish();
    546                     }
    547                 });
    548                 builder.setNegativeButton(R.string.dlg_cancel, null);
    549                 return builder.create();
    550             }
    551             default:
    552                 return super.onCreateDialog(id, args);
    553 
    554         }
    555     }
    556 
    557     void updateInterface() {
    558         findViewById(R.id.restricted_icon).setVisibility(View.GONE);
    559         mAdminIcon.setImageDrawable(mDeviceAdmin.loadIcon(getPackageManager()));
    560         mAdminName.setText(mDeviceAdmin.loadLabel(getPackageManager()));
    561         try {
    562             mAdminDescription.setText(
    563                     mDeviceAdmin.loadDescription(getPackageManager()));
    564             mAdminDescription.setVisibility(View.VISIBLE);
    565         } catch (Resources.NotFoundException e) {
    566             mAdminDescription.setVisibility(View.GONE);
    567         }
    568         if (mAddingProfileOwner) {
    569             mProfileOwnerWarning.setVisibility(View.VISIBLE);
    570         }
    571         if (mAddMsgText != null) {
    572             mAddMsg.setText(mAddMsgText);
    573             mAddMsg.setVisibility(View.VISIBLE);
    574         } else {
    575             mAddMsg.setVisibility(View.GONE);
    576             mAddMsgExpander.setVisibility(View.GONE);
    577         }
    578         if (!mRefreshing && !mAddingProfileOwner
    579                 && mDPM.isAdminActive(mDeviceAdmin.getComponent())) {
    580             mAdding = false;
    581             final boolean isProfileOwner =
    582                     mDeviceAdmin.getComponent().equals(mDPM.getProfileOwner());
    583             final boolean isManagedProfile = isManagedProfile(mDeviceAdmin);
    584             if (isProfileOwner && isManagedProfile) {
    585                 // Profile owner in a managed profile, user can remove profile to disable admin.
    586                 mAdminWarning.setText(R.string.admin_profile_owner_message);
    587                 mActionButton.setText(R.string.remove_managed_profile_label);
    588 
    589                 final EnforcedAdmin admin = getAdminEnforcingCantRemoveProfile();
    590                 final boolean hasBaseRestriction = hasBaseCantRemoveProfileRestriction();
    591                 if (admin != null && !hasBaseRestriction) {
    592                     findViewById(R.id.restricted_icon).setVisibility(View.VISIBLE);
    593                 }
    594                 mActionButton.setEnabled(admin == null && !hasBaseRestriction);
    595             } else if (isProfileOwner || mDeviceAdmin.getComponent().equals(
    596                             mDPM.getDeviceOwnerComponentOnCallingUser())) {
    597                 // Profile owner in a user or device owner, user can't disable admin.
    598                 if (isProfileOwner) {
    599                     // Show profile owner in a user description.
    600                     mAdminWarning.setText(R.string.admin_profile_owner_user_message);
    601                 } else {
    602                     // Show device owner description.
    603                     mAdminWarning.setText(R.string.admin_device_owner_message);
    604                 }
    605                 mActionButton.setText(R.string.remove_device_admin);
    606                 mActionButton.setEnabled(false);
    607             } else {
    608                 addDeviceAdminPolicies(false /* showDescription */);
    609                 mAdminWarning.setText(getString(R.string.device_admin_status,
    610                         mDeviceAdmin.getActivityInfo().applicationInfo.loadLabel(
    611                         getPackageManager())));
    612                 setTitle(R.string.active_device_admin_msg);
    613                 if (mUninstalling) {
    614                     mActionButton.setText(R.string.remove_and_uninstall_device_admin);
    615                 } else {
    616                     mActionButton.setText(R.string.remove_device_admin);
    617                 }
    618             }
    619             CharSequence supportMessage = mDPM.getLongSupportMessageForUser(
    620                     mDeviceAdmin.getComponent(), UserHandle.myUserId());
    621             if (!TextUtils.isEmpty(supportMessage)) {
    622                 mSupportMessage.setText(supportMessage);
    623                 mSupportMessage.setVisibility(View.VISIBLE);
    624             } else {
    625                 mSupportMessage.setVisibility(View.GONE);
    626             }
    627         } else {
    628             addDeviceAdminPolicies(true /* showDescription */);
    629             mAdminWarning.setText(getString(R.string.device_admin_warning,
    630                     mDeviceAdmin.getActivityInfo().applicationInfo.loadLabel(getPackageManager())));
    631             if (mAddingProfileOwner) {
    632                 setTitle(getText(R.string.profile_owner_add_title));
    633             } else {
    634                 setTitle(getText(R.string.add_device_admin_msg));
    635             }
    636             mActionButton.setText(getText(R.string.add_device_admin));
    637             if (isAdminUninstallable()) {
    638                 mUninstallButton.setVisibility(View.VISIBLE);
    639             }
    640             mSupportMessage.setVisibility(View.GONE);
    641             mAdding = true;
    642         }
    643     }
    644 
    645     private EnforcedAdmin getAdminEnforcingCantRemoveProfile() {
    646         // Removing a managed profile is disallowed if DISALLOW_REMOVE_MANAGED_PROFILE
    647         // is set in the parent rather than the user itself.
    648         return RestrictedLockUtils.checkIfRestrictionEnforced(this,
    649                 UserManager.DISALLOW_REMOVE_MANAGED_PROFILE, getParentUserId());
    650     }
    651 
    652     private boolean hasBaseCantRemoveProfileRestriction() {
    653         return RestrictedLockUtils.hasBaseUserRestriction(this,
    654                 UserManager.DISALLOW_REMOVE_MANAGED_PROFILE, getParentUserId());
    655     }
    656 
    657     private int getParentUserId() {
    658         return UserManager.get(this).getProfileParent(UserHandle.myUserId()).id;
    659     }
    660 
    661     private void addDeviceAdminPolicies(boolean showDescription) {
    662         if (!mAdminPoliciesInitialized) {
    663             boolean isAdminUser = UserManager.get(this).isAdminUser();
    664             for (DeviceAdminInfo.PolicyInfo pi : mDeviceAdmin.getUsedPolicies()) {
    665                 int descriptionId = isAdminUser ? pi.description : pi.descriptionForSecondaryUsers;
    666                 int labelId = isAdminUser ? pi.label : pi.labelForSecondaryUsers;
    667                 View view = AppSecurityPermissions.getPermissionItemView(this, getText(labelId),
    668                         showDescription ? getText(descriptionId) : "", true);
    669                 mAdminPolicies.addView(view);
    670             }
    671             mAdminPoliciesInitialized = true;
    672         }
    673     }
    674 
    675     void toggleMessageEllipsis(View v) {
    676         TextView tv = (TextView) v;
    677 
    678         mAddMsgEllipsized = ! mAddMsgEllipsized;
    679         tv.setEllipsize(mAddMsgEllipsized ? TruncateAt.END : null);
    680         tv.setMaxLines(mAddMsgEllipsized ? getEllipsizedLines() : MAX_ADD_MSG_LINES);
    681 
    682         mAddMsgExpander.setImageResource(mAddMsgEllipsized ?
    683             com.android.internal.R.drawable.expander_ic_minimized :
    684             com.android.internal.R.drawable.expander_ic_maximized);
    685     }
    686 
    687     int getEllipsizedLines() {
    688         Display d = ((WindowManager) getSystemService(Context.WINDOW_SERVICE))
    689                     .getDefaultDisplay();
    690 
    691         return d.getHeight() > d.getWidth() ?
    692             MAX_ADD_MSG_LINES_PORTRAIT : MAX_ADD_MSG_LINES_LANDSCAPE;
    693     }
    694 
    695     /**
    696      * @return true if adminInfo is running in a managed profile.
    697      */
    698     private boolean isManagedProfile(DeviceAdminInfo adminInfo) {
    699         UserManager um = UserManager.get(this);
    700         UserInfo info = um.getUserInfo(
    701                 UserHandle.getUserId(adminInfo.getActivityInfo().applicationInfo.uid));
    702         return info != null ? info.isManagedProfile() : false;
    703     }
    704 
    705     /**
    706      * @return an {@link Optional} containing the admin with a given package name, if it exists,
    707      *         or {@link Optional#empty()} otherwise.
    708      */
    709     private Optional<ComponentName> findAdminWithPackageName(String packageName) {
    710         List<ComponentName> admins = mDPM.getActiveAdmins();
    711         if (admins == null) {
    712             return Optional.empty();
    713         }
    714         return admins.stream().filter(i -> i.getPackageName().equals(packageName)).findAny();
    715     }
    716 
    717     private boolean isAdminUninstallable() {
    718         // System apps can't be uninstalled.
    719         return !mDeviceAdmin.getActivityInfo().applicationInfo.isSystemApp();
    720     }
    721 }
    722