Home | History | Annotate | Download | only in settings
      1 /*
      2  * Copyright (C) 2008 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.AlertDialog;
     21 import android.app.admin.DevicePolicyManager;
     22 import android.content.BroadcastReceiver;
     23 import android.content.Context;
     24 import android.content.Intent;
     25 import android.content.IntentFilter;
     26 import android.content.res.Resources;
     27 import android.os.BatteryManager;
     28 import android.os.Bundle;
     29 import android.os.UserHandle;
     30 import android.os.storage.StorageManager;
     31 import android.support.v7.preference.Preference;
     32 import android.text.TextUtils;
     33 import android.view.LayoutInflater;
     34 import android.view.View;
     35 import android.view.ViewGroup;
     36 import android.widget.Button;
     37 
     38 import com.android.internal.logging.MetricsProto.MetricsEvent;
     39 
     40 public class CryptKeeperSettings extends InstrumentedFragment {
     41     private static final String TAG = "CryptKeeper";
     42     private static final String TYPE = "type";
     43     private static final String PASSWORD = "password";
     44 
     45     private static final int KEYGUARD_REQUEST = 55;
     46 
     47     // Minimum battery charge level (in percent) to launch encryption.  If the battery charge is
     48     // lower than this, encryption should not be activated.
     49     private static final int MIN_BATTERY_LEVEL = 80;
     50 
     51     private View mContentView;
     52     private Button mInitiateButton;
     53     private View mPowerWarning;
     54     private View mBatteryWarning;
     55     private IntentFilter mIntentFilter;
     56 
     57     private BroadcastReceiver mIntentReceiver = new BroadcastReceiver() {
     58         @Override
     59         public void onReceive(Context context, Intent intent) {
     60             String action = intent.getAction();
     61             if (action.equals(Intent.ACTION_BATTERY_CHANGED)) {
     62                 final int level = intent.getIntExtra(BatteryManager.EXTRA_LEVEL, 0);
     63                 final int plugged = intent.getIntExtra(BatteryManager.EXTRA_PLUGGED, 0);
     64                 final int invalidCharger = intent.getIntExtra(
     65                     BatteryManager.EXTRA_INVALID_CHARGER, 0);
     66 
     67                 final boolean levelOk = level >= MIN_BATTERY_LEVEL;
     68                 final boolean pluggedOk =
     69                     ((plugged & BatteryManager.BATTERY_PLUGGED_ANY) != 0) &&
     70                      invalidCharger == 0;
     71 
     72                 // Update UI elements based on power/battery status
     73                 mInitiateButton.setEnabled(levelOk && pluggedOk);
     74                 mPowerWarning.setVisibility(pluggedOk ? View.GONE : View.VISIBLE );
     75                 mBatteryWarning.setVisibility(levelOk ? View.GONE : View.VISIBLE);
     76             }
     77         }
     78     };
     79 
     80     /**
     81      * If the user clicks to begin the reset sequence, we next require a
     82      * keyguard confirmation if the user has currently enabled one.  If there
     83      * is no keyguard available, we prompt the user to set a password.
     84      */
     85     private Button.OnClickListener mInitiateListener = new Button.OnClickListener() {
     86         @Override
     87         public void onClick(View v) {
     88             if (!runKeyguardConfirmation(KEYGUARD_REQUEST)) {
     89                 // TODO replace (or follow) this dialog with an explicit launch into password UI
     90                 new AlertDialog.Builder(getActivity())
     91                     .setTitle(R.string.crypt_keeper_dialog_need_password_title)
     92                     .setMessage(R.string.crypt_keeper_dialog_need_password_message)
     93                     .setPositiveButton(android.R.string.ok, null)
     94                     .create()
     95                     .show();
     96             }
     97         }
     98     };
     99 
    100     @Override
    101     public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedState) {
    102         mContentView = inflater.inflate(R.layout.crypt_keeper_settings, null);
    103 
    104         mIntentFilter = new IntentFilter();
    105         mIntentFilter.addAction(Intent.ACTION_BATTERY_CHANGED);
    106 
    107         mInitiateButton = (Button) mContentView.findViewById(R.id.initiate_encrypt);
    108         mInitiateButton.setOnClickListener(mInitiateListener);
    109         mInitiateButton.setEnabled(false);
    110 
    111         mPowerWarning = mContentView.findViewById(R.id.warning_unplugged);
    112         mBatteryWarning = mContentView.findViewById(R.id.warning_low_charge);
    113 
    114         return mContentView;
    115     }
    116 
    117     @Override
    118     protected int getMetricsCategory() {
    119         return MetricsEvent.CRYPT_KEEPER;
    120     }
    121 
    122     @Override
    123     public void onResume() {
    124         super.onResume();
    125         getActivity().registerReceiver(mIntentReceiver, mIntentFilter);
    126     }
    127 
    128     @Override
    129     public void onPause() {
    130         super.onPause();
    131         getActivity().unregisterReceiver(mIntentReceiver);
    132     }
    133 
    134     /**
    135      * If encryption is already started, and this launched via a "start encryption" intent,
    136      * then exit immediately - it's already up and running, so there's no point in "starting" it.
    137      */
    138     @Override
    139     public void onActivityCreated(Bundle savedInstanceState) {
    140         super.onActivityCreated(savedInstanceState);
    141         Activity activity = getActivity();
    142         Intent intent = activity.getIntent();
    143         if (DevicePolicyManager.ACTION_START_ENCRYPTION.equals(intent.getAction())) {
    144             DevicePolicyManager dpm = (DevicePolicyManager)
    145                     activity.getSystemService(Context.DEVICE_POLICY_SERVICE);
    146             if (dpm != null) {
    147                 int status = dpm.getStorageEncryptionStatus();
    148                 if (status != DevicePolicyManager.ENCRYPTION_STATUS_INACTIVE) {
    149                     // There is nothing to do here, so simply finish() (which returns to caller)
    150                     activity.finish();
    151                 }
    152             }
    153         }
    154     }
    155 
    156     /**
    157      * Keyguard validation is run using the standard {@link ConfirmLockPattern}
    158      * component as a subactivity
    159      * @param request the request code to be returned once confirmation finishes
    160      * @return true if confirmation launched
    161      */
    162     private boolean runKeyguardConfirmation(int request) {
    163         Resources res = getActivity().getResources();
    164         ChooseLockSettingsHelper helper = new ChooseLockSettingsHelper(getActivity(), this);
    165 
    166         if (helper.utils().getKeyguardStoredPasswordQuality(UserHandle.myUserId())
    167                 == DevicePolicyManager.PASSWORD_QUALITY_UNSPECIFIED) {
    168             showFinalConfirmation(StorageManager.CRYPT_TYPE_DEFAULT, "");
    169             return true;
    170         }
    171 
    172         return helper.launchConfirmationActivity(request,
    173                 res.getText(R.string.crypt_keeper_encrypt_title), true);
    174     }
    175 
    176     @Override
    177     public void onActivityResult(int requestCode, int resultCode, Intent data) {
    178         super.onActivityResult(requestCode, resultCode, data);
    179 
    180         if (requestCode != KEYGUARD_REQUEST) {
    181             return;
    182         }
    183 
    184         // If the user entered a valid keyguard trace, present the final
    185         // confirmation prompt; otherwise, go back to the initial state.
    186         if (resultCode == Activity.RESULT_OK && data != null) {
    187             int type = data.getIntExtra(ChooseLockSettingsHelper.EXTRA_KEY_TYPE, -1);
    188             String password = data.getStringExtra(ChooseLockSettingsHelper.EXTRA_KEY_PASSWORD);
    189             if (!TextUtils.isEmpty(password)) {
    190                 showFinalConfirmation(type, password);
    191             }
    192         }
    193     }
    194 
    195     private void showFinalConfirmation(int type, String password) {
    196         Preference preference = new Preference(getPreferenceManager().getContext());
    197         preference.setFragment(CryptKeeperConfirm.class.getName());
    198         preference.setTitle(R.string.crypt_keeper_confirm_title);
    199         addEncryptionInfoToPreference(preference, type, password);
    200         ((SettingsActivity) getActivity()).onPreferenceStartFragment(null, preference);
    201     }
    202 
    203     private void addEncryptionInfoToPreference(Preference preference, int type, String password) {
    204         Activity activity = getActivity();
    205         DevicePolicyManager dpm = (DevicePolicyManager)
    206                 activity.getSystemService(Context.DEVICE_POLICY_SERVICE);
    207         if (dpm.getDoNotAskCredentialsOnBoot()) {
    208             preference.getExtras().putInt(TYPE, StorageManager.CRYPT_TYPE_DEFAULT);
    209             preference.getExtras().putString(PASSWORD, "");
    210         } else {
    211             preference.getExtras().putInt(TYPE, type);
    212             preference.getExtras().putString(PASSWORD, password);
    213         }
    214     }
    215 }
    216