Settings: External automatic rule settings.
- Add external automatic rule settings page with the common settings for all rules (enabled, name, zen mode). - Pull common rule-instance settings into settings base class, share with existing schedule rule settings. - New page not searchable since it is at the rule-instance level. - Obtain external rule information from existing conditions provider metadata. Includes rule type caption, sub-configuration activity, and default condition id. - If external condition providers exist with the appropriate metadata, display the external rule types as options in the new rule dialog. (max of 3 external types) - Pull common managed service listing code out of common settings base class and into a more reusable helper class. Bug: 20064962 Change-Id: Ibc13607490b7312a7d9f7f3bd61c3cfcf71a2794
This commit is contained in:
@@ -696,6 +696,10 @@
|
|||||||
<action android:name="android.settings.ZEN_MODE_AUTOMATION_SETTINGS" />
|
<action android:name="android.settings.ZEN_MODE_AUTOMATION_SETTINGS" />
|
||||||
<category android:name="android.intent.category.DEFAULT" />
|
<category android:name="android.intent.category.DEFAULT" />
|
||||||
</intent-filter>
|
</intent-filter>
|
||||||
|
<intent-filter android:priority="1">
|
||||||
|
<action android:name="android.settings.ACTION_CONDITION_PROVIDER_SETTINGS" />
|
||||||
|
<category android:name="android.intent.category.DEFAULT" />
|
||||||
|
</intent-filter>
|
||||||
<intent-filter>
|
<intent-filter>
|
||||||
<action android:name="android.intent.action.MAIN" />
|
<action android:name="android.intent.action.MAIN" />
|
||||||
<category android:name="android.intent.category.DEFAULT" />
|
<category android:name="android.intent.category.DEFAULT" />
|
||||||
@@ -729,6 +733,26 @@
|
|||||||
android:value="true" />
|
android:value="true" />
|
||||||
</activity>
|
</activity>
|
||||||
|
|
||||||
|
<activity android:name="Settings$ZenModeExternalRuleSettingsActivity"
|
||||||
|
android:exported="true"
|
||||||
|
android:taskAffinity="">
|
||||||
|
<intent-filter android:priority="1">
|
||||||
|
<action android:name="android.settings.ZEN_MODE_EXTERNAL_RULE_SETTINGS" />
|
||||||
|
<category android:name="android.intent.category.DEFAULT" />
|
||||||
|
</intent-filter>
|
||||||
|
<intent-filter>
|
||||||
|
<action android:name="android.intent.action.MAIN" />
|
||||||
|
<category android:name="android.intent.category.DEFAULT" />
|
||||||
|
<category android:name="com.android.settings.SHORTCUT" />
|
||||||
|
</intent-filter>
|
||||||
|
<meta-data android:name="com.android.settings.FRAGMENT_CLASS"
|
||||||
|
android:value="com.android.settings.notification.ZenModeExternalRuleSettings" />
|
||||||
|
<meta-data android:name="com.android.settings.TOP_LEVEL_HEADER_ID"
|
||||||
|
android:resource="@id/notification_settings" />
|
||||||
|
<meta-data android:name="com.android.settings.PRIMARY_PROFILE_CONTROLLED"
|
||||||
|
android:value="true" />
|
||||||
|
</activity>
|
||||||
|
|
||||||
<activity android:name="Settings$HomeSettingsActivity"
|
<activity android:name="Settings$HomeSettingsActivity"
|
||||||
android:label="@string/home_settings"
|
android:label="@string/home_settings"
|
||||||
android:taskAffinity="">
|
android:taskAffinity="">
|
||||||
|
@@ -13,7 +13,8 @@
|
|||||||
See the License for the specific language governing permissions and
|
See the License for the specific language governing permissions and
|
||||||
limitations under the License.
|
limitations under the License.
|
||||||
-->
|
-->
|
||||||
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
|
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
android:orientation="vertical"
|
||||||
android:layout_width="match_parent"
|
android:layout_width="match_parent"
|
||||||
android:layout_height="wrap_content" >
|
android:layout_height="wrap_content" >
|
||||||
|
|
||||||
@@ -29,4 +30,34 @@
|
|||||||
|
|
||||||
</EditText>
|
</EditText>
|
||||||
|
|
||||||
</FrameLayout>
|
<RadioGroup
|
||||||
|
android:id="@+id/rule_types"
|
||||||
|
android:visibility="gone"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:layout_marginLeft="22dp"
|
||||||
|
android:layout_marginRight="22dp"
|
||||||
|
android:layout_marginTop="16dp"
|
||||||
|
android:orientation="vertical"
|
||||||
|
android:checkedButton="@+id/rule_type_schedule" >
|
||||||
|
|
||||||
|
<RadioButton android:id="@+id/rule_type_schedule"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:text="@string/zen_schedule_rule_type_name" />
|
||||||
|
|
||||||
|
<RadioButton android:id="@+id/rule_type_2"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="wrap_content" />
|
||||||
|
|
||||||
|
<RadioButton android:id="@+id/rule_type_3"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="wrap_content" />
|
||||||
|
|
||||||
|
<RadioButton android:id="@+id/rule_type_4"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="wrap_content" />
|
||||||
|
|
||||||
|
</RadioGroup>
|
||||||
|
|
||||||
|
</LinearLayout>
|
||||||
|
@@ -5908,11 +5908,23 @@
|
|||||||
<!-- [CHAR LIMIT=40] Zen mode settings: Delete rule dialog button caption -->
|
<!-- [CHAR LIMIT=40] Zen mode settings: Delete rule dialog button caption -->
|
||||||
<string name="zen_mode_delete_rule_button">Delete</string>
|
<string name="zen_mode_delete_rule_button">Delete</string>
|
||||||
|
|
||||||
|
<!-- [CHAR LIMIT=40] Zen mode settings: External rule type -->
|
||||||
|
<string name="zen_mode_rule_type">Rule type</string>
|
||||||
|
|
||||||
|
<!-- [CHAR LIMIT=40] Zen mode settings: External rule type name if unknown -->
|
||||||
|
<string name="zen_mode_rule_type_unknown">Unknown</string>
|
||||||
|
|
||||||
|
<!-- [CHAR LIMIT=40] Zen mode settings: Configure external rule -->
|
||||||
|
<string name="zen_mode_configure_rule">Configure rule</string>
|
||||||
|
|
||||||
|
<!-- [CHAR LIMIT=40] Zen mode settings: Schedule rule type name -->
|
||||||
|
<string name="zen_schedule_rule_type_name">Schedule rule</string>
|
||||||
|
|
||||||
<!-- [CHAR LIMIT=40] Zen mode settings: Text to display if rule isn't found -->
|
<!-- [CHAR LIMIT=40] Zen mode settings: Text to display if rule isn't found -->
|
||||||
<string name="zen_mode_rule_not_found_text">Rule not found.</string>
|
<string name="zen_mode_rule_not_found_text">Rule not found.</string>
|
||||||
|
|
||||||
<!-- [CHAR LIMIT=40] Zen mode settings: Rule summary template (when enabled) -->
|
<!-- [CHAR LIMIT=40] Zen mode settings: Rule summary template (when enabled) -->
|
||||||
<string name="zen_mode_rule_summary_template"><xliff:g id="days" example="Sun - Thu">%1$s</xliff:g> / <xliff:g id="timerange" example="10:00 PM to 7:30 AM">%2$s</xliff:g> / <xliff:g id="mode" example="Alarms only">%3$s</xliff:g></string>
|
<string name="zen_mode_rule_summary_combination"><xliff:g id="description" example="Sun - Thu">%1$s</xliff:g> / <xliff:g id="mode" example="Alarms only">%2$s</xliff:g></string>
|
||||||
|
|
||||||
<!-- [CHAR LIMIT=40] Zen mode settings: Timebased rule days option title -->
|
<!-- [CHAR LIMIT=40] Zen mode settings: Timebased rule days option title -->
|
||||||
<string name="zen_mode_schedule_rule_days">Days</string>
|
<string name="zen_mode_schedule_rule_days">Days</string>
|
||||||
|
45
res/xml/zen_mode_external_rule_settings.xml
Normal file
45
res/xml/zen_mode_external_rule_settings.xml
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<!--
|
||||||
|
Copyright (C) 2015 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.
|
||||||
|
-->
|
||||||
|
|
||||||
|
<PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
android:key="zen_mode_external_rule_settings" >
|
||||||
|
|
||||||
|
<!-- Rule name -->
|
||||||
|
<Preference
|
||||||
|
android:key="rule_name"
|
||||||
|
android:title="@string/zen_mode_rule_name"
|
||||||
|
android:persistent="false" />
|
||||||
|
|
||||||
|
<!-- Rule type -->
|
||||||
|
<Preference
|
||||||
|
android:key="type"
|
||||||
|
android:title="@string/zen_mode_rule_type"
|
||||||
|
android:persistent="false" />
|
||||||
|
|
||||||
|
<!-- Configure -->
|
||||||
|
<Preference
|
||||||
|
android:key="configure"
|
||||||
|
android:title="@string/zen_mode_configure_rule"
|
||||||
|
android:persistent="false" />
|
||||||
|
|
||||||
|
<!-- Zen mode -->
|
||||||
|
<com.android.settings.DropDownPreference
|
||||||
|
android:key="zen_mode"
|
||||||
|
android:title="@string/zen_mode_settings_title"
|
||||||
|
android:persistent="false" />
|
||||||
|
|
||||||
|
</PreferenceScreen>
|
@@ -98,6 +98,7 @@ public class Settings extends SettingsActivity {
|
|||||||
public static class ZenModePrioritySettingsActivity extends SettingsActivity { /* empty */ }
|
public static class ZenModePrioritySettingsActivity extends SettingsActivity { /* empty */ }
|
||||||
public static class ZenModeAutomationSettingsActivity extends SettingsActivity { /* empty */ }
|
public static class ZenModeAutomationSettingsActivity extends SettingsActivity { /* empty */ }
|
||||||
public static class ZenModeScheduleRuleSettingsActivity extends SettingsActivity { /* empty */ }
|
public static class ZenModeScheduleRuleSettingsActivity extends SettingsActivity { /* empty */ }
|
||||||
|
public static class ZenModeExternalRuleSettingsActivity extends SettingsActivity { /* empty */ }
|
||||||
public static class NotificationSettingsActivity extends SettingsActivity { /* empty */ }
|
public static class NotificationSettingsActivity extends SettingsActivity { /* empty */ }
|
||||||
public static class NotificationAppListActivity extends SettingsActivity { /* empty */ }
|
public static class NotificationAppListActivity extends SettingsActivity { /* empty */ }
|
||||||
public static class AppNotificationSettingsActivity extends SettingsActivity { /* empty */ }
|
public static class AppNotificationSettingsActivity extends SettingsActivity { /* empty */ }
|
||||||
|
@@ -95,11 +95,11 @@ import com.android.settings.location.LocationSettings;
|
|||||||
import com.android.settings.nfc.AndroidBeam;
|
import com.android.settings.nfc.AndroidBeam;
|
||||||
import com.android.settings.nfc.PaymentSettings;
|
import com.android.settings.nfc.PaymentSettings;
|
||||||
import com.android.settings.notification.AppNotificationSettings;
|
import com.android.settings.notification.AppNotificationSettings;
|
||||||
import com.android.settings.notification.ConditionProviderSettings;
|
|
||||||
import com.android.settings.notification.NotificationAccessSettings;
|
import com.android.settings.notification.NotificationAccessSettings;
|
||||||
import com.android.settings.notification.NotificationSettings;
|
import com.android.settings.notification.NotificationSettings;
|
||||||
import com.android.settings.notification.NotificationStation;
|
import com.android.settings.notification.NotificationStation;
|
||||||
import com.android.settings.notification.OtherSoundSettings;
|
import com.android.settings.notification.OtherSoundSettings;
|
||||||
|
import com.android.settings.notification.ZenModeExternalRuleSettings;
|
||||||
import com.android.settings.notification.ZenModeSettings;
|
import com.android.settings.notification.ZenModeSettings;
|
||||||
import com.android.settings.notification.ZenModeScheduleRuleSettings;
|
import com.android.settings.notification.ZenModeScheduleRuleSettings;
|
||||||
import com.android.settings.print.PrintJobSettingsFragment;
|
import com.android.settings.print.PrintJobSettingsFragment;
|
||||||
@@ -320,7 +320,6 @@ public class SettingsActivity extends Activity
|
|||||||
DreamSettings.class.getName(),
|
DreamSettings.class.getName(),
|
||||||
UserSettings.class.getName(),
|
UserSettings.class.getName(),
|
||||||
NotificationAccessSettings.class.getName(),
|
NotificationAccessSettings.class.getName(),
|
||||||
ConditionProviderSettings.class.getName(),
|
|
||||||
PrintSettingsFragment.class.getName(),
|
PrintSettingsFragment.class.getName(),
|
||||||
PrintJobSettingsFragment.class.getName(),
|
PrintJobSettingsFragment.class.getName(),
|
||||||
TrustedCredentialsSettings.class.getName(),
|
TrustedCredentialsSettings.class.getName(),
|
||||||
@@ -337,6 +336,7 @@ public class SettingsActivity extends Activity
|
|||||||
ApnSettings.class.getName(),
|
ApnSettings.class.getName(),
|
||||||
WifiCallingSettings.class.getName(),
|
WifiCallingSettings.class.getName(),
|
||||||
ZenModeScheduleRuleSettings.class.getName(),
|
ZenModeScheduleRuleSettings.class.getName(),
|
||||||
|
ZenModeExternalRuleSettings.class.getName(),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@@ -1,50 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (C) 2014 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.notification;
|
|
||||||
|
|
||||||
import android.content.Context;
|
|
||||||
import android.content.pm.PackageManager;
|
|
||||||
import android.provider.Settings;
|
|
||||||
import android.service.notification.ConditionProviderService;
|
|
||||||
|
|
||||||
public class ConditionProviderSettings extends ManagedServiceSettings {
|
|
||||||
private static final String TAG = ConditionProviderSettings.class.getSimpleName();
|
|
||||||
private static final Config CONFIG = getConditionProviderConfig();
|
|
||||||
|
|
||||||
private static Config getConditionProviderConfig() {
|
|
||||||
final Config c = new Config();
|
|
||||||
c.tag = TAG;
|
|
||||||
c.setting = Settings.Secure.ENABLED_CONDITION_PROVIDERS;
|
|
||||||
c.intentAction = ConditionProviderService.SERVICE_INTERFACE;
|
|
||||||
c.permission = android.Manifest.permission.BIND_CONDITION_PROVIDER_SERVICE;
|
|
||||||
c.noun = "condition provider";
|
|
||||||
return c;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected Config getConfig() {
|
|
||||||
return CONFIG;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static int getProviderCount(PackageManager pm) {
|
|
||||||
return getServicesCount(CONFIG, pm);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static int getEnabledProviderCount(Context context) {
|
|
||||||
return getEnabledServicesCount(CONFIG, context);
|
|
||||||
}
|
|
||||||
}
|
|
@@ -16,28 +16,17 @@
|
|||||||
|
|
||||||
package com.android.settings.notification;
|
package com.android.settings.notification;
|
||||||
|
|
||||||
import android.app.ActivityManager;
|
|
||||||
import android.app.AlertDialog;
|
import android.app.AlertDialog;
|
||||||
import android.app.Dialog;
|
import android.app.Dialog;
|
||||||
import android.app.DialogFragment;
|
import android.app.DialogFragment;
|
||||||
import android.app.ListFragment;
|
import android.app.ListFragment;
|
||||||
import android.content.BroadcastReceiver;
|
|
||||||
import android.content.ComponentName;
|
import android.content.ComponentName;
|
||||||
import android.content.ContentResolver;
|
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.DialogInterface;
|
import android.content.DialogInterface;
|
||||||
import android.content.Intent;
|
|
||||||
import android.content.IntentFilter;
|
|
||||||
import android.content.pm.PackageItemInfo;
|
import android.content.pm.PackageItemInfo;
|
||||||
import android.content.pm.PackageManager;
|
import android.content.pm.PackageManager;
|
||||||
import android.content.pm.ResolveInfo;
|
|
||||||
import android.content.pm.ServiceInfo;
|
import android.content.pm.ServiceInfo;
|
||||||
import android.database.ContentObserver;
|
|
||||||
import android.net.Uri;
|
|
||||||
import android.os.Bundle;
|
import android.os.Bundle;
|
||||||
import android.os.Handler;
|
|
||||||
import android.provider.Settings;
|
|
||||||
import android.util.Slog;
|
|
||||||
import android.view.LayoutInflater;
|
import android.view.LayoutInflater;
|
||||||
import android.view.View;
|
import android.view.View;
|
||||||
import android.view.ViewGroup;
|
import android.view.ViewGroup;
|
||||||
@@ -49,17 +38,15 @@ import android.widget.TextView;
|
|||||||
|
|
||||||
import com.android.settings.R;
|
import com.android.settings.R;
|
||||||
|
|
||||||
import java.util.HashSet;
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
public abstract class ManagedServiceSettings extends ListFragment {
|
public abstract class ManagedServiceSettings extends ListFragment {
|
||||||
private static final boolean SHOW_PACKAGE_NAME = false;
|
private static final boolean SHOW_PACKAGE_NAME = false;
|
||||||
|
|
||||||
private final Config mConfig;
|
private final Config mConfig;
|
||||||
private PackageManager mPM;
|
|
||||||
private ContentResolver mCR;
|
|
||||||
|
|
||||||
private final HashSet<ComponentName> mEnabledServices = new HashSet<ComponentName>();
|
private PackageManager mPM;
|
||||||
|
private ServiceListing mServiceListing;
|
||||||
private ServiceListAdapter mListAdapter;
|
private ServiceListAdapter mListAdapter;
|
||||||
|
|
||||||
abstract protected Config getConfig();
|
abstract protected Config getConfig();
|
||||||
@@ -68,19 +55,65 @@ public abstract class ManagedServiceSettings extends ListFragment {
|
|||||||
mConfig = getConfig();
|
mConfig = getConfig();
|
||||||
}
|
}
|
||||||
|
|
||||||
private final ContentObserver mSettingsObserver = new ContentObserver(new Handler()) {
|
|
||||||
@Override
|
@Override
|
||||||
public void onChange(boolean selfChange, Uri uri) {
|
public void onCreate(Bundle icicle) {
|
||||||
updateList();
|
super.onCreate(icicle);
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
private final BroadcastReceiver mPackageReceiver = new BroadcastReceiver() {
|
mPM = getActivity().getPackageManager();
|
||||||
|
mServiceListing = new ServiceListing(getActivity(), mConfig);
|
||||||
|
mServiceListing.addCallback(new ServiceListing.Callback() {
|
||||||
@Override
|
@Override
|
||||||
public void onReceive(Context context, Intent intent) {
|
public void onServicesReloaded(List<ServiceInfo> services) {
|
||||||
updateList();
|
updateList(services);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
mListAdapter = new ServiceListAdapter(getActivity());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public View onCreateView(LayoutInflater inflater, ViewGroup container,
|
||||||
|
Bundle savedInstanceState) {
|
||||||
|
View v = inflater.inflate(R.layout.managed_service_settings, container, false);
|
||||||
|
TextView empty = (TextView) v.findViewById(android.R.id.empty);
|
||||||
|
empty.setText(mConfig.emptyText);
|
||||||
|
return v;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onResume() {
|
||||||
|
super.onResume();
|
||||||
|
mServiceListing.reload();
|
||||||
|
mServiceListing.setListening(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onPause() {
|
||||||
|
super.onPause();
|
||||||
|
mServiceListing.setListening(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void updateList(List<ServiceInfo> services) {
|
||||||
|
mListAdapter.clear();
|
||||||
|
mListAdapter.addAll(services);
|
||||||
|
mListAdapter.sort(new PackageItemInfo.DisplayNameComparator(mPM));
|
||||||
|
|
||||||
|
getListView().setAdapter(mListAdapter);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onListItemClick(ListView l, View v, int position, long id) {
|
||||||
|
ServiceInfo info = mListAdapter.getItem(position);
|
||||||
|
final ComponentName cn = new ComponentName(info.packageName, info.name);
|
||||||
|
if (mServiceListing.isEnabled(cn)) {
|
||||||
|
// the simple version: disabling
|
||||||
|
mServiceListing.setEnabled(cn, false);
|
||||||
|
} else {
|
||||||
|
// show a scary dialog
|
||||||
|
new ScaryWarningDialogFragment()
|
||||||
|
.setServiceInfo(cn, info.loadLabel(mPM).toString())
|
||||||
|
.show(getFragmentManager(), "dialog");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
};
|
|
||||||
|
|
||||||
public class ScaryWarningDialogFragment extends DialogFragment {
|
public class ScaryWarningDialogFragment extends DialogFragment {
|
||||||
static final String KEY_COMPONENT = "c";
|
static final String KEY_COMPONENT = "c";
|
||||||
@@ -99,7 +132,8 @@ public abstract class ManagedServiceSettings extends ListFragment {
|
|||||||
super.onCreate(savedInstanceState);
|
super.onCreate(savedInstanceState);
|
||||||
final Bundle args = getArguments();
|
final Bundle args = getArguments();
|
||||||
final String label = args.getString(KEY_LABEL);
|
final String label = args.getString(KEY_LABEL);
|
||||||
final ComponentName cn = ComponentName.unflattenFromString(args.getString(KEY_COMPONENT));
|
final ComponentName cn = ComponentName.unflattenFromString(args
|
||||||
|
.getString(KEY_COMPONENT));
|
||||||
|
|
||||||
final String title = getResources().getString(mConfig.warningDialogTitle, label);
|
final String title = getResources().getString(mConfig.warningDialogTitle, label);
|
||||||
final String summary = getResources().getString(mConfig.warningDialogSummary, label);
|
final String summary = getResources().getString(mConfig.warningDialogSummary, label);
|
||||||
@@ -110,8 +144,7 @@ public abstract class ManagedServiceSettings extends ListFragment {
|
|||||||
.setPositiveButton(android.R.string.ok,
|
.setPositiveButton(android.R.string.ok,
|
||||||
new DialogInterface.OnClickListener() {
|
new DialogInterface.OnClickListener() {
|
||||||
public void onClick(DialogInterface dialog, int id) {
|
public void onClick(DialogInterface dialog, int id) {
|
||||||
mEnabledServices.add(cn);
|
mServiceListing.setEnabled(cn, true);
|
||||||
saveEnabledServices();
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.setNegativeButton(android.R.string.cancel,
|
.setNegativeButton(android.R.string.cancel,
|
||||||
@@ -124,151 +157,6 @@ public abstract class ManagedServiceSettings extends ListFragment {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onCreate(Bundle icicle) {
|
|
||||||
super.onCreate(icicle);
|
|
||||||
|
|
||||||
mPM = getActivity().getPackageManager();
|
|
||||||
mCR = getActivity().getContentResolver();
|
|
||||||
mListAdapter = new ServiceListAdapter(getActivity());
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public View onCreateView(LayoutInflater inflater, ViewGroup container,
|
|
||||||
Bundle savedInstanceState) {
|
|
||||||
View v = inflater.inflate(R.layout.managed_service_settings, container, false);
|
|
||||||
TextView empty = (TextView) v.findViewById(android.R.id.empty);
|
|
||||||
empty.setText(mConfig.emptyText);
|
|
||||||
return v;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onResume() {
|
|
||||||
super.onResume();
|
|
||||||
updateList();
|
|
||||||
|
|
||||||
// listen for package changes
|
|
||||||
IntentFilter filter = new IntentFilter();
|
|
||||||
filter.addAction(Intent.ACTION_PACKAGE_ADDED);
|
|
||||||
filter.addAction(Intent.ACTION_PACKAGE_CHANGED);
|
|
||||||
filter.addAction(Intent.ACTION_PACKAGE_REMOVED);
|
|
||||||
filter.addAction(Intent.ACTION_PACKAGE_REPLACED);
|
|
||||||
filter.addDataScheme("package");
|
|
||||||
getActivity().registerReceiver(mPackageReceiver, filter);
|
|
||||||
|
|
||||||
mCR.registerContentObserver(Settings.Secure.getUriFor(mConfig.setting),
|
|
||||||
false, mSettingsObserver);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onPause() {
|
|
||||||
super.onPause();
|
|
||||||
|
|
||||||
getActivity().unregisterReceiver(mPackageReceiver);
|
|
||||||
mCR.unregisterContentObserver(mSettingsObserver);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void loadEnabledServices() {
|
|
||||||
mEnabledServices.clear();
|
|
||||||
final String flat = Settings.Secure.getString(mCR, mConfig.setting);
|
|
||||||
if (flat != null && !"".equals(flat)) {
|
|
||||||
final String[] names = flat.split(":");
|
|
||||||
for (int i = 0; i < names.length; i++) {
|
|
||||||
final ComponentName cn = ComponentName.unflattenFromString(names[i]);
|
|
||||||
if (cn != null) {
|
|
||||||
mEnabledServices.add(cn);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void saveEnabledServices() {
|
|
||||||
StringBuilder sb = null;
|
|
||||||
for (ComponentName cn : mEnabledServices) {
|
|
||||||
if (sb == null) {
|
|
||||||
sb = new StringBuilder();
|
|
||||||
} else {
|
|
||||||
sb.append(':');
|
|
||||||
}
|
|
||||||
sb.append(cn.flattenToString());
|
|
||||||
}
|
|
||||||
Settings.Secure.putString(mCR,
|
|
||||||
mConfig.setting,
|
|
||||||
sb != null ? sb.toString() : "");
|
|
||||||
}
|
|
||||||
|
|
||||||
private void updateList() {
|
|
||||||
loadEnabledServices();
|
|
||||||
|
|
||||||
getServices(mConfig, mListAdapter, mPM);
|
|
||||||
mListAdapter.sort(new PackageItemInfo.DisplayNameComparator(mPM));
|
|
||||||
|
|
||||||
getListView().setAdapter(mListAdapter);
|
|
||||||
}
|
|
||||||
|
|
||||||
protected static int getEnabledServicesCount(Config config, Context context) {
|
|
||||||
final String flat = Settings.Secure.getString(context.getContentResolver(), config.setting);
|
|
||||||
if (flat == null || "".equals(flat)) return 0;
|
|
||||||
final String[] components = flat.split(":");
|
|
||||||
return components.length;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected static int getServicesCount(Config c, PackageManager pm) {
|
|
||||||
return getServices(c, null, pm);
|
|
||||||
}
|
|
||||||
|
|
||||||
private static int getServices(Config c, ArrayAdapter<ServiceInfo> adapter, PackageManager pm) {
|
|
||||||
int services = 0;
|
|
||||||
if (adapter != null) {
|
|
||||||
adapter.clear();
|
|
||||||
}
|
|
||||||
final int user = ActivityManager.getCurrentUser();
|
|
||||||
|
|
||||||
List<ResolveInfo> installedServices = pm.queryIntentServicesAsUser(
|
|
||||||
new Intent(c.intentAction),
|
|
||||||
PackageManager.GET_SERVICES | PackageManager.GET_META_DATA,
|
|
||||||
user);
|
|
||||||
|
|
||||||
for (int i = 0, count = installedServices.size(); i < count; i++) {
|
|
||||||
ResolveInfo resolveInfo = installedServices.get(i);
|
|
||||||
ServiceInfo info = resolveInfo.serviceInfo;
|
|
||||||
|
|
||||||
if (!c.permission.equals(info.permission)) {
|
|
||||||
Slog.w(c.tag, "Skipping " + c.noun + " service "
|
|
||||||
+ info.packageName + "/" + info.name
|
|
||||||
+ ": it does not require the permission "
|
|
||||||
+ c.permission);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
if (adapter != null) {
|
|
||||||
adapter.add(info);
|
|
||||||
}
|
|
||||||
services++;
|
|
||||||
}
|
|
||||||
return services;
|
|
||||||
}
|
|
||||||
|
|
||||||
private boolean isServiceEnabled(ServiceInfo info) {
|
|
||||||
final ComponentName cn = new ComponentName(info.packageName, info.name);
|
|
||||||
return mEnabledServices.contains(cn);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onListItemClick(ListView l, View v, int position, long id) {
|
|
||||||
ServiceInfo info = mListAdapter.getItem(position);
|
|
||||||
final ComponentName cn = new ComponentName(info.packageName, info.name);
|
|
||||||
if (mEnabledServices.contains(cn)) {
|
|
||||||
// the simple version: disabling
|
|
||||||
mEnabledServices.remove(cn);
|
|
||||||
saveEnabledServices();
|
|
||||||
} else {
|
|
||||||
// show a scary dialog
|
|
||||||
new ScaryWarningDialogFragment()
|
|
||||||
.setServiceInfo(cn, info.loadLabel(mPM).toString())
|
|
||||||
.show(getFragmentManager(), "dialog");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private static class ViewHolder {
|
private static class ViewHolder {
|
||||||
ImageView icon;
|
ImageView icon;
|
||||||
TextView name;
|
TextView name;
|
||||||
@@ -327,7 +215,8 @@ public abstract class ManagedServiceSettings extends ListFragment {
|
|||||||
} else {
|
} else {
|
||||||
vh.description.setVisibility(View.GONE);
|
vh.description.setVisibility(View.GONE);
|
||||||
}
|
}
|
||||||
vh.checkbox.setChecked(isServiceEnabled(info));
|
final ComponentName cn = new ComponentName(info.packageName, info.name);
|
||||||
|
vh.checkbox.setChecked(mServiceListing.isEnabled(cn));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -46,10 +46,10 @@ public class NotificationAccessSettings extends ManagedServiceSettings {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public static int getListenersCount(PackageManager pm) {
|
public static int getListenersCount(PackageManager pm) {
|
||||||
return getServicesCount(CONFIG, pm);
|
return ServiceListing.getServicesCount(CONFIG, pm);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static int getEnabledListenersCount(Context context) {
|
public static int getEnabledListenersCount(Context context) {
|
||||||
return getEnabledServicesCount(CONFIG, context);
|
return ServiceListing.getEnabledServicesCount(CONFIG, context);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
206
src/com/android/settings/notification/ServiceListing.java
Normal file
206
src/com/android/settings/notification/ServiceListing.java
Normal file
@@ -0,0 +1,206 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2015 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.notification;
|
||||||
|
|
||||||
|
import android.app.ActivityManager;
|
||||||
|
import android.content.BroadcastReceiver;
|
||||||
|
import android.content.ComponentName;
|
||||||
|
import android.content.ContentResolver;
|
||||||
|
import android.content.Context;
|
||||||
|
import android.content.Intent;
|
||||||
|
import android.content.IntentFilter;
|
||||||
|
import android.content.pm.PackageManager;
|
||||||
|
import android.content.pm.ResolveInfo;
|
||||||
|
import android.content.pm.ServiceInfo;
|
||||||
|
import android.database.ContentObserver;
|
||||||
|
import android.net.Uri;
|
||||||
|
import android.os.Handler;
|
||||||
|
import android.provider.Settings;
|
||||||
|
import android.util.Slog;
|
||||||
|
|
||||||
|
import com.android.settings.notification.ManagedServiceSettings.Config;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
public class ServiceListing {
|
||||||
|
private final ContentResolver mContentResolver;
|
||||||
|
private final Context mContext;
|
||||||
|
private final Config mConfig;
|
||||||
|
private final HashSet<ComponentName> mEnabledServices = new HashSet<ComponentName>();
|
||||||
|
private final List<ServiceInfo> mServices = new ArrayList<ServiceInfo>();
|
||||||
|
private final List<Callback> mCallbacks = new ArrayList<Callback>();
|
||||||
|
|
||||||
|
private boolean mListening;
|
||||||
|
|
||||||
|
public ServiceListing(Context context, Config config) {
|
||||||
|
mContext = context;
|
||||||
|
mConfig = config;
|
||||||
|
mContentResolver = context.getContentResolver();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void addCallback(Callback callback) {
|
||||||
|
mCallbacks.add(callback);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void removeCallback(Callback callback) {
|
||||||
|
mCallbacks.remove(callback);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setListening(boolean listening) {
|
||||||
|
if (mListening == listening) return;
|
||||||
|
mListening = listening;
|
||||||
|
if (mListening) {
|
||||||
|
// listen for package changes
|
||||||
|
IntentFilter filter = new IntentFilter();
|
||||||
|
filter.addAction(Intent.ACTION_PACKAGE_ADDED);
|
||||||
|
filter.addAction(Intent.ACTION_PACKAGE_CHANGED);
|
||||||
|
filter.addAction(Intent.ACTION_PACKAGE_REMOVED);
|
||||||
|
filter.addAction(Intent.ACTION_PACKAGE_REPLACED);
|
||||||
|
filter.addDataScheme("package");
|
||||||
|
mContext.registerReceiver(mPackageReceiver, filter);
|
||||||
|
mContentResolver.registerContentObserver(Settings.Secure.getUriFor(mConfig.setting),
|
||||||
|
false, mSettingsObserver);
|
||||||
|
} else {
|
||||||
|
mContext.unregisterReceiver(mPackageReceiver);
|
||||||
|
mContentResolver.unregisterContentObserver(mSettingsObserver);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static int getEnabledServicesCount(Config config, Context context) {
|
||||||
|
final String flat = Settings.Secure.getString(context.getContentResolver(), config.setting);
|
||||||
|
if (flat == null || "".equals(flat)) return 0;
|
||||||
|
final String[] components = flat.split(":");
|
||||||
|
return components.length;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static int getServicesCount(Config c, PackageManager pm) {
|
||||||
|
return getServices(c, null, pm);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static ServiceInfo findService(Context context, Config config, final ComponentName cn) {
|
||||||
|
final ServiceListing listing = new ServiceListing(context, config);
|
||||||
|
final List<ServiceInfo> services = listing.reload();
|
||||||
|
for (ServiceInfo service : services) {
|
||||||
|
final ComponentName serviceCN = new ComponentName(service.packageName, service.name);
|
||||||
|
if (serviceCN.equals(cn)) {
|
||||||
|
return service;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static int getServices(Config c, List<ServiceInfo> list, PackageManager pm) {
|
||||||
|
int services = 0;
|
||||||
|
if (list != null) {
|
||||||
|
list.clear();
|
||||||
|
}
|
||||||
|
final int user = ActivityManager.getCurrentUser();
|
||||||
|
|
||||||
|
List<ResolveInfo> installedServices = pm.queryIntentServicesAsUser(
|
||||||
|
new Intent(c.intentAction),
|
||||||
|
PackageManager.GET_SERVICES | PackageManager.GET_META_DATA,
|
||||||
|
user);
|
||||||
|
|
||||||
|
for (int i = 0, count = installedServices.size(); i < count; i++) {
|
||||||
|
ResolveInfo resolveInfo = installedServices.get(i);
|
||||||
|
ServiceInfo info = resolveInfo.serviceInfo;
|
||||||
|
|
||||||
|
if (!c.permission.equals(info.permission)) {
|
||||||
|
Slog.w(c.tag, "Skipping " + c.noun + " service "
|
||||||
|
+ info.packageName + "/" + info.name
|
||||||
|
+ ": it does not require the permission "
|
||||||
|
+ c.permission);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (list != null) {
|
||||||
|
list.add(info);
|
||||||
|
}
|
||||||
|
services++;
|
||||||
|
}
|
||||||
|
return services;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void saveEnabledServices() {
|
||||||
|
StringBuilder sb = null;
|
||||||
|
for (ComponentName cn : mEnabledServices) {
|
||||||
|
if (sb == null) {
|
||||||
|
sb = new StringBuilder();
|
||||||
|
} else {
|
||||||
|
sb.append(':');
|
||||||
|
}
|
||||||
|
sb.append(cn.flattenToString());
|
||||||
|
}
|
||||||
|
Settings.Secure.putString(mContentResolver, mConfig.setting,
|
||||||
|
sb != null ? sb.toString() : "");
|
||||||
|
}
|
||||||
|
|
||||||
|
private void loadEnabledServices() {
|
||||||
|
mEnabledServices.clear();
|
||||||
|
final String flat = Settings.Secure.getString(mContentResolver, mConfig.setting);
|
||||||
|
if (flat != null && !"".equals(flat)) {
|
||||||
|
final String[] names = flat.split(":");
|
||||||
|
for (int i = 0; i < names.length; i++) {
|
||||||
|
final ComponentName cn = ComponentName.unflattenFromString(names[i]);
|
||||||
|
if (cn != null) {
|
||||||
|
mEnabledServices.add(cn);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<ServiceInfo> reload() {
|
||||||
|
loadEnabledServices();
|
||||||
|
getServices(mConfig, mServices, mContext.getPackageManager());
|
||||||
|
for (Callback callback : mCallbacks) {
|
||||||
|
callback.onServicesReloaded(mServices);
|
||||||
|
}
|
||||||
|
return mServices;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isEnabled(ComponentName cn) {
|
||||||
|
return mEnabledServices.contains(cn);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setEnabled(ComponentName cn, boolean enabled) {
|
||||||
|
if (enabled) {
|
||||||
|
mEnabledServices.add(cn);
|
||||||
|
} else {
|
||||||
|
mEnabledServices.remove(cn);
|
||||||
|
}
|
||||||
|
saveEnabledServices();
|
||||||
|
}
|
||||||
|
|
||||||
|
private final ContentObserver mSettingsObserver = new ContentObserver(new Handler()) {
|
||||||
|
@Override
|
||||||
|
public void onChange(boolean selfChange, Uri uri) {
|
||||||
|
reload();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
private final BroadcastReceiver mPackageReceiver = new BroadcastReceiver() {
|
||||||
|
@Override
|
||||||
|
public void onReceive(Context context, Intent intent) {
|
||||||
|
reload();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
public interface Callback {
|
||||||
|
void onServicesReloaded(List<ServiceInfo> services);
|
||||||
|
}
|
||||||
|
}
|
@@ -18,12 +18,16 @@ package com.android.settings.notification;
|
|||||||
|
|
||||||
import static android.service.notification.ZenModeConfig.ALL_DAYS;
|
import static android.service.notification.ZenModeConfig.ALL_DAYS;
|
||||||
|
|
||||||
|
import android.content.ComponentName;
|
||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
|
import android.content.pm.ServiceInfo;
|
||||||
import android.os.Bundle;
|
import android.os.Bundle;
|
||||||
import android.preference.Preference;
|
import android.preference.Preference;
|
||||||
import android.preference.Preference.OnPreferenceClickListener;
|
import android.preference.Preference.OnPreferenceClickListener;
|
||||||
import android.preference.PreferenceScreen;
|
import android.preference.PreferenceScreen;
|
||||||
|
import android.provider.Settings;
|
||||||
import android.provider.Settings.Global;
|
import android.provider.Settings.Global;
|
||||||
|
import android.service.notification.ConditionProviderService;
|
||||||
import android.service.notification.ZenModeConfig;
|
import android.service.notification.ZenModeConfig;
|
||||||
import android.service.notification.ZenModeConfig.ScheduleInfo;
|
import android.service.notification.ZenModeConfig.ScheduleInfo;
|
||||||
import android.service.notification.ZenModeConfig.ZenRule;
|
import android.service.notification.ZenModeConfig.ZenRule;
|
||||||
@@ -35,30 +39,39 @@ import android.view.MenuItem;
|
|||||||
|
|
||||||
import com.android.internal.logging.MetricsLogger;
|
import com.android.internal.logging.MetricsLogger;
|
||||||
import com.android.settings.R;
|
import com.android.settings.R;
|
||||||
|
import com.android.settings.notification.ManagedServiceSettings.Config;
|
||||||
|
import com.android.settings.notification.ZenRuleNameDialog.RuleInfo;
|
||||||
|
|
||||||
import java.text.SimpleDateFormat;
|
import java.text.SimpleDateFormat;
|
||||||
import java.util.Calendar;
|
import java.util.Calendar;
|
||||||
|
import java.util.List;
|
||||||
import java.util.TreeSet;
|
import java.util.TreeSet;
|
||||||
|
|
||||||
public class ZenModeAutomationSettings extends ZenModeSettingsBase {
|
public class ZenModeAutomationSettings extends ZenModeSettingsBase {
|
||||||
private static final SimpleDateFormat DAY_FORMAT = new SimpleDateFormat("EEE");
|
private static final SimpleDateFormat DAY_FORMAT = new SimpleDateFormat("EEE");
|
||||||
|
|
||||||
|
static final Config CONFIG = getConditionProviderConfig();
|
||||||
|
|
||||||
private final Calendar mCalendar = Calendar.getInstance();
|
private final Calendar mCalendar = Calendar.getInstance();
|
||||||
|
|
||||||
|
private ServiceListing mServiceListing;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onCreate(Bundle icicle) {
|
public void onCreate(Bundle icicle) {
|
||||||
super.onCreate(icicle);
|
super.onCreate(icicle);
|
||||||
|
|
||||||
setHasOptionsMenu(true);
|
setHasOptionsMenu(true);
|
||||||
|
|
||||||
addPreferencesFromResource(R.xml.zen_mode_automation_settings);
|
addPreferencesFromResource(R.xml.zen_mode_automation_settings);
|
||||||
|
mServiceListing = new ServiceListing(mContext, CONFIG);
|
||||||
|
mServiceListing.addCallback(mServiceListingCallback);
|
||||||
|
mServiceListing.reload();
|
||||||
|
mServiceListing.setListening(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void showRule(String ruleId, String ruleName) {
|
@Override
|
||||||
if (DEBUG) Log.d(TAG, "showRule " + ruleId + " name=" + ruleName);
|
public void onDestroy() {
|
||||||
mContext.startActivity(new Intent(ZenModeScheduleRuleSettings.ACTION)
|
super.onDestroy();
|
||||||
.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP)
|
mServiceListing.setListening(false);
|
||||||
.putExtra(ZenModeScheduleRuleSettings.EXTRA_RULE_ID, ruleId));
|
mServiceListing.removeCallback(mServiceListingCallback);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@@ -75,29 +88,6 @@ public class ZenModeAutomationSettings extends ZenModeSettingsBase {
|
|||||||
return super.onOptionsItemSelected(item);
|
return super.onOptionsItemSelected(item);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void showAddRuleDialog() {
|
|
||||||
new ZenRuleNameDialog(mContext, "", mConfig.getAutomaticRuleNames()) {
|
|
||||||
@Override
|
|
||||||
public void onOk(String ruleName) {
|
|
||||||
final ScheduleInfo schedule = new ScheduleInfo();
|
|
||||||
schedule.days = ZenModeConfig.ALL_DAYS;
|
|
||||||
schedule.startHour = 22;
|
|
||||||
schedule.endHour = 7;
|
|
||||||
final ZenRule rule = new ZenRule();
|
|
||||||
rule.name = ruleName;
|
|
||||||
rule.enabled = true;
|
|
||||||
rule.zenMode = Global.ZEN_MODE_IMPORTANT_INTERRUPTIONS;
|
|
||||||
rule.conditionId = ZenModeConfig.toScheduleConditionId(schedule);
|
|
||||||
final ZenModeConfig newConfig = mConfig.copy();
|
|
||||||
final String ruleId = newConfig.newRuleId();
|
|
||||||
newConfig.automaticRules.put(ruleId, rule);
|
|
||||||
if (setZenModeConfig(newConfig)) {
|
|
||||||
showRule(ruleId, rule.name);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}.show();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void onZenModeChanged() {
|
protected void onZenModeChanged() {
|
||||||
// don't care
|
// don't care
|
||||||
@@ -114,15 +104,42 @@ public class ZenModeAutomationSettings extends ZenModeSettingsBase {
|
|||||||
updateControls();
|
updateControls();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void showAddRuleDialog() {
|
||||||
|
new ZenRuleNameDialog(mContext, mServiceListing, null, mConfig.getAutomaticRuleNames()) {
|
||||||
|
@Override
|
||||||
|
public void onOk(String ruleName, RuleInfo ri) {
|
||||||
|
final ZenRule rule = new ZenRule();
|
||||||
|
rule.name = ruleName;
|
||||||
|
rule.enabled = true;
|
||||||
|
rule.zenMode = Global.ZEN_MODE_IMPORTANT_INTERRUPTIONS;
|
||||||
|
rule.conditionId = ri.defaultConditionId;
|
||||||
|
rule.component = ri.serviceComponent;
|
||||||
|
final ZenModeConfig newConfig = mConfig.copy();
|
||||||
|
final String ruleId = newConfig.newRuleId();
|
||||||
|
newConfig.automaticRules.put(ruleId, rule);
|
||||||
|
if (setZenModeConfig(newConfig)) {
|
||||||
|
showRule(ri.settingsAction, ri.configurationActivity, ruleId, rule.name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}.show();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void showRule(String settingsAction, ComponentName configurationActivity,
|
||||||
|
String ruleId, String ruleName) {
|
||||||
|
if (DEBUG) Log.d(TAG, "showRule " + ruleId + " name=" + ruleName);
|
||||||
|
mContext.startActivity(new Intent(settingsAction)
|
||||||
|
.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP)
|
||||||
|
.putExtra(ZenModeRuleSettingsBase.EXTRA_RULE_ID, ruleId));
|
||||||
|
}
|
||||||
|
|
||||||
private void updateControls() {
|
private void updateControls() {
|
||||||
final PreferenceScreen root = getPreferenceScreen();
|
final PreferenceScreen root = getPreferenceScreen();
|
||||||
root.removeAll();
|
root.removeAll();
|
||||||
|
|
||||||
if (mConfig == null) return;
|
if (mConfig == null) return;
|
||||||
for (int i = 0; i < mConfig.automaticRules.size(); i++) {
|
for (int i = 0; i < mConfig.automaticRules.size(); i++) {
|
||||||
final String id = mConfig.automaticRules.keyAt(i);
|
final String id = mConfig.automaticRules.keyAt(i);
|
||||||
final ZenRule rule = mConfig.automaticRules.valueAt(i);
|
final ZenRule rule = mConfig.automaticRules.valueAt(i);
|
||||||
if (!ZenModeConfig.isValidScheduleConditionId(rule.conditionId)) continue;
|
final boolean isSchedule = ZenModeConfig.isValidScheduleConditionId(rule.conditionId);
|
||||||
final Preference p = new Preference(mContext);
|
final Preference p = new Preference(mContext);
|
||||||
p.setTitle(rule.name);
|
p.setTitle(rule.name);
|
||||||
p.setSummary(computeRuleSummary(rule));
|
p.setSummary(computeRuleSummary(rule));
|
||||||
@@ -130,7 +147,9 @@ public class ZenModeAutomationSettings extends ZenModeSettingsBase {
|
|||||||
p.setOnPreferenceClickListener(new OnPreferenceClickListener() {
|
p.setOnPreferenceClickListener(new OnPreferenceClickListener() {
|
||||||
@Override
|
@Override
|
||||||
public boolean onPreferenceClick(Preference preference) {
|
public boolean onPreferenceClick(Preference preference) {
|
||||||
showRule(id, rule.name);
|
final String action = isSchedule ? ZenModeScheduleRuleSettings.ACTION
|
||||||
|
: ZenModeExternalRuleSettings.ACTION;
|
||||||
|
showRule(action, null, id, rule.name);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@@ -146,13 +165,16 @@ public class ZenModeAutomationSettings extends ZenModeSettingsBase {
|
|||||||
private String computeRuleSummary(ZenRule rule) {
|
private String computeRuleSummary(ZenRule rule) {
|
||||||
if (rule == null || !rule.enabled) return getString(R.string.switch_off_text);
|
if (rule == null || !rule.enabled) return getString(R.string.switch_off_text);
|
||||||
final ScheduleInfo schedule = ZenModeConfig.tryParseScheduleConditionId(rule.conditionId);
|
final ScheduleInfo schedule = ZenModeConfig.tryParseScheduleConditionId(rule.conditionId);
|
||||||
if (schedule == null) return getString(R.string.switch_on_text);
|
final String mode = ZenModeSettings.computeZenModeCaption(getResources(), rule.zenMode);
|
||||||
|
String summary = getString(R.string.switch_on_text);
|
||||||
|
if (schedule != null) {
|
||||||
final String days = computeContiguousDayRanges(schedule.days);
|
final String days = computeContiguousDayRanges(schedule.days);
|
||||||
final String start = getTime(schedule.startHour, schedule.startMinute);
|
final String start = getTime(schedule.startHour, schedule.startMinute);
|
||||||
final String end = getTime(schedule.endHour, schedule.endMinute);
|
final String end = getTime(schedule.endHour, schedule.endMinute);
|
||||||
final String time = getString(R.string.summary_range_verbal_combination, start, end);
|
final String time = getString(R.string.summary_range_verbal_combination, start, end);
|
||||||
final String mode = ZenModeSettings.computeZenModeCaption(getResources(), rule.zenMode);
|
summary = getString(R.string.zen_mode_rule_summary_combination, days, time);
|
||||||
return getString(R.string.zen_mode_rule_summary_template, days, time, mode);
|
}
|
||||||
|
return getString(R.string.zen_mode_rule_summary_combination, summary, mode);
|
||||||
}
|
}
|
||||||
|
|
||||||
private String getTime(int hour, int minute) {
|
private String getTime(int hour, int minute) {
|
||||||
@@ -199,4 +221,30 @@ public class ZenModeAutomationSettings extends ZenModeSettingsBase {
|
|||||||
return DAY_FORMAT.format(mCalendar.getTime());
|
return DAY_FORMAT.format(mCalendar.getTime());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static Config getConditionProviderConfig() {
|
||||||
|
final Config c = new Config();
|
||||||
|
c.tag = TAG;
|
||||||
|
c.setting = Settings.Secure.ENABLED_CONDITION_PROVIDERS;
|
||||||
|
c.intentAction = ConditionProviderService.SERVICE_INTERFACE;
|
||||||
|
c.permission = android.Manifest.permission.BIND_CONDITION_PROVIDER_SERVICE;
|
||||||
|
c.noun = "condition provider";
|
||||||
|
return c;
|
||||||
|
}
|
||||||
|
|
||||||
|
private final ServiceListing.Callback mServiceListingCallback = new ServiceListing.Callback() {
|
||||||
|
@Override
|
||||||
|
public void onServicesReloaded(List<ServiceInfo> services) {
|
||||||
|
for (ServiceInfo service : services) {
|
||||||
|
final RuleInfo ri = ZenModeExternalRuleSettings.getRuleInfo(service);
|
||||||
|
if (ri != null && ri.serviceComponent != null
|
||||||
|
&& ri.settingsAction == ZenModeExternalRuleSettings.ACTION) {
|
||||||
|
if (!mServiceListing.isEnabled(ri.serviceComponent)) {
|
||||||
|
Log.i(TAG, "Enabling external condition provider: " + ri.serviceComponent);
|
||||||
|
mServiceListing.setEnabled(ri.serviceComponent, true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@@ -0,0 +1,133 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2015 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.notification;
|
||||||
|
|
||||||
|
import android.app.Activity;
|
||||||
|
import android.content.ComponentName;
|
||||||
|
import android.content.Intent;
|
||||||
|
import android.content.pm.ServiceInfo;
|
||||||
|
import android.net.Uri;
|
||||||
|
import android.preference.Preference;
|
||||||
|
import android.preference.Preference.OnPreferenceClickListener;
|
||||||
|
import android.preference.PreferenceScreen;
|
||||||
|
import android.provider.Settings;
|
||||||
|
import android.service.notification.ZenModeConfig.ZenRule;
|
||||||
|
import android.util.Log;
|
||||||
|
|
||||||
|
import com.android.internal.logging.MetricsLogger;
|
||||||
|
import com.android.settings.R;
|
||||||
|
import com.android.settings.notification.ZenRuleNameDialog.RuleInfo;
|
||||||
|
|
||||||
|
public class ZenModeExternalRuleSettings extends ZenModeRuleSettingsBase {
|
||||||
|
private static final String KEY_TYPE = "type";
|
||||||
|
private static final String KEY_CONFIGURE = "configure";
|
||||||
|
|
||||||
|
public static final String ACTION = Settings.ACTION_ZEN_MODE_EXTERNAL_RULE_SETTINGS;
|
||||||
|
private static final int REQUEST_CODE_CONFIGURE = 1;
|
||||||
|
|
||||||
|
private static final String MD_RULE_TYPE = "automatic.ruleType";
|
||||||
|
private static final String MD_DEFAULT_CONDITION_ID = "automatic.defaultConditionId";
|
||||||
|
private static final String MD_CONFIGURATION_ACTIVITY = "automatic.configurationActivity";
|
||||||
|
private static final String EXTRA_CONDITION_ID = "automatic.conditionId";
|
||||||
|
|
||||||
|
private Preference mType;
|
||||||
|
private Preference mConfigure;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected boolean setRule(ZenRule rule) {
|
||||||
|
return rule != null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected String getZenModeDependency() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onCreateInternal() {
|
||||||
|
addPreferencesFromResource(R.xml.zen_mode_external_rule_settings);
|
||||||
|
final PreferenceScreen root = getPreferenceScreen();
|
||||||
|
final ServiceInfo si = ServiceListing.findService(mContext,
|
||||||
|
ZenModeAutomationSettings.CONFIG, mRule.component);
|
||||||
|
if (DEBUG) Log.d(TAG, "ServiceInfo: " + si);
|
||||||
|
final RuleInfo ri = getRuleInfo(si);
|
||||||
|
if (DEBUG) Log.d(TAG, "RuleInfo: " + ri);
|
||||||
|
mType = root.findPreference(KEY_TYPE);
|
||||||
|
if (ri == null) {
|
||||||
|
mType.setSummary(R.string.zen_mode_rule_type_unknown);
|
||||||
|
} else {
|
||||||
|
mType.setSummary(ri.caption);
|
||||||
|
}
|
||||||
|
|
||||||
|
mConfigure = root.findPreference(KEY_CONFIGURE);
|
||||||
|
if (ri == null || ri.configurationActivity == null) {
|
||||||
|
mConfigure.setEnabled(false);
|
||||||
|
} else {
|
||||||
|
mConfigure.setOnPreferenceClickListener(new OnPreferenceClickListener() {
|
||||||
|
@Override
|
||||||
|
public boolean onPreferenceClick(Preference preference) {
|
||||||
|
startActivityForResult(new Intent().setComponent(ri.configurationActivity),
|
||||||
|
REQUEST_CODE_CONFIGURE);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onActivityResult(int requestCode, int resultCode, Intent data) {
|
||||||
|
super.onActivityResult(requestCode, resultCode, data);
|
||||||
|
if (requestCode == REQUEST_CODE_CONFIGURE) {
|
||||||
|
if (resultCode == Activity.RESULT_OK && data != null) {
|
||||||
|
final Uri conditionId = data.getParcelableExtra(EXTRA_CONDITION_ID);
|
||||||
|
if (conditionId != null && !conditionId.equals(mRule.conditionId)) {
|
||||||
|
updateRule(conditionId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static RuleInfo getRuleInfo(ServiceInfo si) {
|
||||||
|
if (si == null || si.metaData == null) return null;
|
||||||
|
final String ruleType = si.metaData.getString(MD_RULE_TYPE);
|
||||||
|
final String defaultConditionId = si.metaData.getString(MD_DEFAULT_CONDITION_ID);
|
||||||
|
final String configurationActivity = si.metaData.getString(MD_CONFIGURATION_ACTIVITY);
|
||||||
|
if (ruleType != null && !ruleType.trim().isEmpty() && defaultConditionId != null) {
|
||||||
|
final RuleInfo ri = new RuleInfo();
|
||||||
|
ri.serviceComponent = new ComponentName(si.packageName, si.name);
|
||||||
|
ri.settingsAction = ZenModeExternalRuleSettings.ACTION;
|
||||||
|
ri.caption = ruleType;
|
||||||
|
ri.defaultConditionId = Uri.parse(defaultConditionId);
|
||||||
|
if (configurationActivity != null) {
|
||||||
|
ri.configurationActivity = ComponentName.unflattenFromString(configurationActivity);
|
||||||
|
}
|
||||||
|
return ri;
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void updateControlsInternal() {
|
||||||
|
// everything done up front
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected int getMetricsCategory() {
|
||||||
|
return MetricsLogger.NOTIFICATION_ZEN_MODE_EXTERNAL_RULE;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@@ -0,0 +1,256 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2015 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.notification;
|
||||||
|
|
||||||
|
import android.app.AlertDialog;
|
||||||
|
import android.content.Context;
|
||||||
|
import android.content.DialogInterface;
|
||||||
|
import android.content.DialogInterface.OnClickListener;
|
||||||
|
import android.content.Intent;
|
||||||
|
import android.net.Uri;
|
||||||
|
import android.os.Bundle;
|
||||||
|
import android.preference.Preference;
|
||||||
|
import android.preference.Preference.OnPreferenceClickListener;
|
||||||
|
import android.preference.PreferenceScreen;
|
||||||
|
import android.provider.Settings.Global;
|
||||||
|
import android.service.notification.ZenModeConfig;
|
||||||
|
import android.service.notification.ZenModeConfig.ZenRule;
|
||||||
|
import android.util.Log;
|
||||||
|
import android.view.Menu;
|
||||||
|
import android.view.MenuInflater;
|
||||||
|
import android.view.MenuItem;
|
||||||
|
import android.widget.Switch;
|
||||||
|
import android.widget.Toast;
|
||||||
|
|
||||||
|
import com.android.settings.DropDownPreference;
|
||||||
|
import com.android.settings.R;
|
||||||
|
import com.android.settings.SettingsActivity;
|
||||||
|
import com.android.settings.widget.SwitchBar;
|
||||||
|
|
||||||
|
public abstract class ZenModeRuleSettingsBase extends ZenModeSettingsBase
|
||||||
|
implements SwitchBar.OnSwitchChangeListener {
|
||||||
|
protected static final String TAG = ZenModeSettingsBase.TAG;
|
||||||
|
protected static final boolean DEBUG = ZenModeSettingsBase.DEBUG;
|
||||||
|
|
||||||
|
public static final String EXTRA_RULE_ID = "rule_id";
|
||||||
|
private static final String KEY_RULE_NAME = "rule_name";
|
||||||
|
private static final String KEY_ZEN_MODE = "zen_mode";
|
||||||
|
|
||||||
|
protected Context mContext;
|
||||||
|
protected boolean mDisableListeners;
|
||||||
|
protected ZenRule mRule;
|
||||||
|
|
||||||
|
private String mRuleId;
|
||||||
|
private boolean mDeleting;
|
||||||
|
private Preference mRuleName;
|
||||||
|
private SwitchBar mSwitchBar;
|
||||||
|
private DropDownPreference mZenMode;
|
||||||
|
|
||||||
|
abstract protected void onCreateInternal();
|
||||||
|
abstract protected boolean setRule(ZenRule rule);
|
||||||
|
abstract protected String getZenModeDependency();
|
||||||
|
abstract protected void updateControlsInternal();
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onCreate(Bundle icicle) {
|
||||||
|
super.onCreate(icicle);
|
||||||
|
|
||||||
|
mContext = getActivity();
|
||||||
|
|
||||||
|
final Intent intent = getActivity().getIntent();
|
||||||
|
if (DEBUG) Log.d(TAG, "onCreate getIntent()=" + intent);
|
||||||
|
if (intent == null) {
|
||||||
|
Log.w(TAG, "No intent");
|
||||||
|
toastAndFinish();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
mRuleId = intent.getStringExtra(EXTRA_RULE_ID);
|
||||||
|
if (DEBUG) Log.d(TAG, "mRuleId=" + mRuleId);
|
||||||
|
if (refreshRuleOrFinish()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
setHasOptionsMenu(true);
|
||||||
|
|
||||||
|
onCreateInternal();
|
||||||
|
|
||||||
|
final PreferenceScreen root = getPreferenceScreen();
|
||||||
|
mRuleName = root.findPreference(KEY_RULE_NAME);
|
||||||
|
mRuleName.setOnPreferenceClickListener(new OnPreferenceClickListener() {
|
||||||
|
@Override
|
||||||
|
public boolean onPreferenceClick(Preference preference) {
|
||||||
|
showRuleNameDialog();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
mZenMode = (DropDownPreference) root.findPreference(KEY_ZEN_MODE);
|
||||||
|
mZenMode.addItem(R.string.zen_mode_option_important_interruptions,
|
||||||
|
Global.ZEN_MODE_IMPORTANT_INTERRUPTIONS);
|
||||||
|
mZenMode.addItem(R.string.zen_mode_option_alarms, Global.ZEN_MODE_ALARMS);
|
||||||
|
mZenMode.addItem(R.string.zen_mode_option_no_interruptions,
|
||||||
|
Global.ZEN_MODE_NO_INTERRUPTIONS);
|
||||||
|
mZenMode.setCallback(new DropDownPreference.Callback() {
|
||||||
|
@Override
|
||||||
|
public boolean onItemSelected(int pos, Object value) {
|
||||||
|
if (mDisableListeners) return true;
|
||||||
|
final int zenMode = (Integer) value;
|
||||||
|
if (zenMode == mRule.zenMode) return true;
|
||||||
|
if (DEBUG) Log.d(TAG, "onPrefChange zenMode=" + zenMode);
|
||||||
|
mRule.zenMode = zenMode;
|
||||||
|
setZenModeConfig(mConfig);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
mZenMode.setOrder(10); // sort at the bottom of the category
|
||||||
|
mZenMode.setDependency(getZenModeDependency());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onResume() {
|
||||||
|
super.onResume();
|
||||||
|
updateControls();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onActivityCreated(Bundle savedInstanceState) {
|
||||||
|
super.onActivityCreated(savedInstanceState);
|
||||||
|
|
||||||
|
final SettingsActivity activity = (SettingsActivity) getActivity();
|
||||||
|
mSwitchBar = activity.getSwitchBar();
|
||||||
|
mSwitchBar.addOnSwitchChangeListener(this);
|
||||||
|
mSwitchBar.show();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onDestroyView() {
|
||||||
|
super.onDestroyView();
|
||||||
|
mSwitchBar.removeOnSwitchChangeListener(this);
|
||||||
|
mSwitchBar.hide();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onSwitchChanged(Switch switchView, boolean isChecked) {
|
||||||
|
if (DEBUG) Log.d(TAG, "onSwitchChanged " + isChecked);
|
||||||
|
if (mDisableListeners) return;
|
||||||
|
final boolean enabled = isChecked;
|
||||||
|
if (enabled == mRule.enabled) return;
|
||||||
|
if (DEBUG) Log.d(TAG, "onSwitchChanged enabled=" + enabled);
|
||||||
|
mRule.enabled = enabled;
|
||||||
|
mRule.snoozing = false;
|
||||||
|
setZenModeConfig(mConfig);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void updateRule(Uri newConditionId) {
|
||||||
|
mRule.conditionId = newConditionId;
|
||||||
|
mRule.condition = null;
|
||||||
|
mRule.snoozing = false;
|
||||||
|
setZenModeConfig(mConfig);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onZenModeChanged() {
|
||||||
|
// noop
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onZenModeConfigChanged() {
|
||||||
|
if (!refreshRuleOrFinish()) {
|
||||||
|
updateControls();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) {
|
||||||
|
if (DEBUG) Log.d(TAG, "onCreateOptionsMenu");
|
||||||
|
inflater.inflate(R.menu.zen_mode_rule, menu);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean onOptionsItemSelected(MenuItem item) {
|
||||||
|
if (DEBUG) Log.d(TAG, "onOptionsItemSelected " + item.getItemId());
|
||||||
|
if (item.getItemId() == R.id.delete) {
|
||||||
|
showDeleteRuleDialog();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return super.onOptionsItemSelected(item);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void showRuleNameDialog() {
|
||||||
|
new ZenRuleNameDialog(mContext, null, mRule.name, mConfig.getAutomaticRuleNames()) {
|
||||||
|
@Override
|
||||||
|
public void onOk(String ruleName, RuleInfo type) {
|
||||||
|
final ZenModeConfig newConfig = mConfig.copy();
|
||||||
|
final ZenRule rule = newConfig.automaticRules.get(mRuleId);
|
||||||
|
if (rule == null) return;
|
||||||
|
rule.name = ruleName;
|
||||||
|
setZenModeConfig(newConfig);
|
||||||
|
}
|
||||||
|
}.show();
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean refreshRuleOrFinish() {
|
||||||
|
mRule = mConfig.automaticRules.get(mRuleId);
|
||||||
|
if (DEBUG) Log.d(TAG, "mRule=" + mRule);
|
||||||
|
if (!setRule(mRule)) {
|
||||||
|
toastAndFinish();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void showDeleteRuleDialog() {
|
||||||
|
new AlertDialog.Builder(mContext)
|
||||||
|
.setMessage(getString(R.string.zen_mode_delete_rule_confirmation, mRule.name))
|
||||||
|
.setNegativeButton(R.string.cancel, null)
|
||||||
|
.setPositiveButton(R.string.zen_mode_delete_rule_button, new OnClickListener() {
|
||||||
|
@Override
|
||||||
|
public void onClick(DialogInterface dialog, int which) {
|
||||||
|
mDeleting = true;
|
||||||
|
mConfig.automaticRules.remove(mRuleId);
|
||||||
|
setZenModeConfig(mConfig);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.show();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void toastAndFinish() {
|
||||||
|
if (!mDeleting) {
|
||||||
|
Toast.makeText(mContext, R.string.zen_mode_rule_not_found_text, Toast.LENGTH_SHORT)
|
||||||
|
.show();
|
||||||
|
}
|
||||||
|
getActivity().finish();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void updateRuleName() {
|
||||||
|
getActivity().setTitle(mRule.name);
|
||||||
|
mRuleName.setSummary(mRule.name);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void updateControls() {
|
||||||
|
mDisableListeners = true;
|
||||||
|
updateRuleName();
|
||||||
|
updateControlsInternal();
|
||||||
|
mZenMode.setSelectedValue(mRule.zenMode);
|
||||||
|
mDisableListeners = false;
|
||||||
|
if (mSwitchBar != null) {
|
||||||
|
mSwitchBar.setChecked(mRule.enabled);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@@ -25,141 +25,58 @@ import android.app.FragmentManager;
|
|||||||
import android.app.TimePickerDialog;
|
import android.app.TimePickerDialog;
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.DialogInterface;
|
import android.content.DialogInterface;
|
||||||
import android.content.DialogInterface.OnClickListener;
|
|
||||||
import android.content.DialogInterface.OnDismissListener;
|
import android.content.DialogInterface.OnDismissListener;
|
||||||
import android.content.Intent;
|
|
||||||
import android.os.Bundle;
|
import android.os.Bundle;
|
||||||
import android.preference.Preference;
|
import android.preference.Preference;
|
||||||
import android.preference.Preference.OnPreferenceClickListener;
|
import android.preference.Preference.OnPreferenceClickListener;
|
||||||
import android.preference.PreferenceScreen;
|
import android.preference.PreferenceScreen;
|
||||||
import android.provider.Settings;
|
import android.provider.Settings;
|
||||||
import android.provider.Settings.Global;
|
|
||||||
import android.service.notification.ZenModeConfig;
|
import android.service.notification.ZenModeConfig;
|
||||||
import android.service.notification.ZenModeConfig.ScheduleInfo;
|
import android.service.notification.ZenModeConfig.ScheduleInfo;
|
||||||
import android.service.notification.ZenModeConfig.ZenRule;
|
import android.service.notification.ZenModeConfig.ZenRule;
|
||||||
import android.text.format.DateFormat;
|
import android.text.format.DateFormat;
|
||||||
import android.util.Log;
|
import android.util.Log;
|
||||||
import android.view.Menu;
|
|
||||||
import android.view.MenuInflater;
|
|
||||||
import android.view.MenuItem;
|
|
||||||
import android.widget.Switch;
|
|
||||||
import android.widget.TimePicker;
|
import android.widget.TimePicker;
|
||||||
import android.widget.Toast;
|
|
||||||
|
|
||||||
import com.android.internal.logging.MetricsLogger;
|
import com.android.internal.logging.MetricsLogger;
|
||||||
import com.android.settings.DropDownPreference;
|
|
||||||
import com.android.settings.R;
|
import com.android.settings.R;
|
||||||
import com.android.settings.SettingsActivity;
|
|
||||||
import com.android.settings.widget.SwitchBar;
|
|
||||||
|
|
||||||
import java.text.SimpleDateFormat;
|
import java.text.SimpleDateFormat;
|
||||||
import java.util.Arrays;
|
import java.util.Arrays;
|
||||||
import java.util.Calendar;
|
import java.util.Calendar;
|
||||||
|
|
||||||
public class ZenModeScheduleRuleSettings extends ZenModeSettingsBase
|
public class ZenModeScheduleRuleSettings extends ZenModeRuleSettingsBase {
|
||||||
implements SwitchBar.OnSwitchChangeListener {
|
|
||||||
private static final String TAG = ZenModeSettingsBase.TAG;
|
|
||||||
private static final boolean DEBUG = ZenModeSettingsBase.DEBUG;
|
|
||||||
|
|
||||||
private static final String KEY_RULE_NAME = "rule_name";
|
|
||||||
private static final String KEY_DAYS = "days";
|
private static final String KEY_DAYS = "days";
|
||||||
private static final String KEY_START_TIME = "start_time";
|
private static final String KEY_START_TIME = "start_time";
|
||||||
private static final String KEY_END_TIME = "end_time";
|
private static final String KEY_END_TIME = "end_time";
|
||||||
private static final String KEY_ZEN_MODE = "zen_mode";
|
|
||||||
|
|
||||||
private static final SimpleDateFormat DAY_FORMAT = new SimpleDateFormat("EEE");
|
private static final SimpleDateFormat DAY_FORMAT = new SimpleDateFormat("EEE");
|
||||||
|
|
||||||
public static final String ACTION = Settings.ACTION_ZEN_MODE_SCHEDULE_RULE_SETTINGS;
|
public static final String ACTION = Settings.ACTION_ZEN_MODE_SCHEDULE_RULE_SETTINGS;
|
||||||
public static final String EXTRA_RULE_ID = "rule_id";
|
|
||||||
|
|
||||||
private Context mContext;
|
|
||||||
private boolean mDisableListeners;
|
|
||||||
private SwitchBar mSwitchBar;
|
|
||||||
private Preference mRuleName;
|
|
||||||
private Preference mDays;
|
private Preference mDays;
|
||||||
private TimePickerPreference mStart;
|
private TimePickerPreference mStart;
|
||||||
private TimePickerPreference mEnd;
|
private TimePickerPreference mEnd;
|
||||||
private DropDownPreference mZenMode;
|
|
||||||
|
|
||||||
private String mRuleId;
|
|
||||||
private ZenRule mRule;
|
|
||||||
private ScheduleInfo mSchedule;
|
private ScheduleInfo mSchedule;
|
||||||
private boolean mDeleting;
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void onZenModeChanged() {
|
protected boolean setRule(ZenRule rule) {
|
||||||
// noop
|
mSchedule = rule != null ? ZenModeConfig.tryParseScheduleConditionId(rule.conditionId)
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected void onZenModeConfigChanged() {
|
|
||||||
if (!refreshRuleOrFinish()) {
|
|
||||||
updateControls();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private boolean refreshRuleOrFinish() {
|
|
||||||
mRule = mConfig.automaticRules.get(mRuleId);
|
|
||||||
if (DEBUG) Log.d(TAG, "mRule=" + mRule);
|
|
||||||
mSchedule = mRule != null ? ZenModeConfig.tryParseScheduleConditionId(mRule.conditionId)
|
|
||||||
: null;
|
: null;
|
||||||
if (mSchedule == null) {
|
return mSchedule != null;
|
||||||
toastAndFinish();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) {
|
protected String getZenModeDependency() {
|
||||||
if (DEBUG) Log.d(TAG, "onCreateOptionsMenu");
|
return mDays.getKey();
|
||||||
inflater.inflate(R.menu.zen_mode_rule, menu);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean onOptionsItemSelected(MenuItem item) {
|
protected void onCreateInternal() {
|
||||||
if (DEBUG) Log.d(TAG, "onOptionsItemSelected " + item.getItemId());
|
|
||||||
if (item.getItemId() == R.id.delete) {
|
|
||||||
showDeleteRuleDialog();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return super.onOptionsItemSelected(item);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onCreate(Bundle icicle) {
|
|
||||||
super.onCreate(icicle);
|
|
||||||
|
|
||||||
mContext = getActivity();
|
|
||||||
|
|
||||||
final Intent intent = getActivity().getIntent();
|
|
||||||
if (DEBUG) Log.d(TAG, "onCreate getIntent()=" + intent);
|
|
||||||
if (intent == null) {
|
|
||||||
Log.w(TAG, "No intent");
|
|
||||||
toastAndFinish();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
mRuleId = intent.getStringExtra(EXTRA_RULE_ID);
|
|
||||||
if (DEBUG) Log.d(TAG, "mRuleId=" + mRuleId);
|
|
||||||
if (refreshRuleOrFinish()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
addPreferencesFromResource(R.xml.zen_mode_schedule_rule_settings);
|
addPreferencesFromResource(R.xml.zen_mode_schedule_rule_settings);
|
||||||
final PreferenceScreen root = getPreferenceScreen();
|
final PreferenceScreen root = getPreferenceScreen();
|
||||||
|
|
||||||
setHasOptionsMenu(true);
|
|
||||||
|
|
||||||
mRuleName = root.findPreference(KEY_RULE_NAME);
|
|
||||||
mRuleName.setOnPreferenceClickListener(new OnPreferenceClickListener() {
|
|
||||||
@Override
|
|
||||||
public boolean onPreferenceClick(Preference preference) {
|
|
||||||
showRuleNameDialog();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
mDays = root.findPreference(KEY_DAYS);
|
mDays = root.findPreference(KEY_DAYS);
|
||||||
mDays.setOnPreferenceClickListener(new OnPreferenceClickListener() {
|
mDays.setOnPreferenceClickListener(new OnPreferenceClickListener() {
|
||||||
@Override
|
@Override
|
||||||
@@ -186,10 +103,7 @@ public class ZenModeScheduleRuleSettings extends ZenModeSettingsBase
|
|||||||
if (DEBUG) Log.d(TAG, "onPrefChange start h=" + hour + " m=" + minute);
|
if (DEBUG) Log.d(TAG, "onPrefChange start h=" + hour + " m=" + minute);
|
||||||
mSchedule.startHour = hour;
|
mSchedule.startHour = hour;
|
||||||
mSchedule.startMinute = minute;
|
mSchedule.startMinute = minute;
|
||||||
mRule.conditionId = ZenModeConfig.toScheduleConditionId(mSchedule);
|
updateRule(ZenModeConfig.toScheduleConditionId(mSchedule));
|
||||||
mRule.condition = null;
|
|
||||||
mRule.snoozing = false;
|
|
||||||
setZenModeConfig(mConfig);
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@@ -211,63 +125,12 @@ public class ZenModeScheduleRuleSettings extends ZenModeSettingsBase
|
|||||||
if (DEBUG) Log.d(TAG, "onPrefChange end h=" + hour + " m=" + minute);
|
if (DEBUG) Log.d(TAG, "onPrefChange end h=" + hour + " m=" + minute);
|
||||||
mSchedule.endHour = hour;
|
mSchedule.endHour = hour;
|
||||||
mSchedule.endMinute = minute;
|
mSchedule.endMinute = minute;
|
||||||
mRule.conditionId = ZenModeConfig.toScheduleConditionId(mSchedule);
|
updateRule(ZenModeConfig.toScheduleConditionId(mSchedule));
|
||||||
mRule.condition = null;
|
|
||||||
mRule.snoozing = false;
|
|
||||||
setZenModeConfig(mConfig);
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
root.addPreference(mEnd);
|
root.addPreference(mEnd);
|
||||||
mEnd.setDependency(mDays.getKey());
|
mEnd.setDependency(mDays.getKey());
|
||||||
|
|
||||||
mZenMode = (DropDownPreference) root.findPreference(KEY_ZEN_MODE);
|
|
||||||
mZenMode.addItem(R.string.zen_mode_option_important_interruptions, Global.ZEN_MODE_IMPORTANT_INTERRUPTIONS);
|
|
||||||
mZenMode.addItem(R.string.zen_mode_option_alarms, Global.ZEN_MODE_ALARMS);
|
|
||||||
mZenMode.addItem(R.string.zen_mode_option_no_interruptions, Global.ZEN_MODE_NO_INTERRUPTIONS);
|
|
||||||
mZenMode.setCallback(new DropDownPreference.Callback() {
|
|
||||||
@Override
|
|
||||||
public boolean onItemSelected(int pos, Object value) {
|
|
||||||
if (mDisableListeners) return true;
|
|
||||||
final int zenMode = (Integer) value;
|
|
||||||
if (zenMode == mRule.zenMode) return true;
|
|
||||||
if (DEBUG) Log.d(TAG, "onPrefChange zenMode=" + zenMode);
|
|
||||||
mRule.zenMode = zenMode;
|
|
||||||
setZenModeConfig(mConfig);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
mZenMode.setOrder(10); // sort at the bottom of the category
|
|
||||||
mZenMode.setDependency(mDays.getKey());
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onActivityCreated(Bundle savedInstanceState) {
|
|
||||||
super.onActivityCreated(savedInstanceState);
|
|
||||||
|
|
||||||
final SettingsActivity activity = (SettingsActivity) getActivity();
|
|
||||||
mSwitchBar = activity.getSwitchBar();
|
|
||||||
mSwitchBar.addOnSwitchChangeListener(this);
|
|
||||||
mSwitchBar.show();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onDestroyView() {
|
|
||||||
super.onDestroyView();
|
|
||||||
mSwitchBar.removeOnSwitchChangeListener(this);
|
|
||||||
mSwitchBar.hide();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onSwitchChanged(Switch switchView, boolean isChecked) {
|
|
||||||
if (DEBUG) Log.d(TAG, "onSwitchChanged " + isChecked);
|
|
||||||
if (mDisableListeners) return;
|
|
||||||
final boolean enabled = isChecked;
|
|
||||||
if (enabled == mRule.enabled) return;
|
|
||||||
if (DEBUG) Log.d(TAG, "onSwitchChanged enabled=" + enabled);
|
|
||||||
mRule.enabled = enabled;
|
|
||||||
mRule.snoozing = false;
|
|
||||||
setZenModeConfig(mConfig);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void updateDays() {
|
private void updateDays() {
|
||||||
@@ -308,28 +171,11 @@ public class ZenModeScheduleRuleSettings extends ZenModeSettingsBase
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onResume() {
|
protected void updateControlsInternal() {
|
||||||
super.onResume();
|
|
||||||
updateControls();
|
|
||||||
}
|
|
||||||
|
|
||||||
private void updateRuleName() {
|
|
||||||
getActivity().setTitle(mRule.name);
|
|
||||||
mRuleName.setSummary(mRule.name);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void updateControls() {
|
|
||||||
mDisableListeners = true;
|
|
||||||
updateRuleName();
|
|
||||||
updateDays();
|
updateDays();
|
||||||
mStart.setTime(mSchedule.startHour, mSchedule.startMinute);
|
mStart.setTime(mSchedule.startHour, mSchedule.startMinute);
|
||||||
mEnd.setTime(mSchedule.endHour, mSchedule.endMinute);
|
mEnd.setTime(mSchedule.endHour, mSchedule.endMinute);
|
||||||
mZenMode.setSelectedValue(mRule.zenMode);
|
|
||||||
mDisableListeners = false;
|
|
||||||
updateEndSummary();
|
updateEndSummary();
|
||||||
if (mSwitchBar != null) {
|
|
||||||
mSwitchBar.setChecked(mRule.enabled);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@@ -337,34 +183,6 @@ public class ZenModeScheduleRuleSettings extends ZenModeSettingsBase
|
|||||||
return MetricsLogger.NOTIFICATION_ZEN_MODE_SCHEDULE_RULE;
|
return MetricsLogger.NOTIFICATION_ZEN_MODE_SCHEDULE_RULE;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void showDeleteRuleDialog() {
|
|
||||||
new AlertDialog.Builder(mContext)
|
|
||||||
.setMessage(getString(R.string.zen_mode_delete_rule_confirmation, mRule.name))
|
|
||||||
.setNegativeButton(R.string.cancel, null)
|
|
||||||
.setPositiveButton(R.string.zen_mode_delete_rule_button, new OnClickListener() {
|
|
||||||
@Override
|
|
||||||
public void onClick(DialogInterface dialog, int which) {
|
|
||||||
mDeleting = true;
|
|
||||||
mConfig.automaticRules.remove(mRuleId);
|
|
||||||
setZenModeConfig(mConfig);
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.show();
|
|
||||||
}
|
|
||||||
|
|
||||||
private void showRuleNameDialog() {
|
|
||||||
new ZenRuleNameDialog(mContext, mRule.name, mConfig.getAutomaticRuleNames()) {
|
|
||||||
@Override
|
|
||||||
public void onOk(String ruleName) {
|
|
||||||
final ZenModeConfig newConfig = mConfig.copy();
|
|
||||||
final ZenRule rule = newConfig.automaticRules.get(mRuleId);
|
|
||||||
if (rule == null) return;
|
|
||||||
rule.name = ruleName;
|
|
||||||
setZenModeConfig(newConfig);
|
|
||||||
}
|
|
||||||
}.show();
|
|
||||||
}
|
|
||||||
|
|
||||||
private void showDaysDialog() {
|
private void showDaysDialog() {
|
||||||
new AlertDialog.Builder(mContext)
|
new AlertDialog.Builder(mContext)
|
||||||
.setTitle(R.string.zen_mode_schedule_rule_days)
|
.setTitle(R.string.zen_mode_schedule_rule_days)
|
||||||
@@ -375,10 +193,7 @@ public class ZenModeScheduleRuleSettings extends ZenModeSettingsBase
|
|||||||
if (Arrays.equals(days, mSchedule.days)) return;
|
if (Arrays.equals(days, mSchedule.days)) return;
|
||||||
if (DEBUG) Log.d(TAG, "days.onChanged days=" + Arrays.asList(days));
|
if (DEBUG) Log.d(TAG, "days.onChanged days=" + Arrays.asList(days));
|
||||||
mSchedule.days = days;
|
mSchedule.days = days;
|
||||||
mRule.conditionId = ZenModeConfig.toScheduleConditionId(mSchedule);
|
updateRule(ZenModeConfig.toScheduleConditionId(mSchedule));
|
||||||
mRule.condition = null;
|
|
||||||
mRule.snoozing = false;
|
|
||||||
setZenModeConfig(mConfig);
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.setOnDismissListener(new OnDismissListener() {
|
.setOnDismissListener(new OnDismissListener() {
|
||||||
@@ -391,14 +206,6 @@ public class ZenModeScheduleRuleSettings extends ZenModeSettingsBase
|
|||||||
.show();
|
.show();
|
||||||
}
|
}
|
||||||
|
|
||||||
private void toastAndFinish() {
|
|
||||||
if (!mDeleting) {
|
|
||||||
Toast.makeText(mContext, R.string.zen_mode_rule_not_found_text, Toast.LENGTH_SHORT)
|
|
||||||
.show();
|
|
||||||
}
|
|
||||||
getActivity().finish();
|
|
||||||
}
|
|
||||||
|
|
||||||
private static class TimePickerPreference extends Preference {
|
private static class TimePickerPreference extends Preference {
|
||||||
private final Context mContext;
|
private final Context mContext;
|
||||||
|
|
||||||
@@ -474,4 +281,5 @@ public class ZenModeScheduleRuleSettings extends ZenModeSettingsBase
|
|||||||
boolean onSetTime(int hour, int minute);
|
boolean onSetTime(int hour, int minute);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@@ -17,35 +17,71 @@
|
|||||||
package com.android.settings.notification;
|
package com.android.settings.notification;
|
||||||
|
|
||||||
import android.app.AlertDialog;
|
import android.app.AlertDialog;
|
||||||
|
import android.content.ComponentName;
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.DialogInterface;
|
import android.content.DialogInterface;
|
||||||
|
import android.content.DialogInterface.OnDismissListener;
|
||||||
|
import android.content.pm.ServiceInfo;
|
||||||
|
import android.net.Uri;
|
||||||
|
import android.service.notification.ZenModeConfig;
|
||||||
|
import android.service.notification.ZenModeConfig.ScheduleInfo;
|
||||||
import android.text.Editable;
|
import android.text.Editable;
|
||||||
import android.text.TextUtils;
|
import android.text.TextUtils;
|
||||||
import android.text.TextWatcher;
|
import android.text.TextWatcher;
|
||||||
import android.util.ArraySet;
|
import android.util.ArraySet;
|
||||||
|
import android.util.Log;
|
||||||
import android.view.LayoutInflater;
|
import android.view.LayoutInflater;
|
||||||
import android.view.View;
|
import android.view.View;
|
||||||
import android.widget.EditText;
|
import android.widget.EditText;
|
||||||
|
import android.widget.RadioButton;
|
||||||
|
import android.widget.RadioGroup;
|
||||||
|
|
||||||
import com.android.settings.R;
|
import com.android.settings.R;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
public abstract class ZenRuleNameDialog {
|
public abstract class ZenRuleNameDialog {
|
||||||
|
private static final String TAG = ZenModeSettings.TAG;
|
||||||
|
private static final boolean DEBUG = ZenModeSettings.DEBUG;
|
||||||
|
|
||||||
private final AlertDialog mDialog;
|
private final AlertDialog mDialog;
|
||||||
private final EditText mEditText;
|
private final EditText mEditText;
|
||||||
|
private final RadioGroup mTypes;
|
||||||
private final ArraySet<String> mExistingNames;
|
private final ArraySet<String> mExistingNames;
|
||||||
|
private final ServiceListing mServiceListing;
|
||||||
|
private final RuleInfo[] mExternalRules = new RuleInfo[3];
|
||||||
|
|
||||||
public ZenRuleNameDialog(Context context, String ruleName, ArraySet<String> existingNames) {
|
public ZenRuleNameDialog(Context context, ServiceListing serviceListing, String ruleName,
|
||||||
|
ArraySet<String> existingNames) {
|
||||||
|
mServiceListing = serviceListing;
|
||||||
final View v = LayoutInflater.from(context).inflate(R.layout.zen_rule_name, null, false);
|
final View v = LayoutInflater.from(context).inflate(R.layout.zen_rule_name, null, false);
|
||||||
mEditText = (EditText) v.findViewById(R.id.rule_name);
|
mEditText = (EditText) v.findViewById(R.id.rule_name);
|
||||||
|
if (ruleName != null) {
|
||||||
mEditText.setText(ruleName);
|
mEditText.setText(ruleName);
|
||||||
|
}
|
||||||
mEditText.setSelectAllOnFocus(true);
|
mEditText.setSelectAllOnFocus(true);
|
||||||
|
mTypes = (RadioGroup) v.findViewById(R.id.rule_types);
|
||||||
|
if (mServiceListing != null) {
|
||||||
|
bindType(R.id.rule_type_schedule, defaultNewSchedule());
|
||||||
|
bindExternalRules();
|
||||||
|
mServiceListing.addCallback(mServiceListingCallback);
|
||||||
|
mServiceListing.reload();
|
||||||
|
}
|
||||||
mDialog = new AlertDialog.Builder(context)
|
mDialog = new AlertDialog.Builder(context)
|
||||||
.setTitle(R.string.zen_mode_rule_name)
|
.setTitle(R.string.zen_mode_rule_name)
|
||||||
.setView(v)
|
.setView(v)
|
||||||
.setPositiveButton(R.string.okay, new DialogInterface.OnClickListener() {
|
.setPositiveButton(R.string.okay, new DialogInterface.OnClickListener() {
|
||||||
@Override
|
@Override
|
||||||
public void onClick(DialogInterface dialog, int which) {
|
public void onClick(DialogInterface dialog, int which) {
|
||||||
onOk(trimmedText());
|
onOk(trimmedText(), selectedRuleInfo());
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.setOnDismissListener(new OnDismissListener() {
|
||||||
|
@Override
|
||||||
|
public void onDismiss(DialogInterface dialog) {
|
||||||
|
if (mServiceListing != null) {
|
||||||
|
mServiceListing.removeCallback(mServiceListingCallback);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.setNegativeButton(R.string.cancel, null)
|
.setNegativeButton(R.string.cancel, null)
|
||||||
@@ -72,17 +108,37 @@ public abstract class ZenRuleNameDialog {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
abstract public void onOk(String ruleName);
|
abstract public void onOk(String ruleName, RuleInfo ruleInfo);
|
||||||
|
|
||||||
private String trimmedText() {
|
|
||||||
return mEditText.getText() == null ? null : mEditText.getText().toString().trim();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void show() {
|
public void show() {
|
||||||
mDialog.show();
|
mDialog.show();
|
||||||
updatePositiveButton();
|
updatePositiveButton();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void bindType(int id, RuleInfo ri) {
|
||||||
|
final RadioButton rb = (RadioButton) mTypes.findViewById(id);
|
||||||
|
if (ri == null) {
|
||||||
|
rb.setVisibility(View.GONE);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
rb.setVisibility(View.VISIBLE);
|
||||||
|
if (ri.caption != null) {
|
||||||
|
rb.setText(ri.caption);
|
||||||
|
}
|
||||||
|
rb.setTag(ri);
|
||||||
|
}
|
||||||
|
|
||||||
|
private RuleInfo selectedRuleInfo() {
|
||||||
|
final int id = mTypes.getCheckedRadioButtonId();
|
||||||
|
if (id == -1) return null;
|
||||||
|
final RadioButton rb = (RadioButton) mTypes.findViewById(id);
|
||||||
|
return (RuleInfo) rb.getTag();
|
||||||
|
}
|
||||||
|
|
||||||
|
private String trimmedText() {
|
||||||
|
return mEditText.getText() == null ? null : mEditText.getText().toString().trim();
|
||||||
|
}
|
||||||
|
|
||||||
private void updatePositiveButton() {
|
private void updatePositiveButton() {
|
||||||
final String name = trimmedText();
|
final String name = trimmedText();
|
||||||
final boolean validName = !TextUtils.isEmpty(name)
|
final boolean validName = !TextUtils.isEmpty(name)
|
||||||
@@ -90,4 +146,51 @@ public abstract class ZenRuleNameDialog {
|
|||||||
mDialog.getButton(DialogInterface.BUTTON_POSITIVE).setEnabled(validName);
|
mDialog.getButton(DialogInterface.BUTTON_POSITIVE).setEnabled(validName);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static RuleInfo defaultNewSchedule() {
|
||||||
|
final ScheduleInfo schedule = new ScheduleInfo();
|
||||||
|
schedule.days = ZenModeConfig.ALL_DAYS;
|
||||||
|
schedule.startHour = 22;
|
||||||
|
schedule.endHour = 7;
|
||||||
|
final RuleInfo rt = new RuleInfo();
|
||||||
|
rt.settingsAction = ZenModeScheduleRuleSettings.ACTION;
|
||||||
|
rt.defaultConditionId = ZenModeConfig.toScheduleConditionId(schedule);
|
||||||
|
return rt;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void bindExternalRules() {
|
||||||
|
bindType(R.id.rule_type_2, mExternalRules[0]);
|
||||||
|
bindType(R.id.rule_type_3, mExternalRules[1]);
|
||||||
|
bindType(R.id.rule_type_4, mExternalRules[2]);
|
||||||
|
// show radio group if we have at least one external rule type
|
||||||
|
mTypes.setVisibility(mExternalRules[0] != null ? View.VISIBLE : View.GONE);
|
||||||
|
}
|
||||||
|
|
||||||
|
private final ServiceListing.Callback mServiceListingCallback = new ServiceListing.Callback() {
|
||||||
|
@Override
|
||||||
|
public void onServicesReloaded(List<ServiceInfo> services) {
|
||||||
|
if (DEBUG) Log.d(TAG, "Services reloaded: count=" + services.size());
|
||||||
|
mExternalRules[0] = mExternalRules[1] = mExternalRules[2] = null;
|
||||||
|
int i = 0;
|
||||||
|
for (ServiceInfo si : services) {
|
||||||
|
final RuleInfo ri = ZenModeExternalRuleSettings.getRuleInfo(si);
|
||||||
|
if (ri != null) {
|
||||||
|
mExternalRules[i] = ri;
|
||||||
|
i++;
|
||||||
|
if (i == mExternalRules.length) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
bindExternalRules();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
public static class RuleInfo {
|
||||||
|
public String caption;
|
||||||
|
public String settingsAction;
|
||||||
|
public Uri defaultConditionId;
|
||||||
|
public ComponentName serviceComponent;
|
||||||
|
public ComponentName configurationActivity;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
Reference in New Issue
Block a user