1 /* 2 * Copyright (C) 2017 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License 15 */ 16 17 package android.telephony.ims.stub; 18 19 import android.annotation.NonNull; 20 import android.annotation.SystemApi; 21 import android.annotation.TestApi; 22 import android.os.Message; 23 import android.os.RemoteException; 24 import android.telephony.ims.ImsCallProfile; 25 import android.telephony.ims.ImsCallSession; 26 import android.telephony.ims.ImsCallSessionListener; 27 import android.telephony.ims.ImsReasonInfo; 28 import android.telephony.ims.ImsStreamMediaProfile; 29 import android.telephony.ims.ImsVideoCallProvider; 30 import android.telephony.ims.aidl.IImsCallSessionListener; 31 32 import com.android.ims.internal.IImsCallSession; 33 import com.android.ims.internal.IImsVideoCallProvider; 34 /** 35 * Base implementation of IImsCallSession, which implements stub versions of the methods available. 36 * 37 * Override the methods that your implementation of ImsCallSession supports. 38 * 39 * @hide 40 */ 41 @SystemApi 42 @TestApi 43 // DO NOT remove or change the existing APIs, only add new ones to this Base implementation or you 44 // will break other implementations of ImsCallSession maintained by other ImsServices. 45 public class ImsCallSessionImplBase implements AutoCloseable { 46 /** 47 * Notify USSD Mode. 48 */ 49 public static final int USSD_MODE_NOTIFY = 0; 50 /** 51 * Request USSD Mode 52 */ 53 public static final int USSD_MODE_REQUEST = 1; 54 55 /** 56 * Defines IMS call session state. 57 */ 58 public static class State { 59 public static final int IDLE = 0; 60 public static final int INITIATED = 1; 61 public static final int NEGOTIATING = 2; 62 public static final int ESTABLISHING = 3; 63 public static final int ESTABLISHED = 4; 64 65 public static final int RENEGOTIATING = 5; 66 public static final int REESTABLISHING = 6; 67 68 public static final int TERMINATING = 7; 69 public static final int TERMINATED = 8; 70 71 public static final int INVALID = (-1); 72 73 /** 74 * Converts the state to string. 75 */ toString(int state)76 public static String toString(int state) { 77 switch (state) { 78 case IDLE: 79 return "IDLE"; 80 case INITIATED: 81 return "INITIATED"; 82 case NEGOTIATING: 83 return "NEGOTIATING"; 84 case ESTABLISHING: 85 return "ESTABLISHING"; 86 case ESTABLISHED: 87 return "ESTABLISHED"; 88 case RENEGOTIATING: 89 return "RENEGOTIATING"; 90 case REESTABLISHING: 91 return "REESTABLISHING"; 92 case TERMINATING: 93 return "TERMINATING"; 94 case TERMINATED: 95 return "TERMINATED"; 96 default: 97 return "UNKNOWN"; 98 } 99 } 100 101 /** 102 * @hide 103 */ State()104 private State() { 105 } 106 } 107 108 // Non-final for injection by tests 109 private IImsCallSession mServiceImpl = new IImsCallSession.Stub() { 110 @Override 111 public void close() { 112 ImsCallSessionImplBase.this.close(); 113 } 114 115 @Override 116 public String getCallId() { 117 return ImsCallSessionImplBase.this.getCallId(); 118 } 119 120 @Override 121 public ImsCallProfile getCallProfile() { 122 return ImsCallSessionImplBase.this.getCallProfile(); 123 } 124 125 @Override 126 public ImsCallProfile getLocalCallProfile() { 127 return ImsCallSessionImplBase.this.getLocalCallProfile(); 128 } 129 130 @Override 131 public ImsCallProfile getRemoteCallProfile() { 132 return ImsCallSessionImplBase.this.getRemoteCallProfile(); 133 } 134 135 @Override 136 public String getProperty(String name) { 137 return ImsCallSessionImplBase.this.getProperty(name); 138 } 139 140 @Override 141 public int getState() { 142 return ImsCallSessionImplBase.this.getState(); 143 } 144 145 @Override 146 public boolean isInCall() { 147 return ImsCallSessionImplBase.this.isInCall(); 148 } 149 150 @Override 151 public void setListener(IImsCallSessionListener listener) { 152 ImsCallSessionImplBase.this.setListener(new ImsCallSessionListener(listener)); 153 } 154 155 @Override 156 public void setMute(boolean muted) { 157 ImsCallSessionImplBase.this.setMute(muted); 158 } 159 160 @Override 161 public void start(String callee, ImsCallProfile profile) { 162 ImsCallSessionImplBase.this.start(callee, profile); 163 } 164 165 @Override 166 public void startConference(String[] participants, ImsCallProfile profile) throws 167 RemoteException { 168 ImsCallSessionImplBase.this.startConference(participants, profile); 169 } 170 171 @Override 172 public void accept(int callType, ImsStreamMediaProfile profile) { 173 ImsCallSessionImplBase.this.accept(callType, profile); 174 } 175 176 @Override 177 public void deflect(String deflectNumber) { 178 ImsCallSessionImplBase.this.deflect(deflectNumber); 179 } 180 181 @Override 182 public void reject(int reason) { 183 ImsCallSessionImplBase.this.reject(reason); 184 } 185 186 @Override 187 public void transfer(@NonNull String number, boolean isConfirmationRequired) { 188 ImsCallSessionImplBase.this.transfer(number, isConfirmationRequired); 189 } 190 191 @Override 192 public void consultativeTransfer(@NonNull IImsCallSession transferToSession) { 193 ImsCallSessionImplBase otherSession = new ImsCallSessionImplBase(); 194 otherSession.setServiceImpl(transferToSession); 195 ImsCallSessionImplBase.this.transfer(otherSession); 196 } 197 198 @Override 199 public void terminate(int reason) { 200 ImsCallSessionImplBase.this.terminate(reason); 201 } 202 203 @Override 204 public void hold(ImsStreamMediaProfile profile) { 205 ImsCallSessionImplBase.this.hold(profile); 206 } 207 208 @Override 209 public void resume(ImsStreamMediaProfile profile) { 210 ImsCallSessionImplBase.this.resume(profile); 211 } 212 213 @Override 214 public void merge() { 215 ImsCallSessionImplBase.this.merge(); 216 } 217 218 @Override 219 public void update(int callType, ImsStreamMediaProfile profile) { 220 ImsCallSessionImplBase.this.update(callType, profile); 221 } 222 223 @Override 224 public void extendToConference(String[] participants) { 225 ImsCallSessionImplBase.this.extendToConference(participants); 226 } 227 228 @Override 229 public void inviteParticipants(String[] participants) { 230 ImsCallSessionImplBase.this.inviteParticipants(participants); 231 } 232 233 @Override 234 public void removeParticipants(String[] participants) { 235 ImsCallSessionImplBase.this.removeParticipants(participants); 236 } 237 238 @Override 239 public void sendDtmf(char c, Message result) { 240 ImsCallSessionImplBase.this.sendDtmf(c, result); 241 } 242 243 @Override 244 public void startDtmf(char c) { 245 ImsCallSessionImplBase.this.startDtmf(c); 246 } 247 248 @Override 249 public void stopDtmf() { 250 ImsCallSessionImplBase.this.stopDtmf(); 251 } 252 253 @Override 254 public void sendUssd(String ussdMessage) { 255 ImsCallSessionImplBase.this.sendUssd(ussdMessage); 256 } 257 258 @Override 259 public IImsVideoCallProvider getVideoCallProvider() { 260 return ImsCallSessionImplBase.this.getVideoCallProvider(); 261 } 262 263 @Override 264 public boolean isMultiparty() { 265 return ImsCallSessionImplBase.this.isMultiparty(); 266 } 267 268 @Override 269 public void sendRttModifyRequest(ImsCallProfile toProfile) { 270 ImsCallSessionImplBase.this.sendRttModifyRequest(toProfile); 271 } 272 273 @Override 274 public void sendRttModifyResponse(boolean status) { 275 ImsCallSessionImplBase.this.sendRttModifyResponse(status); 276 } 277 278 @Override 279 public void sendRttMessage(String rttMessage) { 280 ImsCallSessionImplBase.this.sendRttMessage(rttMessage); 281 } 282 }; 283 284 /** 285 * @hide 286 */ setListener(IImsCallSessionListener listener)287 public final void setListener(IImsCallSessionListener listener) throws RemoteException { 288 setListener(new ImsCallSessionListener(listener)); 289 } 290 291 /** 292 * Sets the listener to listen to the session events. An {@link ImsCallSession} 293 * can only hold one listener at a time. Subsequent calls to this method 294 * override the previous listener. 295 * 296 * @param listener {@link ImsCallSessionListener} used to notify the framework of updates 297 * to the ImsCallSession 298 */ setListener(ImsCallSessionListener listener)299 public void setListener(ImsCallSessionListener listener) { 300 } 301 302 /** 303 * Closes the object. This {@link ImsCallSessionImplBase} is not usable after being closed. 304 */ 305 @Override close()306 public void close() { 307 308 } 309 310 /** 311 * @return A String containing the unique call ID of this {@link ImsCallSessionImplBase}. 312 */ getCallId()313 public String getCallId() { 314 return null; 315 } 316 317 /** 318 * @return The {@link ImsCallProfile} that this {@link ImsCallSessionImplBase} is associated 319 * with. 320 */ getCallProfile()321 public ImsCallProfile getCallProfile() { 322 return null; 323 } 324 325 /** 326 * @return The local {@link ImsCallProfile} that this {@link ImsCallSessionImplBase} is 327 * associated with. 328 */ getLocalCallProfile()329 public ImsCallProfile getLocalCallProfile() { 330 return null; 331 } 332 333 /** 334 * @return The remote {@link ImsCallProfile} that this {@link ImsCallSessionImplBase} is 335 * associated with. 336 */ getRemoteCallProfile()337 public ImsCallProfile getRemoteCallProfile() { 338 return null; 339 } 340 341 /** 342 * @param name The String extra key. 343 * @return The string extra value associated with the specified property. 344 */ getProperty(String name)345 public String getProperty(String name) { 346 return null; 347 } 348 349 /** 350 * @return The {@link ImsCallSessionImplBase} state, defined in 351 * {@link ImsCallSessionImplBase.State}. 352 */ getState()353 public int getState() { 354 return ImsCallSessionImplBase.State.INVALID; 355 } 356 357 /** 358 * @return true if the {@link ImsCallSessionImplBase} is in a call, false otherwise. 359 */ isInCall()360 public boolean isInCall() { 361 return false; 362 } 363 364 /** 365 * Mutes or unmutes the mic for the active call. 366 * 367 * @param muted true if the call should be muted, false otherwise. 368 */ setMute(boolean muted)369 public void setMute(boolean muted) { 370 } 371 372 /** 373 * Initiates an IMS call with the specified number and call profile. 374 * The session listener set in {@link #setListener(ImsCallSessionListener)} is called back upon 375 * defined session events. 376 * Only valid to call when the session state is in 377 * {@link ImsCallSession.State#IDLE}. 378 * 379 * @param callee dialed string to make the call to 380 * @param profile call profile to make the call with the specified service type, 381 * call type and media information 382 * @see {@link ImsCallSession.Listener#callSessionStarted}, 383 * {@link ImsCallSession.Listener#callSessionStartFailed} 384 */ start(String callee, ImsCallProfile profile)385 public void start(String callee, ImsCallProfile profile) { 386 } 387 388 /** 389 * Initiates an IMS call with the specified participants and call profile. 390 * The session listener set in {@link #setListener(ImsCallSessionListener)} is called back upon 391 * defined session events. 392 * The method is only valid to call when the session state is in 393 * {@link ImsCallSession.State#IDLE}. 394 * 395 * @param participants participant list to initiate an IMS conference call 396 * @param profile call profile to make the call with the specified service type, 397 * call type and media information 398 * @see {@link ImsCallSession.Listener#callSessionStarted}, 399 * {@link ImsCallSession.Listener#callSessionStartFailed} 400 */ startConference(String[] participants, ImsCallProfile profile)401 public void startConference(String[] participants, ImsCallProfile profile) { 402 } 403 404 /** 405 * Accepts an incoming call or session update. 406 * 407 * @param callType call type specified in {@link ImsCallProfile} to be answered 408 * @param profile stream media profile {@link ImsStreamMediaProfile} to be answered 409 * @see {@link ImsCallSession.Listener#callSessionStarted} 410 */ accept(int callType, ImsStreamMediaProfile profile)411 public void accept(int callType, ImsStreamMediaProfile profile) { 412 } 413 414 /** 415 * Deflects an incoming call. 416 * 417 * @param deflectNumber number to deflect the call 418 */ deflect(String deflectNumber)419 public void deflect(String deflectNumber) { 420 } 421 422 /** 423 * Rejects an incoming call or session update. 424 * 425 * @param reason reason code to reject an incoming call, defined in {@link ImsReasonInfo}. 426 * The {@link android.telecom.InCallService} (dialer app) can use the 427 * {@link android.telecom.Call#reject(int)} API to reject a call while specifying 428 * a user-indicated reason for rejecting the call. 429 * Normal call declines ({@link android.telecom.Call#REJECT_REASON_DECLINED}) will 430 * map to {@link ImsReasonInfo#CODE_USER_DECLINE}. 431 * Unwanted calls ({@link android.telecom.Call#REJECT_REASON_UNWANTED}) will map 432 * to {@link ImsReasonInfo#CODE_SIP_USER_MARKED_UNWANTED}. 433 * {@link ImsCallSession.Listener#callSessionStartFailed} 434 */ reject(int reason)435 public void reject(int reason) { 436 } 437 438 /** 439 * Transfer an established call to given number 440 * 441 * @param number number to transfer the call 442 * @param isConfirmationRequired if {@code True}, indicates a confirmed transfer, 443 * if {@code False} it indicates an unconfirmed transfer. 444 * @hide 445 */ transfer(@onNull String number, boolean isConfirmationRequired)446 public void transfer(@NonNull String number, boolean isConfirmationRequired) { 447 } 448 449 /** 450 * Transfer an established call to another call session 451 * 452 * @param otherSession The other ImsCallSession to transfer the ongoing session to. 453 * @hide 454 */ transfer(@onNull ImsCallSessionImplBase otherSession)455 public void transfer(@NonNull ImsCallSessionImplBase otherSession) { 456 } 457 458 /** 459 * Terminates a call. 460 * 461 * @param reason reason code to terminate a call, defined in {@link ImsReasonInfo}. 462 * 463 * @see {@link ImsCallSession.Listener#callSessionTerminated} 464 */ terminate(int reason)465 public void terminate(int reason) { 466 } 467 468 /** 469 * Puts a call on hold. When it succeeds, {@link ImsCallSession.Listener#callSessionHeld} is 470 * called. 471 * 472 * @param profile stream media profile {@link ImsStreamMediaProfile} to hold the call 473 * @see {@link ImsCallSession.Listener#callSessionHeld}, 474 * {@link ImsCallSession.Listener#callSessionHoldFailed} 475 */ hold(ImsStreamMediaProfile profile)476 public void hold(ImsStreamMediaProfile profile) { 477 } 478 479 /** 480 * Continues a call that's on hold. When it succeeds, 481 * {@link ImsCallSession.Listener#callSessionResumed} is called. 482 * 483 * @param profile stream media profile with {@link ImsStreamMediaProfile} to resume the call 484 * @see {@link ImsCallSession.Listener#callSessionResumed}, 485 * {@link ImsCallSession.Listener#callSessionResumeFailed} 486 */ resume(ImsStreamMediaProfile profile)487 public void resume(ImsStreamMediaProfile profile) { 488 } 489 490 /** 491 * Merges the active and held call. When the merge starts, 492 * {@link ImsCallSession.Listener#callSessionMergeStarted} is called. 493 * {@link ImsCallSession.Listener#callSessionMergeComplete} is called if the merge is 494 * successful, and {@link ImsCallSession.Listener#callSessionMergeFailed} is called if the merge 495 * fails. 496 * 497 * @see {@link ImsCallSession.Listener#callSessionMergeStarted}, 498 * {@link ImsCallSession.Listener#callSessionMergeComplete}, 499 * {@link ImsCallSession.Listener#callSessionMergeFailed} 500 */ merge()501 public void merge() { 502 } 503 504 /** 505 * Updates the current call's properties (ex. call mode change: video upgrade / downgrade). 506 * 507 * @param callType call type specified in {@link ImsCallProfile} to be updated 508 * @param profile stream media profile {@link ImsStreamMediaProfile} to be updated 509 * @see {@link ImsCallSession.Listener#callSessionUpdated}, 510 * {@link ImsCallSession.Listener#callSessionUpdateFailed} 511 */ update(int callType, ImsStreamMediaProfile profile)512 public void update(int callType, ImsStreamMediaProfile profile) { 513 } 514 515 /** 516 * Extends this call to the conference call with the specified recipients. 517 * 518 * @param participants participant list to be invited to the conference call after extending the 519 * call 520 * @see {@link ImsCallSession.Listener#callSessionConferenceExtended}, 521 * {@link ImsCallSession.Listener#callSessionConferenceExtendFailed} 522 */ extendToConference(String[] participants)523 public void extendToConference(String[] participants) { 524 } 525 526 /** 527 * Requests the conference server to invite an additional participants to the conference. 528 * 529 * @param participants participant list to be invited to the conference call 530 * @see {@link ImsCallSession.Listener#callSessionInviteParticipantsRequestDelivered}, 531 * {@link ImsCallSession.Listener#callSessionInviteParticipantsRequestFailed} 532 */ inviteParticipants(String[] participants)533 public void inviteParticipants(String[] participants) { 534 } 535 536 /** 537 * Requests the conference server to remove the specified participants from the conference. 538 * 539 * @param participants participant list to be removed from the conference call 540 * @see {@link ImsCallSession.Listener#callSessionRemoveParticipantsRequestDelivered}, 541 * {@link ImsCallSession.Listener#callSessionRemoveParticipantsRequestFailed} 542 */ removeParticipants(String[] participants)543 public void removeParticipants(String[] participants) { 544 } 545 546 /** 547 * Sends a DTMF code. According to <a href="http://tools.ietf.org/html/rfc2833">RFC 2833</a>, 548 * event 0 ~ 9 maps to decimal value 0 ~ 9, '*' to 10, '#' to 11, event 'A' ~ 'D' to 12 ~ 15, 549 * and event flash to 16. Currently, event flash is not supported. 550 * 551 * @param c the DTMF to send. '0' ~ '9', 'A' ~ 'D', '*', '#' are valid inputs. 552 * @param result If non-null, the {@link Message} to send when the operation is complete. This 553 * is done by using the associated {@link android.os.Messenger} in 554 * {@link Message#replyTo}. For example: 555 * {@code 556 * // Send DTMF and other operations... 557 * try { 558 * // Notify framework that the DTMF was sent. 559 * Messenger dtmfMessenger = result.replyTo; 560 * if (dtmfMessenger != null) { 561 * dtmfMessenger.send(result); 562 * } 563 * } catch (RemoteException e) { 564 * // Remote side is dead 565 * } 566 * } 567 */ sendDtmf(char c, Message result)568 public void sendDtmf(char c, Message result) { 569 } 570 571 /** 572 * Start a DTMF code. According to <a href="http://tools.ietf.org/html/rfc2833">RFC 2833</a>, 573 * event 0 ~ 9 maps to decimal value 0 ~ 9, '*' to 10, '#' to 11, event 'A' ~ 'D' to 12 ~ 15, 574 * and event flash to 16. Currently, event flash is not supported. 575 * 576 * @param c the DTMF to send. '0' ~ '9', 'A' ~ 'D', '*', '#' are valid inputs. 577 */ startDtmf(char c)578 public void startDtmf(char c) { 579 } 580 581 /** 582 * Stop a DTMF code. 583 */ stopDtmf()584 public void stopDtmf() { 585 } 586 587 /** 588 * Sends an USSD message. 589 * 590 * @param ussdMessage USSD message to send 591 */ sendUssd(String ussdMessage)592 public void sendUssd(String ussdMessage) { 593 } 594 595 /** 596 * See {@link #getImsVideoCallProvider()}, used directly in older ImsService implementations. 597 * @hide 598 */ getVideoCallProvider()599 public IImsVideoCallProvider getVideoCallProvider() { 600 ImsVideoCallProvider provider = getImsVideoCallProvider(); 601 return provider != null ? provider.getInterface() : null; 602 } 603 604 /** 605 * @return The {@link ImsVideoCallProvider} implementation contained within the IMS service 606 * process. 607 */ getImsVideoCallProvider()608 public ImsVideoCallProvider getImsVideoCallProvider() { 609 return null; 610 } 611 612 /** 613 * Determines if the current session is multiparty. 614 * @return {@code True} if the session is multiparty. 615 */ isMultiparty()616 public boolean isMultiparty() { 617 return false; 618 } 619 620 /** 621 * Device issues RTT modify request 622 * @param toProfile The profile with requested changes made 623 */ sendRttModifyRequest(ImsCallProfile toProfile)624 public void sendRttModifyRequest(ImsCallProfile toProfile) { 625 } 626 627 /** 628 * Device responds to Remote RTT modify request 629 * @param status true if the the request was accepted or false of the request is defined. 630 */ sendRttModifyResponse(boolean status)631 public void sendRttModifyResponse(boolean status) { 632 } 633 634 /** 635 * Device sends RTT message 636 * @param rttMessage RTT message to be sent 637 */ sendRttMessage(String rttMessage)638 public void sendRttMessage(String rttMessage) { 639 } 640 641 /** @hide */ getServiceImpl()642 public IImsCallSession getServiceImpl() { 643 return mServiceImpl; 644 } 645 646 /** @hide */ setServiceImpl(IImsCallSession serviceImpl)647 public void setServiceImpl(IImsCallSession serviceImpl) { 648 mServiceImpl = serviceImpl; 649 } 650 } 651