| /* |
| * Copyright (C) 2013 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.incallui; |
| |
| import android.content.Context; |
| import android.content.Intent; |
| import android.graphics.Point; |
| import android.os.Bundle; |
| import android.os.Trace; |
| import android.provider.BlockedNumberContract; |
| import android.telecom.Call.Details; |
| import android.telecom.CallAudioState; |
| import android.telecom.DisconnectCause; |
| import android.telecom.PhoneAccount; |
| import android.telecom.PhoneAccountHandle; |
| import android.telecom.TelecomManager; |
| import android.telecom.VideoProfile; |
| import android.telephony.PhoneStateListener; |
| import android.telephony.TelephonyManager; |
| import android.util.ArraySet; |
| import android.view.Window; |
| import android.view.WindowManager; |
| import android.widget.Toast; |
| |
| import androidx.annotation.MainThread; |
| import androidx.annotation.NonNull; |
| import androidx.annotation.Nullable; |
| |
| import com.android.contacts.common.compat.CallCompat; |
| import com.android.dialer.CallConfiguration; |
| import com.android.dialer.Mode; |
| import com.android.dialer.R; |
| import com.android.dialer.blocking.FilteredNumberAsyncQueryHandler; |
| import com.android.dialer.blocking.FilteredNumberAsyncQueryHandler.OnCheckBlockedListener; |
| import com.android.dialer.blocking.FilteredNumbersUtil; |
| import com.android.dialer.common.Assert; |
| import com.android.dialer.common.LogUtil; |
| import com.android.dialer.common.concurrent.DialerExecutorComponent; |
| import com.android.dialer.postcall.PostCall; |
| import com.android.dialer.telecom.TelecomUtil; |
| import com.android.dialer.util.TouchPointManager; |
| import com.android.incallui.InCallOrientationEventListener.ScreenOrientation; |
| import com.android.incallui.answerproximitysensor.PseudoScreenState; |
| import com.android.incallui.audiomode.AudioModeProvider; |
| import com.android.incallui.call.CallList; |
| import com.android.incallui.call.DialerCall; |
| import com.android.incallui.call.ExternalCallList; |
| import com.android.incallui.call.TelecomAdapter; |
| import com.android.incallui.call.state.DialerCallState; |
| import com.android.incallui.disconnectdialog.DisconnectMessage; |
| import com.android.incallui.incalluilock.InCallUiLock; |
| import com.android.incallui.spam.SpamCallListListener; |
| import com.android.incallui.telecomeventui.InternationalCallOnWifiDialogActivity; |
| import com.android.incallui.telecomeventui.InternationalCallOnWifiDialogFragment; |
| import com.android.incallui.videosurface.bindings.VideoSurfaceBindings; |
| import com.android.incallui.videosurface.protocol.VideoSurfaceTexture; |
| import com.android.incallui.videotech.utils.VideoUtils; |
| import com.google.protobuf.InvalidProtocolBufferException; |
| import java.util.Collections; |
| import java.util.List; |
| import java.util.Objects; |
| import java.util.Set; |
| import java.util.concurrent.ConcurrentHashMap; |
| import java.util.concurrent.CopyOnWriteArrayList; |
| |
| /** |
| * Takes updates from the CallList and notifies the InCallActivity (UI) of the changes. Responsible |
| * for starting the activity for a new call and finishing the activity when all calls are |
| * disconnected. Creates and manages the in-call state and provides a listener pattern for the |
| * presenters that want to listen in on the in-call state changes. TODO: This class has become more |
| * of a state machine at this point. Consider renaming. |
| */ |
| public class InCallPresenter implements CallList.Listener, AudioModeProvider.AudioModeListener { |
| |
| private static final Bundle EMPTY_EXTRAS = new Bundle(); |
| |
| private static InCallPresenter inCallPresenter; |
| |
| /** |
| * ConcurrentHashMap constructor params: 8 is initial table size, 0.9f is load factor before |
| * resizing, 1 means we only expect a single thread to access the map so make only a single shard |
| */ |
| private final Set<InCallStateListener> listeners = |
| Collections.newSetFromMap(new ConcurrentHashMap<InCallStateListener, Boolean>(8, 0.9f, 1)); |
| |
| private final List<IncomingCallListener> incomingCallListeners = new CopyOnWriteArrayList<>(); |
| private final Set<InCallDetailsListener> detailsListeners = |
| Collections.newSetFromMap(new ConcurrentHashMap<InCallDetailsListener, Boolean>(8, 0.9f, 1)); |
| private final Set<CanAddCallListener> canAddCallListeners = |
| Collections.newSetFromMap(new ConcurrentHashMap<CanAddCallListener, Boolean>(8, 0.9f, 1)); |
| private final Set<InCallUiListener> inCallUiListeners = |
| Collections.newSetFromMap(new ConcurrentHashMap<InCallUiListener, Boolean>(8, 0.9f, 1)); |
| private final Set<InCallOrientationListener> orientationListeners = |
| Collections.newSetFromMap( |
| new ConcurrentHashMap<InCallOrientationListener, Boolean>(8, 0.9f, 1)); |
| private final Set<InCallEventListener> inCallEventListeners = |
| Collections.newSetFromMap(new ConcurrentHashMap<InCallEventListener, Boolean>(8, 0.9f, 1)); |
| |
| private StatusBarNotifier statusBarNotifier; |
| private ExternalCallNotifier externalCallNotifier; |
| private InCallVibrationHandler vibrationHandler; |
| private InCallDndHandler dndHandler; |
| private ContactInfoCache contactInfoCache; |
| private Context context; |
| private final OnCheckBlockedListener onCheckBlockedListener = |
| new OnCheckBlockedListener() { |
| @Override |
| public void onCheckComplete(final Integer id) { |
| if (id != null && id != FilteredNumberAsyncQueryHandler.INVALID_ID) { |
| // Silence the ringer now to prevent ringing and vibration before the call is |
| // terminated when Telecom attempts to add it. |
| TelecomUtil.silenceRinger(context); |
| } |
| } |
| }; |
| private CallList callList; |
| private ExternalCallList externalCallList; |
| private InCallActivity inCallActivity; |
| private ManageConferenceActivity manageConferenceActivity; |
| private final android.telecom.Call.Callback callCallback = |
| new android.telecom.Call.Callback() { |
| @Override |
| public void onPostDialWait( |
| android.telecom.Call telecomCall, String remainingPostDialSequence) { |
| final DialerCall call = callList.getDialerCallFromTelecomCall(telecomCall); |
| if (call == null) { |
| LogUtil.w( |
| "InCallPresenter.onPostDialWait", |
| "DialerCall not found in call list: " + telecomCall); |
| return; |
| } |
| onPostDialCharWait(call.getId(), remainingPostDialSequence); |
| } |
| |
| @Override |
| public void onDetailsChanged( |
| android.telecom.Call telecomCall, android.telecom.Call.Details details) { |
| final DialerCall call = callList.getDialerCallFromTelecomCall(telecomCall); |
| if (call == null) { |
| LogUtil.w( |
| "InCallPresenter.onDetailsChanged", |
| "DialerCall not found in call list: " + telecomCall); |
| return; |
| } |
| |
| if (details.hasProperty(Details.PROPERTY_IS_EXTERNAL_CALL) |
| && !externalCallList.isCallTracked(telecomCall)) { |
| |
| // A regular call became an external call so swap call lists. |
| LogUtil.i("InCallPresenter.onDetailsChanged", "Call became external: " + telecomCall); |
| callList.onInternalCallMadeExternal(context, telecomCall); |
| externalCallList.onCallAdded(telecomCall); |
| return; |
| } |
| |
| for (InCallDetailsListener listener : detailsListeners) { |
| listener.onDetailsChanged(call, details); |
| } |
| } |
| |
| @Override |
| public void onConferenceableCallsChanged( |
| android.telecom.Call telecomCall, List<android.telecom.Call> conferenceableCalls) { |
| LogUtil.i( |
| "InCallPresenter.onConferenceableCallsChanged", |
| "onConferenceableCallsChanged: " + telecomCall); |
| onDetailsChanged(telecomCall, telecomCall.getDetails()); |
| } |
| }; |
| private InCallState inCallState = InCallState.NO_CALLS; |
| private ProximitySensor proximitySensor; |
| private final PseudoScreenState pseudoScreenState = new PseudoScreenState(); |
| private boolean serviceConnected; |
| private InCallCameraManager inCallCameraManager; |
| private CallList.Listener spamCallListListener; |
| private CallList.Listener activeCallsListener; |
| /** Whether or not we are currently bound and waiting for Telecom to send us a new call. */ |
| private boolean boundAndWaitingForOutgoingCall; |
| /** Determines if the InCall UI is in fullscreen mode or not. */ |
| private boolean isFullScreen = false; |
| |
| private boolean screenTimeoutEnabled = true; |
| |
| private PhoneStateListener phoneStateListener = |
| new PhoneStateListener() { |
| @Override |
| public void onCallStateChanged(int state, String incomingNumber) { |
| if (state == TelephonyManager.CALL_STATE_RINGING) { |
| if (FilteredNumbersUtil.hasRecentEmergencyCall(context)) { |
| return; |
| } |
| // Check if the number is blocked, to silence the ringer. |
| if (BlockedNumberContract.isBlocked(context, incomingNumber)) { |
| TelecomUtil.silenceRinger(context); |
| } |
| } |
| } |
| }; |
| |
| /** Whether or not InCallService is bound to Telecom. */ |
| private boolean serviceBound = false; |
| |
| /** |
| * When configuration changes Android kills the current activity and starts a new one. The flag is |
| * used to check if full clean up is necessary (activity is stopped and new activity won't be |
| * started), or if a new activity will be started right after the current one is destroyed, and |
| * therefore no need in release all resources. |
| */ |
| private boolean isChangingConfigurations = false; |
| |
| private boolean awaitingCallListUpdate = false; |
| |
| private ExternalCallList.ExternalCallListener externalCallListener = |
| new ExternalCallList.ExternalCallListener() { |
| |
| @Override |
| public void onExternalCallPulled(android.telecom.Call call) { |
| // Note: External calls do not require spam checking. |
| callList.onCallAdded(context, call); |
| call.registerCallback(callCallback); |
| } |
| |
| @Override |
| public void onExternalCallAdded(android.telecom.Call call) { |
| // No-op |
| } |
| |
| @Override |
| public void onExternalCallRemoved(android.telecom.Call call) { |
| // No-op |
| } |
| |
| @Override |
| public void onExternalCallUpdated(android.telecom.Call call) { |
| // No-op |
| } |
| }; |
| |
| private ThemeColorManager themeColorManager; |
| private VideoSurfaceTexture localVideoSurfaceTexture; |
| private VideoSurfaceTexture remoteVideoSurfaceTexture; |
| |
| private boolean addCallClicked = false; |
| private boolean automaticallyMutedByAddCall = false; |
| |
| /** Inaccessible constructor. Must use getRunningInstance() to get this singleton. */ |
| InCallPresenter() {} |
| |
| public static synchronized InCallPresenter getInstance() { |
| if (inCallPresenter == null) { |
| Trace.beginSection("InCallPresenter.Constructor"); |
| inCallPresenter = new InCallPresenter(); |
| Trace.endSection(); |
| } |
| return inCallPresenter; |
| } |
| |
| /** |
| * Determines whether or not a call has no valid phone accounts that can be used to make the call |
| * with. Emergency calls do not require a phone account. |
| * |
| * @param call to check accounts for. |
| * @return {@code true} if the call has no call capable phone accounts set, {@code false} if the |
| * call contains a phone account that could be used to initiate it with, or is an emergency |
| * call. |
| */ |
| public static boolean isCallWithNoValidAccounts(DialerCall call) { |
| if (call != null && !call.isEmergencyCall()) { |
| Bundle extras = call.getIntentExtras(); |
| |
| if (extras == null) { |
| extras = EMPTY_EXTRAS; |
| } |
| |
| final List<PhoneAccountHandle> phoneAccountHandles = |
| extras.getParcelableArrayList(android.telecom.Call.AVAILABLE_PHONE_ACCOUNTS); |
| |
| if ((call.getAccountHandle() == null |
| && (phoneAccountHandles == null || phoneAccountHandles.isEmpty()))) { |
| LogUtil.i( |
| "InCallPresenter.isCallWithNoValidAccounts", "No valid accounts for call " + call); |
| return true; |
| } |
| } |
| return false; |
| } |
| |
| public InCallState getInCallState() { |
| return inCallState; |
| } |
| |
| public CallList getCallList() { |
| return callList; |
| } |
| |
| public void setUp( |
| @NonNull Context context, |
| CallList callList, |
| ExternalCallList externalCallList, |
| StatusBarNotifier statusBarNotifier, |
| ExternalCallNotifier externalCallNotifier, |
| ContactInfoCache contactInfoCache, |
| ProximitySensor proximitySensor) { |
| Trace.beginSection("InCallPresenter.setUp"); |
| if (serviceConnected) { |
| LogUtil.i("InCallPresenter.setUp", "New service connection replacing existing one."); |
| if (context != this.context || callList != this.callList) { |
| throw new IllegalStateException(); |
| } |
| Trace.endSection(); |
| return; |
| } |
| |
| Objects.requireNonNull(context); |
| this.context = context; |
| |
| this.contactInfoCache = contactInfoCache; |
| |
| this.statusBarNotifier = statusBarNotifier; |
| this.externalCallNotifier = externalCallNotifier; |
| addListener(this.statusBarNotifier); |
| |
| vibrationHandler = new InCallVibrationHandler(context); |
| addListener(vibrationHandler); |
| |
| dndHandler = new InCallDndHandler(context); |
| addListener(dndHandler); |
| |
| this.proximitySensor = proximitySensor; |
| addListener(this.proximitySensor); |
| |
| if (themeColorManager == null) { |
| themeColorManager = new ThemeColorManager(new InCallUIMaterialColorMapUtils(this.context)); |
| } |
| |
| this.callList = callList; |
| this.externalCallList = externalCallList; |
| externalCallList.addExternalCallListener(this.externalCallNotifier); |
| externalCallList.addExternalCallListener(externalCallListener); |
| |
| // This only gets called by the service so this is okay. |
| serviceConnected = true; |
| |
| // The final thing we do in this set up is add ourselves as a listener to CallList. This |
| // will kick off an update and the whole process can start. |
| this.callList.addListener(this); |
| |
| // Create spam call list listener and add it to the list of listeners |
| spamCallListListener = |
| new SpamCallListListener( |
| context, DialerExecutorComponent.get(context).dialerExecutorFactory()); |
| this.callList.addListener(spamCallListListener); |
| activeCallsListener = new ActiveCallsCallListListener(context); |
| this.callList.addListener(activeCallsListener); |
| |
| VideoPauseController.getInstance().setUp(this); |
| this.context |
| .getSystemService(TelephonyManager.class) |
| .listen(phoneStateListener, PhoneStateListener.LISTEN_CALL_STATE); |
| |
| AudioModeProvider.getInstance().addListener(this); |
| |
| // Add listener to notify Telephony process when the incoming call screen is started or |
| // finished. This is for hiding USSD dialog because the incoming call screen should have |
| // higher precedence over this dialog. |
| MotorolaInCallUiNotifier motorolaInCallUiNotifier = new MotorolaInCallUiNotifier(context); |
| addInCallUiListener(motorolaInCallUiNotifier); |
| addListener(motorolaInCallUiNotifier); |
| |
| LogUtil.d("InCallPresenter.setUp", "Finished InCallPresenter.setUp"); |
| Trace.endSection(); |
| } |
| |
| /** |
| * Called when the telephony service has disconnected from us. This will happen when there are no |
| * more active calls. However, we may still want to continue showing the UI for certain cases like |
| * showing "Call Ended". What we really want is to wait for the activity and the service to both |
| * disconnect before we tear things down. This method sets a serviceConnected boolean and calls a |
| * secondary method that performs the aforementioned logic. |
| */ |
| public void tearDown() { |
| LogUtil.d("InCallPresenter.tearDown", "tearDown"); |
| callList.clearOnDisconnect(); |
| |
| serviceConnected = false; |
| |
| context |
| .getSystemService(TelephonyManager.class) |
| .listen(phoneStateListener, PhoneStateListener.LISTEN_NONE); |
| |
| attemptCleanup(); |
| VideoPauseController.getInstance().tearDown(); |
| AudioModeProvider.getInstance().removeListener(this); |
| } |
| |
| private void attemptFinishActivity() { |
| screenTimeoutEnabled = true; |
| final boolean doFinish = (inCallActivity != null && isActivityStarted()); |
| LogUtil.i("InCallPresenter.attemptFinishActivity", "Hide in call UI: " + doFinish); |
| if (doFinish) { |
| inCallActivity.setExcludeFromRecents(true); |
| inCallActivity.finish(); |
| } |
| } |
| |
| /** |
| * Called when the UI ends. Attempts to tear down everything if necessary. See {@link #tearDown()} |
| * for more insight on the tear-down process. |
| */ |
| public void unsetActivity(InCallActivity inCallActivity) { |
| if (inCallActivity == null) { |
| throw new IllegalArgumentException("unregisterActivity cannot be called with null"); |
| } |
| if (this.inCallActivity == null) { |
| LogUtil.i( |
| "InCallPresenter.unsetActivity", "No InCallActivity currently set, no need to unset."); |
| return; |
| } |
| if (this.inCallActivity != inCallActivity) { |
| LogUtil.w( |
| "InCallPresenter.unsetActivity", |
| "Second instance of InCallActivity is trying to unregister when another" |
| + " instance is active. Ignoring."); |
| return; |
| } |
| updateActivity(null); |
| } |
| |
| /** |
| * Updates the current instance of {@link InCallActivity} with the provided one. If a {@code null} |
| * activity is provided, it means that the activity was finished and we should attempt to cleanup. |
| */ |
| private void updateActivity(InCallActivity inCallActivity) { |
| Trace.beginSection("InCallPresenter.updateActivity"); |
| boolean updateListeners = false; |
| boolean doAttemptCleanup = false; |
| |
| if (inCallActivity != null) { |
| if (this.inCallActivity == null) { |
| context = inCallActivity.getApplicationContext(); |
| updateListeners = true; |
| LogUtil.i("InCallPresenter.updateActivity", "UI Initialized"); |
| } else { |
| // since setActivity is called onStart(), it can be called multiple times. |
| // This is fine and ignorable, but we do not want to update the world every time |
| // this happens (like going to/from background) so we do not set updateListeners. |
| } |
| |
| this.inCallActivity = inCallActivity; |
| this.inCallActivity.setExcludeFromRecents(false); |
| |
| // By the time the UI finally comes up, the call may already be disconnected. |
| // If that's the case, we may need to show an error dialog. |
| if (callList != null && callList.getDisconnectedCall() != null) { |
| showDialogOrToastForDisconnectedCall(callList.getDisconnectedCall()); |
| } |
| |
| // When the UI comes up, we need to first check the in-call state. |
| // If we are showing NO_CALLS, that means that a call probably connected and |
| // then immediately disconnected before the UI was able to come up. |
| // If we dont have any calls, start tearing down the UI instead. |
| // NOTE: This code relies on {@link #mInCallActivity} being set so we run it after |
| // it has been set. |
| if (inCallState == InCallState.NO_CALLS) { |
| LogUtil.i("InCallPresenter.updateActivity", "UI Initialized, but no calls left. Shut down"); |
| attemptFinishActivity(); |
| Trace.endSection(); |
| return; |
| } |
| } else { |
| LogUtil.i("InCallPresenter.updateActivity", "UI Destroyed"); |
| updateListeners = true; |
| this.inCallActivity = null; |
| |
| // We attempt cleanup for the destroy case but only after we recalculate the state |
| // to see if we need to come back up or stay shut down. This is why we do the |
| // cleanup after the call to onCallListChange() instead of directly here. |
| doAttemptCleanup = true; |
| } |
| |
| // Messages can come from the telephony layer while the activity is coming up |
| // and while the activity is going down. So in both cases we need to recalculate what |
| // state we should be in after they complete. |
| // Examples: (1) A new incoming call could come in and then get disconnected before |
| // the activity is created. |
| // (2) All calls could disconnect and then get a new incoming call before the |
| // activity is destroyed. |
| // |
| // a bug - We previously had a check for mServiceConnected here as well, but there are |
| // cases where we need to recalculate the current state even if the service in not |
| // connected. In particular the case where startOrFinish() is called while the app is |
| // already finish()ing. In that case, we skip updating the state with the knowledge that |
| // we will check again once the activity has finished. That means we have to recalculate the |
| // state here even if the service is disconnected since we may not have finished a state |
| // transition while finish()ing. |
| if (updateListeners) { |
| onCallListChange(callList); |
| } |
| |
| if (doAttemptCleanup) { |
| attemptCleanup(); |
| } |
| Trace.endSection(); |
| } |
| |
| public void setManageConferenceActivity( |
| @Nullable ManageConferenceActivity manageConferenceActivity) { |
| this.manageConferenceActivity = manageConferenceActivity; |
| } |
| |
| public void onBringToForeground(boolean showDialpad) { |
| LogUtil.i("InCallPresenter.onBringToForeground", "Bringing UI to foreground."); |
| bringToForeground(showDialpad); |
| } |
| |
| public void onCallAdded(final android.telecom.Call call) { |
| Trace.beginSection("InCallPresenter.onCallAdded"); |
| if (call.getDetails().hasProperty(CallCompat.Details.PROPERTY_IS_EXTERNAL_CALL)) { |
| externalCallList.onCallAdded(call); |
| } else { |
| callList.onCallAdded(context, call); |
| } |
| |
| // Since a call has been added we are no longer waiting for Telecom to send us a call. |
| setBoundAndWaitingForOutgoingCall(false, null); |
| call.registerCallback(callCallback); |
| Trace.endSection(); |
| } |
| |
| public void onCallRemoved(android.telecom.Call call) { |
| if (call.getDetails().hasProperty(CallCompat.Details.PROPERTY_IS_EXTERNAL_CALL)) { |
| externalCallList.onCallRemoved(call); |
| } else { |
| callList.onCallRemoved(context, call); |
| call.unregisterCallback(callCallback); |
| } |
| } |
| |
| public void onCanAddCallChanged(boolean canAddCall) { |
| for (CanAddCallListener listener : canAddCallListeners) { |
| listener.onCanAddCallChanged(canAddCall); |
| } |
| } |
| |
| @Override |
| public void onWiFiToLteHandover(DialerCall call) { |
| if (call.hasShownWiFiToLteHandoverToast()) { |
| return; |
| } |
| |
| Toast.makeText(context, R.string.video_call_wifi_to_lte_handover_toast, Toast.LENGTH_LONG) |
| .show(); |
| call.setHasShownWiFiToLteHandoverToast(); |
| } |
| |
| @Override |
| public void onHandoverToWifiFailed(DialerCall call) { |
| if (inCallActivity != null) { |
| inCallActivity.showDialogOrToastForWifiHandoverFailure(call); |
| } else { |
| Toast.makeText(context, R.string.video_call_lte_to_wifi_failed_message, Toast.LENGTH_SHORT) |
| .show(); |
| } |
| } |
| |
| @Override |
| public void onInternationalCallOnWifi(@NonNull DialerCall call) { |
| LogUtil.enterBlock("InCallPresenter.onInternationalCallOnWifi"); |
| |
| if (!InternationalCallOnWifiDialogFragment.shouldShow(context)) { |
| LogUtil.i( |
| "InCallPresenter.onInternationalCallOnWifi", |
| "InternationalCallOnWifiDialogFragment.shouldShow returned false"); |
| return; |
| } |
| |
| if (inCallActivity != null) { |
| inCallActivity.showDialogForInternationalCallOnWifi(call); |
| } else { |
| Intent intent = new Intent(context, InternationalCallOnWifiDialogActivity.class); |
| // Prevent showing MainActivity with InternationalCallOnWifiDialogActivity on above |
| intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_MULTIPLE_TASK); |
| intent.putExtra(InternationalCallOnWifiDialogActivity.EXTRA_CALL_ID, call.getId()); |
| context.startActivity(intent); |
| } |
| } |
| |
| /** |
| * Called when there is a change to the call list. Sets the In-Call state for the entire in-call |
| * app based on the information it gets from CallList. Dispatches the in-call state to all |
| * listeners. Can trigger the creation or destruction of the UI based on the states that is |
| * calculates. |
| */ |
| @Override |
| public void onCallListChange(CallList callList) { |
| Trace.beginSection("InCallPresenter.onCallListChange"); |
| if (inCallActivity != null && inCallActivity.isInCallScreenAnimating()) { |
| awaitingCallListUpdate = true; |
| Trace.endSection(); |
| return; |
| } |
| if (callList == null) { |
| Trace.endSection(); |
| return; |
| } |
| |
| awaitingCallListUpdate = false; |
| |
| InCallState newState = getPotentialStateFromCallList(callList); |
| InCallState oldState = inCallState; |
| LogUtil.d( |
| "InCallPresenter.onCallListChange", |
| "onCallListChange oldState= " + oldState + " newState=" + newState); |
| |
| // If the user placed a call and was asked to choose the account, but then pressed "Home", the |
| // incall activity for that call will still exist (even if it's not visible). In the case of |
| // an incoming call in that situation, just disconnect that "waiting for account" call and |
| // dismiss the dialog. The same activity will be reused to handle the new incoming call. See |
| // a bug for more details. |
| DialerCall waitingForAccountCall; |
| if (newState == InCallState.INCOMING |
| && (waitingForAccountCall = callList.getWaitingForAccountCall()) != null) { |
| waitingForAccountCall.disconnect(); |
| // The InCallActivity might be destroyed or not started yet at this point. |
| if (isActivityStarted()) { |
| inCallActivity.dismissPendingDialogs(); |
| } |
| } |
| |
| newState = startOrFinishUi(newState); |
| LogUtil.d( |
| "InCallPresenter.onCallListChange", "onCallListChange newState changed to " + newState); |
| |
| // Set the new state before announcing it to the world |
| LogUtil.i( |
| "InCallPresenter.onCallListChange", |
| "Phone switching state: " + oldState + " -> " + newState); |
| inCallState = newState; |
| |
| // Foreground call changed |
| DialerCall primary = null; |
| if (newState == InCallState.INCOMING) { |
| primary = callList.getIncomingCall(); |
| } else if (newState == InCallState.PENDING_OUTGOING || newState == InCallState.OUTGOING) { |
| primary = callList.getOutgoingCall(); |
| if (primary == null) { |
| primary = callList.getPendingOutgoingCall(); |
| } |
| } else if (newState == InCallState.INCALL) { |
| primary = getCallToDisplay(callList, null, false); |
| } |
| if (primary != null) { |
| onForegroundCallChanged(primary); |
| } |
| |
| // notify listeners of new state |
| for (InCallStateListener listener : listeners) { |
| LogUtil.d( |
| "InCallPresenter.onCallListChange", |
| "Notify " + listener + " of state " + inCallState.toString()); |
| listener.onStateChange(oldState, inCallState, callList); |
| } |
| |
| if (isActivityStarted()) { |
| final boolean hasCall = |
| callList.getActiveOrBackgroundCall() != null || callList.getOutgoingCall() != null; |
| inCallActivity.dismissKeyguard(hasCall); |
| } |
| |
| Trace.endSection(); |
| } |
| |
| /** |
| * Get the highest priority call to display. Goes through the calls and chooses which to return |
| * based on priority of which type of call to display to the user. Callers can use the "ignore" |
| * feature to get the second best call by passing a previously found primary call as ignore. |
| * |
| * @param ignore A call to ignore if found. |
| */ |
| static DialerCall getCallToDisplay( |
| CallList callList, DialerCall ignore, boolean skipDisconnected) { |
| // Active calls come second. An active call always gets precedent. |
| DialerCall retval = callList.getActiveCall(); |
| if (retval != null && retval != ignore) { |
| return retval; |
| } |
| |
| // Sometimes there is intemediate state that two calls are in active even one is about |
| // to be on hold. |
| retval = callList.getSecondActiveCall(); |
| if (retval != null && retval != ignore) { |
| return retval; |
| } |
| |
| // Disconnected calls get primary position if there are no active calls |
| // to let user know quickly what call has disconnected. Disconnected |
| // calls are very short lived. |
| if (!skipDisconnected) { |
| retval = callList.getDisconnectingCall(); |
| if (retval != null && retval != ignore) { |
| return retval; |
| } |
| retval = callList.getDisconnectedCall(); |
| if (retval != null && retval != ignore) { |
| return retval; |
| } |
| } |
| |
| // Then we go to background call (calls on hold) |
| retval = callList.getBackgroundCall(); |
| if (retval != null && retval != ignore) { |
| return retval; |
| } |
| |
| // Lastly, we go to a second background call. |
| retval = callList.getSecondBackgroundCall(); |
| |
| return retval; |
| } |
| |
| /** Called when there is a new incoming call. */ |
| @Override |
| public void onIncomingCall(DialerCall call) { |
| Trace.beginSection("InCallPresenter.onIncomingCall"); |
| InCallState newState = startOrFinishUi(InCallState.INCOMING); |
| InCallState oldState = inCallState; |
| |
| LogUtil.i( |
| "InCallPresenter.onIncomingCall", "Phone switching state: " + oldState + " -> " + newState); |
| inCallState = newState; |
| |
| Trace.beginSection("listener.onIncomingCall"); |
| for (IncomingCallListener listener : incomingCallListeners) { |
| listener.onIncomingCall(oldState, inCallState, call); |
| } |
| Trace.endSection(); |
| |
| Trace.beginSection("onPrimaryCallStateChanged"); |
| if (inCallActivity != null) { |
| // Re-evaluate which fragment is being shown. |
| inCallActivity.onPrimaryCallStateChanged(); |
| } |
| Trace.endSection(); |
| Trace.endSection(); |
| } |
| |
| @Override |
| public void onUpgradeToVideo(DialerCall call) { |
| if (VideoUtils.hasReceivedVideoUpgradeRequest(call.getVideoTech().getSessionModificationState()) |
| && inCallState == InCallPresenter.InCallState.INCOMING) { |
| LogUtil.i( |
| "InCallPresenter.onUpgradeToVideo", |
| "rejecting upgrade request due to existing incoming call"); |
| call.getVideoTech().declineVideoRequest(); |
| } |
| |
| if (inCallActivity != null) { |
| // Re-evaluate which fragment is being shown. |
| inCallActivity.onPrimaryCallStateChanged(); |
| } |
| } |
| |
| @Override |
| public void onUpgradeToRtt(DialerCall call, int rttRequestId) { |
| if (inCallActivity != null) { |
| inCallActivity.showDialogForRttRequest(call, rttRequestId); |
| } |
| } |
| |
| @Override |
| public void onSessionModificationStateChange(DialerCall call) { |
| int newState = call.getVideoTech().getSessionModificationState(); |
| LogUtil.i("InCallPresenter.onSessionModificationStateChange", "state: %d", newState); |
| if (proximitySensor == null) { |
| LogUtil.i("InCallPresenter.onSessionModificationStateChange", "proximitySensor is null"); |
| return; |
| } |
| proximitySensor.setIsAttemptingVideoCall( |
| call.hasSentVideoUpgradeRequest() || call.hasReceivedVideoUpgradeRequest()); |
| if (inCallActivity != null) { |
| // Re-evaluate which fragment is being shown. |
| inCallActivity.onPrimaryCallStateChanged(); |
| } |
| } |
| |
| /** |
| * Called when a call becomes disconnected. Called everytime an existing call changes from being |
| * connected (incoming/outgoing/active) to disconnected. |
| */ |
| @Override |
| public void onDisconnect(DialerCall call) { |
| showDialogOrToastForDisconnectedCall(call); |
| |
| // We need to do the run the same code as onCallListChange. |
| onCallListChange(callList); |
| |
| if (isActivityStarted()) { |
| inCallActivity.dismissKeyguard(false); |
| } |
| |
| if (call.isEmergencyCall()) { |
| FilteredNumbersUtil.recordLastEmergencyCallTime(context); |
| } |
| |
| if (!callList.hasLiveCall() |
| && !call.getLogState().isIncoming |
| && !isSecretCode(call.getNumber()) |
| && !call.isVoiceMailNumber()) { |
| PostCall.onCallDisconnected(context, call.getNumber(), call.getConnectTimeMillis()); |
| } |
| } |
| |
| private boolean isSecretCode(@Nullable String number) { |
| return number != null |
| && (number.length() <= 8 || number.startsWith("*#*#") || number.endsWith("#*#*")); |
| } |
| |
| /** Given the call list, return the state in which the in-call screen should be. */ |
| public InCallState getPotentialStateFromCallList(CallList callList) { |
| |
| InCallState newState = InCallState.NO_CALLS; |
| |
| if (callList == null) { |
| return newState; |
| } |
| if (callList.getIncomingCall() != null) { |
| newState = InCallState.INCOMING; |
| } else if (callList.getWaitingForAccountCall() != null) { |
| newState = InCallState.WAITING_FOR_ACCOUNT; |
| } else if (callList.getPendingOutgoingCall() != null) { |
| newState = InCallState.PENDING_OUTGOING; |
| } else if (callList.getOutgoingCall() != null) { |
| newState = InCallState.OUTGOING; |
| } else if (callList.getActiveCall() != null |
| || callList.getBackgroundCall() != null |
| || callList.getDisconnectedCall() != null |
| || callList.getDisconnectingCall() != null) { |
| newState = InCallState.INCALL; |
| } |
| |
| if (newState == InCallState.NO_CALLS) { |
| if (boundAndWaitingForOutgoingCall) { |
| return InCallState.PENDING_OUTGOING; |
| } |
| } |
| |
| return newState; |
| } |
| |
| public boolean isBoundAndWaitingForOutgoingCall() { |
| return boundAndWaitingForOutgoingCall; |
| } |
| |
| public void setBoundAndWaitingForOutgoingCall(boolean isBound, PhoneAccountHandle handle) { |
| LogUtil.i( |
| "InCallPresenter.setBoundAndWaitingForOutgoingCall", |
| "setBoundAndWaitingForOutgoingCall: " + isBound); |
| boundAndWaitingForOutgoingCall = isBound; |
| themeColorManager.setPendingPhoneAccountHandle(handle); |
| if (isBound && inCallState == InCallState.NO_CALLS) { |
| inCallState = InCallState.PENDING_OUTGOING; |
| } |
| } |
| |
| public void onShrinkAnimationComplete() { |
| if (awaitingCallListUpdate) { |
| onCallListChange(callList); |
| } |
| } |
| |
| public void addIncomingCallListener(IncomingCallListener listener) { |
| Objects.requireNonNull(listener); |
| incomingCallListeners.add(listener); |
| } |
| |
| public void removeIncomingCallListener(IncomingCallListener listener) { |
| if (listener != null) { |
| incomingCallListeners.remove(listener); |
| } |
| } |
| |
| public void addListener(InCallStateListener listener) { |
| Objects.requireNonNull(listener); |
| listeners.add(listener); |
| } |
| |
| public void removeListener(InCallStateListener listener) { |
| if (listener != null) { |
| listeners.remove(listener); |
| } |
| } |
| |
| public void addDetailsListener(InCallDetailsListener listener) { |
| Objects.requireNonNull(listener); |
| detailsListeners.add(listener); |
| } |
| |
| public void removeDetailsListener(InCallDetailsListener listener) { |
| if (listener != null) { |
| detailsListeners.remove(listener); |
| } |
| } |
| |
| public void addCanAddCallListener(CanAddCallListener listener) { |
| Objects.requireNonNull(listener); |
| canAddCallListeners.add(listener); |
| } |
| |
| public void removeCanAddCallListener(CanAddCallListener listener) { |
| if (listener != null) { |
| canAddCallListeners.remove(listener); |
| } |
| } |
| |
| public void addOrientationListener(InCallOrientationListener listener) { |
| Objects.requireNonNull(listener); |
| orientationListeners.add(listener); |
| } |
| |
| public void removeOrientationListener(InCallOrientationListener listener) { |
| if (listener != null) { |
| orientationListeners.remove(listener); |
| } |
| } |
| |
| public void addInCallEventListener(InCallEventListener listener) { |
| Objects.requireNonNull(listener); |
| inCallEventListeners.add(listener); |
| } |
| |
| public void removeInCallEventListener(InCallEventListener listener) { |
| if (listener != null) { |
| inCallEventListeners.remove(listener); |
| } |
| } |
| |
| public ProximitySensor getProximitySensor() { |
| return proximitySensor; |
| } |
| |
| public PseudoScreenState getPseudoScreenState() { |
| return pseudoScreenState; |
| } |
| |
| /** Returns true if the incall app is the foreground application. */ |
| public boolean isShowingInCallUi() { |
| if (!isActivityStarted()) { |
| return false; |
| } |
| if (manageConferenceActivity != null && manageConferenceActivity.isVisible()) { |
| return true; |
| } |
| return inCallActivity.isVisible(); |
| } |
| |
| /** |
| * Returns true if the activity has been created and is running. Returns true as long as activity |
| * is not destroyed or finishing. This ensures that we return true even if the activity is paused |
| * (not in foreground). |
| */ |
| public boolean isActivityStarted() { |
| return (inCallActivity != null |
| && !inCallActivity.isDestroyed() |
| && !inCallActivity.isFinishing()); |
| } |
| |
| /** |
| * Determines if the In-Call app is currently changing configuration. |
| * |
| * @return {@code true} if the In-Call app is changing configuration. |
| */ |
| public boolean isChangingConfigurations() { |
| return isChangingConfigurations; |
| } |
| |
| /** |
| * Tracks whether the In-Call app is currently in the process of changing configuration (i.e. |
| * screen orientation). |
| */ |
| /*package*/ |
| void updateIsChangingConfigurations() { |
| isChangingConfigurations = false; |
| if (inCallActivity != null) { |
| isChangingConfigurations = inCallActivity.isChangingConfigurations(); |
| } |
| LogUtil.v( |
| "InCallPresenter.updateIsChangingConfigurations", |
| "updateIsChangingConfigurations = " + isChangingConfigurations); |
| } |
| |
| /** Called when the activity goes in/out of the foreground. */ |
| public void onUiShowing(boolean showing) { |
| if (proximitySensor != null) { |
| proximitySensor.onInCallShowing(showing); |
| } |
| |
| if (showing) { |
| refreshMuteState(); |
| } else { |
| updateIsChangingConfigurations(); |
| } |
| |
| for (InCallUiListener listener : inCallUiListeners) { |
| listener.onUiShowing(showing); |
| } |
| |
| if (inCallActivity != null) { |
| // Re-evaluate which fragment is being shown. |
| inCallActivity.onPrimaryCallStateChanged(); |
| } |
| } |
| |
| public void refreshUi() { |
| if (inCallActivity != null) { |
| // Re-evaluate which fragment is being shown. |
| inCallActivity.onPrimaryCallStateChanged(); |
| } |
| } |
| |
| public void addInCallUiListener(InCallUiListener listener) { |
| inCallUiListeners.add(listener); |
| } |
| |
| public boolean removeInCallUiListener(InCallUiListener listener) { |
| return inCallUiListeners.remove(listener); |
| } |
| |
| /*package*/ |
| void onActivityStarted() { |
| LogUtil.d("InCallPresenter.onActivityStarted", "onActivityStarted"); |
| notifyVideoPauseController(true); |
| applyScreenTimeout(); |
| } |
| |
| /*package*/ |
| void onActivityStopped() { |
| LogUtil.d("InCallPresenter.onActivityStopped", "onActivityStopped"); |
| notifyVideoPauseController(false); |
| } |
| |
| private void notifyVideoPauseController(boolean showing) { |
| LogUtil.d( |
| "InCallPresenter.notifyVideoPauseController", |
| "mIsChangingConfigurations=" + isChangingConfigurations); |
| if (!isChangingConfigurations) { |
| VideoPauseController.getInstance().onUiShowing(showing); |
| } |
| } |
| |
| /** Brings the app into the foreground if possible. */ |
| public void bringToForeground(boolean showDialpad) { |
| // Before we bring the incall UI to the foreground, we check to see if: |
| // 1. It is not currently in the foreground |
| // 2. We are in a state where we want to show the incall ui (i.e. there are calls to |
| // be displayed) |
| // If the activity hadn't actually been started previously, yet there are still calls |
| // present (e.g. a call was accepted by a bluetooth or wired headset), we want to |
| // bring it up the UI regardless. |
| if (!isShowingInCallUi() && inCallState != InCallState.NO_CALLS) { |
| showInCall(showDialpad, false /* newOutgoingCall */); |
| } |
| } |
| |
| public void onPostDialCharWait(String callId, String chars) { |
| // If not visible, inCallActivity is stopped. Starting from P, calling recreate() will destroy |
| // the old activity instance and create a new instance immediately. Previously, the old activity |
| // went through its lifecycle from create to destroy before creating a new instance. |
| // So this case doesn't work now: make a call with char WAIT, leave in call UI, call gets |
| // connected, and go back to in call UI to see the dialog. |
| // So we should show dialog in an empty activity if inCallActivity is not visible. And it also |
| // helps with background calling. |
| if (isActivityStarted() && inCallActivity.isVisible()) { |
| inCallActivity.showDialogForPostCharWait(callId, chars); |
| } else { |
| Intent intent = new Intent(context, PostCharDialogActivity.class); |
| // Prevent showing MainActivity with PostCharDialogActivity on above |
| intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_MULTIPLE_TASK); |
| intent.putExtra(PostCharDialogActivity.EXTRA_CALL_ID, callId); |
| intent.putExtra(PostCharDialogActivity.EXTRA_POST_DIAL_STRING, chars); |
| context.startActivity(intent); |
| } |
| } |
| |
| /** |
| * Handles the green CALL key while in-call. |
| * |
| * @return true if we consumed the event. |
| */ |
| public boolean handleCallKey() { |
| LogUtil.v("InCallPresenter.handleCallKey", null); |
| |
| // The green CALL button means either "Answer", "Unhold", or |
| // "Swap calls", or can be a no-op, depending on the current state |
| // of the Phone. |
| |
| /** INCOMING CALL */ |
| final CallList calls = callList; |
| final DialerCall incomingCall = calls.getIncomingCall(); |
| LogUtil.v("InCallPresenter.handleCallKey", "incomingCall: " + incomingCall); |
| |
| // (1) Attempt to answer a call |
| if (incomingCall != null) { |
| incomingCall.answer(VideoProfile.STATE_AUDIO_ONLY); |
| return true; |
| } |
| |
| /** STATE_ACTIVE CALL */ |
| final DialerCall activeCall = calls.getActiveCall(); |
| if (activeCall != null) { |
| // TODO: This logic is repeated from CallButtonPresenter.java. We should |
| // consolidate this logic. |
| final boolean canMerge = |
| activeCall.can(android.telecom.Call.Details.CAPABILITY_MERGE_CONFERENCE); |
| final boolean canSwap = |
| activeCall.can(android.telecom.Call.Details.CAPABILITY_SWAP_CONFERENCE); |
| |
| LogUtil.v( |
| "InCallPresenter.handleCallKey", |
| "activeCall: " + activeCall + ", canMerge: " + canMerge + ", canSwap: " + canSwap); |
| |
| // (2) Attempt actions on conference calls |
| if (canMerge) { |
| TelecomAdapter.getInstance().merge(activeCall.getId()); |
| return true; |
| } else if (canSwap) { |
| TelecomAdapter.getInstance().swap(activeCall.getId()); |
| return true; |
| } |
| } |
| |
| /** BACKGROUND CALL */ |
| final DialerCall heldCall = calls.getBackgroundCall(); |
| if (heldCall != null) { |
| // We have a hold call so presumeable it will always support HOLD...but |
| // there is no harm in double checking. |
| final boolean canHold = heldCall.can(android.telecom.Call.Details.CAPABILITY_HOLD); |
| |
| LogUtil.v("InCallPresenter.handleCallKey", "heldCall: " + heldCall + ", canHold: " + canHold); |
| |
| // (4) unhold call |
| if (heldCall.getState() == DialerCallState.ONHOLD && canHold) { |
| heldCall.unhold(); |
| return true; |
| } |
| } |
| |
| // Always consume hard keys |
| return true; |
| } |
| |
| /** Clears the previous fullscreen state. */ |
| public void clearFullscreen() { |
| isFullScreen = false; |
| } |
| |
| /** |
| * Changes the fullscreen mode of the in-call UI. |
| * |
| * @param isFullScreen {@code true} if in-call should be in fullscreen mode, {@code false} |
| * otherwise. |
| */ |
| public void setFullScreen(boolean isFullScreen) { |
| setFullScreen(isFullScreen, false /* force */); |
| } |
| |
| /** |
| * Changes the fullscreen mode of the in-call UI. |
| * |
| * @param isFullScreen {@code true} if in-call should be in fullscreen mode, {@code false} |
| * otherwise. |
| * @param force {@code true} if fullscreen mode should be set regardless of its current state. |
| */ |
| public void setFullScreen(boolean isFullScreen, boolean force) { |
| LogUtil.i("InCallPresenter.setFullScreen", "setFullScreen = " + isFullScreen); |
| |
| // As a safeguard, ensure we cannot enter fullscreen if the dialpad is shown. |
| if (isDialpadVisible()) { |
| isFullScreen = false; |
| LogUtil.v( |
| "InCallPresenter.setFullScreen", |
| "setFullScreen overridden as dialpad is shown = " + isFullScreen); |
| } |
| |
| if (this.isFullScreen == isFullScreen && !force) { |
| LogUtil.v("InCallPresenter.setFullScreen", "setFullScreen ignored as already in that state."); |
| return; |
| } |
| this.isFullScreen = isFullScreen; |
| notifyFullscreenModeChange(this.isFullScreen); |
| } |
| |
| /** |
| * @return {@code true} if the in-call ui is currently in fullscreen mode, {@code false} |
| * otherwise. |
| */ |
| public boolean isFullscreen() { |
| return isFullScreen; |
| } |
| |
| /** |
| * Called by the {@link VideoCallPresenter} to inform of a change in full screen video status. |
| * |
| * @param isFullscreenMode {@code True} if entering full screen mode. |
| */ |
| public void notifyFullscreenModeChange(boolean isFullscreenMode) { |
| for (InCallEventListener listener : inCallEventListeners) { |
| listener.onFullscreenModeChanged(isFullscreenMode); |
| } |
| } |
| |
| /** Instruct the in-call activity to show an error dialog or toast for a disconnected call. */ |
| private void showDialogOrToastForDisconnectedCall(DialerCall call) { |
| if (call.getState() != DialerCallState.DISCONNECTED) { |
| return; |
| } |
| |
| // For newly disconnected calls, we may want to show a dialog on specific error conditions |
| if (call.getAccountHandle() == null && !call.isConferenceCall()) { |
| setDisconnectCauseForMissingAccounts(call); |
| } |
| |
| if (isActivityStarted()) { |
| inCallActivity.showDialogOrToastForDisconnectedCall( |
| new DisconnectMessage(inCallActivity, call)); |
| } else { |
| CharSequence message = new DisconnectMessage(context, call).toastMessage; |
| if (message != null) { |
| Toast.makeText(context, message, Toast.LENGTH_LONG).show(); |
| } |
| } |
| } |
| |
| /** |
| * When the state of in-call changes, this is the first method to get called. It determines if the |
| * UI needs to be started or finished depending on the new state and does it. |
| */ |
| private InCallState startOrFinishUi(InCallState newState) { |
| Trace.beginSection("InCallPresenter.startOrFinishUi"); |
| LogUtil.d( |
| "InCallPresenter.startOrFinishUi", "startOrFinishUi: " + inCallState + " -> " + newState); |
| |
| // TODO: Consider a proper state machine implementation |
| |
| // If the state isn't changing we have already done any starting/stopping of activities in |
| // a previous pass...so lets cut out early |
| if (newState == inCallState) { |
| Trace.endSection(); |
| return newState; |
| } |
| |
| // A new Incoming call means that the user needs to be notified of the the call (since |
| // it wasn't them who initiated it). We do this through full screen notifications and |
| // happens indirectly through {@link StatusBarNotifier}. |
| // |
| // The process for incoming calls is as follows: |
| // |
| // 1) CallList - Announces existence of new INCOMING call |
| // 2) InCallPresenter - Gets announcement and calculates that the new InCallState |
| // - should be set to INCOMING. |
| // 3) InCallPresenter - This method is called to see if we need to start or finish |
| // the app given the new state. |
| // 4) StatusBarNotifier - Listens to InCallState changes. InCallPresenter calls |
| // StatusBarNotifier explicitly to issue a FullScreen Notification |
| // that will either start the InCallActivity or show the user a |
| // top-level notification dialog if the user is in an immersive app. |
| // That notification can also start the InCallActivity. |
| // 5) InCallActivity - Main activity starts up and at the end of its onCreate will |
| // call InCallPresenter::setActivity() to let the presenter |
| // know that start-up is complete. |
| // |
| // [ AND NOW YOU'RE IN THE CALL. voila! ] |
| |
| // A dialog to show on top of the InCallUI to select a PhoneAccount |
| final boolean showAccountPicker = (InCallState.WAITING_FOR_ACCOUNT == newState); |
| |
| // A new outgoing call indicates that the user just now dialed a number and when that |
| // happens we need to display the screen immediately or show an account picker dialog if |
| // no default is set. However, if the main InCallUI is already visible, we do not want to |
| // re-initiate the start-up animation, so we do not need to do anything here. |
| // |
| // It is also possible to go into an intermediate state where the call has been initiated |
| // but Telecom has not yet returned with the details of the call (handle, gateway, etc.). |
| // This pending outgoing state can also launch the call screen. |
| // |
| // This is different from the incoming call sequence because we do not need to shock the |
| // user with a top-level notification. Just show the call UI normally. |
| boolean callCardFragmentVisible = |
| inCallActivity != null && inCallActivity.getCallCardFragmentVisible(); |
| final boolean mainUiNotVisible = !isShowingInCallUi() || !callCardFragmentVisible; |
| boolean showCallUi = InCallState.OUTGOING == newState && mainUiNotVisible; |
| |
| // Direct transition from PENDING_OUTGOING -> INCALL means that there was an error in the |
| // outgoing call process, so the UI should be brought up to show an error dialog. |
| showCallUi |= |
| (InCallState.PENDING_OUTGOING == inCallState |
| && InCallState.INCALL == newState |
| && !isShowingInCallUi()); |
| |
| // Another exception - InCallActivity is in charge of disconnecting a call with no |
| // valid accounts set. Bring the UI up if this is true for the current pending outgoing |
| // call so that: |
| // 1) The call can be disconnected correctly |
| // 2) The UI comes up and correctly displays the error dialog. |
| // TODO: Remove these special case conditions by making InCallPresenter a true state |
| // machine. Telecom should also be the component responsible for disconnecting a call |
| // with no valid accounts. |
| showCallUi |= |
| InCallState.PENDING_OUTGOING == newState |
| && mainUiNotVisible |
| && isCallWithNoValidAccounts(callList.getPendingOutgoingCall()); |
| |
| // The only time that we have an instance of mInCallActivity and it isn't started is |
| // when it is being destroyed. In that case, lets avoid bringing up another instance of |
| // the activity. When it is finally destroyed, we double check if we should bring it back |
| // up so we aren't going to lose anything by avoiding a second startup here. |
| boolean activityIsFinishing = inCallActivity != null && !isActivityStarted(); |
| if (activityIsFinishing) { |
| LogUtil.i( |
| "InCallPresenter.startOrFinishUi", |
| "Undo the state change: " + newState + " -> " + inCallState); |
| Trace.endSection(); |
| return inCallState; |
| } |
| |
| // We're about the bring up the in-call UI for outgoing and incoming call. If we still have |
| // dialogs up, we need to clear them out before showing in-call screen. This is necessary |
| // to fix the bug that dialog will show up when data reaches limit even after makeing new |
| // outgoing call after user ignore it by pressing home button. |
| if ((newState == InCallState.INCOMING || newState == InCallState.PENDING_OUTGOING) |
| && !showCallUi |
| && isActivityStarted()) { |
| inCallActivity.dismissPendingDialogs(); |
| } |
| |
| if (showCallUi || showAccountPicker) { |
| LogUtil.i("InCallPresenter.startOrFinishUi", "Start in call UI"); |
| showInCall(false /* showDialpad */, !showAccountPicker /* newOutgoingCall */); |
| } else if (newState == InCallState.NO_CALLS) { |
| // The new state is the no calls state. Tear everything down. |
| inCallState = newState; |
| attemptFinishActivity(); |
| attemptCleanup(); |
| } |
| |
| Trace.endSection(); |
| return newState; |
| } |
| |
| /** |
| * Sets the DisconnectCause for a call that was disconnected because it was missing a PhoneAccount |
| * or PhoneAccounts to select from. |
| */ |
| private void setDisconnectCauseForMissingAccounts(DialerCall call) { |
| |
| Bundle extras = call.getIntentExtras(); |
| // Initialize the extras bundle to avoid NPE |
| if (extras == null) { |
| extras = new Bundle(); |
| } |
| |
| final List<PhoneAccountHandle> phoneAccountHandles = |
| extras.getParcelableArrayList(android.telecom.Call.AVAILABLE_PHONE_ACCOUNTS); |
| |
| if (phoneAccountHandles == null || phoneAccountHandles.isEmpty()) { |
| String scheme = call.getHandle().getScheme(); |
| final String errorMsg = |
| PhoneAccount.SCHEME_TEL.equals(scheme) |
| ? context.getString(R.string.callFailed_simError) |
| : context.getString(R.string.incall_error_supp_service_unknown); |
| DisconnectCause disconnectCause = |
| new DisconnectCause(DisconnectCause.ERROR, null, errorMsg, errorMsg); |
| call.setDisconnectCause(disconnectCause); |
| } |
| } |
| |
| /** |
| * @return {@code true} if the InCallPresenter is ready to be torn down, {@code false} otherwise. |
| * Calling classes should use this as an indication whether to interact with the |
| * InCallPresenter or not. |
| */ |
| public boolean isReadyForTearDown() { |
| return inCallActivity == null && !serviceConnected && inCallState == InCallState.NO_CALLS; |
| } |
| |
| /** |
| * Checks to see if both the UI is gone and the service is disconnected. If so, tear it all down. |
| */ |
| private void attemptCleanup() { |
| if (isReadyForTearDown()) { |
| LogUtil.i("InCallPresenter.attemptCleanup", "Cleaning up"); |
| |
| cleanupSurfaces(); |
| |
| isChangingConfigurations = false; |
| |
| // blow away stale contact info so that we get fresh data on |
| // the next set of calls |
| if (contactInfoCache != null) { |
| contactInfoCache.clearCache(); |
| } |
| contactInfoCache = null; |
| |
| if (proximitySensor != null) { |
| removeListener(proximitySensor); |
| proximitySensor.tearDown(); |
| } |
| proximitySensor = null; |
| |
| if (statusBarNotifier != null) { |
| removeListener(statusBarNotifier); |
| } |
| |
| if (externalCallNotifier != null && externalCallList != null) { |
| externalCallList.removeExternalCallListener(externalCallNotifier); |
| } |
| statusBarNotifier = null; |
| |
| if (vibrationHandler != null) { |
| removeListener(vibrationHandler); |
| } |
| vibrationHandler = null; |
| |
| if (dndHandler != null) { |
| removeListener(dndHandler); |
| } |
| dndHandler = null; |
| |
| if (callList != null) { |
| callList.removeListener(this); |
| callList.removeListener(spamCallListListener); |
| } |
| callList = null; |
| |
| context = null; |
| inCallActivity = null; |
| manageConferenceActivity = null; |
| |
| listeners.clear(); |
| incomingCallListeners.clear(); |
| detailsListeners.clear(); |
| canAddCallListeners.clear(); |
| orientationListeners.clear(); |
| inCallEventListeners.clear(); |
| inCallUiListeners.clear(); |
| if (!inCallUiLocks.isEmpty()) { |
| LogUtil.e("InCallPresenter.attemptCleanup", "held in call locks: " + inCallUiLocks); |
| inCallUiLocks.clear(); |
| } |
| LogUtil.d("InCallPresenter.attemptCleanup", "finished"); |
| } |
| } |
| |
| public void showInCall(boolean showDialpad, boolean newOutgoingCall) { |
| LogUtil.i("InCallPresenter.showInCall", "Showing InCallActivity"); |
| context.startActivity( |
| InCallActivity.getIntent(context, showDialpad, newOutgoingCall, false /* forFullScreen */)); |
| } |
| |
| public void onServiceBind() { |
| serviceBound = true; |
| } |
| |
| public void onServiceUnbind() { |
| InCallPresenter.getInstance().setBoundAndWaitingForOutgoingCall(false, null); |
| serviceBound = false; |
| } |
| |
| public boolean isServiceBound() { |
| return serviceBound; |
| } |
| |
| public void maybeStartRevealAnimation(Intent intent) { |
| if (intent == null || inCallActivity != null) { |
| return; |
| } |
| final Bundle extras = intent.getBundleExtra(TelecomManager.EXTRA_OUTGOING_CALL_EXTRAS); |
| if (extras == null) { |
| // Incoming call, just show the in-call UI directly. |
| return; |
| } |
| |
| if (extras.containsKey(android.telecom.Call.AVAILABLE_PHONE_ACCOUNTS)) { |
| // Account selection dialog will show up so don't show the animation. |
| return; |
| } |
| |
| final PhoneAccountHandle accountHandle = |
| intent.getParcelableExtra(TelecomManager.EXTRA_PHONE_ACCOUNT_HANDLE); |
| final Point touchPoint = extras.getParcelable(TouchPointManager.TOUCH_POINT); |
| |
| setBoundAndWaitingForOutgoingCall(true, accountHandle); |
| |
| final Intent activityIntent = |
| InCallActivity.getIntent(context, false, true, false /* forFullScreen */); |
| activityIntent.putExtra(TouchPointManager.TOUCH_POINT, touchPoint); |
| context.startActivity(activityIntent); |
| } |
| |
| /** |
| * Retrieves the current in-call camera manager instance, creating if necessary. |
| * |
| * @return The {@link InCallCameraManager}. |
| */ |
| public InCallCameraManager getInCallCameraManager() { |
| synchronized (this) { |
| if (inCallCameraManager == null) { |
| inCallCameraManager = new InCallCameraManager(context); |
| } |
| |
| return inCallCameraManager; |
| } |
| } |
| |
| /** |
| * Notifies listeners of changes in orientation and notify calls of rotation angle change. |
| * |
| * @param orientation The screen orientation of the device (one of: {@link |
| * InCallOrientationEventListener#SCREEN_ORIENTATION_0}, {@link |
| * InCallOrientationEventListener#SCREEN_ORIENTATION_90}, {@link |
| * InCallOrientationEventListener#SCREEN_ORIENTATION_180}, {@link |
| * InCallOrientationEventListener#SCREEN_ORIENTATION_270}). |
| */ |
| public void onDeviceOrientationChange(@ScreenOrientation int orientation) { |
| LogUtil.d( |
| "InCallPresenter.onDeviceOrientationChange", |
| "onDeviceOrientationChange: orientation= " + orientation); |
| |
| if (callList != null) { |
| callList.notifyCallsOfDeviceRotation(orientation); |
| } else { |
| LogUtil.w("InCallPresenter.onDeviceOrientationChange", "CallList is null."); |
| } |
| |
| // Notify listeners of device orientation changed. |
| for (InCallOrientationListener listener : orientationListeners) { |
| listener.onDeviceOrientationChanged(orientation); |
| } |
| } |
| |
| /** |
| * Configures the in-call UI activity so it can change orientations or not. Enables the |
| * orientation event listener if allowOrientationChange is true, disables it if false. |
| * |
| * @param allowOrientationChange {@code true} if the in-call UI can change between portrait and |
| * landscape. {@code false} if the in-call UI should be locked in portrait. |
| */ |
| public void setInCallAllowsOrientationChange(boolean allowOrientationChange) { |
| if (inCallActivity == null) { |
| LogUtil.e( |
| "InCallPresenter.setInCallAllowsOrientationChange", |
| "InCallActivity is null. Can't set requested orientation."); |
| return; |
| } |
| inCallActivity.setAllowOrientationChange(allowOrientationChange); |
| } |
| |
| public void enableScreenTimeout(boolean enable) { |
| LogUtil.v("InCallPresenter.enableScreenTimeout", "enableScreenTimeout: value=" + enable); |
| screenTimeoutEnabled = enable; |
| applyScreenTimeout(); |
| } |
| |
| private void applyScreenTimeout() { |
| if (inCallActivity == null) { |
| LogUtil.e("InCallPresenter.applyScreenTimeout", "InCallActivity is null."); |
| return; |
| } |
| |
| final Window window = inCallActivity.getWindow(); |
| if (screenTimeoutEnabled) { |
| window.clearFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON); |
| } else { |
| window.addFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON); |
| } |
| } |
| |
| /** |
| * Hides or shows the conference manager fragment. |
| * |
| * @param show {@code true} if the conference manager should be shown, {@code false} if it should |
| * be hidden. |
| */ |
| public void showConferenceCallManager(boolean show) { |
| if (inCallActivity != null) { |
| inCallActivity.showConferenceFragment(show); |
| } |
| if (!show && manageConferenceActivity != null) { |
| manageConferenceActivity.finish(); |
| } |
| } |
| |
| /** |
| * Determines if the dialpad is visible. |
| * |
| * @return {@code true} if the dialpad is visible, {@code false} otherwise. |
| */ |
| public boolean isDialpadVisible() { |
| if (inCallActivity == null) { |
| return false; |
| } |
| return inCallActivity.isDialpadVisible(); |
| } |
| |
| public ThemeColorManager getThemeColorManager() { |
| return themeColorManager; |
| } |
| |
| /** Called when the foreground call changes. */ |
| public void onForegroundCallChanged(DialerCall newForegroundCall) { |
| themeColorManager.onForegroundCallChanged(context, newForegroundCall); |
| if (inCallActivity != null) { |
| inCallActivity.onForegroundCallChanged(newForegroundCall); |
| } |
| } |
| |
| public InCallActivity getActivity() { |
| return inCallActivity; |
| } |
| |
| /** Called when the UI begins, and starts the callstate callbacks if necessary. */ |
| public void setActivity(InCallActivity inCallActivity) { |
| if (inCallActivity == null) { |
| throw new IllegalArgumentException("registerActivity cannot be called with null"); |
| } |
| if (this.inCallActivity != null && this.inCallActivity != inCallActivity) { |
| LogUtil.w( |
| "InCallPresenter.setActivity", "Setting a second activity before destroying the first."); |
| } |
| updateActivity(inCallActivity); |
| } |
| |
| ExternalCallNotifier getExternalCallNotifier() { |
| return externalCallNotifier; |
| } |
| |
| VideoSurfaceTexture getLocalVideoSurfaceTexture() { |
| if (localVideoSurfaceTexture == null) { |
| localVideoSurfaceTexture = VideoSurfaceBindings.createLocalVideoSurfaceTexture(); |
| } |
| return localVideoSurfaceTexture; |
| } |
| |
| VideoSurfaceTexture getRemoteVideoSurfaceTexture() { |
| if (remoteVideoSurfaceTexture == null) { |
| remoteVideoSurfaceTexture = VideoSurfaceBindings.createRemoteVideoSurfaceTexture(); |
| } |
| return remoteVideoSurfaceTexture; |
| } |
| |
| void cleanupSurfaces() { |
| if (remoteVideoSurfaceTexture != null) { |
| remoteVideoSurfaceTexture.setDoneWithSurface(); |
| remoteVideoSurfaceTexture = null; |
| } |
| if (localVideoSurfaceTexture != null) { |
| localVideoSurfaceTexture.setDoneWithSurface(); |
| localVideoSurfaceTexture = null; |
| } |
| } |
| |
| @Override |
| public void onAudioStateChanged(CallAudioState audioState) { |
| if (statusBarNotifier != null) { |
| statusBarNotifier.updateNotification(); |
| } |
| } |
| |
| /** All the main states of InCallActivity. */ |
| public enum InCallState { |
| // InCall Screen is off and there are no calls |
| NO_CALLS, |
| |
| // Incoming-call screen is up |
| INCOMING, |
| |
| // In-call experience is showing |
| INCALL, |
| |
| // Waiting for user input before placing outgoing call |
| WAITING_FOR_ACCOUNT, |
| |
| // UI is starting up but no call has been initiated yet. |
| // The UI is waiting for Telecom to respond. |
| PENDING_OUTGOING, |
| |
| // User is dialing out |
| OUTGOING; |
| |
| public boolean isIncoming() { |
| return (this == INCOMING); |
| } |
| |
| public boolean isConnectingOrConnected() { |
| return (this == INCOMING || this == OUTGOING || this == INCALL); |
| } |
| } |
| |
| /** Interface implemented by classes that need to know about the InCall State. */ |
| public interface InCallStateListener { |
| |
| // TODO: Enhance state to contain the call objects instead of passing CallList |
| void onStateChange(InCallState oldState, InCallState newState, CallList callList); |
| } |
| |
| public interface IncomingCallListener { |
| |
| void onIncomingCall(InCallState oldState, InCallState newState, DialerCall call); |
| } |
| |
| public interface CanAddCallListener { |
| |
| void onCanAddCallChanged(boolean canAddCall); |
| } |
| |
| public interface InCallDetailsListener { |
| |
| void onDetailsChanged(DialerCall call, android.telecom.Call.Details details); |
| } |
| |
| public interface InCallOrientationListener { |
| |
| void onDeviceOrientationChanged(@ScreenOrientation int orientation); |
| } |
| |
| /** |
| * Interface implemented by classes that need to know about events which occur within the In-Call |
| * UI. Used as a means of communicating between fragments that make up the UI. |
| */ |
| public interface InCallEventListener { |
| |
| void onFullscreenModeChanged(boolean isFullscreenMode); |
| } |
| |
| public interface InCallUiListener { |
| |
| void onUiShowing(boolean showing); |
| } |
| |
| private class InCallUiLockImpl implements InCallUiLock { |
| private final String tag; |
| |
| private InCallUiLockImpl(String tag) { |
| this.tag = tag; |
| } |
| |
| @MainThread |
| @Override |
| public void release() { |
| Assert.isMainThread(); |
| releaseInCallUiLock(InCallUiLockImpl.this); |
| } |
| |
| @Override |
| public String toString() { |
| return "InCallUiLock[" + tag + "]"; |
| } |
| } |
| |
| @MainThread |
| public InCallUiLock acquireInCallUiLock(String tag) { |
| Assert.isMainThread(); |
| InCallUiLock lock = new InCallUiLockImpl(tag); |
| inCallUiLocks.add(lock); |
| return lock; |
| } |
| |
| @MainThread |
| private void releaseInCallUiLock(InCallUiLock lock) { |
| Assert.isMainThread(); |
| LogUtil.i("InCallPresenter.releaseInCallUiLock", "releasing %s", lock); |
| inCallUiLocks.remove(lock); |
| if (inCallUiLocks.isEmpty()) { |
| LogUtil.i("InCallPresenter.releaseInCallUiLock", "all locks released"); |
| if (inCallState == InCallState.NO_CALLS) { |
| LogUtil.i("InCallPresenter.releaseInCallUiLock", "no more calls, finishing UI"); |
| attemptFinishActivity(); |
| attemptCleanup(); |
| } |
| } |
| } |
| |
| @MainThread |
| public boolean isInCallUiLocked() { |
| Assert.isMainThread(); |
| if (inCallUiLocks.isEmpty()) { |
| return false; |
| } |
| for (InCallUiLock lock : inCallUiLocks) { |
| LogUtil.i("InCallPresenter.isInCallUiLocked", "still locked by %s", lock); |
| } |
| return true; |
| } |
| |
| public void addCallClicked() { |
| if (addCallClicked) { |
| // Since clicking add call button brings user to MainActivity and coming back refreshes mute |
| // state, add call button should only be clicked once during InCallActivity shows. |
| return; |
| } |
| addCallClicked = true; |
| if (!AudioModeProvider.getInstance().getAudioState().isMuted()) { |
| // Automatically mute the current call |
| TelecomAdapter.getInstance().mute(true); |
| automaticallyMutedByAddCall = true; |
| } |
| TelecomAdapter.getInstance().addCall(); |
| } |
| |
| /** Refresh mute state after call UI resuming from add call screen. */ |
| public void refreshMuteState() { |
| LogUtil.i( |
| "InCallPresenter.refreshMuteState", |
| "refreshMuteStateAfterAddCall: %b addCallClicked: %b", |
| automaticallyMutedByAddCall, |
| addCallClicked); |
| if (!addCallClicked) { |
| return; |
| } |
| if (automaticallyMutedByAddCall) { |
| // Restore the previous mute state |
| TelecomAdapter.getInstance().mute(false); |
| automaticallyMutedByAddCall = false; |
| } |
| addCallClicked = false; |
| } |
| |
| private final Set<InCallUiLock> inCallUiLocks = new ArraySet<>(); |
| } |