blob: ca02f41eb09fae1fbf3c0ad233afbe51c7b1236e [file] [log] [blame]
/*
* 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;
}
}