| /* |
| * 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.ViewTreeObserver; |
| 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; |
| |
| boolean mAdding; |
| boolean mRefreshing; |
| boolean mWaitingForRemoveMsg; |
| boolean mAddingProfileOwner; |
| boolean mAdminPoliciesInitialized; |
| 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<ResolveInfo> 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<count; i++) { |
| ResolveInfo ri = avail.get(i); |
| if (ai.packageName.equals(ri.activityInfo.packageName) |
| && ai.name.equals(ri.activityInfo.name)) { |
| try { |
| // We didn't retrieve the meta data for all possible matches, so |
| // need to use the activity info of this specific one that was retrieved. |
| ri.activityInfo = ai; |
| DeviceAdminInfo dpi = new DeviceAdminInfo(this, ri); |
| found = true; |
| } catch (XmlPullParserException e) { |
| Log.w(TAG, "Bad " + ri.activityInfo, e); |
| } catch (IOException e) { |
| Log.w(TAG, "Bad " + ri.activityInfo, e); |
| } |
| break; |
| } |
| } |
| if (!found) { |
| Log.w(TAG, "Request to add invalid device admin: " + who); |
| finish(); |
| return; |
| } |
| } |
| |
| ResolveInfo ri = new ResolveInfo(); |
| ri.activityInfo = ai; |
| try { |
| mDeviceAdmin = new DeviceAdminInfo(this, ri); |
| } catch (XmlPullParserException e) { |
| Log.w(TAG, "Unable to retrieve device policy " + who, e); |
| finish(); |
| return; |
| } catch (IOException e) { |
| Log.w(TAG, "Unable to retrieve device policy " + who, e); |
| finish(); |
| return; |
| } |
| |
| // This admin already exists, an we have two options at this point. If new policy |
| // bits are set, show the user the new list. If nothing has changed, simply return |
| // "OK" immediately. |
| if (DevicePolicyManager.ACTION_ADD_DEVICE_ADMIN.equals(getIntent().getAction())) { |
| mRefreshing = false; |
| if (mDPM.isAdminActive(who)) { |
| ArrayList<DeviceAdminInfo.PolicyInfo> 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); |
| final View.OnClickListener onClickListener = new View.OnClickListener() { |
| @Override |
| public void onClick(View v) { |
| toggleMessageEllipsis(mAddMsg); |
| } |
| }; |
| mAddMsgExpander.setOnClickListener(onClickListener); |
| mAddMsg.setOnClickListener(onClickListener); |
| |
| // Determine whether the message can be collapsed - getLineCount() gives the correct |
| // number of lines only after a layout pass. |
| mAddMsg.getViewTreeObserver().addOnGlobalLayoutListener( |
| new ViewTreeObserver.OnGlobalLayoutListener() { |
| @Override |
| public void onGlobalLayout() { |
| final int maxLines = getEllipsizedLines(); |
| // hide the icon if number of visible lines does not exceed maxLines |
| boolean hideMsgExpander = mAddMsg.getLineCount() <= maxLines; |
| mAddMsgExpander.setVisibility(hideMsgExpander ? View.GONE : View.VISIBLE); |
| if (hideMsgExpander) { |
| mAddMsg.setOnClickListener(null); |
| ((View)mAddMsgExpander.getParent()).invalidate(); |
| } |
| mAddMsg.getViewTreeObserver().removeOnGlobalLayoutListener(this); |
| } |
| }); |
| |
| // 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.setFilterTouchesWhenObscured(true); |
| 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.setFilterTouchesWhenObscured(true); |
| 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(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); |
| |
| } |
| } |
| |
| void updateInterface() { |
| mAdminIcon.setImageDrawable(mDeviceAdmin.loadIcon(getPackageManager())); |
| mAdminName.setText(mDeviceAdmin.loadLabel(getPackageManager())); |
| try { |
| mAdminDescription.setText( |
| mDeviceAdmin.loadDescription(getPackageManager())); |
| mAdminDescription.setVisibility(View.VISIBLE); |
| } catch (Resources.NotFoundException e) { |
| mAdminDescription.setVisibility(View.GONE); |
| } |
| if (mAddingProfileOwner) { |
| mProfileOwnerWarning.setVisibility(View.VISIBLE); |
| } |
| if (mAddMsgText != null) { |
| mAddMsg.setText(mAddMsgText); |
| mAddMsg.setVisibility(View.VISIBLE); |
| } else { |
| mAddMsg.setVisibility(View.GONE); |
| mAddMsgExpander.setVisibility(View.GONE); |
| } |
| if (!mRefreshing && !mAddingProfileOwner |
| && mDPM.isAdminActive(mDeviceAdmin.getComponent())) { |
| addDeviceAdminPolicies(false /* showDescription */); |
| mAdminWarning.setText(getString(R.string.device_admin_status, |
| mDeviceAdmin.getActivityInfo().applicationInfo.loadLabel(getPackageManager()))); |
| setTitle(getText(R.string.active_device_admin_msg)); |
| mActionButton.setText(getText(R.string.remove_device_admin)); |
| mAdding = false; |
| } else { |
| addDeviceAdminPolicies(true /* showDescription */); |
| mAdminWarning.setText(getString(R.string.device_admin_warning, |
| mDeviceAdmin.getActivityInfo().applicationInfo.loadLabel(getPackageManager()))); |
| if (mAddingProfileOwner) { |
| setTitle(getText(R.string.profile_owner_add_title)); |
| } else { |
| setTitle(getText(R.string.add_device_admin_msg)); |
| } |
| mActionButton.setText(getText(R.string.add_device_admin)); |
| mAdding = true; |
| } |
| } |
| |
| private void addDeviceAdminPolicies(boolean showDescription) { |
| if (!mAdminPoliciesInitialized) { |
| boolean isOwner = UserHandle.getCallingUserHandle().isOwner(); |
| for (DeviceAdminInfo.PolicyInfo pi : mDeviceAdmin.getUsedPolicies()) { |
| int descriptionId = isOwner ? pi.description : pi.descriptionForSecondaryUsers; |
| int labelId = isOwner ? pi.label : pi.labelForSecondaryUsers; |
| View view = AppSecurityPermissions.getPermissionItemView(this, getText(labelId), |
| showDescription ? getText(descriptionId) : "", true); |
| mAdminPolicies.addView(view); |
| } |
| mAdminPoliciesInitialized = true; |
| } |
| } |
| |
| void toggleMessageEllipsis(View v) { |
| TextView tv = (TextView) v; |
| |
| mAddMsgEllipsized = ! mAddMsgEllipsized; |
| tv.setEllipsize(mAddMsgEllipsized ? TruncateAt.END : null); |
| tv.setMaxLines(mAddMsgEllipsized ? getEllipsizedLines() : MAX_ADD_MSG_LINES); |
| |
| mAddMsgExpander.setImageResource(mAddMsgEllipsized ? |
| com.android.internal.R.drawable.expander_ic_minimized : |
| com.android.internal.R.drawable.expander_ic_maximized); |
| } |
| |
| int getEllipsizedLines() { |
| Display d = ((WindowManager) getSystemService(Context.WINDOW_SERVICE)) |
| .getDefaultDisplay(); |
| |
| return d.getHeight() > d.getWidth() ? |
| MAX_ADD_MSG_LINES_PORTRAIT : MAX_ADD_MSG_LINES_LANDSCAPE; |
| } |
| |
| } |