• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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 org.xmlpull.v1.XmlPullParserException;
20 
21 import android.app.Activity;
22 import android.app.ActivityManagerNative;
23 import android.app.AlertDialog;
24 import android.app.Dialog;
25 import android.app.admin.DeviceAdminInfo;
26 import android.app.admin.DeviceAdminReceiver;
27 import android.app.admin.DevicePolicyManager;
28 import android.content.ComponentName;
29 import android.content.Context;
30 import android.content.DialogInterface;
31 import android.content.Intent;
32 import android.content.pm.ActivityInfo;
33 import android.content.pm.PackageManager;
34 import android.content.pm.ResolveInfo;
35 import android.content.res.Resources;
36 import android.os.Bundle;
37 import android.os.Handler;
38 import android.os.RemoteCallback;
39 import android.os.RemoteException;
40 import android.text.TextUtils.TruncateAt;
41 import android.util.Log;
42 import android.view.Display;
43 import android.view.View;
44 import android.view.ViewGroup;
45 import android.view.WindowManager;
46 import android.widget.AppSecurityPermissions;
47 import android.widget.Button;
48 import android.widget.ImageView;
49 import android.widget.TextView;
50 
51 import java.io.IOException;
52 import java.util.ArrayList;
53 import java.util.HashSet;
54 import java.util.List;
55 
56 public class DeviceAdminAdd extends Activity {
57     static final String TAG = "DeviceAdminAdd";
58 
59     static final int DIALOG_WARNING = 1;
60 
61     private static final int MAX_ADD_MSG_LINES_PORTRAIT = 5;
62     private static final int MAX_ADD_MSG_LINES_LANDSCAPE = 2;
63     private static final int MAX_ADD_MSG_LINES = 15;
64 
65     Handler mHandler;
66 
67     DevicePolicyManager mDPM;
68     DeviceAdminInfo mDeviceAdmin;
69     CharSequence mAddMsgText;
70 
71     ImageView mAdminIcon;
72     TextView mAdminName;
73     TextView mAdminDescription;
74     TextView mAddMsg;
75     ImageView mAddMsgExpander;
76     boolean mAddMsgEllipsized = true;
77     TextView mAdminWarning;
78     ViewGroup mAdminPolicies;
79     Button mActionButton;
80     Button mCancelButton;
81 
82     final ArrayList<View> mAddingPolicies = new ArrayList<View>();
83     final ArrayList<View> mActivePolicies = new ArrayList<View>();
84 
85     boolean mAdding;
86     boolean mRefreshing;
87 
88     @Override
onCreate(Bundle icicle)89     protected void onCreate(Bundle icicle) {
90         super.onCreate(icicle);
91 
92         mHandler = new Handler(getMainLooper());
93 
94         mDPM = (DevicePolicyManager)getSystemService(Context.DEVICE_POLICY_SERVICE);
95 
96         if ((getIntent().getFlags()&Intent.FLAG_ACTIVITY_NEW_TASK) != 0) {
97             Log.w(TAG, "Cannot start ADD_DEVICE_ADMIN as a new task");
98             finish();
99             return;
100         }
101 
102         ComponentName cn = (ComponentName)getIntent().getParcelableExtra(
103                 DevicePolicyManager.EXTRA_DEVICE_ADMIN);
104         if (cn == null) {
105             Log.w(TAG, "No component specified in " + getIntent().getAction());
106             finish();
107             return;
108         }
109 
110         ActivityInfo ai;
111         try {
112             ai = getPackageManager().getReceiverInfo(cn, PackageManager.GET_META_DATA);
113         } catch (PackageManager.NameNotFoundException e) {
114             Log.w(TAG, "Unable to retrieve device policy " + cn, e);
115             finish();
116             return;
117         }
118 
119         // When activating, make sure the given component name is actually a valid device admin.
120         // No need to check this when deactivating, because it is safe to deactivate an active
121         // invalid device admin.
122         if (!mDPM.isAdminActive(cn)) {
123             List<ResolveInfo> avail = getPackageManager().queryBroadcastReceivers(
124                     new Intent(DeviceAdminReceiver.ACTION_DEVICE_ADMIN_ENABLED),
125                     PackageManager.GET_DISABLED_UNTIL_USED_COMPONENTS);
126             int count = avail == null ? 0 : avail.size();
127             boolean found = false;
128             for (int i=0; i<count; i++) {
129                 ResolveInfo ri = avail.get(i);
130                 if (ai.packageName.equals(ri.activityInfo.packageName)
131                         && ai.name.equals(ri.activityInfo.name)) {
132                     try {
133                         // We didn't retrieve the meta data for all possible matches, so
134                         // need to use the activity info of this specific one that was retrieved.
135                         ri.activityInfo = ai;
136                         DeviceAdminInfo dpi = new DeviceAdminInfo(this, ri);
137                         found = true;
138                     } catch (XmlPullParserException e) {
139                         Log.w(TAG, "Bad " + ri.activityInfo, e);
140                     } catch (IOException e) {
141                         Log.w(TAG, "Bad " + ri.activityInfo, e);
142                     }
143                     break;
144                 }
145             }
146             if (!found) {
147                 Log.w(TAG, "Request to add invalid device admin: " + cn);
148                 finish();
149                 return;
150             }
151         }
152 
153         ResolveInfo ri = new ResolveInfo();
154         ri.activityInfo = ai;
155         try {
156             mDeviceAdmin = new DeviceAdminInfo(this, ri);
157         } catch (XmlPullParserException e) {
158             Log.w(TAG, "Unable to retrieve device policy " + cn, e);
159             finish();
160             return;
161         } catch (IOException e) {
162             Log.w(TAG, "Unable to retrieve device policy " + cn, e);
163             finish();
164             return;
165         }
166 
167         // This admin already exists, an we have two options at this point.  If new policy
168         // bits are set, show the user the new list.  If nothing has changed, simply return
169         // "OK" immediately.
170         if (DevicePolicyManager.ACTION_ADD_DEVICE_ADMIN.equals(getIntent().getAction())) {
171             mRefreshing = false;
172             if (mDPM.isAdminActive(cn)) {
173                 ArrayList<DeviceAdminInfo.PolicyInfo> newPolicies = mDeviceAdmin.getUsedPolicies();
174                 for (int i = 0; i < newPolicies.size(); i++) {
175                     DeviceAdminInfo.PolicyInfo pi = newPolicies.get(i);
176                     if (!mDPM.hasGrantedPolicy(cn, pi.ident)) {
177                         mRefreshing = true;
178                         break;
179                     }
180                 }
181                 if (!mRefreshing) {
182                     // Nothing changed (or policies were removed) - return immediately
183                     setResult(Activity.RESULT_OK);
184                     finish();
185                     return;
186                 }
187             }
188         }
189         mAddMsgText = getIntent().getCharSequenceExtra(DevicePolicyManager.EXTRA_ADD_EXPLANATION);
190 
191         setContentView(R.layout.device_admin_add);
192 
193         mAdminIcon = (ImageView)findViewById(R.id.admin_icon);
194         mAdminName = (TextView)findViewById(R.id.admin_name);
195         mAdminDescription = (TextView)findViewById(R.id.admin_description);
196 
197         mAddMsg = (TextView)findViewById(R.id.add_msg);
198         mAddMsgExpander = (ImageView) findViewById(R.id.add_msg_expander);
199         mAddMsg.setOnClickListener(new View.OnClickListener() {
200             public void onClick(View v) {
201                 toggleMessageEllipsis(v);
202             }
203         });
204 
205         // toggleMessageEllipsis also handles initial layout:
206         toggleMessageEllipsis(mAddMsg);
207 
208         mAdminWarning = (TextView) findViewById(R.id.admin_warning);
209         mAdminPolicies = (ViewGroup) findViewById(R.id.admin_policies);
210         mCancelButton = (Button) findViewById(R.id.cancel_button);
211         mCancelButton.setOnClickListener(new View.OnClickListener() {
212             public void onClick(View v) {
213                 finish();
214             }
215         });
216         mActionButton = (Button) findViewById(R.id.action_button);
217         mActionButton.setOnClickListener(new View.OnClickListener() {
218             public void onClick(View v) {
219                 if (mAdding) {
220                     try {
221                         mDPM.setActiveAdmin(mDeviceAdmin.getComponent(), mRefreshing);
222                         setResult(Activity.RESULT_OK);
223                     } catch (RuntimeException e) {
224                         // Something bad happened...  could be that it was
225                         // already set, though.
226                         Log.w(TAG, "Exception trying to activate admin "
227                                 + mDeviceAdmin.getComponent(), e);
228                         if (mDPM.isAdminActive(mDeviceAdmin.getComponent())) {
229                             setResult(Activity.RESULT_OK);
230                         }
231                     }
232                     finish();
233                 } else {
234                     try {
235                         // Don't allow the admin to put a dialog up in front
236                         // of us while we interact with the user.
237                         ActivityManagerNative.getDefault().stopAppSwitches();
238                     } catch (RemoteException e) {
239                     }
240                     mDPM.getRemoveWarning(mDeviceAdmin.getComponent(),
241                             new RemoteCallback(mHandler) {
242                         @Override
243                         protected void onResult(Bundle bundle) {
244                             CharSequence msg = bundle != null
245                                     ? bundle.getCharSequence(
246                                             DeviceAdminReceiver.EXTRA_DISABLE_WARNING)
247                                     : null;
248                             if (msg == null) {
249                                 try {
250                                     ActivityManagerNative.getDefault().resumeAppSwitches();
251                                 } catch (RemoteException e) {
252                                 }
253                                 mDPM.removeActiveAdmin(mDeviceAdmin.getComponent());
254                                 finish();
255                             } else {
256                                 Bundle args = new Bundle();
257                                 args.putCharSequence(
258                                         DeviceAdminReceiver.EXTRA_DISABLE_WARNING, msg);
259                                 showDialog(DIALOG_WARNING, args);
260                             }
261                         }
262                     });
263                 }
264             }
265         });
266     }
267 
268     @Override
onResume()269     protected void onResume() {
270         super.onResume();
271         updateInterface();
272     }
273 
274     @Override
onCreateDialog(int id, Bundle args)275     protected Dialog onCreateDialog(int id, Bundle args) {
276         switch (id) {
277             case DIALOG_WARNING: {
278                 CharSequence msg = args.getCharSequence(DeviceAdminReceiver.EXTRA_DISABLE_WARNING);
279                 AlertDialog.Builder builder = new AlertDialog.Builder(
280                         DeviceAdminAdd.this);
281                 builder.setMessage(msg);
282                 builder.setPositiveButton(R.string.dlg_ok,
283                         new DialogInterface.OnClickListener() {
284                     public void onClick(DialogInterface dialog, int which) {
285                         mDPM.removeActiveAdmin(mDeviceAdmin.getComponent());
286                         finish();
287                     }
288                 });
289                 builder.setNegativeButton(R.string.dlg_cancel, null);
290                 return builder.create();
291             }
292             default:
293                 return super.onCreateDialog(id, args);
294 
295         }
296     }
297 
setViewVisibility(ArrayList<View> views, int visibility)298     static void setViewVisibility(ArrayList<View> views, int visibility) {
299         final int N = views.size();
300         for (int i=0; i<N; i++) {
301             views.get(i).setVisibility(visibility);
302         }
303     }
304 
updateInterface()305     void updateInterface() {
306         mAdminIcon.setImageDrawable(mDeviceAdmin.loadIcon(getPackageManager()));
307         mAdminName.setText(mDeviceAdmin.loadLabel(getPackageManager()));
308         try {
309             mAdminDescription.setText(
310                     mDeviceAdmin.loadDescription(getPackageManager()));
311             mAdminDescription.setVisibility(View.VISIBLE);
312         } catch (Resources.NotFoundException e) {
313             mAdminDescription.setVisibility(View.GONE);
314         }
315         if (mAddMsgText != null) {
316             mAddMsg.setText(mAddMsgText);
317             mAddMsg.setVisibility(View.VISIBLE);
318         } else {
319             mAddMsg.setVisibility(View.GONE);
320             mAddMsgExpander.setVisibility(View.GONE);
321         }
322         if (!mRefreshing && mDPM.isAdminActive(mDeviceAdmin.getComponent())) {
323             if (mActivePolicies.size() == 0) {
324                 ArrayList<DeviceAdminInfo.PolicyInfo> policies = mDeviceAdmin.getUsedPolicies();
325                 for (int i=0; i<policies.size(); i++) {
326                     DeviceAdminInfo.PolicyInfo pi = policies.get(i);
327                     View view = AppSecurityPermissions.getPermissionItemView(
328                             this, getText(pi.label), "", true);
329                     mActivePolicies.add(view);
330                     mAdminPolicies.addView(view);
331                 }
332             }
333             setViewVisibility(mActivePolicies, View.VISIBLE);
334             setViewVisibility(mAddingPolicies, View.GONE);
335             mAdminWarning.setText(getString(R.string.device_admin_status,
336                     mDeviceAdmin.getActivityInfo().applicationInfo.loadLabel(getPackageManager())));
337             setTitle(getText(R.string.active_device_admin_msg));
338             mActionButton.setText(getText(R.string.remove_device_admin));
339             mAdding = false;
340         } else {
341             if (mAddingPolicies.size() == 0) {
342                 ArrayList<DeviceAdminInfo.PolicyInfo> policies = mDeviceAdmin.getUsedPolicies();
343                 for (int i=0; i<policies.size(); i++) {
344                     DeviceAdminInfo.PolicyInfo pi = policies.get(i);
345                     View view = AppSecurityPermissions.getPermissionItemView(
346                             this, getText(pi.label), getText(pi.description), true);
347                     mAddingPolicies.add(view);
348                     mAdminPolicies.addView(view);
349                 }
350             }
351             setViewVisibility(mAddingPolicies, View.VISIBLE);
352             setViewVisibility(mActivePolicies, View.GONE);
353             mAdminWarning.setText(getString(R.string.device_admin_warning,
354                     mDeviceAdmin.getActivityInfo().applicationInfo.loadLabel(getPackageManager())));
355             setTitle(getText(R.string.add_device_admin_msg));
356             mActionButton.setText(getText(R.string.add_device_admin));
357             mAdding = true;
358         }
359     }
360 
361 
toggleMessageEllipsis(View v)362     void toggleMessageEllipsis(View v) {
363         TextView tv = (TextView) v;
364 
365         mAddMsgEllipsized = ! mAddMsgEllipsized;
366         tv.setEllipsize(mAddMsgEllipsized ? TruncateAt.END : null);
367         tv.setMaxLines(mAddMsgEllipsized ? getEllipsizedLines() : MAX_ADD_MSG_LINES);
368 
369         mAddMsgExpander.setImageResource(mAddMsgEllipsized ?
370             com.android.internal.R.drawable.expander_ic_minimized :
371             com.android.internal.R.drawable.expander_ic_maximized);
372     }
373 
getEllipsizedLines()374     int getEllipsizedLines() {
375         Display d = ((WindowManager) getSystemService(Context.WINDOW_SERVICE))
376                     .getDefaultDisplay();
377 
378         return d.getHeight() > d.getWidth() ?
379             MAX_ADD_MSG_LINES_PORTRAIT : MAX_ADD_MSG_LINES_LANDSCAPE;
380     }
381 
382 }
383