summaryrefslogtreecommitdiff
path: root/java/com/android/dialer/speeddial/DisambigDialog.java
diff options
context:
space:
mode:
Diffstat (limited to 'java/com/android/dialer/speeddial/DisambigDialog.java')
-rw-r--r--java/com/android/dialer/speeddial/DisambigDialog.java244
1 files changed, 244 insertions, 0 deletions
diff --git a/java/com/android/dialer/speeddial/DisambigDialog.java b/java/com/android/dialer/speeddial/DisambigDialog.java
new file mode 100644
index 000000000..ca02f41eb
--- /dev/null
+++ b/java/com/android/dialer/speeddial/DisambigDialog.java
@@ -0,0 +1,244 @@
+/*
+ * Copyright (C) 2017 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.dialer.speeddial;
+
+import android.app.AlertDialog;
+import android.app.Dialog;
+import android.app.DialogFragment;
+import android.app.FragmentManager;
+import android.content.ContentResolver;
+import android.content.res.Resources;
+import android.database.Cursor;
+import android.os.Bundle;
+import android.provider.ContactsContract.CommonDataKinds.Phone;
+import android.support.annotation.Nullable;
+import android.support.annotation.VisibleForTesting;
+import android.text.TextUtils;
+import android.util.ArraySet;
+import android.view.LayoutInflater;
+import android.view.View;
+import android.widget.LinearLayout;
+import android.widget.TextView;
+import com.android.dialer.callintent.CallInitiationType;
+import com.android.dialer.callintent.CallIntentBuilder;
+import com.android.dialer.common.LogUtil;
+import com.android.dialer.common.concurrent.DialerExecutor.Worker;
+import com.android.dialer.common.concurrent.DialerExecutorComponent;
+import com.android.dialer.duo.DuoComponent;
+import com.android.dialer.precall.PreCall;
+import java.util.ArrayList;
+import java.util.Arrays;
+import java.util.Set;
+
+/** Disambiguation dialog for favorite contacts in {@link SpeedDialFragment}. */
+public class DisambigDialog extends DialogFragment {
+
+ @VisibleForTesting public static final String DISAMBIG_DIALOG_TAG = "disambig_dialog";
+ private static final String DISAMBIG_DIALOG_WORKER_TAG = "disambig_dialog_worker";
+
+ private final Set<String> phoneNumbers = new ArraySet<>();
+ private LinearLayout container;
+ private String lookupKey;
+
+ /** Show a disambiguation dialog for a starred contact without a favorite communication avenue. */
+ public static DisambigDialog show(String lookupKey, FragmentManager manager) {
+ DisambigDialog dialog = new DisambigDialog();
+ dialog.lookupKey = lookupKey;
+ dialog.show(manager, DISAMBIG_DIALOG_TAG);
+ return dialog;
+ }
+
+ @Override
+ public Dialog onCreateDialog(Bundle savedInstanceState) {
+ LayoutInflater inflater = getActivity().getLayoutInflater();
+ View view = inflater.inflate(R.layout.disambig_dialog_layout, null, false);
+ container = view.findViewById(R.id.communication_avenue_container);
+ return new AlertDialog.Builder(getActivity()).setView(view).create();
+ }
+
+ @Override
+ public void onResume() {
+ super.onResume();
+ lookupContactInfo();
+ }
+
+ @Override
+ public void onPause() {
+ super.onPause();
+ // TODO(calderwoodra): for simplicity, just dismiss the dialog on configuration change and
+ // consider changing this later.
+ dismiss();
+ }
+
+ private void lookupContactInfo() {
+ DialerExecutorComponent.get(getContext())
+ .dialerExecutorFactory()
+ .createUiTaskBuilder(
+ getFragmentManager(),
+ DISAMBIG_DIALOG_WORKER_TAG,
+ new LookupContactInfoWorker(getContext().getContentResolver()))
+ .onSuccess(this::insertOptions)
+ .onFailure(this::onLookupFailed)
+ .build()
+ .executeParallel(lookupKey);
+ }
+
+ /**
+ * Inflates and inserts the following in the dialog:
+ *
+ * <ul>
+ * <li>Header for each unique phone number
+ * <li>Clickable video option if the phone number is video reachable (ViLTE, Duo)
+ * <li>Clickable voice option
+ * </ul>
+ */
+ private void insertOptions(Cursor cursor) {
+ if (!cursorIsValid(cursor)) {
+ dismiss();
+ return;
+ }
+
+ do {
+ String number = cursor.getString(LookupContactInfoWorker.NUMBER_INDEX);
+ // TODO(calderwoodra): improve this to include fuzzy matching
+ if (phoneNumbers.add(number)) {
+ insertOption(
+ number,
+ getLabel(getContext().getResources(), cursor),
+ isVideoReachable(cursor, number));
+ }
+ } while (cursor.moveToNext());
+ cursor.close();
+ // TODO(calderwoodra): set max height of the scrollview. Might need to override onMeasure.
+ }
+
+ /** Returns true if the given number is ViLTE reachable or Duo reachable. */
+ private boolean isVideoReachable(Cursor cursor, String number) {
+ boolean isVideoReachable = cursor.getInt(LookupContactInfoWorker.PHONE_PRESENCE_INDEX) == 1;
+ if (!isVideoReachable) {
+ isVideoReachable = DuoComponent.get(getContext()).getDuo().isReachable(getContext(), number);
+ }
+ return isVideoReachable;
+ }
+
+ /** Inserts a group of options for a specific phone number. */
+ private void insertOption(String number, String phoneType, boolean isVideoReachable) {
+ View view =
+ getActivity()
+ .getLayoutInflater()
+ .inflate(R.layout.disambig_option_layout, container, false);
+ ((TextView) view.findViewById(R.id.phone_type)).setText(phoneType);
+ ((TextView) view.findViewById(R.id.phone_number)).setText(number);
+
+ if (isVideoReachable) {
+ View videoOption = view.findViewById(R.id.video_call_container);
+ videoOption.setOnClickListener(v -> onVideoOptionClicked(number));
+ videoOption.setVisibility(View.VISIBLE);
+ }
+ View voiceOption = view.findViewById(R.id.voice_call_container);
+ voiceOption.setOnClickListener(v -> onVoiceOptionClicked(number));
+ container.addView(view);
+ }
+
+ private void onVideoOptionClicked(String number) {
+ // TODO(calderwoodra): save this option if remember is checked
+ // TODO(calderwoodra): place a duo call if possible
+ PreCall.start(
+ getContext(),
+ new CallIntentBuilder(number, CallInitiationType.Type.SPEED_DIAL).setIsVideoCall(true));
+ }
+
+ private void onVoiceOptionClicked(String number) {
+ // TODO(calderwoodra): save this option if remember is checked
+ PreCall.start(getContext(), new CallIntentBuilder(number, CallInitiationType.Type.SPEED_DIAL));
+ }
+
+ // TODO(calderwoodra): handle CNAP and cequint types.
+ // TODO(calderwoodra): unify this into a utility method with CallLogAdapter#getNumberType
+ private static String getLabel(Resources resources, Cursor cursor) {
+ int numberType = cursor.getInt(LookupContactInfoWorker.PHONE_TYPE_INDEX);
+ String numberLabel = cursor.getString(LookupContactInfoWorker.PHONE_LABEL_INDEX);
+
+ // Returns empty label instead of "custom" if the custom label is empty.
+ if (numberType == Phone.TYPE_CUSTOM && TextUtils.isEmpty(numberLabel)) {
+ return "";
+ }
+ return (String) Phone.getTypeLabel(resources, numberType, numberLabel);
+ }
+
+ // Checks if the cursor is valid and logs an error if there are any issues.
+ private static boolean cursorIsValid(Cursor cursor) {
+ if (cursor == null) {
+ LogUtil.e("DisambigDialog.insertOptions", "cursor null.");
+ return false;
+ } else if (cursor.isClosed()) {
+ LogUtil.e("DisambigDialog.insertOptions", "cursor closed.");
+ cursor.close();
+ return false;
+ } else if (!cursor.moveToFirst()) {
+ LogUtil.e("DisambigDialog.insertOptions", "cursor empty.");
+ cursor.close();
+ return false;
+ }
+ return true;
+ }
+
+ private void onLookupFailed(Throwable throwable) {
+ LogUtil.e("DisambigDialog.onLookupFailed", null, throwable);
+ insertOptions(null);
+ }
+
+ private static class LookupContactInfoWorker implements Worker<String, Cursor> {
+
+ static final int NUMBER_INDEX = 0;
+ static final int PHONE_TYPE_INDEX = 1;
+ static final int PHONE_LABEL_INDEX = 2;
+ static final int PHONE_PRESENCE_INDEX = 3;
+
+ private static final String[] projection =
+ new String[] {Phone.NUMBER, Phone.TYPE, Phone.LABEL, Phone.CARRIER_PRESENCE};
+ private final ContentResolver resolver;
+
+ LookupContactInfoWorker(ContentResolver resolver) {
+ this.resolver = resolver;
+ }
+
+ @Nullable
+ @Override
+ public Cursor doInBackground(@Nullable String lookupKey) throws Throwable {
+ if (TextUtils.isEmpty(lookupKey)) {
+ LogUtil.e("LookupConctactInfoWorker.doInBackground", "contact id unsest.");
+ return null;
+ }
+ return resolver.query(
+ Phone.CONTENT_URI, projection, Phone.LOOKUP_KEY + " = ?", new String[] {lookupKey}, null);
+ }
+ }
+
+ @VisibleForTesting
+ public static String[] getProjectionForTesting() {
+ ArrayList<String> projection =
+ new ArrayList<>(Arrays.asList(LookupContactInfoWorker.projection));
+ projection.add(Phone.LOOKUP_KEY);
+ return projection.toArray(new String[projection.size()]);
+ }
+
+ @VisibleForTesting
+ public LinearLayout getContainer() {
+ return container;
+ }
+}