• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2017 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 package com.android.settingslib.suggestions;
17 
18 import android.Manifest;
19 import android.accounts.Account;
20 import android.accounts.AccountManager;
21 import android.annotation.RequiresPermission;
22 import android.content.Context;
23 import android.content.Intent;
24 import android.content.SharedPreferences;
25 import android.content.pm.PackageManager;
26 import android.content.pm.UserInfo;
27 import android.content.res.Resources;
28 import android.net.ConnectivityManager;
29 import android.net.NetworkInfo;
30 import android.os.UserHandle;
31 import android.os.UserManager;
32 import android.provider.Settings;
33 import android.support.annotation.VisibleForTesting;
34 import android.text.TextUtils;
35 import android.text.format.DateUtils;
36 import android.util.ArrayMap;
37 import android.util.AttributeSet;
38 import android.util.Log;
39 import android.util.Pair;
40 import android.util.Xml;
41 import android.view.InflateException;
42 
43 import com.android.settingslib.drawer.Tile;
44 import com.android.settingslib.drawer.TileUtils;
45 
46 import org.xmlpull.v1.XmlPullParser;
47 import org.xmlpull.v1.XmlPullParserException;
48 
49 import java.io.IOException;
50 import java.util.ArrayList;
51 import java.util.List;
52 
53 public class SuggestionParser {
54 
55     private static final String TAG = "SuggestionParser";
56 
57     // If defined, only returns this suggestion if the feature is supported.
58     public static final String META_DATA_REQUIRE_FEATURE = "com.android.settings.require_feature";
59 
60     // If defined, only display this optional step if an account of that type exists.
61     private static final String META_DATA_REQUIRE_ACCOUNT = "com.android.settings.require_account";
62 
63     // If defined and not true, do not should optional step.
64     private static final String META_DATA_IS_SUPPORTED = "com.android.settings.is_supported";
65 
66     // If defined, only display this optional step if the current user is of that type.
67     private static final String META_DATA_REQUIRE_USER_TYPE =
68             "com.android.settings.require_user_type";
69 
70     // If defined, only display this optional step if a connection is available.
71     private static final String META_DATA_IS_CONNECTION_REQUIRED =
72             "com.android.settings.require_connection";
73 
74     // The valid values that setup wizard recognizes for differentiating user types.
75     private static final String META_DATA_PRIMARY_USER_TYPE_VALUE = "primary";
76     private static final String META_DATA_ADMIN_USER_TYPE_VALUE = "admin";
77     private static final String META_DATA_GUEST_USER_TYPE_VALUE = "guest";
78     private static final String META_DATA_RESTRICTED_USER_TYPE_VALUE = "restricted";
79 
80     /**
81      * Allows suggestions to appear after a certain number of days, and to re-appear if dismissed.
82      * For instance:
83      * 0,10
84      * Will appear immediately, but if the user removes it, it will come back after 10 days.
85      *
86      * Another example:
87      * 10,30
88      * Will only show up after 10 days, and then again after 30.
89      */
90     public static final String META_DATA_DISMISS_CONTROL = "com.android.settings.dismiss";
91 
92     // Shared prefs keys for storing dismissed state.
93     // Index into current dismissed state.
94     public static final String SETUP_TIME = "_setup_time";
95     private static final String IS_DISMISSED = "_is_dismissed";
96 
97     // Default dismiss control for smart suggestions.
98     private static final String DEFAULT_SMART_DISMISS_CONTROL = "0";
99 
100     private final Context mContext;
101     private final List<SuggestionCategory> mSuggestionList;
102     private final ArrayMap<Pair<String, String>, Tile> mAddCache = new ArrayMap<>();
103     private final SharedPreferences mSharedPrefs;
104     private final String mDefaultDismissControl;
105 
SuggestionParser(Context context, SharedPreferences sharedPrefs, int orderXml, String defaultDismissControl)106     public SuggestionParser(Context context, SharedPreferences sharedPrefs, int orderXml,
107             String defaultDismissControl) {
108         this(
109                 context,
110                 sharedPrefs,
111                 (List<SuggestionCategory>) new SuggestionOrderInflater(context).parse(orderXml),
112                 defaultDismissControl);
113     }
114 
SuggestionParser(Context context, SharedPreferences sharedPrefs, int orderXml)115     public SuggestionParser(Context context, SharedPreferences sharedPrefs, int orderXml) {
116         this(context, sharedPrefs, orderXml, DEFAULT_SMART_DISMISS_CONTROL);
117     }
118 
119     @VisibleForTesting
SuggestionParser( Context context, SharedPreferences sharedPrefs, List<SuggestionCategory> suggestionList, String defaultDismissControl)120     public SuggestionParser(
121             Context context,
122             SharedPreferences sharedPrefs,
123             List<SuggestionCategory> suggestionList,
124             String defaultDismissControl) {
125         mContext = context;
126         mSuggestionList = suggestionList;
127         mSharedPrefs = sharedPrefs;
128         mDefaultDismissControl = defaultDismissControl;
129     }
130 
getSuggestions(boolean isSmartSuggestionEnabled)131     public SuggestionList getSuggestions(boolean isSmartSuggestionEnabled) {
132         final SuggestionList suggestionList = new SuggestionList();
133         final int N = mSuggestionList.size();
134         for (int i = 0; i < N; i++) {
135             final SuggestionCategory category = mSuggestionList.get(i);
136             if (category.exclusive && !isExclusiveCategoryExpired(category)) {
137                 // If suggestions from an exclusive category are present, parsing is stopped
138                 // and only suggestions from that category are displayed. Note that subsequent
139                 // exclusive categories are also ignored.
140                 final List<Tile> exclusiveSuggestions = new ArrayList<>();
141 
142                 // Read suggestion and force isSmartSuggestion to be false so the rule defined
143                 // from each suggestion itself is used.
144                 readSuggestions(category, exclusiveSuggestions, false /* isSmartSuggestion */);
145                 if (!exclusiveSuggestions.isEmpty()) {
146                     final SuggestionList exclusiveList = new SuggestionList();
147                     exclusiveList.addSuggestions(category, exclusiveSuggestions);
148                     return exclusiveList;
149                 }
150             } else {
151                 // Either the category is not exclusive, or the exclusiveness expired so we should
152                 // treat it as a normal category.
153                 final List<Tile> suggestions = new ArrayList<>();
154                 readSuggestions(category, suggestions, isSmartSuggestionEnabled);
155                 suggestionList.addSuggestions(category, suggestions);
156             }
157         }
158         return suggestionList;
159     }
160 
161     /**
162      * Dismisses a suggestion, returns true if the suggestion has no more dismisses left and should
163      * be disabled.
164      */
dismissSuggestion(Tile suggestion)165     public boolean dismissSuggestion(Tile suggestion) {
166         final String keyBase = suggestion.intent.getComponent().flattenToShortString();
167         mSharedPrefs.edit()
168                 .putBoolean(keyBase + IS_DISMISSED, true)
169                 .commit();
170         return true;
171     }
172 
173     @VisibleForTesting
filterSuggestions( List<Tile> suggestions, int countBefore, boolean isSmartSuggestionEnabled)174     public void filterSuggestions(
175             List<Tile> suggestions, int countBefore, boolean isSmartSuggestionEnabled) {
176         for (int i = countBefore; i < suggestions.size(); i++) {
177             if (!isAvailable(suggestions.get(i)) ||
178                     !isSupported(suggestions.get(i)) ||
179                     !satisifesRequiredUserType(suggestions.get(i)) ||
180                     !satisfiesRequiredAccount(suggestions.get(i)) ||
181                     !satisfiesConnectivity(suggestions.get(i)) ||
182                     isDismissed(suggestions.get(i), isSmartSuggestionEnabled)) {
183                 suggestions.remove(i--);
184             }
185         }
186     }
187 
188     @VisibleForTesting
readSuggestions( SuggestionCategory category, List<Tile> suggestions, boolean isSmartSuggestionEnabled)189     void readSuggestions(
190             SuggestionCategory category, List<Tile> suggestions, boolean isSmartSuggestionEnabled) {
191         int countBefore = suggestions.size();
192         Intent intent = new Intent(Intent.ACTION_MAIN);
193         intent.addCategory(category.category);
194         if (category.pkg != null) {
195             intent.setPackage(category.pkg);
196         }
197         TileUtils.getTilesForIntent(mContext, new UserHandle(UserHandle.myUserId()), intent,
198                 mAddCache, null, suggestions, true, false, false, true /* shouldUpdateTiles */);
199         filterSuggestions(suggestions, countBefore, isSmartSuggestionEnabled);
200         if (!category.multiple && suggestions.size() > (countBefore + 1)) {
201             // If there are too many, remove them all and only re-add the one with the highest
202             // priority.
203             Tile item = suggestions.remove(suggestions.size() - 1);
204             while (suggestions.size() > countBefore) {
205                 Tile last = suggestions.remove(suggestions.size() - 1);
206                 if (last.priority > item.priority) {
207                     item = last;
208                 }
209             }
210             // If category is marked as done, do not add any item.
211             if (!isCategoryDone(category.category)) {
212                 suggestions.add(item);
213             }
214         }
215     }
216 
isAvailable(Tile suggestion)217     private boolean isAvailable(Tile suggestion) {
218         final String featuresRequired = suggestion.metaData.getString(META_DATA_REQUIRE_FEATURE);
219         if (featuresRequired != null) {
220             for (String feature : featuresRequired.split(",")) {
221                 if (TextUtils.isEmpty(feature)) {
222                     Log.w(TAG, "Found empty substring when parsing required features: "
223                             + featuresRequired);
224                 } else if (!mContext.getPackageManager().hasSystemFeature(feature)) {
225                     Log.i(TAG, suggestion.title + " requires unavailable feature " + feature);
226                     return false;
227                 }
228             }
229         }
230         return true;
231     }
232 
233     @RequiresPermission(Manifest.permission.MANAGE_USERS)
satisifesRequiredUserType(Tile suggestion)234     private boolean satisifesRequiredUserType(Tile suggestion) {
235         final String requiredUser = suggestion.metaData.getString(META_DATA_REQUIRE_USER_TYPE);
236         if (requiredUser != null) {
237             final UserManager userManager = mContext.getSystemService(UserManager.class);
238             UserInfo userInfo = userManager.getUserInfo(UserHandle.myUserId());
239             for (String userType : requiredUser.split("\\|")) {
240                 final boolean primaryUserCondtionMet = userInfo.isPrimary()
241                         && META_DATA_PRIMARY_USER_TYPE_VALUE.equals(userType);
242                 final boolean adminUserConditionMet = userInfo.isAdmin()
243                         && META_DATA_ADMIN_USER_TYPE_VALUE.equals(userType);
244                 final boolean guestUserCondtionMet = userInfo.isGuest()
245                         && META_DATA_GUEST_USER_TYPE_VALUE.equals(userType);
246                 final boolean restrictedUserCondtionMet = userInfo.isRestricted()
247                         && META_DATA_RESTRICTED_USER_TYPE_VALUE.equals(userType);
248                 if (primaryUserCondtionMet || adminUserConditionMet || guestUserCondtionMet
249                         || restrictedUserCondtionMet) {
250                     return true;
251                 }
252             }
253             Log.i(TAG, suggestion.title + " requires user type " + requiredUser);
254             return false;
255         }
256         return true;
257     }
258 
satisfiesRequiredAccount(Tile suggestion)259     public boolean satisfiesRequiredAccount(Tile suggestion) {
260         final String requiredAccountType = suggestion.metaData.getString(META_DATA_REQUIRE_ACCOUNT);
261         if (requiredAccountType == null) {
262             return true;
263         }
264         AccountManager accountManager = mContext.getSystemService(AccountManager.class);
265         Account[] accounts = accountManager.getAccountsByType(requiredAccountType);
266         boolean satisfiesRequiredAccount = accounts.length > 0;
267         if (!satisfiesRequiredAccount) {
268             Log.i(TAG, suggestion.title + " requires unavailable account type "
269                     + requiredAccountType);
270         }
271         return satisfiesRequiredAccount;
272     }
273 
isSupported(Tile suggestion)274     public boolean isSupported(Tile suggestion) {
275         final int isSupportedResource = suggestion.metaData.getInt(META_DATA_IS_SUPPORTED);
276         try {
277             if (suggestion.intent == null) {
278                 return false;
279             }
280             final Resources res = mContext.getPackageManager().getResourcesForActivity(
281                     suggestion.intent.getComponent());
282             boolean isSupported =
283                     isSupportedResource != 0 ? res.getBoolean(isSupportedResource) : true;
284             if (!isSupported) {
285                 Log.i(TAG, suggestion.title + " requires unsupported resource "
286                         + isSupportedResource);
287             }
288             return isSupported;
289         } catch (PackageManager.NameNotFoundException e) {
290             Log.w(TAG, "Cannot find resources for " + suggestion.intent.getComponent());
291             return false;
292         } catch (Resources.NotFoundException e) {
293             Log.w(TAG, "Cannot find resources for " + suggestion.intent.getComponent(), e);
294             return false;
295         }
296     }
297 
satisfiesConnectivity(Tile suggestion)298     private boolean satisfiesConnectivity(Tile suggestion) {
299         final boolean isConnectionRequired =
300                 suggestion.metaData.getBoolean(META_DATA_IS_CONNECTION_REQUIRED);
301         if (!isConnectionRequired) {
302             return true;
303         }
304         ConnectivityManager cm =
305                 (ConnectivityManager) mContext.getSystemService(Context.CONNECTIVITY_SERVICE);
306         NetworkInfo netInfo = cm.getActiveNetworkInfo();
307         boolean satisfiesConnectivity = netInfo != null && netInfo.isConnectedOrConnecting();
308         if (!satisfiesConnectivity) {
309             Log.i(TAG, suggestion.title + " is missing required connection.");
310         }
311         return satisfiesConnectivity;
312     }
313 
isCategoryDone(String category)314     public boolean isCategoryDone(String category) {
315         String name = Settings.Secure.COMPLETED_CATEGORY_PREFIX + category;
316         return Settings.Secure.getInt(mContext.getContentResolver(), name, 0) != 0;
317     }
318 
markCategoryDone(String category)319     public void markCategoryDone(String category) {
320         String name = Settings.Secure.COMPLETED_CATEGORY_PREFIX + category;
321         Settings.Secure.putInt(mContext.getContentResolver(), name, 1);
322     }
323 
324     /**
325      * Whether or not the category's exclusiveness has expired.
326      */
isExclusiveCategoryExpired(SuggestionCategory category)327     private boolean isExclusiveCategoryExpired(SuggestionCategory category) {
328         final String keySetupTime = category.category + SETUP_TIME;
329         final long currentTime = System.currentTimeMillis();
330         if (!mSharedPrefs.contains(keySetupTime)) {
331             mSharedPrefs.edit()
332                     .putLong(keySetupTime, currentTime)
333                     .commit();
334         }
335         if (category.exclusiveExpireDaysInMillis < 0) {
336             // negative means never expires
337             return false;
338         }
339         final long setupTime = mSharedPrefs.getLong(keySetupTime, 0);
340         final long elapsedTime = currentTime - setupTime;
341         Log.d(TAG, "Day " + elapsedTime / DateUtils.DAY_IN_MILLIS + " for " + category.category);
342         return elapsedTime > category.exclusiveExpireDaysInMillis;
343     }
344 
345     @VisibleForTesting
isDismissed(Tile suggestion, boolean isSmartSuggestionEnabled)346     boolean isDismissed(Tile suggestion, boolean isSmartSuggestionEnabled) {
347         String dismissControl = getDismissControl(suggestion, isSmartSuggestionEnabled);
348         String keyBase = suggestion.intent.getComponent().flattenToShortString();
349         if (!mSharedPrefs.contains(keyBase + SETUP_TIME)) {
350             mSharedPrefs.edit()
351                     .putLong(keyBase + SETUP_TIME, System.currentTimeMillis())
352                     .commit();
353         }
354         // Check if it's already manually dismissed
355         final boolean isDismissed = mSharedPrefs.getBoolean(keyBase + IS_DISMISSED, false);
356         if (isDismissed) {
357             return true;
358         }
359         if (dismissControl == null) {
360             return false;
361         }
362         // Parse when suggestion should first appear. return true to artificially hide suggestion
363         // before then.
364         int firstAppearDay = parseDismissString(dismissControl);
365         long firstAppearDayInMs = getEndTime(mSharedPrefs.getLong(keyBase + SETUP_TIME, 0),
366                 firstAppearDay);
367         if (System.currentTimeMillis() >= firstAppearDayInMs) {
368             // Dismiss timeout has passed, undismiss it.
369             mSharedPrefs.edit()
370                     .putBoolean(keyBase + IS_DISMISSED, false)
371                     .commit();
372             return false;
373         }
374         return true;
375     }
376 
getEndTime(long startTime, int daysDelay)377     private long getEndTime(long startTime, int daysDelay) {
378         long days = daysDelay * DateUtils.DAY_IN_MILLIS;
379         return startTime + days;
380     }
381 
382     /**
383      * Parse the first int from a string formatted as "0,1,2..."
384      * The value means suggestion should first appear on Day X.
385      */
parseDismissString(String dismissControl)386     private int parseDismissString(String dismissControl) {
387         final String[] dismissStrs = dismissControl.split(",");
388         return Integer.parseInt(dismissStrs[0]);
389     }
390 
getDismissControl(Tile suggestion, boolean isSmartSuggestionEnabled)391     private String getDismissControl(Tile suggestion, boolean isSmartSuggestionEnabled) {
392         if (isSmartSuggestionEnabled) {
393             return mDefaultDismissControl;
394         } else {
395             return suggestion.metaData.getString(META_DATA_DISMISS_CONTROL);
396         }
397     }
398 
399     private static class SuggestionOrderInflater {
400         private static final String TAG_LIST = "optional-steps";
401         private static final String TAG_ITEM = "step";
402 
403         private static final String ATTR_CATEGORY = "category";
404         private static final String ATTR_PACKAGE = "package";
405         private static final String ATTR_MULTIPLE = "multiple";
406         private static final String ATTR_EXCLUSIVE = "exclusive";
407         private static final String ATTR_EXCLUSIVE_EXPIRE_DAYS = "exclusiveExpireDays";
408 
409         private final Context mContext;
410 
SuggestionOrderInflater(Context context)411         public SuggestionOrderInflater(Context context) {
412             mContext = context;
413         }
414 
parse(int resource)415         public Object parse(int resource) {
416             XmlPullParser parser = mContext.getResources().getXml(resource);
417             final AttributeSet attrs = Xml.asAttributeSet(parser);
418             try {
419                 // Look for the root node.
420                 int type;
421                 do {
422                     type = parser.next();
423                 } while (type != XmlPullParser.START_TAG && type != XmlPullParser.END_DOCUMENT);
424 
425                 if (type != XmlPullParser.START_TAG) {
426                     throw new InflateException(parser.getPositionDescription()
427                             + ": No start tag found!");
428                 }
429 
430                 // Temp is the root that was found in the xml
431                 Object xmlRoot = onCreateItem(parser.getName(), attrs);
432 
433                 // Inflate all children under temp
434                 rParse(parser, xmlRoot, attrs);
435                 return xmlRoot;
436             } catch (XmlPullParserException | IOException e) {
437                 Log.w(TAG, "Problem parser resource " + resource, e);
438                 return null;
439             }
440         }
441 
442         /**
443          * Recursive method used to descend down the xml hierarchy and instantiate
444          * items, instantiate their children.
445          */
rParse(XmlPullParser parser, Object parent, final AttributeSet attrs)446         private void rParse(XmlPullParser parser, Object parent, final AttributeSet attrs)
447                 throws XmlPullParserException, IOException {
448             final int depth = parser.getDepth();
449 
450             int type;
451             while (((type = parser.next()) != XmlPullParser.END_TAG ||
452                     parser.getDepth() > depth) && type != XmlPullParser.END_DOCUMENT) {
453                 if (type != XmlPullParser.START_TAG) {
454                     continue;
455                 }
456 
457                 final String name = parser.getName();
458 
459                 Object item = onCreateItem(name, attrs);
460                 onAddChildItem(parent, item);
461                 rParse(parser, item, attrs);
462             }
463         }
464 
onAddChildItem(Object parent, Object child)465         protected void onAddChildItem(Object parent, Object child) {
466             if (parent instanceof List<?> && child instanceof SuggestionCategory) {
467                 ((List<SuggestionCategory>) parent).add((SuggestionCategory) child);
468             } else {
469                 throw new IllegalArgumentException("Parent was not a list");
470             }
471         }
472 
onCreateItem(String name, AttributeSet attrs)473         protected Object onCreateItem(String name, AttributeSet attrs) {
474             if (name.equals(TAG_LIST)) {
475                 return new ArrayList<SuggestionCategory>();
476             } else if (name.equals(TAG_ITEM)) {
477                 SuggestionCategory category = new SuggestionCategory();
478                 category.category = attrs.getAttributeValue(null, ATTR_CATEGORY);
479                 category.pkg = attrs.getAttributeValue(null, ATTR_PACKAGE);
480                 String multiple = attrs.getAttributeValue(null, ATTR_MULTIPLE);
481                 category.multiple = !TextUtils.isEmpty(multiple) && Boolean.parseBoolean(multiple);
482                 String exclusive = attrs.getAttributeValue(null, ATTR_EXCLUSIVE);
483                 category.exclusive =
484                         !TextUtils.isEmpty(exclusive) && Boolean.parseBoolean(exclusive);
485                 String expireDaysAttr = attrs.getAttributeValue(null,
486                         ATTR_EXCLUSIVE_EXPIRE_DAYS);
487                 long expireDays = !TextUtils.isEmpty(expireDaysAttr)
488                         ? Integer.parseInt(expireDaysAttr)
489                         : -1;
490                 category.exclusiveExpireDaysInMillis = DateUtils.DAY_IN_MILLIS * expireDays;
491                 return category;
492             } else {
493                 throw new IllegalArgumentException("Unknown item " + name);
494             }
495         }
496     }
497 }
498 
499