/* * Copyright (C) 2008 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.security; import android.app.Activity; import android.app.admin.DevicePolicyManager; import android.app.settings.SettingsEnums; import android.content.BroadcastReceiver; import android.content.Context; import android.content.Intent; import android.content.IntentFilter; import android.content.res.Resources; import android.os.BatteryManager; import android.os.Bundle; import android.os.UserHandle; import android.os.storage.StorageManager; import android.text.TextUtils; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.widget.Button; import androidx.appcompat.app.AlertDialog; import androidx.preference.Preference; import com.android.internal.widget.LockscreenCredential; import com.android.settings.CryptKeeperConfirm; import com.android.settings.R; import com.android.settings.SettingsActivity; import com.android.settings.core.InstrumentedPreferenceFragment; import com.android.settings.password.ChooseLockSettingsHelper; import com.android.settings.password.ConfirmLockPattern; public class CryptKeeperSettings extends InstrumentedPreferenceFragment { private static final String TAG = "CryptKeeper"; private static final String TYPE = "type"; private static final String PASSWORD = "password"; private static final int KEYGUARD_REQUEST = 55; // Minimum battery charge level (in percent) to launch encryption. If the battery charge is // lower than this, encryption should not be activated. private static final int MIN_BATTERY_LEVEL = 80; private View mContentView; private Button mInitiateButton; private View mPowerWarning; private View mBatteryWarning; private IntentFilter mIntentFilter; private BroadcastReceiver mIntentReceiver = new BroadcastReceiver() { @Override public void onReceive(Context context, Intent intent) { String action = intent.getAction(); if (action.equals(Intent.ACTION_BATTERY_CHANGED)) { final int level = intent.getIntExtra(BatteryManager.EXTRA_LEVEL, 0); final int plugged = intent.getIntExtra(BatteryManager.EXTRA_PLUGGED, 0); final int invalidCharger = intent.getIntExtra( BatteryManager.EXTRA_INVALID_CHARGER, 0); final boolean levelOk = level >= MIN_BATTERY_LEVEL; final boolean pluggedOk = ((plugged & BatteryManager.BATTERY_PLUGGED_ANY) != 0) && invalidCharger == 0; // Update UI elements based on power/battery status mInitiateButton.setEnabled(levelOk && pluggedOk); mPowerWarning.setVisibility(pluggedOk ? View.GONE : View.VISIBLE ); mBatteryWarning.setVisibility(levelOk ? View.GONE : View.VISIBLE); } } }; /** * If the user clicks to begin the reset sequence, we next require a * keyguard confirmation if the user has currently enabled one. If there * is no keyguard available, we prompt the user to set a password. */ private Button.OnClickListener mInitiateListener = new Button.OnClickListener() { @Override public void onClick(View v) { if (!runKeyguardConfirmation(KEYGUARD_REQUEST)) { // TODO replace (or follow) this dialog with an explicit launch into password UI new AlertDialog.Builder(getActivity()) .setTitle(R.string.crypt_keeper_dialog_need_password_title) .setMessage(R.string.crypt_keeper_dialog_need_password_message) .setPositiveButton(android.R.string.ok, null) .create() .show(); } } }; @Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedState) { mContentView = inflater.inflate(R.layout.crypt_keeper_settings, null); mIntentFilter = new IntentFilter(); mIntentFilter.addAction(Intent.ACTION_BATTERY_CHANGED); mInitiateButton = (Button) mContentView.findViewById(R.id.initiate_encrypt); mInitiateButton.setOnClickListener(mInitiateListener); mInitiateButton.setEnabled(false); mPowerWarning = mContentView.findViewById(R.id.warning_unplugged); mBatteryWarning = mContentView.findViewById(R.id.warning_low_charge); return mContentView; } @Override public int getMetricsCategory() { return SettingsEnums.CRYPT_KEEPER; } @Override public void onResume() { super.onResume(); getActivity().registerReceiver(mIntentReceiver, mIntentFilter); } @Override public void onPause() { super.onPause(); getActivity().unregisterReceiver(mIntentReceiver); } /** * If encryption is already started, and this launched via a "start encryption" intent, * then exit immediately - it's already up and running, so there's no point in "starting" it. */ @Override public void onActivityCreated(Bundle savedInstanceState) { super.onActivityCreated(savedInstanceState); Activity activity = getActivity(); Intent intent = activity.getIntent(); if (DevicePolicyManager.ACTION_START_ENCRYPTION.equals(intent.getAction())) { DevicePolicyManager dpm = (DevicePolicyManager) activity.getSystemService(Context.DEVICE_POLICY_SERVICE); if (dpm != null) { int status = dpm.getStorageEncryptionStatus(); if (status != DevicePolicyManager.ENCRYPTION_STATUS_INACTIVE) { // There is nothing to do here, so simply finish() (which returns to caller) activity.finish(); } } } activity.setTitle(R.string.crypt_keeper_encrypt_title); } /** * Keyguard validation is run using the standard {@link ConfirmLockPattern} * component as a subactivity * @param request the request code to be returned once confirmation finishes * @return true if confirmation launched */ private boolean runKeyguardConfirmation(int request) { Resources res = getActivity().getResources(); ChooseLockSettingsHelper helper = new ChooseLockSettingsHelper(getActivity(), this); if (helper.utils().getKeyguardStoredPasswordQuality(UserHandle.myUserId()) == DevicePolicyManager.PASSWORD_QUALITY_UNSPECIFIED) { showFinalConfirmation(StorageManager.CRYPT_TYPE_DEFAULT, "".getBytes()); return true; } return helper.launchConfirmationActivity(request, res.getText(R.string.crypt_keeper_encrypt_title), true); } @Override public void onActivityResult(int requestCode, int resultCode, Intent data) { super.onActivityResult(requestCode, resultCode, data); if (requestCode != KEYGUARD_REQUEST) { return; } // If the user entered a valid keyguard trace, present the final // confirmation prompt; otherwise, go back to the initial state. if (resultCode == Activity.RESULT_OK && data != null) { int type = data.getIntExtra(ChooseLockSettingsHelper.EXTRA_KEY_TYPE, -1); LockscreenCredential password = data.getParcelableExtra( ChooseLockSettingsHelper.EXTRA_KEY_PASSWORD); if (password != null && !password.isNone()) { showFinalConfirmation(type, password.getCredential()); } } } private void showFinalConfirmation(int type, byte[] password) { Preference preference = new Preference(getPreferenceManager().getContext()); preference.setFragment(CryptKeeperConfirm.class.getName()); preference.setTitle(R.string.crypt_keeper_confirm_title); addEncryptionInfoToPreference(preference, type, password); ((SettingsActivity) getActivity()).onPreferenceStartFragment(null, preference); } private void addEncryptionInfoToPreference(Preference preference, int type, byte[] password) { Activity activity = getActivity(); DevicePolicyManager dpm = (DevicePolicyManager) activity.getSystemService(Context.DEVICE_POLICY_SERVICE); if (dpm.getDoNotAskCredentialsOnBoot()) { preference.getExtras().putInt(TYPE, StorageManager.CRYPT_TYPE_DEFAULT); preference.getExtras().putByteArray(PASSWORD, "".getBytes()); } else { preference.getExtras().putInt(TYPE, type); preference.getExtras().putByteArray(PASSWORD, password); } } }