1 /* 2 * Copyright (C) 2018 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; 18 19 import android.annotation.NonNull; 20 import android.annotation.SystemApi; 21 import android.os.Bundle; 22 import android.os.Parcel; 23 import android.os.Parcelable; 24 import android.telecom.Call; 25 import android.telecom.Connection; 26 27 import com.android.telephony.Rlog; 28 29 import java.util.HashMap; 30 import java.util.Iterator; 31 import java.util.Map.Entry; 32 import java.util.Set; 33 34 /** 35 * Provides the conference information (defined in RFC 4575) for IMS conference call. 36 * 37 * @hide 38 */ 39 @SystemApi 40 public final class ImsConferenceState implements Parcelable { 41 private static final String TAG = "ImsConferenceState"; 42 /** 43 * conference-info : user 44 */ 45 // user (String) : Tel or SIP URI 46 public static final String USER = "user"; 47 // user > display text (String) 48 public static final String DISPLAY_TEXT = "display-text"; 49 // user > endpoint (String) : URI or GRUU or Phone number 50 public static final String ENDPOINT = "endpoint"; 51 // user > endpoint > status 52 public static final String STATUS = "status"; 53 54 /** 55 * status-type (String) : 56 * "pending" : Endpoint is not yet in the session, but it is anticipated that he/she will 57 * join in the near future. 58 * "dialing-out" : Focus has dialed out to connect the endpoint to the conference, 59 * but the endpoint is not yet in the roster (probably being authenticated). 60 * "dialing-in" : Endpoint is dialing into the conference, not yet in the roster 61 * (probably being authenticated). 62 * "alerting" : PSTN alerting or SIP 180 Ringing was returned for the outbound call; 63 * endpoint is being alerted. 64 * "on-hold" : Active signaling dialog exists between an endpoint and a focus, 65 * but endpoint is "on-hold" for this conference, i.e., he/she is neither "hearing" 66 * the conference mix nor is his/her media being mixed in the conference. 67 * "connected" : Endpoint is a participant in the conference. Depending on the media policies, 68 * he/she can send and receive media to and from other participants. 69 * "disconnecting" : Focus is in the process of disconnecting the endpoint 70 * (e.g. in SIP a DISCONNECT or BYE was sent to the endpoint). 71 * "disconnected" : Endpoint is not a participant in the conference, and no active dialog 72 * exists between the endpoint and the focus. 73 * "muted-via-focus" : Active signaling dialog exists beween an endpoint and a focus and 74 * the endpoint can "listen" to the conference, but the endpoint's media is not being 75 * mixed into the conference. 76 * "connect-fail" : Endpoint fails to join the conference by rejecting the conference call. 77 */ 78 public static final String STATUS_PENDING = "pending"; 79 public static final String STATUS_DIALING_OUT = "dialing-out"; 80 public static final String STATUS_DIALING_IN = "dialing-in"; 81 public static final String STATUS_ALERTING = "alerting"; 82 public static final String STATUS_ON_HOLD = "on-hold"; 83 public static final String STATUS_CONNECTED = "connected"; 84 public static final String STATUS_DISCONNECTING = "disconnecting"; 85 public static final String STATUS_DISCONNECTED = "disconnected"; 86 public static final String STATUS_MUTED_VIA_FOCUS = "muted-via-focus"; 87 public static final String STATUS_CONNECT_FAIL = "connect-fail"; 88 public static final String STATUS_SEND_ONLY = "sendonly"; 89 public static final String STATUS_SEND_RECV = "sendrecv"; 90 91 /** 92 * conference-info : SIP status code (integer) 93 */ 94 public static final String SIP_STATUS_CODE = "sipstatuscode"; 95 96 public final HashMap<String, Bundle> mParticipants = new HashMap<String, Bundle>(); 97 98 /** @hide */ ImsConferenceState()99 public ImsConferenceState() { 100 } 101 ImsConferenceState(Parcel in)102 private ImsConferenceState(Parcel in) { 103 readFromParcel(in); 104 } 105 106 @Override describeContents()107 public int describeContents() { 108 return 0; 109 } 110 111 @Override writeToParcel(Parcel out, int flags)112 public void writeToParcel(Parcel out, int flags) { 113 out.writeInt(mParticipants.size()); 114 115 if (mParticipants.size() > 0) { 116 Set<Entry<String, Bundle>> entries = mParticipants.entrySet(); 117 118 if (entries != null) { 119 Iterator<Entry<String, Bundle>> iterator = entries.iterator(); 120 121 while (iterator.hasNext()) { 122 Entry<String, Bundle> entry = iterator.next(); 123 124 out.writeString(entry.getKey()); 125 out.writeParcelable(entry.getValue(), 0); 126 } 127 } 128 } 129 } 130 readFromParcel(Parcel in)131 private void readFromParcel(Parcel in) { 132 int size = in.readInt(); 133 134 for (int i = 0; i < size; ++i) { 135 String user = in.readString(); 136 Bundle state = in.readParcelable(null, android.os.Bundle.class); 137 mParticipants.put(user, state); 138 } 139 } 140 141 public static final @android.annotation.NonNull Creator<ImsConferenceState> CREATOR = 142 new Creator<ImsConferenceState>() { 143 @Override 144 public ImsConferenceState createFromParcel(Parcel in) { 145 return new ImsConferenceState(in); 146 } 147 148 @Override 149 public ImsConferenceState[] newArray(int size) { 150 return new ImsConferenceState[size]; 151 } 152 }; 153 154 /** 155 * Translates an {@code ImsConferenceState} status type to a telecom connection state. 156 * 157 * @param status The status type. 158 * @return The corresponding {@link android.telecom.Connection} state. 159 */ getConnectionStateForStatus(String status)160 public static int getConnectionStateForStatus(String status) { 161 if (status.equals(STATUS_PENDING)) { 162 return Connection.STATE_INITIALIZING; 163 } else if (status.equals(STATUS_DIALING_IN)) { 164 return Connection.STATE_RINGING; 165 } else if (status.equals(STATUS_ALERTING) || 166 status.equals(STATUS_DIALING_OUT)) { 167 return Connection.STATE_DIALING; 168 } else if (status.equals(STATUS_ON_HOLD) || 169 status.equals(STATUS_SEND_ONLY)) { 170 return Connection.STATE_HOLDING; 171 } else if (status.equals(STATUS_CONNECTED) || 172 status.equals(STATUS_MUTED_VIA_FOCUS) || 173 status.equals(STATUS_DISCONNECTING) || 174 status.equals(STATUS_SEND_RECV)) { 175 return Connection.STATE_ACTIVE; 176 } else if (status.equals(STATUS_DISCONNECTED)) { 177 return Connection.STATE_DISCONNECTED; 178 } 179 return Call.STATE_ACTIVE; 180 } 181 182 @NonNull 183 @Override toString()184 public String toString() { 185 StringBuilder sb = new StringBuilder(); 186 sb.append("["); 187 sb.append(ImsConferenceState.class.getSimpleName()); 188 sb.append(" "); 189 if (mParticipants.size() > 0) { 190 Set<Entry<String, Bundle>> entries = mParticipants.entrySet(); 191 192 if (entries != null) { 193 Iterator<Entry<String, Bundle>> iterator = entries.iterator(); 194 sb.append("<"); 195 while (iterator.hasNext()) { 196 Entry<String, Bundle> entry = iterator.next(); 197 sb.append(Rlog.pii(TAG, entry.getKey())); 198 sb.append(": "); 199 Bundle participantData = entry.getValue(); 200 201 for (String key : participantData.keySet()) { 202 sb.append(key); 203 sb.append("="); 204 if (STATUS.equals(key)) { 205 sb.append(participantData.get(key)); 206 } else { 207 sb.append(Rlog.pii(TAG, participantData.get(key))); 208 } 209 sb.append(", "); 210 } 211 } 212 sb.append(">"); 213 } 214 } 215 sb.append("]"); 216 return sb.toString(); 217 } 218 } 219