Enforce Keyguard policy in Notification Settings

- Applied getKeyguardDisabledFeatures for notification settings and
    notification setup page (after settings a screenlock)
- If a notification settings is disabled, the next least secure setting
    will be chosen
- Although KEYGUARD_DISABLE_UNREDACTED_NOTIFICATIONS can be set be
    profile, it will not be reflected in both settings page. This is
    because it does not affect the owner (user 0), as mentioned in
    DevicePolicyManagerService.PROFILE_KEYGUARD_FEATURES_AFFECT_OWNER
- Skip RedactionInterstitial if there is <= 1 options for the user
- Tested with both Setup wizard and settings case, both pattern and
    password, as well as toggling the policy on and off

Bug: 19307118
Bug: 17099898

Change-Id: If640d5576caa0163e9942569f7b4643a30bbfe0a
This commit is contained in:
Alex Chau
2015-06-12 11:45:08 +01:00
parent 2c55f738c4
commit ccade40278
10 changed files with 129 additions and 43 deletions

View File

@@ -33,6 +33,7 @@
android:textAppearance="?android:attr/textAppearanceMedium" /> android:textAppearance="?android:attr/textAppearanceMedium" />
<RadioGroup <RadioGroup
android:id="@+id/radio_group"
android:layout_width="wrap_content" android:layout_width="wrap_content"
android:layout_height="wrap_content" android:layout_height="wrap_content"
android:layout_marginTop="@dimen/redaction_vertical_margins" android:layout_marginTop="@dimen/redaction_vertical_margins"

View File

@@ -43,6 +43,7 @@
android:text="@string/lock_screen_notifications_interstitial_message" /> android:text="@string/lock_screen_notifications_interstitial_message" />
<RadioGroup <RadioGroup
android:id="@+id/radio_group"
android:layout_width="wrap_content" android:layout_width="wrap_content"
android:layout_height="wrap_content" android:layout_height="wrap_content"
android:layout_marginTop="@dimen/redaction_vertical_margins" android:layout_marginTop="@dimen/redaction_vertical_margins"

View File

@@ -559,7 +559,10 @@ public class ChooseLockPassword extends SettingsActivity {
getActivity().finish(); getActivity().finish();
mDone = true; mDone = true;
if (!wasSecureBefore) { if (!wasSecureBefore) {
startActivity(getRedactionInterstitialIntent(getActivity())); Intent intent = getRedactionInterstitialIntent(getActivity());
if (intent != null) {
startActivity(intent);
}
} }
} }

View File

@@ -641,7 +641,10 @@ public class ChooseLockPattern extends SettingsActivity {
startVerifyPattern(utils, wasSecureBefore); startVerifyPattern(utils, wasSecureBefore);
} else { } else {
if (!wasSecureBefore) { if (!wasSecureBefore) {
startActivity(getRedactionInterstitialIntent(getActivity())); Intent intent = getRedactionInterstitialIntent(getActivity());
if (intent != null) {
startActivity(intent);
}
} }
getActivity().setResult(RESULT_FINISHED); getActivity().setResult(RESULT_FINISHED);
doFinish(); doFinish();
@@ -670,7 +673,10 @@ public class ChooseLockPattern extends SettingsActivity {
mPendingLockCheck = null; mPendingLockCheck = null;
if (!wasSecureBefore) { if (!wasSecureBefore) {
startActivity(getRedactionInterstitialIntent(getActivity())); Intent intent = getRedactionInterstitialIntent(getActivity());
if (intent != null) {
startActivity(intent);
}
} }
Intent intent = new Intent(); Intent intent = new Intent();

View File

@@ -126,6 +126,10 @@ public class DropDownPreference extends Preference {
mValues.add(value); mValues.add(value);
} }
public int getItemCount() {
return mAdapter.getCount();
}
public void clearItems(){ public void clearItems(){
mAdapter.clear(); mAdapter.clear();
mValues.clear(); mValues.clear();

View File

@@ -110,7 +110,9 @@ public class SetupChooseLockPassword extends ChooseLockPassword {
@Override @Override
protected Intent getRedactionInterstitialIntent(Context context) { protected Intent getRedactionInterstitialIntent(Context context) {
Intent intent = SetupRedactionInterstitial.createStartIntent(context); Intent intent = SetupRedactionInterstitial.createStartIntent(context);
SetupWizardUtils.copySetupExtras(getActivity().getIntent(), intent); if (intent != null) {
SetupWizardUtils.copySetupExtras(getActivity().getIntent(), intent);
}
return intent; return intent;
} }

View File

@@ -103,7 +103,9 @@ public class SetupChooseLockPattern extends ChooseLockPattern {
@Override @Override
protected Intent getRedactionInterstitialIntent(Context context) { protected Intent getRedactionInterstitialIntent(Context context) {
Intent intent = SetupRedactionInterstitial.createStartIntent(context); Intent intent = SetupRedactionInterstitial.createStartIntent(context);
SetupWizardUtils.copySetupExtras(getActivity().getIntent(), intent); if (intent != null) {
SetupWizardUtils.copySetupExtras(getActivity().getIntent(), intent);
}
return intent; return intent;
} }

View File

@@ -40,9 +40,11 @@ public class SetupRedactionInterstitial extends RedactionInterstitial {
public static Intent createStartIntent(Context ctx) { public static Intent createStartIntent(Context ctx) {
Intent startIntent = RedactionInterstitial.createStartIntent(ctx); Intent startIntent = RedactionInterstitial.createStartIntent(ctx);
startIntent.setClass(ctx, SetupRedactionInterstitial.class); if (startIntent != null) {
startIntent.putExtra(EXTRA_PREFS_SHOW_BUTTON_BAR, false) startIntent.setClass(ctx, SetupRedactionInterstitial.class);
.putExtra(EXTRA_SHOW_FRAGMENT_TITLE_RESID, -1); startIntent.putExtra(EXTRA_PREFS_SHOW_BUTTON_BAR, false)
.putExtra(EXTRA_SHOW_FRAGMENT_TITLE_RESID, -1);
}
return startIntent; return startIntent;
} }

View File

@@ -17,6 +17,7 @@
package com.android.settings.notification; package com.android.settings.notification;
import android.app.NotificationManager; import android.app.NotificationManager;
import android.app.admin.DevicePolicyManager;
import android.content.BroadcastReceiver; import android.content.BroadcastReceiver;
import android.content.ComponentName; import android.content.ComponentName;
import android.content.ContentResolver; import android.content.ContentResolver;
@@ -427,32 +428,56 @@ public class NotificationSettings extends SettingsPreferenceFragment implements
return; return;
} }
mLockscreen.addItem(R.string.lock_screen_notifications_summary_show, boolean isSecureNotificationsDisabled = isSecureNotificationsDisabled();
R.string.lock_screen_notifications_summary_show); boolean isUnredactedNotificationsDisabled = isUnredactedNotificationsDisabled();
if (mSecure) { if (!isSecureNotificationsDisabled && !isUnredactedNotificationsDisabled) {
mLockscreen.addItem(R.string.lock_screen_notifications_summary_show,
R.string.lock_screen_notifications_summary_show);
}
if (mSecure && !isSecureNotificationsDisabled) {
mLockscreen.addItem(R.string.lock_screen_notifications_summary_hide, mLockscreen.addItem(R.string.lock_screen_notifications_summary_hide,
R.string.lock_screen_notifications_summary_hide); R.string.lock_screen_notifications_summary_hide);
} }
mLockscreen.addItem(R.string.lock_screen_notifications_summary_disable, mLockscreen.addItem(R.string.lock_screen_notifications_summary_disable,
R.string.lock_screen_notifications_summary_disable); R.string.lock_screen_notifications_summary_disable);
updateLockscreenNotifications(); updateLockscreenNotifications();
mLockscreen.setCallback(new DropDownPreference.Callback() { if (mLockscreen.getItemCount() > 1) {
@Override mLockscreen.setCallback(new DropDownPreference.Callback() {
public boolean onItemSelected(int pos, Object value) { @Override
final int val = (Integer) value; public boolean onItemSelected(int pos, Object value) {
if (val == mLockscreenSelectedValue) { final int val = (Integer) value;
if (val == mLockscreenSelectedValue) {
return true;
}
final boolean enabled =
val != R.string.lock_screen_notifications_summary_disable;
final boolean show = val == R.string.lock_screen_notifications_summary_show;
Settings.Secure.putInt(getContentResolver(),
Settings.Secure.LOCK_SCREEN_ALLOW_PRIVATE_NOTIFICATIONS, show ? 1 : 0);
Settings.Secure.putInt(getContentResolver(),
Settings.Secure.LOCK_SCREEN_SHOW_NOTIFICATIONS, enabled ? 1 : 0);
mLockscreenSelectedValue = val;
return true; return true;
} }
final boolean enabled = val != R.string.lock_screen_notifications_summary_disable; });
final boolean show = val == R.string.lock_screen_notifications_summary_show; } else {
Settings.Secure.putInt(getContentResolver(), // There is one or less option for the user, disable the drop down.
Settings.Secure.LOCK_SCREEN_ALLOW_PRIVATE_NOTIFICATIONS, show ? 1 : 0); mLockscreen.setEnabled(false);
Settings.Secure.putInt(getContentResolver(), }
Settings.Secure.LOCK_SCREEN_SHOW_NOTIFICATIONS, enabled ? 1 : 0); }
mLockscreenSelectedValue = val;
return true; private boolean isSecureNotificationsDisabled() {
} final DevicePolicyManager dpm =
}); (DevicePolicyManager) getSystemService(Context.DEVICE_POLICY_SERVICE);
return dpm != null && (dpm.getKeyguardDisabledFeatures(null)
& DevicePolicyManager.KEYGUARD_DISABLE_SECURE_NOTIFICATIONS) != 0;
}
private boolean isUnredactedNotificationsDisabled() {
final DevicePolicyManager dpm =
(DevicePolicyManager) getSystemService(Context.DEVICE_POLICY_SERVICE);
return dpm != null && (dpm.getKeyguardDisabledFeatures(null)
& DevicePolicyManager.KEYGUARD_DISABLE_UNREDACTED_NOTIFICATIONS) != 0;
} }
private void updateLockscreenNotifications() { private void updateLockscreenNotifications() {

View File

@@ -21,6 +21,7 @@ import com.android.settings.R;
import com.android.settings.SettingsActivity; import com.android.settings.SettingsActivity;
import com.android.settings.SettingsPreferenceFragment; import com.android.settings.SettingsPreferenceFragment;
import android.app.admin.DevicePolicyManager;
import android.content.Context; import android.content.Context;
import android.content.Intent; import android.content.Intent;
import android.os.Bundle; import android.os.Bundle;
@@ -29,6 +30,7 @@ import android.view.LayoutInflater;
import android.view.View; import android.view.View;
import android.view.ViewGroup; import android.view.ViewGroup;
import android.widget.RadioButton; import android.widget.RadioButton;
import android.widget.RadioGroup;
public class RedactionInterstitial extends SettingsActivity { public class RedactionInterstitial extends SettingsActivity {
@@ -44,20 +46,44 @@ public class RedactionInterstitial extends SettingsActivity {
return RedactionInterstitialFragment.class.getName().equals(fragmentName); return RedactionInterstitialFragment.class.getName().equals(fragmentName);
} }
/**
* Create an intent for launching RedactionInterstitial.
* @return An intent to launch the activity is if is available, @null if the activity is not
* available to be launched.
*/
public static Intent createStartIntent(Context ctx) { public static Intent createStartIntent(Context ctx) {
return new Intent(ctx, RedactionInterstitial.class) if (isSecureNotificationsDisabled(ctx)) {
.putExtra(EXTRA_PREFS_SHOW_BUTTON_BAR, true) // If there is no choices for the user, we should not start the activity.
.putExtra(EXTRA_PREFS_SET_BACK_TEXT, (String) null) return null;
.putExtra(EXTRA_PREFS_SET_NEXT_TEXT, ctx.getString( } else {
R.string.app_notifications_dialog_done)); return new Intent(ctx, RedactionInterstitial.class)
.putExtra(EXTRA_PREFS_SHOW_BUTTON_BAR, true)
.putExtra(EXTRA_PREFS_SET_BACK_TEXT, (String) null)
.putExtra(EXTRA_PREFS_SET_NEXT_TEXT, ctx.getString(
R.string.app_notifications_dialog_done));
}
}
private static boolean isSecureNotificationsDisabled(Context context) {
final DevicePolicyManager dpm =
(DevicePolicyManager) context.getSystemService(Context.DEVICE_POLICY_SERVICE);
return dpm != null && (dpm.getKeyguardDisabledFeatures(null)
& DevicePolicyManager.KEYGUARD_DISABLE_SECURE_NOTIFICATIONS) != 0;
}
private static boolean isUnredactedNotificationsDisabled(Context context) {
final DevicePolicyManager dpm =
(DevicePolicyManager) context.getSystemService(Context.DEVICE_POLICY_SERVICE);
return dpm != null && (dpm.getKeyguardDisabledFeatures(null)
& DevicePolicyManager.KEYGUARD_DISABLE_UNREDACTED_NOTIFICATIONS) != 0;
} }
public static class RedactionInterstitialFragment extends SettingsPreferenceFragment public static class RedactionInterstitialFragment extends SettingsPreferenceFragment
implements View.OnClickListener { implements RadioGroup.OnCheckedChangeListener {
private RadioGroup mRadioGroup;
private RadioButton mShowAllButton; private RadioButton mShowAllButton;
private RadioButton mRedactSensitiveButton; private RadioButton mRedactSensitiveButton;
private RadioButton mHideAllButton;
@Override @Override
protected int getMetricsCategory() { protected int getMetricsCategory() {
@@ -73,13 +99,19 @@ public class RedactionInterstitial extends SettingsActivity {
@Override @Override
public void onViewCreated(View view, Bundle savedInstanceState) { public void onViewCreated(View view, Bundle savedInstanceState) {
super.onViewCreated(view, savedInstanceState); super.onViewCreated(view, savedInstanceState);
mRadioGroup = (RadioGroup) view.findViewById(R.id.radio_group);
mShowAllButton = (RadioButton) view.findViewById(R.id.show_all); mShowAllButton = (RadioButton) view.findViewById(R.id.show_all);
mRedactSensitiveButton = (RadioButton) view.findViewById(R.id.redact_sensitive); mRedactSensitiveButton = (RadioButton) view.findViewById(R.id.redact_sensitive);
mHideAllButton = (RadioButton) view.findViewById(R.id.hide_all);
mShowAllButton.setOnClickListener(this); mRadioGroup.setOnCheckedChangeListener(this);
mRedactSensitiveButton.setOnClickListener(this);
mHideAllButton.setOnClickListener(this); // Disable buttons according to policy.
if (isSecureNotificationsDisabled(getActivity())) {
mShowAllButton.setEnabled(false);
mRedactSensitiveButton.setEnabled(false);
} else if (isUnredactedNotificationsDisabled(getActivity())) {
mShowAllButton.setEnabled(false);
}
} }
@Override @Override
@@ -93,15 +125,23 @@ public class RedactionInterstitial extends SettingsActivity {
Settings.Secure.LOCK_SCREEN_SHOW_NOTIFICATIONS, 0) != 0; Settings.Secure.LOCK_SCREEN_SHOW_NOTIFICATIONS, 0) != 0;
final boolean show = Settings.Secure.getInt(getContentResolver(), final boolean show = Settings.Secure.getInt(getContentResolver(),
Settings.Secure.LOCK_SCREEN_ALLOW_PRIVATE_NOTIFICATIONS, 1) != 0; Settings.Secure.LOCK_SCREEN_ALLOW_PRIVATE_NOTIFICATIONS, 1) != 0;
mShowAllButton.setChecked(enabled && show);
mRedactSensitiveButton.setChecked(enabled && !show); int checkedButtonId = R.id.hide_all;
mHideAllButton.setChecked(!enabled); if (enabled) {
if (show && mShowAllButton.isEnabled()) {
checkedButtonId = R.id.show_all;
} else if (mRedactSensitiveButton.isEnabled()) {
checkedButtonId = R.id.redact_sensitive;
}
}
mRadioGroup.check(checkedButtonId);
} }
@Override @Override
public void onClick(View v) { public void onCheckedChanged(RadioGroup group, int checkedId) {
final boolean show = (v == mShowAllButton); final boolean show = (checkedId == R.id.show_all);
final boolean enabled = (v != mHideAllButton); final boolean enabled = (checkedId != R.id.hide_all);
Settings.Secure.putInt(getContentResolver(), Settings.Secure.putInt(getContentResolver(),
Settings.Secure.LOCK_SCREEN_ALLOW_PRIVATE_NOTIFICATIONS, show ? 1 : 0); Settings.Secure.LOCK_SCREEN_ALLOW_PRIVATE_NOTIFICATIONS, show ? 1 : 0);