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 
17 package com.android.settings.applications.defaultapps;
18 
19 import android.Manifest;
20 import android.app.Activity;
21 import android.app.settings.SettingsEnums;
22 import android.content.ComponentName;
23 import android.content.Context;
24 import android.content.DialogInterface;
25 import android.content.Intent;
26 import android.content.pm.PackageManager;
27 import android.content.pm.ResolveInfo;
28 import android.content.pm.ServiceInfo;
29 import android.net.Uri;
30 import android.os.Bundle;
31 import android.os.UserHandle;
32 import android.provider.Settings;
33 import android.service.autofill.AutofillService;
34 import android.service.autofill.AutofillServiceInfo;
35 import android.text.Html;
36 import android.text.TextUtils;
37 import android.util.Log;
38 
39 import androidx.preference.Preference;
40 
41 import com.android.internal.content.PackageMonitor;
42 import com.android.settings.R;
43 import com.android.settingslib.applications.DefaultAppInfo;
44 import com.android.settingslib.utils.ThreadUtils;
45 import com.android.settingslib.widget.CandidateInfo;
46 
47 import java.util.ArrayList;
48 import java.util.List;
49 
50 public class DefaultAutofillPicker extends DefaultAppPickerFragment {
51 
52     private static final String TAG = "DefaultAutofillPicker";
53 
54     static final String SETTING = Settings.Secure.AUTOFILL_SERVICE;
55     static final Intent AUTOFILL_PROBE = new Intent(AutofillService.SERVICE_INTERFACE);
56 
57     /**
58      * Extra set when the fragment is implementing ACTION_REQUEST_SET_AUTOFILL_SERVICE.
59      */
60     public static final String EXTRA_PACKAGE_NAME = "package_name";
61 
62     /**
63      * Set when the fragment is implementing ACTION_REQUEST_SET_AUTOFILL_SERVICE.
64      */
65     private DialogInterface.OnClickListener mCancelListener;
66 
67     @Override
onCreate(Bundle savedInstanceState)68     public void onCreate(Bundle savedInstanceState) {
69         super.onCreate(savedInstanceState);
70 
71         final Activity activity = getActivity();
72         if (activity != null && activity.getIntent().getStringExtra(EXTRA_PACKAGE_NAME) != null) {
73             mCancelListener = (d, w) -> {
74                 activity.setResult(Activity.RESULT_CANCELED);
75                 activity.finish();
76             };
77             // If mCancelListener is not null, fragment is started from
78             // ACTION_REQUEST_SET_AUTOFILL_SERVICE and we should always use the calling uid.
79             mUserId = UserHandle.myUserId();
80         }
81         mSettingsPackageMonitor.register(activity, activity.getMainLooper(), false);
82         update();
83     }
84 
85     @Override
newConfirmationDialogFragment(String selectedKey, CharSequence confirmationMessage)86     protected ConfirmationDialogFragment newConfirmationDialogFragment(String selectedKey,
87             CharSequence confirmationMessage) {
88         final AutofillPickerConfirmationDialogFragment fragment =
89                 new AutofillPickerConfirmationDialogFragment();
90         fragment.init(this, selectedKey, confirmationMessage);
91         return fragment;
92     }
93 
94     /**
95      * Custom dialog fragment that has a cancel listener used to propagate the result back to
96      * caller (for the cases where the picker is launched by
97      * {@code android.settings.REQUEST_SET_AUTOFILL_SERVICE}.
98      */
99     public static class AutofillPickerConfirmationDialogFragment
100             extends ConfirmationDialogFragment {
101 
102         @Override
onCreate(Bundle savedInstanceState)103         public void onCreate(Bundle savedInstanceState) {
104             final DefaultAutofillPicker target = (DefaultAutofillPicker) getTargetFragment();
105             setCancelListener(target.mCancelListener);
106             super.onCreate(savedInstanceState);
107         }
108     }
109 
110     @Override
getPreferenceScreenResId()111     protected int getPreferenceScreenResId() {
112         return R.xml.default_autofill_settings;
113     }
114 
115     @Override
getMetricsCategory()116     public int getMetricsCategory() {
117         return SettingsEnums.DEFAULT_AUTOFILL_PICKER;
118     }
119 
120     @Override
shouldShowItemNone()121     protected boolean shouldShowItemNone() {
122         return true;
123     }
124 
125     /**
126      * Monitor coming and going auto fill services and calls {@link #update()} when necessary
127      */
128     private final PackageMonitor mSettingsPackageMonitor = new PackageMonitor() {
129         @Override
130         public void onPackageAdded(String packageName, int uid) {
131             ThreadUtils.postOnMainThread(() -> update());
132         }
133 
134         @Override
135         public void onPackageModified(String packageName) {
136             ThreadUtils.postOnMainThread(() -> update());
137         }
138 
139         @Override
140         public void onPackageRemoved(String packageName, int uid) {
141             ThreadUtils.postOnMainThread(() -> update());
142         }
143     };
144 
145     /**
146      * Update the data in this UI.
147      */
update()148     private void update() {
149         updateCandidates();
150         addAddServicePreference();
151     }
152 
153     @Override
onDestroy()154     public void onDestroy() {
155         mSettingsPackageMonitor.unregister();
156         super.onDestroy();
157     }
158 
159     /**
160      * Gets the preference that allows to add a new autofill service.
161      *
162      * @return The preference or {@code null} if no service can be added
163      */
newAddServicePreferenceOrNull()164     private Preference newAddServicePreferenceOrNull() {
165         final String searchUri = Settings.Secure.getStringForUser(
166                 getActivity().getContentResolver(),
167                 Settings.Secure.AUTOFILL_SERVICE_SEARCH_URI,
168                 mUserId);
169         if (TextUtils.isEmpty(searchUri)) {
170             return null;
171         }
172 
173         final Intent addNewServiceIntent = new Intent(Intent.ACTION_VIEW, Uri.parse(searchUri));
174         final Context context = getPrefContext();
175         final Preference preference = new Preference(context);
176         preference.setOnPreferenceClickListener(p -> {
177                     context.startActivityAsUser(addNewServiceIntent, UserHandle.of(mUserId));
178                     return true;
179                 });
180         preference.setTitle(R.string.print_menu_item_add_service);
181         preference.setIcon(R.drawable.ic_add_24dp);
182         preference.setOrder(Integer.MAX_VALUE -1);
183         preference.setPersistent(false);
184         return preference;
185     }
186 
187     /**
188      * Add a preference that allows the user to add a service if the market link for that is
189      * configured.
190      */
addAddServicePreference()191     private void addAddServicePreference() {
192         final Preference addNewServicePreference = newAddServicePreferenceOrNull();
193         if (addNewServicePreference != null) {
194             getPreferenceScreen().addPreference(addNewServicePreference);
195         }
196     }
197 
198     @Override
getCandidates()199     protected List<DefaultAppInfo> getCandidates() {
200         final List<DefaultAppInfo> candidates = new ArrayList<>();
201         final List<ResolveInfo> resolveInfos = mPm.queryIntentServicesAsUser(
202                 AUTOFILL_PROBE, PackageManager.GET_META_DATA, mUserId);
203         final Context context = getContext();
204         for (ResolveInfo info : resolveInfos) {
205             final String permission = info.serviceInfo.permission;
206             if (Manifest.permission.BIND_AUTOFILL_SERVICE.equals(permission)) {
207                 candidates.add(new DefaultAppInfo(context, mPm, mUserId, new ComponentName(
208                         info.serviceInfo.packageName, info.serviceInfo.name)));
209             }
210             if (Manifest.permission.BIND_AUTOFILL.equals(permission)) {
211                 // Let it go for now...
212                 Log.w(TAG, "AutofillService from '" + info.serviceInfo.packageName
213                         + "' uses unsupported permission " + Manifest.permission.BIND_AUTOFILL
214                         + ". It works for now, but might not be supported on future releases");
215                 candidates.add(new DefaultAppInfo(context, mPm, mUserId, new ComponentName(
216                         info.serviceInfo.packageName, info.serviceInfo.name)));
217             }
218         }
219         return candidates;
220     }
221 
getDefaultKey(Context context, int userId)222     public static String getDefaultKey(Context context, int userId) {
223         String setting = Settings.Secure.getStringForUser(
224                 context.getContentResolver(), SETTING, userId);
225         if (setting != null) {
226             ComponentName componentName = ComponentName.unflattenFromString(setting);
227             if (componentName != null) {
228                 return componentName.flattenToString();
229             }
230         }
231         return null;
232     }
233 
234     @Override
getDefaultKey()235     protected String getDefaultKey() {
236         return getDefaultKey(getContext(), mUserId);
237     }
238 
239     @Override
getConfirmationMessage(CandidateInfo appInfo)240     protected CharSequence getConfirmationMessage(CandidateInfo appInfo) {
241         if (appInfo == null) {
242             return null;
243         }
244         final CharSequence appName = appInfo.loadLabel();
245         final String message = getContext().getString(
246                 R.string.autofill_confirmation_message, Html.escapeHtml(appName));
247         return Html.fromHtml(message);
248     }
249 
250     @Override
setDefaultKey(String key)251     protected boolean setDefaultKey(String key) {
252         Settings.Secure.putStringForUser(getContext().getContentResolver(), SETTING, key, mUserId);
253 
254         // Check if activity was launched from Settings.ACTION_REQUEST_SET_AUTOFILL_SERVICE
255         // intent, and set proper result if so...
256         final Activity activity = getActivity();
257         if (activity != null) {
258             final String packageName = activity.getIntent().getStringExtra(EXTRA_PACKAGE_NAME);
259             if (packageName != null) {
260                 final int result = key != null && key.startsWith(packageName) ? Activity.RESULT_OK
261                         : Activity.RESULT_CANCELED;
262                 activity.setResult(result);
263                 activity.finish();
264             }
265         }
266         return true;
267     }
268 
269     /**
270      * Provides Intent to setting activity for the specified autofill service.
271      */
272     static final class AutofillSettingIntentProvider implements SettingIntentProvider {
273 
274         private final String mSelectedKey;
275         private final Context mContext;
276         private final int mUserId;
277 
AutofillSettingIntentProvider(Context context, int userId, String key)278         public AutofillSettingIntentProvider(Context context, int userId, String key) {
279             mSelectedKey = key;
280             mContext = context;
281             mUserId = userId;
282         }
283 
284         @Override
getIntent()285         public Intent getIntent() {
286             final List<ResolveInfo> resolveInfos = mContext.getPackageManager()
287                     .queryIntentServicesAsUser(
288                             AUTOFILL_PROBE, PackageManager.GET_META_DATA, mUserId);
289 
290             for (ResolveInfo resolveInfo : resolveInfos) {
291                 final ServiceInfo serviceInfo = resolveInfo.serviceInfo;
292                 final String flattenKey = new ComponentName(
293                         serviceInfo.packageName, serviceInfo.name).flattenToString();
294                 if (TextUtils.equals(mSelectedKey, flattenKey)) {
295                     final String settingsActivity;
296                     try {
297                         settingsActivity = new AutofillServiceInfo(mContext, serviceInfo)
298                                 .getSettingsActivity();
299                     } catch (SecurityException e) {
300                         // Service does not declare the proper permission, ignore it.
301                         Log.w(TAG, "Error getting info for " + serviceInfo + ": " + e);
302                         return null;
303                     }
304                     if (TextUtils.isEmpty(settingsActivity)) {
305                         return null;
306                     }
307                     return new Intent(Intent.ACTION_MAIN).setComponent(
308                             new ComponentName(serviceInfo.packageName, settingsActivity));
309                 }
310             }
311 
312             return null;
313         }
314     }
315 }
316