blob: 9772cb1df15c981416f72995be9d3af66b8bdc4d [file] [log] [blame]
Eric Erfanianccca3152017-02-22 16:32:36 -08001/*
2 * Copyright (C) 2013 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.incallui;
18
19import static com.android.contacts.common.compat.CallCompat.Details.PROPERTY_ENTERPRISE_CALL;
20
21import android.Manifest;
Eric Erfanianccca3152017-02-22 16:32:36 -080022import android.content.Context;
23import android.content.Intent;
24import android.content.IntentFilter;
25import android.content.pm.ApplicationInfo;
26import android.content.pm.PackageManager;
27import android.graphics.drawable.Drawable;
28import android.hardware.display.DisplayManager;
29import android.os.BatteryManager;
30import android.os.Handler;
Eric Erfanian2ca43182017-08-31 06:57:16 -070031import android.os.Trace;
Eric Erfaniand5e47f62017-03-15 14:41:07 -070032import android.support.annotation.NonNull;
Eric Erfanianccca3152017-02-22 16:32:36 -080033import android.support.annotation.Nullable;
34import android.support.v4.app.Fragment;
35import android.support.v4.content.ContextCompat;
36import android.telecom.Call.Details;
37import android.telecom.StatusHints;
38import android.telecom.TelecomManager;
wangqia4623702018-04-02 14:23:38 -070039import android.text.BidiFormatter;
40import android.text.TextDirectionHeuristics;
Eric Erfanianccca3152017-02-22 16:32:36 -080041import android.text.TextUtils;
42import android.view.Display;
43import android.view.View;
44import android.view.accessibility.AccessibilityEvent;
45import android.view.accessibility.AccessibilityManager;
46import com.android.contacts.common.ContactsUtils;
47import com.android.contacts.common.preference.ContactsPreferences;
48import com.android.contacts.common.util.ContactDisplayUtils;
49import com.android.dialer.common.Assert;
Eric Erfanianccca3152017-02-22 16:32:36 -080050import com.android.dialer.common.LogUtil;
Eric Erfanian2ca43182017-08-31 06:57:16 -070051import com.android.dialer.configprovider.ConfigProviderBindings;
Eric Erfanian8369df02017-05-03 10:27:13 -070052import com.android.dialer.logging.DialerImpression;
Eric Erfaniand8046e52017-04-06 09:41:50 -070053import com.android.dialer.logging.Logger;
Eric Erfanianccca3152017-02-22 16:32:36 -080054import com.android.dialer.multimedia.MultimediaData;
Eric Erfaniand5e47f62017-03-15 14:41:07 -070055import com.android.dialer.oem.MotorolaUtils;
wangqi97539352017-09-25 11:15:16 -070056import com.android.dialer.phonenumberutil.PhoneNumberHelper;
Eric Erfanian2ca43182017-08-31 06:57:16 -070057import com.android.dialer.postcall.PostCall;
twyen4fde0ac2018-03-22 18:04:23 -070058import com.android.dialer.preferredsim.suggestion.SuggestionProvider;
Eric Erfanianccca3152017-02-22 16:32:36 -080059import com.android.incallui.ContactInfoCache.ContactCacheEntry;
60import com.android.incallui.ContactInfoCache.ContactInfoCacheCallback;
61import com.android.incallui.InCallPresenter.InCallDetailsListener;
62import com.android.incallui.InCallPresenter.InCallEventListener;
63import com.android.incallui.InCallPresenter.InCallState;
64import com.android.incallui.InCallPresenter.InCallStateListener;
65import com.android.incallui.InCallPresenter.IncomingCallListener;
66import com.android.incallui.call.CallList;
67import com.android.incallui.call.DialerCall;
Eric Erfanianccca3152017-02-22 16:32:36 -080068import com.android.incallui.call.DialerCallListener;
wangqibb94ca62018-04-27 14:34:04 -070069import com.android.incallui.call.state.DialerCallState;
Eric Erfaniand5e47f62017-03-15 14:41:07 -070070import com.android.incallui.calllocation.CallLocation;
71import com.android.incallui.calllocation.CallLocationComponent;
Eric Erfanianccca3152017-02-22 16:32:36 -080072import com.android.incallui.incall.protocol.ContactPhotoType;
73import com.android.incallui.incall.protocol.InCallScreen;
74import com.android.incallui.incall.protocol.InCallScreenDelegate;
75import com.android.incallui.incall.protocol.PrimaryCallState;
Eric Erfanian2ca43182017-08-31 06:57:16 -070076import com.android.incallui.incall.protocol.PrimaryCallState.ButtonState;
Eric Erfanianccca3152017-02-22 16:32:36 -080077import com.android.incallui.incall.protocol.PrimaryInfo;
78import com.android.incallui.incall.protocol.SecondaryInfo;
Eric Erfanian90508232017-03-24 09:31:16 -070079import com.android.incallui.videotech.utils.SessionModificationState;
Eric Erfanianccca3152017-02-22 16:32:36 -080080import java.lang.ref.WeakReference;
81
82/**
83 * Controller for the Call Card Fragment. This class listens for changes to InCallState and passes
84 * it along to the fragment.
85 */
86public class CallCardPresenter
87 implements InCallStateListener,
88 IncomingCallListener,
89 InCallDetailsListener,
90 InCallEventListener,
91 InCallScreenDelegate,
Eric Erfanian2ca43182017-08-31 06:57:16 -070092 DialerCallListener {
Eric Erfanianccca3152017-02-22 16:32:36 -080093
94 /**
95 * Amount of time to wait before sending an announcement via the accessibility manager. When the
96 * call state changes to an outgoing or incoming state for the first time, the UI can often be
97 * changing due to call updates or contact lookup. This allows the UI to settle to a stable state
98 * to ensure that the correct information is announced.
99 */
100 private static final long ACCESSIBILITY_ANNOUNCEMENT_DELAY_MILLIS = 500;
101
102 /** Flag to allow the user's current location to be shown during emergency calls. */
103 private static final String CONFIG_ENABLE_EMERGENCY_LOCATION = "config_enable_emergency_location";
104
105 private static final boolean CONFIG_ENABLE_EMERGENCY_LOCATION_DEFAULT = true;
106
107 /**
108 * Make it possible to not get location during an emergency call if the battery is too low, since
109 * doing so could trigger gps and thus potentially cause the phone to die in the middle of the
110 * call.
111 */
112 private static final String CONFIG_MIN_BATTERY_PERCENT_FOR_EMERGENCY_LOCATION =
113 "min_battery_percent_for_emergency_location";
114
115 private static final long CONFIG_MIN_BATTERY_PERCENT_FOR_EMERGENCY_LOCATION_DEFAULT = 10;
116
linyuh183cb712017-12-27 17:02:37 -0800117 private final Context context;
Eric Erfanianccca3152017-02-22 16:32:36 -0800118 private final Handler handler = new Handler();
119
linyuh183cb712017-12-27 17:02:37 -0800120 private DialerCall primary;
wangqidd5f1a52018-01-23 18:01:00 -0800121 private String primaryNumber;
linyuh183cb712017-12-27 17:02:37 -0800122 private DialerCall secondary;
wangqidd5f1a52018-01-23 18:01:00 -0800123 private String secondaryNumber;
linyuh183cb712017-12-27 17:02:37 -0800124 private ContactCacheEntry primaryContactInfo;
125 private ContactCacheEntry secondaryContactInfo;
126 @Nullable private ContactsPreferences contactsPreferences;
127 private boolean isFullscreen = false;
128 private InCallScreen inCallScreen;
Eric Erfanianccca3152017-02-22 16:32:36 -0800129 private boolean isInCallScreenReady;
130 private boolean shouldSendAccessibilityEvent;
Eric Erfaniand5e47f62017-03-15 14:41:07 -0700131
132 @NonNull private final CallLocation callLocation;
Eric Erfanianccca3152017-02-22 16:32:36 -0800133 private final Runnable sendAccessibilityEventRunnable =
134 new Runnable() {
135 @Override
136 public void run() {
linyuh183cb712017-12-27 17:02:37 -0800137 shouldSendAccessibilityEvent = !sendAccessibilityEvent(context, getUi());
Eric Erfanianccca3152017-02-22 16:32:36 -0800138 LogUtil.i(
139 "CallCardPresenter.sendAccessibilityEventRunnable",
140 "still should send: %b",
141 shouldSendAccessibilityEvent);
142 if (!shouldSendAccessibilityEvent) {
143 handler.removeCallbacks(this);
144 }
145 }
146 };
147
148 public CallCardPresenter(Context context) {
wangqi385a5a12017-09-28 10:44:54 -0700149 LogUtil.i("CallCardPresenter.constructor", null);
linyuh183cb712017-12-27 17:02:37 -0800150 this.context = Assert.isNotNull(context).getApplicationContext();
151 callLocation = CallLocationComponent.get(this.context).getCallLocation();
Eric Erfanianccca3152017-02-22 16:32:36 -0800152 }
153
154 private static boolean hasCallSubject(DialerCall call) {
155 return !TextUtils.isEmpty(call.getCallSubject());
156 }
157
158 @Override
159 public void onInCallScreenDelegateInit(InCallScreen inCallScreen) {
160 Assert.isNotNull(inCallScreen);
linyuh183cb712017-12-27 17:02:37 -0800161 this.inCallScreen = inCallScreen;
162 contactsPreferences = ContactsPreferencesFactory.newContactsPreferences(context);
Eric Erfanianccca3152017-02-22 16:32:36 -0800163
164 // Call may be null if disconnect happened already.
165 DialerCall call = CallList.getInstance().getFirstCall();
166 if (call != null) {
linyuh183cb712017-12-27 17:02:37 -0800167 primary = call;
168 if (shouldShowNoteSentToast(primary)) {
169 this.inCallScreen.showNoteSentToast();
Eric Erfanianccca3152017-02-22 16:32:36 -0800170 }
171 call.addListener(this);
Eric Erfanianccca3152017-02-22 16:32:36 -0800172 // start processing lookups right away.
173 if (!call.isConferenceCall()) {
wangqibb94ca62018-04-27 14:34:04 -0700174 startContactInfoSearch(call, true, call.getState() == DialerCallState.INCOMING);
Eric Erfanianccca3152017-02-22 16:32:36 -0800175 } else {
176 updateContactEntry(null, true);
177 }
178 }
179
180 onStateChange(null, InCallPresenter.getInstance().getInCallState(), CallList.getInstance());
181 }
182
183 @Override
184 public void onInCallScreenReady() {
wangqi385a5a12017-09-28 10:44:54 -0700185 LogUtil.i("CallCardPresenter.onInCallScreenReady", null);
Eric Erfanianccca3152017-02-22 16:32:36 -0800186 Assert.checkState(!isInCallScreenReady);
linyuh183cb712017-12-27 17:02:37 -0800187 if (contactsPreferences != null) {
188 contactsPreferences.refreshValue(ContactsPreferences.DISPLAY_ORDER_KEY);
Eric Erfanianccca3152017-02-22 16:32:36 -0800189 }
190
Eric Erfanianccca3152017-02-22 16:32:36 -0800191 // Contact search may have completed before ui is ready.
linyuh183cb712017-12-27 17:02:37 -0800192 if (primaryContactInfo != null) {
Eric Erfanianccca3152017-02-22 16:32:36 -0800193 updatePrimaryDisplayInfo();
194 }
195
196 // Register for call state changes last
197 InCallPresenter.getInstance().addListener(this);
198 InCallPresenter.getInstance().addIncomingCallListener(this);
199 InCallPresenter.getInstance().addDetailsListener(this);
200 InCallPresenter.getInstance().addInCallEventListener(this);
201 isInCallScreenReady = true;
Eric Erfaniand5e47f62017-03-15 14:41:07 -0700202
Eric Erfaniand8046e52017-04-06 09:41:50 -0700203 // Log location impressions
linyuh183cb712017-12-27 17:02:37 -0800204 if (isOutgoingEmergencyCall(primary)) {
205 Logger.get(context).logImpression(DialerImpression.Type.EMERGENCY_NEW_EMERGENCY_CALL);
206 } else if (isIncomingEmergencyCall(primary) || isIncomingEmergencyCall(secondary)) {
207 Logger.get(context).logImpression(DialerImpression.Type.EMERGENCY_CALLBACK);
Eric Erfaniand8046e52017-04-06 09:41:50 -0700208 }
209
Eric Erfaniand5e47f62017-03-15 14:41:07 -0700210 // Showing the location may have been skipped if the UI wasn't ready during previous layout.
211 if (shouldShowLocation()) {
linyuh183cb712017-12-27 17:02:37 -0800212 inCallScreen.showLocationUi(getLocationFragment());
Eric Erfaniand8046e52017-04-06 09:41:50 -0700213
214 // Log location impressions
215 if (!hasLocationPermission()) {
linyuh183cb712017-12-27 17:02:37 -0800216 Logger.get(context).logImpression(DialerImpression.Type.EMERGENCY_NO_LOCATION_PERMISSION);
Eric Erfaniand8046e52017-04-06 09:41:50 -0700217 } else if (isBatteryTooLowForEmergencyLocation()) {
linyuh183cb712017-12-27 17:02:37 -0800218 Logger.get(context)
Eric Erfaniand8046e52017-04-06 09:41:50 -0700219 .logImpression(DialerImpression.Type.EMERGENCY_BATTERY_TOO_LOW_TO_GET_LOCATION);
linyuh183cb712017-12-27 17:02:37 -0800220 } else if (!callLocation.canGetLocation(context)) {
221 Logger.get(context).logImpression(DialerImpression.Type.EMERGENCY_CANT_GET_LOCATION);
Eric Erfaniand8046e52017-04-06 09:41:50 -0700222 }
Eric Erfaniand5e47f62017-03-15 14:41:07 -0700223 }
Eric Erfanianccca3152017-02-22 16:32:36 -0800224 }
225
226 @Override
227 public void onInCallScreenUnready() {
wangqi385a5a12017-09-28 10:44:54 -0700228 LogUtil.i("CallCardPresenter.onInCallScreenUnready", null);
Eric Erfanianccca3152017-02-22 16:32:36 -0800229 Assert.checkState(isInCallScreenReady);
230
Eric Erfanianccca3152017-02-22 16:32:36 -0800231 // stop getting call state changes
232 InCallPresenter.getInstance().removeListener(this);
233 InCallPresenter.getInstance().removeIncomingCallListener(this);
234 InCallPresenter.getInstance().removeDetailsListener(this);
235 InCallPresenter.getInstance().removeInCallEventListener(this);
linyuh183cb712017-12-27 17:02:37 -0800236 if (primary != null) {
237 primary.removeListener(this);
Eric Erfanianccca3152017-02-22 16:32:36 -0800238 }
239
Eric Erfaniand5e47f62017-03-15 14:41:07 -0700240 callLocation.close();
241
linyuh183cb712017-12-27 17:02:37 -0800242 primary = null;
243 primaryContactInfo = null;
244 secondaryContactInfo = null;
Eric Erfanianccca3152017-02-22 16:32:36 -0800245 isInCallScreenReady = false;
246 }
247
248 @Override
249 public void onIncomingCall(InCallState oldState, InCallState newState, DialerCall call) {
250 // same logic should happen as with onStateChange()
251 onStateChange(oldState, newState, CallList.getInstance());
252 }
253
254 @Override
255 public void onStateChange(InCallState oldState, InCallState newState, CallList callList) {
Eric Erfanian2ca43182017-08-31 06:57:16 -0700256 Trace.beginSection("CallCardPresenter.onStateChange");
257 LogUtil.v("CallCardPresenter.onStateChange", "oldState: %s, newState: %s", oldState, newState);
linyuh183cb712017-12-27 17:02:37 -0800258 if (inCallScreen == null) {
Eric Erfanian2ca43182017-08-31 06:57:16 -0700259 Trace.endSection();
Eric Erfanianccca3152017-02-22 16:32:36 -0800260 return;
261 }
262
263 DialerCall primary = null;
264 DialerCall secondary = null;
265
266 if (newState == InCallState.INCOMING) {
267 primary = callList.getIncomingCall();
268 } else if (newState == InCallState.PENDING_OUTGOING || newState == InCallState.OUTGOING) {
269 primary = callList.getOutgoingCall();
270 if (primary == null) {
271 primary = callList.getPendingOutgoingCall();
272 }
273
274 // getCallToDisplay doesn't go through outgoing or incoming calls. It will return the
275 // highest priority call to display as the secondary call.
yuegb9103042018-03-30 12:12:25 -0700276 secondary = InCallPresenter.getCallToDisplay(callList, null, true);
Eric Erfanianccca3152017-02-22 16:32:36 -0800277 } else if (newState == InCallState.INCALL) {
yuegb9103042018-03-30 12:12:25 -0700278 primary = InCallPresenter.getCallToDisplay(callList, null, false);
279 secondary = InCallPresenter.getCallToDisplay(callList, primary, true);
Eric Erfanianccca3152017-02-22 16:32:36 -0800280 }
281
282 LogUtil.v("CallCardPresenter.onStateChange", "primary call: " + primary);
283 LogUtil.v("CallCardPresenter.onStateChange", "secondary call: " + secondary);
wangqidd5f1a52018-01-23 18:01:00 -0800284 String primaryNumber = null;
285 String secondaryNumber = null;
286 if (primary != null) {
287 primaryNumber = primary.getNumber();
288 }
289 if (secondary != null) {
290 secondaryNumber = secondary.getNumber();
291 }
Eric Erfanianccca3152017-02-22 16:32:36 -0800292
293 final boolean primaryChanged =
linyuh183cb712017-12-27 17:02:37 -0800294 !(DialerCall.areSame(this.primary, primary)
wangqidd5f1a52018-01-23 18:01:00 -0800295 && TextUtils.equals(this.primaryNumber, primaryNumber));
Eric Erfanianccca3152017-02-22 16:32:36 -0800296 final boolean secondaryChanged =
linyuh183cb712017-12-27 17:02:37 -0800297 !(DialerCall.areSame(this.secondary, secondary)
wangqidd5f1a52018-01-23 18:01:00 -0800298 && TextUtils.equals(this.secondaryNumber, secondaryNumber));
Eric Erfanianccca3152017-02-22 16:32:36 -0800299
linyuh183cb712017-12-27 17:02:37 -0800300 this.secondary = secondary;
wangqidd5f1a52018-01-23 18:01:00 -0800301 this.secondaryNumber = secondaryNumber;
linyuh183cb712017-12-27 17:02:37 -0800302 DialerCall previousPrimary = this.primary;
303 this.primary = primary;
wangqidd5f1a52018-01-23 18:01:00 -0800304 this.primaryNumber = primaryNumber;
Eric Erfanianccca3152017-02-22 16:32:36 -0800305
linyuh183cb712017-12-27 17:02:37 -0800306 if (this.primary != null) {
linyuh183cb712017-12-27 17:02:37 -0800307 inCallScreen.updateInCallScreenColors();
Eric Erfanianccca3152017-02-22 16:32:36 -0800308 }
309
310 if (primaryChanged && shouldShowNoteSentToast(primary)) {
linyuh183cb712017-12-27 17:02:37 -0800311 inCallScreen.showNoteSentToast();
Eric Erfanianccca3152017-02-22 16:32:36 -0800312 }
313
314 // Refresh primary call information if either:
315 // 1. Primary call changed.
316 // 2. The call's ability to manage conference has changed.
317 if (shouldRefreshPrimaryInfo(primaryChanged)) {
318 // primary call has changed
319 if (previousPrimary != null) {
320 previousPrimary.removeListener(this);
321 }
linyuh183cb712017-12-27 17:02:37 -0800322 this.primary.addListener(this);
Eric Erfanianccca3152017-02-22 16:32:36 -0800323
twyena40a9ef2018-05-22 17:35:59 -0700324 primaryContactInfo = ContactInfoCache.buildCacheEntryFromCall(context, this.primary);
Eric Erfanianccca3152017-02-22 16:32:36 -0800325 updatePrimaryDisplayInfo();
linyuh183cb712017-12-27 17:02:37 -0800326 maybeStartSearch(this.primary, true);
Eric Erfanianccca3152017-02-22 16:32:36 -0800327 }
328
linyuh183cb712017-12-27 17:02:37 -0800329 if (previousPrimary != null && this.primary == null) {
Eric Erfanianccca3152017-02-22 16:32:36 -0800330 previousPrimary.removeListener(this);
331 }
332
wangqic8cf79e2017-10-17 09:21:00 -0700333 if (secondaryChanged) {
linyuh183cb712017-12-27 17:02:37 -0800334 if (this.secondary == null) {
wangqic8cf79e2017-10-17 09:21:00 -0700335 // Secondary call may have ended. Update the ui.
linyuh183cb712017-12-27 17:02:37 -0800336 secondaryContactInfo = null;
wangqic8cf79e2017-10-17 09:21:00 -0700337 updateSecondaryDisplayInfo();
338 } else {
339 // secondary call has changed
twyena40a9ef2018-05-22 17:35:59 -0700340 secondaryContactInfo = ContactInfoCache.buildCacheEntryFromCall(context, this.secondary);
wangqic8cf79e2017-10-17 09:21:00 -0700341 updateSecondaryDisplayInfo();
linyuh183cb712017-12-27 17:02:37 -0800342 maybeStartSearch(this.secondary, false);
wangqic8cf79e2017-10-17 09:21:00 -0700343 }
Eric Erfanianccca3152017-02-22 16:32:36 -0800344 }
345
346 // Set the call state
wangqibb94ca62018-04-27 14:34:04 -0700347 int callState = DialerCallState.IDLE;
linyuh183cb712017-12-27 17:02:37 -0800348 if (this.primary != null) {
349 callState = this.primary.getState();
Eric Erfanianccca3152017-02-22 16:32:36 -0800350 updatePrimaryCallState();
351 } else {
Android Dialer31fe9982018-02-26 13:29:09 -0800352 getUi().setCallState(PrimaryCallState.empty());
Eric Erfanianccca3152017-02-22 16:32:36 -0800353 }
354
355 maybeShowManageConferenceCallButton();
356
357 // Hide the end call button instantly if we're receiving an incoming call.
358 getUi()
359 .setEndCallButtonEnabled(
linyuh183cb712017-12-27 17:02:37 -0800360 shouldShowEndCallButton(this.primary, callState),
wangqibb94ca62018-04-27 14:34:04 -0700361 callState != DialerCallState.INCOMING /* animate */);
Eric Erfanianccca3152017-02-22 16:32:36 -0800362
363 maybeSendAccessibilityEvent(oldState, newState, primaryChanged);
Eric Erfanian2ca43182017-08-31 06:57:16 -0700364 Trace.endSection();
Eric Erfanianccca3152017-02-22 16:32:36 -0800365 }
366
367 @Override
368 public void onDetailsChanged(DialerCall call, Details details) {
369 updatePrimaryCallState();
370
371 if (call.can(Details.CAPABILITY_MANAGE_CONFERENCE)
372 != details.can(Details.CAPABILITY_MANAGE_CONFERENCE)) {
373 maybeShowManageConferenceCallButton();
374 }
375 }
376
377 @Override
378 public void onDialerCallDisconnect() {}
379
380 @Override
381 public void onDialerCallUpdate() {
382 // No-op; specific call updates handled elsewhere.
383 }
384
385 @Override
386 public void onWiFiToLteHandover() {}
387
388 @Override
389 public void onHandoverToWifiFailure() {}
390
Eric Erfanianc857f902017-05-15 14:05:33 -0700391 @Override
392 public void onInternationalCallOnWifi() {}
393
Eric Erfanian2ca43182017-08-31 06:57:16 -0700394 @Override
395 public void onEnrichedCallSessionUpdate() {
396 LogUtil.enterBlock("CallCardPresenter.onEnrichedCallSessionUpdate");
397 updatePrimaryDisplayInfo();
398 }
399
Eric Erfanianccca3152017-02-22 16:32:36 -0800400 /** Handles a change to the child number by refreshing the primary call info. */
401 @Override
402 public void onDialerCallChildNumberChange() {
403 LogUtil.v("CallCardPresenter.onDialerCallChildNumberChange", "");
404
linyuh183cb712017-12-27 17:02:37 -0800405 if (primary == null) {
Eric Erfanianccca3152017-02-22 16:32:36 -0800406 return;
407 }
408 updatePrimaryDisplayInfo();
409 }
410
411 /** Handles a change to the last forwarding number by refreshing the primary call info. */
412 @Override
413 public void onDialerCallLastForwardedNumberChange() {
414 LogUtil.v("CallCardPresenter.onDialerCallLastForwardedNumberChange", "");
415
linyuh183cb712017-12-27 17:02:37 -0800416 if (primary == null) {
Eric Erfanianccca3152017-02-22 16:32:36 -0800417 return;
418 }
419 updatePrimaryDisplayInfo();
420 updatePrimaryCallState();
421 }
422
423 @Override
424 public void onDialerCallUpgradeToVideo() {}
425
Eric Erfaniand5e47f62017-03-15 14:41:07 -0700426 /** Handles a change to the session modification state for a call. */
Eric Erfanianccca3152017-02-22 16:32:36 -0800427 @Override
Eric Erfaniand5e47f62017-03-15 14:41:07 -0700428 public void onDialerCallSessionModificationStateChange() {
429 LogUtil.enterBlock("CallCardPresenter.onDialerCallSessionModificationStateChange");
Eric Erfanianccca3152017-02-22 16:32:36 -0800430
linyuh183cb712017-12-27 17:02:37 -0800431 if (primary == null) {
Eric Erfanianccca3152017-02-22 16:32:36 -0800432 return;
433 }
434 getUi()
435 .setEndCallButtonEnabled(
linyuh183cb712017-12-27 17:02:37 -0800436 primary.getVideoTech().getSessionModificationState()
Eric Erfanian90508232017-03-24 09:31:16 -0700437 != SessionModificationState.RECEIVED_UPGRADE_TO_VIDEO_REQUEST,
Eric Erfanianccca3152017-02-22 16:32:36 -0800438 true /* shouldAnimate */);
439 updatePrimaryCallState();
440 }
441
Eric Erfanianccca3152017-02-22 16:32:36 -0800442 private boolean shouldRefreshPrimaryInfo(boolean primaryChanged) {
linyuh183cb712017-12-27 17:02:37 -0800443 if (primary == null) {
Eric Erfanianccca3152017-02-22 16:32:36 -0800444 return false;
445 }
446 return primaryChanged
linyuh183cb712017-12-27 17:02:37 -0800447 || inCallScreen.isManageConferenceVisible() != shouldShowManageConference();
Eric Erfanianccca3152017-02-22 16:32:36 -0800448 }
449
450 private void updatePrimaryCallState() {
linyuh183cb712017-12-27 17:02:37 -0800451 if (getUi() != null && primary != null) {
Eric Erfanianccca3152017-02-22 16:32:36 -0800452 boolean isWorkCall =
linyuh183cb712017-12-27 17:02:37 -0800453 primary.hasProperty(PROPERTY_ENTERPRISE_CALL)
454 || (primaryContactInfo != null
455 && primaryContactInfo.userType == ContactsUtils.USER_TYPE_WORK);
Eric Erfanianccca3152017-02-22 16:32:36 -0800456 boolean isHdAudioCall =
linyuh183cb712017-12-27 17:02:37 -0800457 isPrimaryCallActive() && primary.hasProperty(Details.PROPERTY_HIGH_DEF_AUDIO);
Eric Erfaniand5e47f62017-03-15 14:41:07 -0700458 boolean isAttemptingHdAudioCall =
459 !isHdAudioCall
linyuh183cb712017-12-27 17:02:37 -0800460 && !primary.hasProperty(DialerCall.PROPERTY_CODEC_KNOWN)
461 && MotorolaUtils.shouldBlinkHdIconWhenConnectingCall(context);
Eric Erfaniand5e47f62017-03-15 14:41:07 -0700462
linyuh183cb712017-12-27 17:02:37 -0800463 boolean isBusiness = primaryContactInfo != null && primaryContactInfo.isBusiness;
Eric Erfaniand5e47f62017-03-15 14:41:07 -0700464
Eric Erfanianccca3152017-02-22 16:32:36 -0800465 // Check for video state change and update the visibility of the contact photo. The contact
466 // photo is hidden when the incoming video surface is shown.
467 // The contact photo visibility can also change in setPrimary().
468 boolean shouldShowContactPhoto =
linyuh183cb712017-12-27 17:02:37 -0800469 !VideoCallPresenter.showIncomingVideo(primary.getVideoState(), primary.getState());
Eric Erfanianccca3152017-02-22 16:32:36 -0800470 getUi()
471 .setCallState(
Android Dialer31fe9982018-02-26 13:29:09 -0800472 PrimaryCallState.builder()
473 .setState(primary.getState())
474 .setIsVideoCall(primary.isVideoCall())
475 .setSessionModificationState(primary.getVideoTech().getSessionModificationState())
476 .setDisconnectCause(primary.getDisconnectCause())
477 .setConnectionLabel(getConnectionLabel())
twyen4fde0ac2018-03-22 18:04:23 -0700478 .setSimSuggestionReason(getSimSuggestionReason())
Android Dialer31fe9982018-02-26 13:29:09 -0800479 .setConnectionIcon(getCallStateIcon())
480 .setGatewayNumber(getGatewayNumber())
481 .setCallSubject(shouldShowCallSubject(primary) ? primary.getCallSubject() : null)
482 .setCallbackNumber(
483 PhoneNumberHelper.formatNumber(
linyuhb06d0092018-03-01 15:05:36 -0800484 context, primary.getCallbackNumber(), primary.getSimCountryIso()))
Android Dialer31fe9982018-02-26 13:29:09 -0800485 .setIsWifi(primary.hasProperty(Details.PROPERTY_WIFI))
486 .setIsConference(
487 primary.isConferenceCall()
488 && !primary.hasProperty(Details.PROPERTY_GENERIC_CONFERENCE))
489 .setIsWorkCall(isWorkCall)
490 .setIsHdAttempting(isAttemptingHdAudioCall)
491 .setIsHdAudioCall(isHdAudioCall)
492 .setIsForwardedNumber(
493 !TextUtils.isEmpty(primary.getLastForwardedNumber())
494 || primary.isCallForwarded())
495 .setShouldShowContactPhoto(shouldShowContactPhoto)
496 .setConnectTimeMillis(primary.getConnectTimeMillis())
497 .setIsVoiceMailNumber(primary.isVoiceMailNumber())
498 .setIsRemotelyHeld(primary.isRemotelyHeld())
499 .setIsBusinessNumber(isBusiness)
500 .setSupportsCallOnHold(supports2ndCallOnHold())
501 .setSwapToSecondaryButtonState(getSwapToSecondaryButtonState())
502 .setIsAssistedDialed(primary.isAssistedDialed())
503 .setCustomLabel(null)
504 .setAssistedDialingExtras(primary.getAssistedDialingExtras())
505 .build());
Eric Erfanianccca3152017-02-22 16:32:36 -0800506
507 InCallActivity activity =
linyuh183cb712017-12-27 17:02:37 -0800508 (InCallActivity) (inCallScreen.getInCallScreenFragment().getActivity());
Eric Erfanianccca3152017-02-22 16:32:36 -0800509 if (activity != null) {
510 activity.onPrimaryCallStateChanged();
511 }
512 }
513 }
514
Eric Erfanian2ca43182017-08-31 06:57:16 -0700515 private @ButtonState int getSwapToSecondaryButtonState() {
linyuh183cb712017-12-27 17:02:37 -0800516 if (secondary == null) {
Eric Erfanian2ca43182017-08-31 06:57:16 -0700517 return ButtonState.NOT_SUPPORT;
518 }
wangqibb94ca62018-04-27 14:34:04 -0700519 if (primary.getState() == DialerCallState.ACTIVE) {
Eric Erfanian2ca43182017-08-31 06:57:16 -0700520 return ButtonState.ENABLED;
521 }
522 return ButtonState.DISABLED;
523 }
524
Eric Erfanianccca3152017-02-22 16:32:36 -0800525 /** Only show the conference call button if we can manage the conference. */
526 private void maybeShowManageConferenceCallButton() {
527 getUi().showManageConferenceCallButton(shouldShowManageConference());
528 }
529
530 /**
531 * Determines if the manage conference button should be visible, based on the current primary
532 * call.
533 *
534 * @return {@code True} if the manage conference button should be visible.
535 */
536 private boolean shouldShowManageConference() {
linyuh183cb712017-12-27 17:02:37 -0800537 if (primary == null) {
Eric Erfanianccca3152017-02-22 16:32:36 -0800538 return false;
539 }
540
linyuh183cb712017-12-27 17:02:37 -0800541 return primary.can(android.telecom.Call.Details.CAPABILITY_MANAGE_CONFERENCE) && !isFullscreen;
Eric Erfanianccca3152017-02-22 16:32:36 -0800542 }
543
Eric Erfanian2ca43182017-08-31 06:57:16 -0700544 private boolean supports2ndCallOnHold() {
545 DialerCall firstCall = CallList.getInstance().getActiveOrBackgroundCall();
546 DialerCall incomingCall = CallList.getInstance().getIncomingCall();
547 if (firstCall != null && incomingCall != null && firstCall != incomingCall) {
548 return incomingCall.can(Details.CAPABILITY_HOLD);
549 }
550 return true;
551 }
552
Eric Erfanianccca3152017-02-22 16:32:36 -0800553 @Override
554 public void onCallStateButtonClicked() {
linyuh183cb712017-12-27 17:02:37 -0800555 Intent broadcastIntent = Bindings.get(context).getCallStateButtonBroadcastIntent(context);
Eric Erfanianccca3152017-02-22 16:32:36 -0800556 if (broadcastIntent != null) {
557 LogUtil.v(
558 "CallCardPresenter.onCallStateButtonClicked",
559 "sending call state button broadcast: " + broadcastIntent);
linyuh183cb712017-12-27 17:02:37 -0800560 context.sendBroadcast(broadcastIntent, Manifest.permission.READ_PHONE_STATE);
Eric Erfanianccca3152017-02-22 16:32:36 -0800561 }
562 }
563
564 @Override
565 public void onManageConferenceClicked() {
566 InCallActivity activity =
linyuh183cb712017-12-27 17:02:37 -0800567 (InCallActivity) (inCallScreen.getInCallScreenFragment().getActivity());
Eric Erfanianccca3152017-02-22 16:32:36 -0800568 activity.showConferenceFragment(true);
569 }
570
571 @Override
572 public void onShrinkAnimationComplete() {
573 InCallPresenter.getInstance().onShrinkAnimationComplete();
574 }
575
Eric Erfanianccca3152017-02-22 16:32:36 -0800576 private void maybeStartSearch(DialerCall call, boolean isPrimary) {
577 // no need to start search for conference calls which show generic info.
578 if (call != null && !call.isConferenceCall()) {
wangqibb94ca62018-04-27 14:34:04 -0700579 startContactInfoSearch(call, isPrimary, call.getState() == DialerCallState.INCOMING);
Eric Erfanianccca3152017-02-22 16:32:36 -0800580 }
581 }
582
Eric Erfanianccca3152017-02-22 16:32:36 -0800583 /** Starts a query for more contact data for the save primary and secondary calls. */
584 private void startContactInfoSearch(
585 final DialerCall call, final boolean isPrimary, boolean isIncoming) {
linyuh183cb712017-12-27 17:02:37 -0800586 final ContactInfoCache cache = ContactInfoCache.getInstance(context);
Eric Erfanianccca3152017-02-22 16:32:36 -0800587
588 cache.findInfo(call, isIncoming, new ContactLookupCallback(this, isPrimary));
589 }
590
591 private void onContactInfoComplete(String callId, ContactCacheEntry entry, boolean isPrimary) {
592 final boolean entryMatchesExistingCall =
linyuh183cb712017-12-27 17:02:37 -0800593 (isPrimary && primary != null && TextUtils.equals(callId, primary.getId()))
594 || (!isPrimary && secondary != null && TextUtils.equals(callId, secondary.getId()));
Eric Erfanianccca3152017-02-22 16:32:36 -0800595 if (entryMatchesExistingCall) {
596 updateContactEntry(entry, isPrimary);
597 } else {
598 LogUtil.e(
599 "CallCardPresenter.onContactInfoComplete",
600 "dropping stale contact lookup info for " + callId);
601 }
602
603 final DialerCall call = CallList.getInstance().getCallById(callId);
604 if (call != null) {
605 call.getLogState().contactLookupResult = entry.contactLookupResult;
606 }
Eric Erfanian2ca43182017-08-31 06:57:16 -0700607 if (entry.lookupUri != null) {
linyuh183cb712017-12-27 17:02:37 -0800608 CallerInfoUtils.sendViewNotification(context, entry.lookupUri);
Eric Erfanianccca3152017-02-22 16:32:36 -0800609 }
610 }
611
612 private void onImageLoadComplete(String callId, ContactCacheEntry entry) {
613 if (getUi() == null) {
614 return;
615 }
616
617 if (entry.photo != null) {
linyuh183cb712017-12-27 17:02:37 -0800618 if (primary != null && callId.equals(primary.getId())) {
Eric Erfanianccca3152017-02-22 16:32:36 -0800619 updateContactEntry(entry, true /* isPrimary */);
linyuh183cb712017-12-27 17:02:37 -0800620 } else if (secondary != null && callId.equals(secondary.getId())) {
Eric Erfanianccca3152017-02-22 16:32:36 -0800621 updateContactEntry(entry, false /* isPrimary */);
622 }
623 }
624 }
625
626 private void updateContactEntry(ContactCacheEntry entry, boolean isPrimary) {
627 if (isPrimary) {
linyuh183cb712017-12-27 17:02:37 -0800628 primaryContactInfo = entry;
Eric Erfanianccca3152017-02-22 16:32:36 -0800629 updatePrimaryDisplayInfo();
630 } else {
linyuh183cb712017-12-27 17:02:37 -0800631 secondaryContactInfo = entry;
Eric Erfanianccca3152017-02-22 16:32:36 -0800632 updateSecondaryDisplayInfo();
633 }
634 }
635
Eric Erfanianccca3152017-02-22 16:32:36 -0800636 private void updatePrimaryDisplayInfo() {
linyuh183cb712017-12-27 17:02:37 -0800637 if (inCallScreen == null) {
Eric Erfanianccca3152017-02-22 16:32:36 -0800638 // TODO: May also occur if search result comes back after ui is destroyed. Look into
639 // removing that case completely.
640 LogUtil.v(
641 "CallCardPresenter.updatePrimaryDisplayInfo",
642 "updatePrimaryDisplayInfo called but ui is null!");
643 return;
644 }
645
linyuh183cb712017-12-27 17:02:37 -0800646 if (primary == null) {
Eric Erfanianccca3152017-02-22 16:32:36 -0800647 // Clear the primary display info.
Android Dialerf9439102018-02-21 08:05:14 -0800648 inCallScreen.setPrimary(PrimaryInfo.empty());
Eric Erfanianccca3152017-02-22 16:32:36 -0800649 return;
650 }
651
652 // Hide the contact photo if we are in a video call and the incoming video surface is
653 // showing.
654 boolean showContactPhoto =
linyuh183cb712017-12-27 17:02:37 -0800655 !VideoCallPresenter.showIncomingVideo(primary.getVideoState(), primary.getState());
Eric Erfanianccca3152017-02-22 16:32:36 -0800656
657 // DialerCall placed through a work phone account.
linyuh183cb712017-12-27 17:02:37 -0800658 boolean hasWorkCallProperty = primary.hasProperty(PROPERTY_ENTERPRISE_CALL);
Eric Erfanianccca3152017-02-22 16:32:36 -0800659
Eric Erfaniand5e47f62017-03-15 14:41:07 -0700660 MultimediaData multimediaData = null;
linyuh183cb712017-12-27 17:02:37 -0800661 if (primary.getEnrichedCallSession() != null) {
662 multimediaData = primary.getEnrichedCallSession().getMultimediaData();
Eric Erfaniand5e47f62017-03-15 14:41:07 -0700663 }
Eric Erfanianccca3152017-02-22 16:32:36 -0800664
linyuh183cb712017-12-27 17:02:37 -0800665 if (primary.isConferenceCall()) {
Eric Erfanianccca3152017-02-22 16:32:36 -0800666 LogUtil.v(
667 "CallCardPresenter.updatePrimaryDisplayInfo",
668 "update primary display info for conference call.");
669
linyuh183cb712017-12-27 17:02:37 -0800670 inCallScreen.setPrimary(
Android Dialerf9439102018-02-21 08:05:14 -0800671 PrimaryInfo.builder()
672 .setName(
673 CallerInfoUtils.getConferenceString(
674 context, primary.hasProperty(Details.PROPERTY_GENERIC_CONFERENCE)))
675 .setNameIsNumber(false)
676 .setPhotoType(ContactPhotoType.DEFAULT_PLACEHOLDER)
677 .setIsSipCall(false)
678 .setIsContactPhotoShown(showContactPhoto)
679 .setIsWorkCall(hasWorkCallProperty)
680 .setIsSpam(false)
681 .setIsLocalContact(false)
682 .setAnsweringDisconnectsOngoingCall(false)
683 .setShouldShowLocation(shouldShowLocation())
684 .setShowInCallButtonGrid(true)
685 .setNumberPresentation(primary.getNumberPresentation())
686 .build());
linyuh183cb712017-12-27 17:02:37 -0800687 } else if (primaryContactInfo != null) {
Eric Erfanianccca3152017-02-22 16:32:36 -0800688 LogUtil.v(
689 "CallCardPresenter.updatePrimaryDisplayInfo",
linyuh183cb712017-12-27 17:02:37 -0800690 "update primary display info for " + primaryContactInfo);
Eric Erfanianccca3152017-02-22 16:32:36 -0800691
linyuh183cb712017-12-27 17:02:37 -0800692 String name = getNameForCall(primaryContactInfo);
Eric Erfanianccca3152017-02-22 16:32:36 -0800693 String number;
694
linyuh183cb712017-12-27 17:02:37 -0800695 boolean isChildNumberShown = !TextUtils.isEmpty(primary.getChildNumber());
696 boolean isForwardedNumberShown = !TextUtils.isEmpty(primary.getLastForwardedNumber());
697 boolean isCallSubjectShown = shouldShowCallSubject(primary);
Eric Erfanianccca3152017-02-22 16:32:36 -0800698
699 if (isCallSubjectShown) {
700 number = null;
701 } else if (isChildNumberShown) {
linyuh183cb712017-12-27 17:02:37 -0800702 number = context.getString(R.string.child_number, primary.getChildNumber());
Eric Erfanianccca3152017-02-22 16:32:36 -0800703 } else if (isForwardedNumberShown) {
704 // Use last forwarded number instead of second line, if present.
linyuh183cb712017-12-27 17:02:37 -0800705 number = primary.getLastForwardedNumber();
Eric Erfanianccca3152017-02-22 16:32:36 -0800706 } else {
linyuh183cb712017-12-27 17:02:37 -0800707 number = primaryContactInfo.number;
Eric Erfanianccca3152017-02-22 16:32:36 -0800708 }
709
linyuh183cb712017-12-27 17:02:37 -0800710 boolean nameIsNumber = name != null && name.equals(primaryContactInfo.number);
Eric Erfaniand5e47f62017-03-15 14:41:07 -0700711
Eric Erfanianccca3152017-02-22 16:32:36 -0800712 // DialerCall with caller that is a work contact.
linyuh183cb712017-12-27 17:02:37 -0800713 boolean isWorkContact = (primaryContactInfo.userType == ContactsUtils.USER_TYPE_WORK);
714 inCallScreen.setPrimary(
Android Dialerf9439102018-02-21 08:05:14 -0800715 PrimaryInfo.builder()
716 .setNumber(number)
717 .setName(primary.updateNameIfRestricted(name))
718 .setNameIsNumber(nameIsNumber)
twyencb2a9812018-03-23 18:21:13 -0700719 .setLocation(
Android Dialerf9439102018-02-21 08:05:14 -0800720 shouldShowLocationAsLabel(nameIsNumber, primaryContactInfo.shouldShowLocation)
721 ? primaryContactInfo.location
722 : null)
twyencb2a9812018-03-23 18:21:13 -0700723 .setLabel(isChildNumberShown || isCallSubjectShown ? null : primaryContactInfo.label)
Android Dialerf9439102018-02-21 08:05:14 -0800724 .setPhoto(primaryContactInfo.photo)
725 .setPhotoType(primaryContactInfo.photoType)
726 .setIsSipCall(primaryContactInfo.isSipCall)
727 .setIsContactPhotoShown(showContactPhoto)
728 .setIsWorkCall(hasWorkCallProperty || isWorkContact)
729 .setIsSpam(primary.isSpam())
730 .setIsLocalContact(primaryContactInfo.isLocalContact())
731 .setAnsweringDisconnectsOngoingCall(primary.answeringDisconnectsForegroundVideoCall())
732 .setShouldShowLocation(shouldShowLocation())
733 .setContactInfoLookupKey(primaryContactInfo.lookupKey)
734 .setMultimediaData(multimediaData)
735 .setShowInCallButtonGrid(true)
736 .setNumberPresentation(primary.getNumberPresentation())
737 .build());
Eric Erfanianccca3152017-02-22 16:32:36 -0800738 } else {
739 // Clear the primary display info.
Android Dialerf9439102018-02-21 08:05:14 -0800740 inCallScreen.setPrimary(PrimaryInfo.empty());
Eric Erfanianccca3152017-02-22 16:32:36 -0800741 }
742
Eric Erfaniand5e47f62017-03-15 14:41:07 -0700743 if (isInCallScreenReady) {
linyuh183cb712017-12-27 17:02:37 -0800744 inCallScreen.showLocationUi(getLocationFragment());
Eric Erfaniand5e47f62017-03-15 14:41:07 -0700745 } else {
746 LogUtil.i("CallCardPresenter.updatePrimaryDisplayInfo", "UI not ready, not showing location");
747 }
748 }
749
Eric Erfaniand8046e52017-04-06 09:41:50 -0700750 private static boolean shouldShowLocationAsLabel(
751 boolean nameIsNumber, boolean shouldShowLocation) {
752 if (nameIsNumber) {
753 return true;
754 }
755 if (shouldShowLocation) {
756 return true;
757 }
758 return false;
759 }
760
Eric Erfaniand5e47f62017-03-15 14:41:07 -0700761 private Fragment getLocationFragment() {
Eric Erfaniand5e47f62017-03-15 14:41:07 -0700762 if (!shouldShowLocation()) {
Eric Erfaniand5e47f62017-03-15 14:41:07 -0700763 return null;
764 }
765 LogUtil.i("CallCardPresenter.getLocationFragment", "returning location fragment");
linyuh183cb712017-12-27 17:02:37 -0800766 return callLocation.getLocationFragment(context);
Eric Erfanianccca3152017-02-22 16:32:36 -0800767 }
768
769 private boolean shouldShowLocation() {
linyuh183cb712017-12-27 17:02:37 -0800770 if (!ConfigProviderBindings.get(context)
wangqi1d62ab22017-11-29 14:29:31 -0800771 .getBoolean(CONFIG_ENABLE_EMERGENCY_LOCATION, CONFIG_ENABLE_EMERGENCY_LOCATION_DEFAULT)) {
772 LogUtil.i("CallCardPresenter.getLocationFragment", "disabled by config.");
773 return false;
774 }
775 if (!isPotentialEmergencyCall()) {
776 LogUtil.i("CallCardPresenter.getLocationFragment", "shouldn't show location");
777 return false;
778 }
779 if (!hasLocationPermission()) {
780 LogUtil.i("CallCardPresenter.getLocationFragment", "no location permission.");
781 return false;
782 }
783 if (isBatteryTooLowForEmergencyLocation()) {
784 LogUtil.i("CallCardPresenter.getLocationFragment", "low battery.");
785 return false;
786 }
linyuh437ae952018-03-26 12:46:18 -0700787 if (inCallScreen.getInCallScreenFragment().getActivity().isInMultiWindowMode()) {
wangqi1d62ab22017-11-29 14:29:31 -0800788 LogUtil.i("CallCardPresenter.getLocationFragment", "in multi-window mode");
789 return false;
790 }
linyuh183cb712017-12-27 17:02:37 -0800791 if (primary.isVideoCall()) {
wangqi1d62ab22017-11-29 14:29:31 -0800792 LogUtil.i("CallCardPresenter.getLocationFragment", "emergency video calls not supported");
793 return false;
794 }
linyuh183cb712017-12-27 17:02:37 -0800795 if (!callLocation.canGetLocation(context)) {
wangqi1d62ab22017-11-29 14:29:31 -0800796 LogUtil.i("CallCardPresenter.getLocationFragment", "can't get current location");
797 return false;
798 }
799 return true;
800 }
801
802 private boolean isPotentialEmergencyCall() {
linyuh183cb712017-12-27 17:02:37 -0800803 if (isOutgoingEmergencyCall(primary)) {
Eric Erfanianccca3152017-02-22 16:32:36 -0800804 LogUtil.i("CallCardPresenter.shouldShowLocation", "new emergency call");
805 return true;
linyuh183cb712017-12-27 17:02:37 -0800806 } else if (isIncomingEmergencyCall(primary)) {
Eric Erfanianccca3152017-02-22 16:32:36 -0800807 LogUtil.i("CallCardPresenter.shouldShowLocation", "potential emergency callback");
808 return true;
linyuh183cb712017-12-27 17:02:37 -0800809 } else if (isIncomingEmergencyCall(secondary)) {
Eric Erfanianccca3152017-02-22 16:32:36 -0800810 LogUtil.i("CallCardPresenter.shouldShowLocation", "has potential emergency callback");
811 return true;
812 }
813 return false;
814 }
815
816 private static boolean isOutgoingEmergencyCall(@Nullable DialerCall call) {
817 return call != null && !call.isIncoming() && call.isEmergencyCall();
818 }
819
820 private static boolean isIncomingEmergencyCall(@Nullable DialerCall call) {
821 return call != null && call.isIncoming() && call.isPotentialEmergencyCallback();
822 }
823
824 private boolean hasLocationPermission() {
linyuh183cb712017-12-27 17:02:37 -0800825 return ContextCompat.checkSelfPermission(context, Manifest.permission.ACCESS_FINE_LOCATION)
Eric Erfanianccca3152017-02-22 16:32:36 -0800826 == PackageManager.PERMISSION_GRANTED;
827 }
828
829 private boolean isBatteryTooLowForEmergencyLocation() {
830 Intent batteryStatus =
linyuh183cb712017-12-27 17:02:37 -0800831 context.registerReceiver(null, new IntentFilter(Intent.ACTION_BATTERY_CHANGED));
Eric Erfanianccca3152017-02-22 16:32:36 -0800832 int status = batteryStatus.getIntExtra(BatteryManager.EXTRA_STATUS, -1);
833 if (status == BatteryManager.BATTERY_STATUS_CHARGING
834 || status == BatteryManager.BATTERY_STATUS_FULL) {
835 // Plugged in or full battery
836 return false;
837 }
838 int level = batteryStatus.getIntExtra(BatteryManager.EXTRA_LEVEL, -1);
839 int scale = batteryStatus.getIntExtra(BatteryManager.EXTRA_SCALE, -1);
840 float batteryPercent = (100f * level) / scale;
841 long threshold =
linyuh183cb712017-12-27 17:02:37 -0800842 ConfigProviderBindings.get(context)
Eric Erfanianccca3152017-02-22 16:32:36 -0800843 .getLong(
844 CONFIG_MIN_BATTERY_PERCENT_FOR_EMERGENCY_LOCATION,
845 CONFIG_MIN_BATTERY_PERCENT_FOR_EMERGENCY_LOCATION_DEFAULT);
846 LogUtil.i(
847 "CallCardPresenter.isBatteryTooLowForEmergencyLocation",
848 "percent charged: " + batteryPercent + ", min required charge: " + threshold);
849 return batteryPercent < threshold;
850 }
851
852 private void updateSecondaryDisplayInfo() {
linyuh183cb712017-12-27 17:02:37 -0800853 if (inCallScreen == null) {
Eric Erfanianccca3152017-02-22 16:32:36 -0800854 return;
855 }
856
linyuh183cb712017-12-27 17:02:37 -0800857 if (secondary == null) {
Eric Erfanianccca3152017-02-22 16:32:36 -0800858 // Clear the secondary display info.
Android Dialer9e335e22018-03-01 08:27:39 -0800859 inCallScreen.setSecondary(SecondaryInfo.builder().setIsFullscreen(isFullscreen).build());
Eric Erfanianccca3152017-02-22 16:32:36 -0800860 return;
861 }
862
linyuh183cb712017-12-27 17:02:37 -0800863 if (secondary.isMergeInProcess()) {
Eric Erfanian2ca43182017-08-31 06:57:16 -0700864 LogUtil.i(
865 "CallCardPresenter.updateSecondaryDisplayInfo",
866 "secondary call is merge in process, clearing info");
Android Dialer9e335e22018-03-01 08:27:39 -0800867 inCallScreen.setSecondary(SecondaryInfo.builder().setIsFullscreen(isFullscreen).build());
Eric Erfanian2ca43182017-08-31 06:57:16 -0700868 return;
869 }
870
linyuh183cb712017-12-27 17:02:37 -0800871 if (secondary.isConferenceCall()) {
872 inCallScreen.setSecondary(
Android Dialer9e335e22018-03-01 08:27:39 -0800873 SecondaryInfo.builder()
874 .setShouldShow(true)
875 .setName(
876 CallerInfoUtils.getConferenceString(
877 context, secondary.hasProperty(Details.PROPERTY_GENERIC_CONFERENCE)))
878 .setProviderLabel(secondary.getCallProviderLabel())
879 .setIsConference(true)
880 .setIsVideoCall(secondary.isVideoCall())
881 .setIsFullscreen(isFullscreen)
882 .build());
linyuh183cb712017-12-27 17:02:37 -0800883 } else if (secondaryContactInfo != null) {
884 LogUtil.v("CallCardPresenter.updateSecondaryDisplayInfo", "" + secondaryContactInfo);
885 String name = getNameForCall(secondaryContactInfo);
886 boolean nameIsNumber = name != null && name.equals(secondaryContactInfo.number);
887 inCallScreen.setSecondary(
Android Dialer9e335e22018-03-01 08:27:39 -0800888 SecondaryInfo.builder()
889 .setShouldShow(true)
890 .setName(secondary.updateNameIfRestricted(name))
891 .setNameIsNumber(nameIsNumber)
892 .setLabel(secondaryContactInfo.label)
893 .setProviderLabel(secondary.getCallProviderLabel())
894 .setIsVideoCall(secondary.isVideoCall())
895 .setIsFullscreen(isFullscreen)
896 .build());
Eric Erfanianccca3152017-02-22 16:32:36 -0800897 } else {
898 // Clear the secondary display info.
Android Dialer9e335e22018-03-01 08:27:39 -0800899 inCallScreen.setSecondary(SecondaryInfo.builder().setIsFullscreen(isFullscreen).build());
Eric Erfanianccca3152017-02-22 16:32:36 -0800900 }
901 }
902
903 /** Returns the gateway number for any existing outgoing call. */
904 private String getGatewayNumber() {
905 if (hasOutgoingGatewayCall()) {
linyuh183cb712017-12-27 17:02:37 -0800906 return DialerCall.getNumberFromHandle(primary.getGatewayInfo().getGatewayAddress());
Eric Erfanianccca3152017-02-22 16:32:36 -0800907 }
908 return null;
909 }
910
911 /**
912 * Returns the label (line of text above the number/name) for any given call. For example,
913 * "calling via [Account/Google Voice]" for outgoing calls.
914 */
915 private String getConnectionLabel() {
linyuh183cb712017-12-27 17:02:37 -0800916 if (ContextCompat.checkSelfPermission(context, Manifest.permission.READ_PHONE_STATE)
Eric Erfanianccca3152017-02-22 16:32:36 -0800917 != PackageManager.PERMISSION_GRANTED) {
918 return null;
919 }
linyuh183cb712017-12-27 17:02:37 -0800920 StatusHints statusHints = primary.getStatusHints();
Eric Erfanianccca3152017-02-22 16:32:36 -0800921 if (statusHints != null && !TextUtils.isEmpty(statusHints.getLabel())) {
922 return statusHints.getLabel().toString();
923 }
924
925 if (hasOutgoingGatewayCall() && getUi() != null) {
926 // Return the label for the gateway app on outgoing calls.
linyuh183cb712017-12-27 17:02:37 -0800927 final PackageManager pm = context.getPackageManager();
Eric Erfanianccca3152017-02-22 16:32:36 -0800928 try {
929 ApplicationInfo info =
linyuh183cb712017-12-27 17:02:37 -0800930 pm.getApplicationInfo(primary.getGatewayInfo().getGatewayProviderPackageName(), 0);
Eric Erfanianccca3152017-02-22 16:32:36 -0800931 return pm.getApplicationLabel(info).toString();
932 } catch (PackageManager.NameNotFoundException e) {
933 LogUtil.e("CallCardPresenter.getConnectionLabel", "gateway Application Not Found.", e);
934 return null;
935 }
936 }
linyuh183cb712017-12-27 17:02:37 -0800937 return primary.getCallProviderLabel();
Eric Erfanianccca3152017-02-22 16:32:36 -0800938 }
939
twyen4fde0ac2018-03-22 18:04:23 -0700940 @Nullable
941 private SuggestionProvider.Reason getSimSuggestionReason() {
942 String value =
943 primary.getIntentExtras().getString(SuggestionProvider.EXTRA_SIM_SUGGESTION_REASON);
944 if (value == null) {
945 return null;
946 }
947 try {
948 return SuggestionProvider.Reason.valueOf(value);
949 } catch (IllegalArgumentException e) {
950 LogUtil.e("CallCardPresenter.getConnectionLabel", "unknown reason " + value);
951 return null;
952 }
953 }
954
Eric Erfanianccca3152017-02-22 16:32:36 -0800955 private Drawable getCallStateIcon() {
956 // Return connection icon if one exists.
linyuh183cb712017-12-27 17:02:37 -0800957 StatusHints statusHints = primary.getStatusHints();
Eric Erfanianccca3152017-02-22 16:32:36 -0800958 if (statusHints != null && statusHints.getIcon() != null) {
linyuh183cb712017-12-27 17:02:37 -0800959 Drawable icon = statusHints.getIcon().loadDrawable(context);
Eric Erfanianccca3152017-02-22 16:32:36 -0800960 if (icon != null) {
961 return icon;
962 }
963 }
964
965 return null;
966 }
967
968 private boolean hasOutgoingGatewayCall() {
969 // We only display the gateway information while STATE_DIALING so return false for any other
970 // call state.
971 // TODO: mPrimary can be null because this is called from updatePrimaryDisplayInfo which
972 // is also called after a contact search completes (call is not present yet). Split the
973 // UI update so it can receive independent updates.
linyuh183cb712017-12-27 17:02:37 -0800974 if (primary == null) {
Eric Erfanianccca3152017-02-22 16:32:36 -0800975 return false;
976 }
wangqibb94ca62018-04-27 14:34:04 -0700977 return DialerCallState.isDialing(primary.getState())
linyuh183cb712017-12-27 17:02:37 -0800978 && primary.getGatewayInfo() != null
979 && !primary.getGatewayInfo().isEmpty();
Eric Erfanianccca3152017-02-22 16:32:36 -0800980 }
981
982 /** Gets the name to display for the call. */
Eric Erfanian2ca43182017-08-31 06:57:16 -0700983 private String getNameForCall(ContactCacheEntry contactInfo) {
Eric Erfanianccca3152017-02-22 16:32:36 -0800984 String preferredName =
985 ContactDisplayUtils.getPreferredDisplayName(
linyuh183cb712017-12-27 17:02:37 -0800986 contactInfo.namePrimary, contactInfo.nameAlternative, contactsPreferences);
Eric Erfanianccca3152017-02-22 16:32:36 -0800987 if (TextUtils.isEmpty(preferredName)) {
wangqia4623702018-04-02 14:23:38 -0700988 return TextUtils.isEmpty(contactInfo.number)
989 ? null
990 : BidiFormatter.getInstance()
991 .unicodeWrap(contactInfo.number, TextDirectionHeuristics.LTR);
Eric Erfanianccca3152017-02-22 16:32:36 -0800992 }
993 return preferredName;
994 }
995
Eric Erfanianccca3152017-02-22 16:32:36 -0800996 @Override
997 public void onSecondaryInfoClicked() {
linyuh183cb712017-12-27 17:02:37 -0800998 if (secondary == null) {
Eric Erfanianccca3152017-02-22 16:32:36 -0800999 LogUtil.e(
1000 "CallCardPresenter.onSecondaryInfoClicked",
1001 "secondary info clicked but no secondary call.");
1002 return;
1003 }
1004
linyuh183cb712017-12-27 17:02:37 -08001005 Logger.get(context)
Eric Erfanian2ca43182017-08-31 06:57:16 -07001006 .logCallImpression(
1007 DialerImpression.Type.IN_CALL_SWAP_SECONDARY_BUTTON_PRESSED,
linyuh183cb712017-12-27 17:02:37 -08001008 primary.getUniqueCallId(),
1009 primary.getTimeAddedMs());
Eric Erfanianccca3152017-02-22 16:32:36 -08001010 LogUtil.i(
linyuh183cb712017-12-27 17:02:37 -08001011 "CallCardPresenter.onSecondaryInfoClicked", "swapping call to foreground: " + secondary);
1012 secondary.unhold();
Eric Erfanianccca3152017-02-22 16:32:36 -08001013 }
1014
1015 @Override
1016 public void onEndCallClicked() {
linyuh183cb712017-12-27 17:02:37 -08001017 LogUtil.i("CallCardPresenter.onEndCallClicked", "disconnecting call: " + primary);
1018 if (primary != null) {
1019 primary.disconnect();
Eric Erfanianccca3152017-02-22 16:32:36 -08001020 }
linyuh183cb712017-12-27 17:02:37 -08001021 PostCall.onDisconnectPressed(context);
Eric Erfanianccca3152017-02-22 16:32:36 -08001022 }
1023
1024 /**
1025 * Handles a change to the fullscreen mode of the in-call UI.
1026 *
1027 * @param isFullscreenMode {@code True} if the in-call UI is entering full screen mode.
1028 */
1029 @Override
1030 public void onFullscreenModeChanged(boolean isFullscreenMode) {
linyuh183cb712017-12-27 17:02:37 -08001031 isFullscreen = isFullscreenMode;
1032 if (inCallScreen == null) {
Eric Erfanianccca3152017-02-22 16:32:36 -08001033 return;
1034 }
1035 maybeShowManageConferenceCallButton();
1036 }
1037
1038 private boolean isPrimaryCallActive() {
wangqibb94ca62018-04-27 14:34:04 -07001039 return primary != null && primary.getState() == DialerCallState.ACTIVE;
Eric Erfanianccca3152017-02-22 16:32:36 -08001040 }
1041
Eric Erfanianccca3152017-02-22 16:32:36 -08001042 private boolean shouldShowEndCallButton(DialerCall primary, int callState) {
1043 if (primary == null) {
1044 return false;
1045 }
wangqibb94ca62018-04-27 14:34:04 -07001046 if ((!DialerCallState.isConnectingOrConnected(callState)
1047 && callState != DialerCallState.DISCONNECTING
1048 && callState != DialerCallState.DISCONNECTED)
1049 || callState == DialerCallState.INCOMING) {
Eric Erfanianccca3152017-02-22 16:32:36 -08001050 return false;
1051 }
linyuh183cb712017-12-27 17:02:37 -08001052 if (this.primary.getVideoTech().getSessionModificationState()
Eric Erfanian90508232017-03-24 09:31:16 -07001053 == SessionModificationState.RECEIVED_UPGRADE_TO_VIDEO_REQUEST) {
Eric Erfanianccca3152017-02-22 16:32:36 -08001054 return false;
1055 }
1056 return true;
1057 }
1058
1059 @Override
1060 public void onInCallScreenResumed() {
Eric Erfaniand8046e52017-04-06 09:41:50 -07001061 updatePrimaryDisplayInfo();
1062
Eric Erfanianccca3152017-02-22 16:32:36 -08001063 if (shouldSendAccessibilityEvent) {
1064 handler.postDelayed(sendAccessibilityEventRunnable, ACCESSIBILITY_ANNOUNCEMENT_DELAY_MILLIS);
1065 }
1066 }
1067
Eric Erfaniand8046e52017-04-06 09:41:50 -07001068 @Override
Eric Erfanian2ca43182017-08-31 06:57:16 -07001069 public void onInCallScreenPaused() {}
Eric Erfaniand8046e52017-04-06 09:41:50 -07001070
Eric Erfanianccca3152017-02-22 16:32:36 -08001071 static boolean sendAccessibilityEvent(Context context, InCallScreen inCallScreen) {
1072 AccessibilityManager am =
1073 (AccessibilityManager) context.getSystemService(Context.ACCESSIBILITY_SERVICE);
1074 if (!am.isEnabled()) {
1075 LogUtil.w("CallCardPresenter.sendAccessibilityEvent", "accessibility is off");
1076 return false;
1077 }
1078 if (inCallScreen == null) {
1079 LogUtil.w("CallCardPresenter.sendAccessibilityEvent", "incallscreen is null");
1080 return false;
1081 }
1082 Fragment fragment = inCallScreen.getInCallScreenFragment();
1083 if (fragment == null || fragment.getView() == null || fragment.getView().getParent() == null) {
1084 LogUtil.w("CallCardPresenter.sendAccessibilityEvent", "fragment/view/parent is null");
1085 return false;
1086 }
1087
1088 DisplayManager displayManager =
1089 (DisplayManager) context.getSystemService(Context.DISPLAY_SERVICE);
1090 Display display = displayManager.getDisplay(Display.DEFAULT_DISPLAY);
1091 boolean screenIsOn = display.getState() == Display.STATE_ON;
1092 LogUtil.d("CallCardPresenter.sendAccessibilityEvent", "screen is on: %b", screenIsOn);
1093 if (!screenIsOn) {
1094 return false;
1095 }
1096
1097 AccessibilityEvent event = AccessibilityEvent.obtain(AccessibilityEvent.TYPE_ANNOUNCEMENT);
1098 inCallScreen.dispatchPopulateAccessibilityEvent(event);
1099 View view = inCallScreen.getInCallScreenFragment().getView();
1100 view.getParent().requestSendAccessibilityEvent(view, event);
1101 return true;
1102 }
1103
1104 private void maybeSendAccessibilityEvent(
1105 InCallState oldState, final InCallState newState, boolean primaryChanged) {
1106 shouldSendAccessibilityEvent = false;
linyuh183cb712017-12-27 17:02:37 -08001107 if (context == null) {
Eric Erfanianccca3152017-02-22 16:32:36 -08001108 return;
1109 }
1110 final AccessibilityManager am =
linyuh183cb712017-12-27 17:02:37 -08001111 (AccessibilityManager) context.getSystemService(Context.ACCESSIBILITY_SERVICE);
Eric Erfanianccca3152017-02-22 16:32:36 -08001112 if (!am.isEnabled()) {
1113 return;
1114 }
1115 // Announce the current call if it's new incoming/outgoing call or primary call is changed
1116 // due to switching calls between two ongoing calls (one is on hold).
1117 if ((oldState != InCallState.OUTGOING && newState == InCallState.OUTGOING)
1118 || (oldState != InCallState.INCOMING && newState == InCallState.INCOMING)
1119 || primaryChanged) {
1120 LogUtil.i(
1121 "CallCardPresenter.maybeSendAccessibilityEvent", "schedule accessibility announcement");
1122 shouldSendAccessibilityEvent = true;
1123 handler.postDelayed(sendAccessibilityEventRunnable, ACCESSIBILITY_ANNOUNCEMENT_DELAY_MILLIS);
1124 }
1125 }
1126
1127 /**
1128 * Determines whether the call subject should be visible on the UI. For the call subject to be
1129 * visible, the call has to be in an incoming or waiting state, and the subject must not be empty.
1130 *
1131 * @param call The call.
1132 * @return {@code true} if the subject should be shown, {@code false} otherwise.
1133 */
1134 private boolean shouldShowCallSubject(DialerCall call) {
1135 if (call == null) {
1136 return false;
1137 }
1138
1139 boolean isIncomingOrWaiting =
wangqibb94ca62018-04-27 14:34:04 -07001140 primary.getState() == DialerCallState.INCOMING
1141 || primary.getState() == DialerCallState.CALL_WAITING;
Eric Erfanianccca3152017-02-22 16:32:36 -08001142 return isIncomingOrWaiting
1143 && !TextUtils.isEmpty(call.getCallSubject())
1144 && call.getNumberPresentation() == TelecomManager.PRESENTATION_ALLOWED
1145 && call.isCallSubjectSupported();
1146 }
1147
1148 /**
1149 * Determines whether the "note sent" toast should be shown. It should be shown for a new outgoing
1150 * call with a subject.
1151 *
1152 * @param call The call
1153 * @return {@code true} if the toast should be shown, {@code false} otherwise.
1154 */
1155 private boolean shouldShowNoteSentToast(DialerCall call) {
1156 return call != null
1157 && hasCallSubject(call)
wangqibb94ca62018-04-27 14:34:04 -07001158 && (call.getState() == DialerCallState.DIALING
1159 || call.getState() == DialerCallState.CONNECTING);
Eric Erfanianccca3152017-02-22 16:32:36 -08001160 }
1161
1162 private InCallScreen getUi() {
linyuh183cb712017-12-27 17:02:37 -08001163 return inCallScreen;
Eric Erfanianccca3152017-02-22 16:32:36 -08001164 }
1165
Android Dialerf9439102018-02-21 08:05:14 -08001166 /** Callback for contact lookup. */
Eric Erfanianccca3152017-02-22 16:32:36 -08001167 public static class ContactLookupCallback implements ContactInfoCacheCallback {
1168
linyuh183cb712017-12-27 17:02:37 -08001169 private final WeakReference<CallCardPresenter> callCardPresenter;
1170 private final boolean isPrimary;
Eric Erfanianccca3152017-02-22 16:32:36 -08001171
1172 public ContactLookupCallback(CallCardPresenter callCardPresenter, boolean isPrimary) {
linyuh183cb712017-12-27 17:02:37 -08001173 this.callCardPresenter = new WeakReference<CallCardPresenter>(callCardPresenter);
1174 this.isPrimary = isPrimary;
Eric Erfanianccca3152017-02-22 16:32:36 -08001175 }
1176
1177 @Override
1178 public void onContactInfoComplete(String callId, ContactCacheEntry entry) {
linyuh183cb712017-12-27 17:02:37 -08001179 CallCardPresenter presenter = callCardPresenter.get();
Eric Erfanianccca3152017-02-22 16:32:36 -08001180 if (presenter != null) {
linyuh183cb712017-12-27 17:02:37 -08001181 presenter.onContactInfoComplete(callId, entry, isPrimary);
Eric Erfanianccca3152017-02-22 16:32:36 -08001182 }
1183 }
1184
1185 @Override
1186 public void onImageLoadComplete(String callId, ContactCacheEntry entry) {
linyuh183cb712017-12-27 17:02:37 -08001187 CallCardPresenter presenter = callCardPresenter.get();
Eric Erfanianccca3152017-02-22 16:32:36 -08001188 if (presenter != null) {
1189 presenter.onImageLoadComplete(callId, entry);
1190 }
1191 }
1192 }
1193}