blob: 63c0bafe389d333d8ac21a1abe913f37beebfc19 [file] [log] [blame]
John Spurlockb2278d62015-04-07 12:47:12 -04001/**
2 * Copyright (c) 2015, The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17package com.android.server.notification;
18
19import android.content.ComponentName;
20import android.net.Uri;
21import android.service.notification.Condition;
22import android.service.notification.IConditionListener;
John Spurlock39581cc2015-04-10 11:59:01 -040023import android.service.notification.IConditionProvider;
John Spurlockb2278d62015-04-07 12:47:12 -040024import android.service.notification.ZenModeConfig;
25import android.service.notification.ZenModeConfig.ZenRule;
26import android.util.ArrayMap;
27import android.util.ArraySet;
28import android.util.Log;
29
30import java.io.PrintWriter;
31import java.util.Objects;
32
33public class ZenModeConditions implements ConditionProviders.Callback {
34 private static final String TAG = ZenModeHelper.TAG;
35 private static final boolean DEBUG = ZenModeHelper.DEBUG;
36
37 private final ZenModeHelper mHelper;
38 private final ConditionProviders mConditionProviders;
39 private final ArrayMap<Uri, ComponentName> mSubscriptions = new ArrayMap<>();
40
John Spurlock39581cc2015-04-10 11:59:01 -040041 private boolean mFirstEvaluation = true;
John Spurlockb2278d62015-04-07 12:47:12 -040042
43 public ZenModeConditions(ZenModeHelper helper, ConditionProviders conditionProviders) {
44 mHelper = helper;
45 mConditionProviders = conditionProviders;
46 if (mConditionProviders.isSystemProviderEnabled(ZenModeConfig.COUNTDOWN_PATH)) {
John Spurlockd60258f2015-04-30 09:30:52 -040047 mConditionProviders.addSystemProvider(new CountdownConditionProvider());
John Spurlockb2278d62015-04-07 12:47:12 -040048 }
49 if (mConditionProviders.isSystemProviderEnabled(ZenModeConfig.SCHEDULE_PATH)) {
John Spurlockd60258f2015-04-30 09:30:52 -040050 mConditionProviders.addSystemProvider(new ScheduleConditionProvider());
51 }
52 if (mConditionProviders.isSystemProviderEnabled(ZenModeConfig.EVENT_PATH)) {
Julia Reynolds7034caf2016-03-09 10:41:00 -050053 mConditionProviders.addSystemProvider(new EventConditionProvider());
John Spurlockb2278d62015-04-07 12:47:12 -040054 }
55 mConditionProviders.setCallback(this);
56 }
57
58 public void dump(PrintWriter pw, String prefix) {
59 pw.print(prefix); pw.print("mSubscriptions="); pw.println(mSubscriptions);
60 }
61
John Spurlock21258a32015-05-27 18:22:55 -040062 public void evaluateConfig(ZenModeConfig config, boolean processSubscriptions) {
John Spurlockb2278d62015-04-07 12:47:12 -040063 if (config == null) return;
John Spurlock39581cc2015-04-10 11:59:01 -040064 if (config.manualRule != null && config.manualRule.condition != null
65 && !config.manualRule.isTrueOrUnknown()) {
John Spurlockb2278d62015-04-07 12:47:12 -040066 if (DEBUG) Log.d(TAG, "evaluateConfig: clearing manual rule");
67 config.manualRule = null;
68 }
69 final ArraySet<Uri> current = new ArraySet<>();
John Spurlock21258a32015-05-27 18:22:55 -040070 evaluateRule(config.manualRule, current, processSubscriptions);
John Spurlockb2278d62015-04-07 12:47:12 -040071 for (ZenRule automaticRule : config.automaticRules.values()) {
John Spurlock21258a32015-05-27 18:22:55 -040072 evaluateRule(automaticRule, current, processSubscriptions);
John Spurlock39581cc2015-04-10 11:59:01 -040073 updateSnoozing(automaticRule);
John Spurlockb2278d62015-04-07 12:47:12 -040074 }
Shibin George23d28ce2016-05-07 16:16:50 +053075
76 synchronized (mSubscriptions) {
77 final int N = mSubscriptions.size();
78 for (int i = N - 1; i >= 0; i--) {
79 final Uri id = mSubscriptions.keyAt(i);
80 final ComponentName component = mSubscriptions.valueAt(i);
81 if (processSubscriptions) {
82 if (!current.contains(id)) {
83 mConditionProviders.unsubscribeIfNecessary(component, id);
84 mSubscriptions.removeAt(i);
85 }
John Spurlocka7d92b12015-05-13 14:48:02 -040086 }
John Spurlockb2278d62015-04-07 12:47:12 -040087 }
88 }
John Spurlock39581cc2015-04-10 11:59:01 -040089 mFirstEvaluation = false;
John Spurlockb2278d62015-04-07 12:47:12 -040090 }
91
92 @Override
93 public void onBootComplete() {
94 // noop
95 }
96
97 @Override
98 public void onUserSwitched() {
99 // noop
100 }
101
102 @Override
John Spurlock39581cc2015-04-10 11:59:01 -0400103 public void onServiceAdded(ComponentName component) {
104 if (DEBUG) Log.d(TAG, "onServiceAdded " + component);
Julia Reynolds89aeab02016-09-15 11:07:50 -0400105 mHelper.setConfig(mHelper.getConfig(), "zmc.onServiceAdded");
John Spurlock39581cc2015-04-10 11:59:01 -0400106 }
107
108 @Override
John Spurlockb2278d62015-04-07 12:47:12 -0400109 public void onConditionChanged(Uri id, Condition condition) {
110 if (DEBUG) Log.d(TAG, "onConditionChanged " + id + " " + condition);
111 ZenModeConfig config = mHelper.getConfig();
112 if (config == null) return;
John Spurlockb2278d62015-04-07 12:47:12 -0400113 boolean updated = updateCondition(id, condition, config.manualRule);
114 for (ZenRule automaticRule : config.automaticRules.values()) {
115 updated |= updateCondition(id, condition, automaticRule);
116 updated |= updateSnoozing(automaticRule);
117 }
118 if (updated) {
Julia Reynolds89aeab02016-09-15 11:07:50 -0400119 mHelper.setConfig(config, "conditionChanged");
John Spurlockb2278d62015-04-07 12:47:12 -0400120 }
121 }
122
John Spurlocka7d92b12015-05-13 14:48:02 -0400123 private void evaluateRule(ZenRule rule, ArraySet<Uri> current, boolean processSubscriptions) {
John Spurlock39581cc2015-04-10 11:59:01 -0400124 if (rule == null || rule.conditionId == null) return;
125 final Uri id = rule.conditionId;
126 boolean isSystemCondition = false;
127 for (SystemConditionProviderService sp : mConditionProviders.getSystemProviders()) {
John Spurlockd60258f2015-04-30 09:30:52 -0400128 if (sp.isValidConditionId(id)) {
John Spurlock39581cc2015-04-10 11:59:01 -0400129 mConditionProviders.ensureRecordExists(sp.getComponent(), id, sp.asInterface());
130 rule.component = sp.getComponent();
131 isSystemCondition = true;
132 }
133 }
134 if (!isSystemCondition) {
135 final IConditionProvider cp = mConditionProviders.findConditionProvider(rule.component);
136 if (DEBUG) Log.d(TAG, "Ensure external rule exists: " + (cp != null) + " for " + id);
137 if (cp != null) {
138 mConditionProviders.ensureRecordExists(rule.component, id, cp);
139 }
140 }
141 if (rule.component == null) {
142 Log.w(TAG, "No component found for automatic rule: " + rule.conditionId);
143 rule.enabled = false;
144 return;
145 }
146 if (current != null) {
147 current.add(id);
148 }
John Spurlocka7d92b12015-05-13 14:48:02 -0400149 if (processSubscriptions) {
150 if (mConditionProviders.subscribeIfNecessary(rule.component, rule.conditionId)) {
Shibin George23d28ce2016-05-07 16:16:50 +0530151 synchronized (mSubscriptions) {
152 mSubscriptions.put(rule.conditionId, rule.component);
153 }
John Spurlocka7d92b12015-05-13 14:48:02 -0400154 } else {
Julia Reynolds6434eb22016-08-08 17:19:26 -0400155 rule.condition = null;
John Spurlocka7d92b12015-05-13 14:48:02 -0400156 if (DEBUG) Log.d(TAG, "zmc failed to subscribe");
157 }
John Spurlock39581cc2015-04-10 11:59:01 -0400158 }
John Spurlock21258a32015-05-27 18:22:55 -0400159 if (rule.condition == null) {
160 rule.condition = mConditionProviders.findCondition(rule.component, rule.conditionId);
161 if (rule.condition != null && DEBUG) Log.d(TAG, "Found existing condition for: "
162 + rule.conditionId);
163 }
John Spurlock39581cc2015-04-10 11:59:01 -0400164 }
165
166 private boolean isAutomaticActive(ComponentName component) {
167 if (component == null) return false;
168 final ZenModeConfig config = mHelper.getConfig();
169 if (config == null) return false;
170 for (ZenRule rule : config.automaticRules.values()) {
171 if (component.equals(rule.component) && rule.isAutomaticActive()) {
172 return true;
173 }
174 }
175 return false;
176 }
177
John Spurlockb2278d62015-04-07 12:47:12 -0400178 private boolean updateSnoozing(ZenRule rule) {
John Spurlock39581cc2015-04-10 11:59:01 -0400179 if (rule != null && rule.snoozing && (mFirstEvaluation || !rule.isTrueOrUnknown())) {
John Spurlockb2278d62015-04-07 12:47:12 -0400180 rule.snoozing = false;
181 if (DEBUG) Log.d(TAG, "Snoozing reset for " + rule.conditionId);
182 return true;
183 }
184 return false;
185 }
186
187 private boolean updateCondition(Uri id, Condition condition, ZenRule rule) {
188 if (id == null || rule == null || rule.conditionId == null) return false;
189 if (!rule.conditionId.equals(id)) return false;
190 if (Objects.equals(condition, rule.condition)) return false;
191 rule.condition = condition;
192 return true;
193 }
John Spurlock39581cc2015-04-10 11:59:01 -0400194
John Spurlockb2278d62015-04-07 12:47:12 -0400195}