/* * Copyright (C) 2014 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 android.bluetooth; import android.annotation.NonNull; import android.compat.annotation.UnsupportedAppUsage; import android.content.AttributionSource; import android.os.Build; import android.os.Parcel; import android.os.Parcelable; import android.os.SystemClock; import java.util.UUID; /** * This class represents a single call, its state and properties. It implements {@link Parcelable} * for inter-process message passing. * * @hide */ public final class BluetoothHeadsetClientCall implements Parcelable, Attributable { /* Call state */ /** Call is active. */ public static final int CALL_STATE_ACTIVE = 0; /** Call is in held state. */ public static final int CALL_STATE_HELD = 1; /** Outgoing call that is being dialed right now. */ public static final int CALL_STATE_DIALING = 2; /** Outgoing call that remote party has already been alerted about. */ public static final int CALL_STATE_ALERTING = 3; /** Incoming call that can be accepted or rejected. */ public static final int CALL_STATE_INCOMING = 4; /** Waiting call state when there is already an active call. */ public static final int CALL_STATE_WAITING = 5; /** * Call that has been held by response and hold (see Bluetooth specification for further * references). */ public static final int CALL_STATE_HELD_BY_RESPONSE_AND_HOLD = 6; /** Call that has been already terminated and should not be referenced as a valid call. */ public static final int CALL_STATE_TERMINATED = 7; private final BluetoothDevice mDevice; private final int mId; private int mState; private String mNumber; private boolean mMultiParty; private final boolean mOutgoing; private final UUID mUUID; private final long mCreationElapsedMilli; private final boolean mInBandRing; /** Creates BluetoothHeadsetClientCall instance. */ public BluetoothHeadsetClientCall( BluetoothDevice device, int id, int state, String number, boolean multiParty, boolean outgoing, boolean inBandRing) { this(device, id, UUID.randomUUID(), state, number, multiParty, outgoing, inBandRing); } public BluetoothHeadsetClientCall( BluetoothDevice device, int id, UUID uuid, int state, String number, boolean multiParty, boolean outgoing, boolean inBandRing) { mDevice = device; mId = id; mUUID = uuid; mState = state; mNumber = number != null ? number : ""; mMultiParty = multiParty; mOutgoing = outgoing; mInBandRing = inBandRing; mCreationElapsedMilli = SystemClock.elapsedRealtime(); } /** @hide */ public void setAttributionSource(@NonNull AttributionSource attributionSource) { Attributable.setAttributionSource(mDevice, attributionSource); } /** * Sets call's state. * *

Note: This is an internal function and shouldn't be exposed * * @param state new call state. */ public void setState(int state) { mState = state; } /** * Sets call's number. * *

Note: This is an internal function and shouldn't be exposed * * @param number String representing phone number. */ public void setNumber(String number) { mNumber = number; } /** * Sets this call as multi party call. * *

Note: This is an internal function and shouldn't be exposed * * @param multiParty if true sets this call as a part of multi party conference. */ public void setMultiParty(boolean multiParty) { mMultiParty = multiParty; } /** * Gets call's device. * * @return call device. */ public BluetoothDevice getDevice() { return mDevice; } /** * Gets call's Id. * * @return call id. */ @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) public int getId() { return mId; } /** * Gets call's UUID. * * @return call uuid * @hide */ public UUID getUUID() { return mUUID; } /** * Gets call's current state. * * @return state of this particular phone call. */ @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) public int getState() { return mState; } /** * Gets call's number. * * @return string representing phone number. */ @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) public String getNumber() { return mNumber; } /** * Gets call's creation time in millis since epoch. * * @return long representing the creation time. */ public long getCreationElapsedMilli() { return mCreationElapsedMilli; } /** * Checks if call is an active call in a conference mode (aka multi party). * * @return true if call is a multi party call, false otherwise. */ @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) public boolean isMultiParty() { return mMultiParty; } /** * Checks if this call is an outgoing call. * * @return true if its outgoing call, false otherwise. */ @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) public boolean isOutgoing() { return mOutgoing; } /** * Checks if the ringtone will be generated by the connected phone * * @return true if in band ring is enabled, false otherwise. */ public boolean isInBandRing() { return mInBandRing; } @Override public String toString() { return toString(false); } /** * Generate a log string for this call * * @param loggable whether device address should be logged * @return log string */ public String toString(boolean loggable) { StringBuilder builder = new StringBuilder("BluetoothHeadsetClientCall{mDevice: "); builder.append(loggable ? mDevice : mDevice.hashCode()); builder.append(", mId: "); builder.append(mId); builder.append(", mUUID: "); builder.append(mUUID); builder.append(", mState: "); switch (mState) { case CALL_STATE_ACTIVE: builder.append("ACTIVE"); break; case CALL_STATE_HELD: builder.append("HELD"); break; case CALL_STATE_DIALING: builder.append("DIALING"); break; case CALL_STATE_ALERTING: builder.append("ALERTING"); break; case CALL_STATE_INCOMING: builder.append("INCOMING"); break; case CALL_STATE_WAITING: builder.append("WAITING"); break; case CALL_STATE_HELD_BY_RESPONSE_AND_HOLD: builder.append("HELD_BY_RESPONSE_AND_HOLD"); break; case CALL_STATE_TERMINATED: builder.append("TERMINATED"); break; default: builder.append(mState); break; } builder.append(", mNumber: "); builder.append(loggable ? mNumber : mNumber.hashCode()); builder.append(", mMultiParty: "); builder.append(mMultiParty); builder.append(", mOutgoing: "); builder.append(mOutgoing); builder.append(", mInBandRing: "); builder.append(mInBandRing); builder.append("}"); return builder.toString(); } /** {@link Parcelable.Creator} interface implementation. */ public static final @NonNull Creator CREATOR = new Creator<>() { @Override public BluetoothHeadsetClientCall createFromParcel(Parcel in) { return new BluetoothHeadsetClientCall( (BluetoothDevice) in.readParcelable(null), in.readInt(), UUID.fromString(in.readString()), in.readInt(), in.readString(), in.readInt() == 1, in.readInt() == 1, in.readInt() == 1); } @Override public BluetoothHeadsetClientCall[] newArray(int size) { return new BluetoothHeadsetClientCall[size]; } }; @Override public void writeToParcel(Parcel out, int flags) { out.writeParcelable(mDevice, 0); out.writeInt(mId); out.writeString(mUUID.toString()); out.writeInt(mState); out.writeString(mNumber); out.writeInt(mMultiParty ? 1 : 0); out.writeInt(mOutgoing ? 1 : 0); out.writeInt(mInBandRing ? 1 : 0); } @Override public int describeContents() { return 0; } }