am 972e0c7b: am 5b8b0df7: Merge "Add handler parameter for callbacks." into mnc-dev

* commit '972e0c7b3e5ba30c844e388eb6736fb745b0e3f9':
  Add handler parameter for callbacks.
This commit is contained in:
Andrew Lee
2015-04-30 22:46:33 +00:00
committed by Android Git Automerger
9 changed files with 510 additions and 126 deletions

View File

@ -30257,6 +30257,7 @@ package android.telecom {
method public void playDtmfTone(char); method public void playDtmfTone(char);
method public void postDialContinue(boolean); method public void postDialContinue(boolean);
method public void registerCallback(android.telecom.Call.Callback); method public void registerCallback(android.telecom.Call.Callback);
method public void registerCallback(android.telecom.Call.Callback, android.os.Handler);
method public void reject(boolean, java.lang.String); method public void reject(boolean, java.lang.String);
method public void splitFromConference(); method public void splitFromConference();
method public void stopDtmfTone(); method public void stopDtmfTone();
@ -30564,6 +30565,7 @@ package android.telecom {
public static abstract class InCallService.VideoCall { public static abstract class InCallService.VideoCall {
ctor public InCallService.VideoCall(); ctor public InCallService.VideoCall();
method public abstract void registerCallback(android.telecom.InCallService.VideoCall.Callback); method public abstract void registerCallback(android.telecom.InCallService.VideoCall.Callback);
method public abstract void registerCallback(android.telecom.InCallService.VideoCall.Callback, android.os.Handler);
method public abstract void requestCallDataUsage(); method public abstract void requestCallDataUsage();
method public abstract void requestCameraCapabilities(); method public abstract void requestCameraCapabilities();
method public abstract void sendSessionModifyRequest(android.telecom.VideoProfile); method public abstract void sendSessionModifyRequest(android.telecom.VideoProfile);
@ -30574,7 +30576,7 @@ package android.telecom {
method public abstract void setPauseImage(java.lang.String); method public abstract void setPauseImage(java.lang.String);
method public abstract void setPreviewSurface(android.view.Surface); method public abstract void setPreviewSurface(android.view.Surface);
method public abstract void setZoom(float); method public abstract void setZoom(float);
method public abstract void unregisterCallback(); method public abstract void unregisterCallback(android.telecom.InCallService.VideoCall.Callback);
} }
public static abstract class InCallService.VideoCall.Callback { public static abstract class InCallService.VideoCall.Callback {
@ -30662,6 +30664,7 @@ package android.telecom {
method public void merge(); method public void merge();
method public void playDtmfTone(char); method public void playDtmfTone(char);
method public final void registerCallback(android.telecom.RemoteConference.Callback); method public final void registerCallback(android.telecom.RemoteConference.Callback);
method public final void registerCallback(android.telecom.RemoteConference.Callback, android.os.Handler);
method public void separate(android.telecom.RemoteConnection); method public void separate(android.telecom.RemoteConnection);
method public void setAudioState(android.telecom.AudioState); method public void setAudioState(android.telecom.AudioState);
method public void stopDtmfTone(); method public void stopDtmfTone();
@ -30701,6 +30704,7 @@ package android.telecom {
method public void playDtmfTone(char); method public void playDtmfTone(char);
method public void postDialContinue(boolean); method public void postDialContinue(boolean);
method public void registerCallback(android.telecom.RemoteConnection.Callback); method public void registerCallback(android.telecom.RemoteConnection.Callback);
method public void registerCallback(android.telecom.RemoteConnection.Callback, android.os.Handler);
method public void reject(); method public void reject();
method public void setAudioState(android.telecom.AudioState); method public void setAudioState(android.telecom.AudioState);
method public void stopDtmfTone(); method public void stopDtmfTone();

View File

@ -32371,6 +32371,7 @@ package android.telecom {
method public void playDtmfTone(char); method public void playDtmfTone(char);
method public void postDialContinue(boolean); method public void postDialContinue(boolean);
method public void registerCallback(android.telecom.Call.Callback); method public void registerCallback(android.telecom.Call.Callback);
method public void registerCallback(android.telecom.Call.Callback, android.os.Handler);
method public void reject(boolean, java.lang.String); method public void reject(boolean, java.lang.String);
method public deprecated void removeListener(android.telecom.Call.Listener); method public deprecated void removeListener(android.telecom.Call.Listener);
method public void splitFromConference(); method public void splitFromConference();
@ -32687,6 +32688,7 @@ package android.telecom {
public static abstract class InCallService.VideoCall { public static abstract class InCallService.VideoCall {
ctor public InCallService.VideoCall(); ctor public InCallService.VideoCall();
method public abstract void registerCallback(android.telecom.InCallService.VideoCall.Callback); method public abstract void registerCallback(android.telecom.InCallService.VideoCall.Callback);
method public abstract void registerCallback(android.telecom.InCallService.VideoCall.Callback, android.os.Handler);
method public abstract void requestCallDataUsage(); method public abstract void requestCallDataUsage();
method public abstract void requestCameraCapabilities(); method public abstract void requestCameraCapabilities();
method public abstract void sendSessionModifyRequest(android.telecom.VideoProfile); method public abstract void sendSessionModifyRequest(android.telecom.VideoProfile);
@ -32697,7 +32699,7 @@ package android.telecom {
method public abstract void setPauseImage(java.lang.String); method public abstract void setPauseImage(java.lang.String);
method public abstract void setPreviewSurface(android.view.Surface); method public abstract void setPreviewSurface(android.view.Surface);
method public abstract void setZoom(float); method public abstract void setZoom(float);
method public abstract void unregisterCallback(); method public abstract void unregisterCallback(android.telecom.InCallService.VideoCall.Callback);
} }
public static abstract class InCallService.VideoCall.Callback { public static abstract class InCallService.VideoCall.Callback {
@ -32805,6 +32807,7 @@ package android.telecom {
method public void merge(); method public void merge();
method public void playDtmfTone(char); method public void playDtmfTone(char);
method public final void registerCallback(android.telecom.RemoteConference.Callback); method public final void registerCallback(android.telecom.RemoteConference.Callback);
method public final void registerCallback(android.telecom.RemoteConference.Callback, android.os.Handler);
method public void separate(android.telecom.RemoteConnection); method public void separate(android.telecom.RemoteConnection);
method public void setAudioState(android.telecom.AudioState); method public void setAudioState(android.telecom.AudioState);
method public void stopDtmfTone(); method public void stopDtmfTone();
@ -32844,6 +32847,7 @@ package android.telecom {
method public void playDtmfTone(char); method public void playDtmfTone(char);
method public void postDialContinue(boolean); method public void postDialContinue(boolean);
method public void registerCallback(android.telecom.RemoteConnection.Callback); method public void registerCallback(android.telecom.RemoteConnection.Callback);
method public void registerCallback(android.telecom.RemoteConnection.Callback, android.os.Handler);
method public void reject(); method public void reject();
method public void setAudioState(android.telecom.AudioState); method public void setAudioState(android.telecom.AudioState);
method public void stopDtmfTone(); method public void stopDtmfTone();

View File

@ -19,10 +19,12 @@ package android.telecom;
import android.annotation.SystemApi; import android.annotation.SystemApi;
import android.net.Uri; import android.net.Uri;
import android.os.Bundle; import android.os.Bundle;
import android.os.Handler;
import java.lang.String; import java.lang.String;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Collections; import java.util.Collections;
import java.util.LinkedList;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.Objects; import java.util.Objects;
@ -608,7 +610,7 @@ public final class Call {
private final List<String> mChildrenIds = new ArrayList<>(); private final List<String> mChildrenIds = new ArrayList<>();
private final List<Call> mChildren = new ArrayList<>(); private final List<Call> mChildren = new ArrayList<>();
private final List<Call> mUnmodifiableChildren = Collections.unmodifiableList(mChildren); private final List<Call> mUnmodifiableChildren = Collections.unmodifiableList(mChildren);
private final List<Callback> mCallbacks = new CopyOnWriteArrayList<>(); private final List<CallbackRecord<Callback>> mCallbackRecords = new CopyOnWriteArrayList<>();
private final List<Call> mConferenceableCalls = new ArrayList<>(); private final List<Call> mConferenceableCalls = new ArrayList<>();
private final List<Call> mUnmodifiableConferenceableCalls = private final List<Call> mUnmodifiableConferenceableCalls =
Collections.unmodifiableList(mConferenceableCalls); Collections.unmodifiableList(mConferenceableCalls);
@ -850,7 +852,20 @@ public final class Call {
* @param callback A {@code Callback}. * @param callback A {@code Callback}.
*/ */
public void registerCallback(Callback callback) { public void registerCallback(Callback callback) {
mCallbacks.add(callback); registerCallback(callback, new Handler());
}
/**
* Registers a callback to this {@code Call}.
*
* @param callback A {@code Callback}.
* @param handler A handler which command and status changes will be delivered to.
*/
public void registerCallback(Callback callback, Handler handler) {
unregisterCallback(callback);
if (callback != null && handler != null) {
mCallbackRecords.add(new CallbackRecord<Callback>(callback, handler));
}
} }
/** /**
@ -860,7 +875,12 @@ public final class Call {
*/ */
public void unregisterCallback(Callback callback) { public void unregisterCallback(Callback callback) {
if (callback != null) { if (callback != null) {
mCallbacks.remove(callback); for (CallbackRecord<Callback> record : mCallbackRecords) {
if (record.getCallback() == callback) {
mCallbackRecords.remove(record);
break;
}
}
} }
} }
@ -1021,57 +1041,120 @@ public final class Call {
} }
} }
private void fireStateChanged(int newState) { private void fireStateChanged(final int newState) {
for (Callback callback : mCallbacks) { for (CallbackRecord<Callback> record : mCallbackRecords) {
callback.onStateChanged(this, newState); final Call call = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onStateChanged(call, newState);
}
});
} }
} }
private void fireParentChanged(Call newParent) { private void fireParentChanged(final Call newParent) {
for (Callback callback : mCallbacks) { for (CallbackRecord<Callback> record : mCallbackRecords) {
callback.onParentChanged(this, newParent); final Call call = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onParentChanged(call, newParent);
}
});
} }
} }
private void fireChildrenChanged(List<Call> children) { private void fireChildrenChanged(final List<Call> children) {
for (Callback callback : mCallbacks) { for (CallbackRecord<Callback> record : mCallbackRecords) {
callback.onChildrenChanged(this, children); final Call call = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onChildrenChanged(call, children);
}
});
} }
} }
private void fireDetailsChanged(Details details) { private void fireDetailsChanged(final Details details) {
for (Callback callback : mCallbacks) { for (CallbackRecord<Callback> record : mCallbackRecords) {
callback.onDetailsChanged(this, details); final Call call = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onDetailsChanged(call, details);
}
});
} }
} }
private void fireCannedTextResponsesLoaded(List<String> cannedTextResponses) { private void fireCannedTextResponsesLoaded(final List<String> cannedTextResponses) {
for (Callback callback : mCallbacks) { for (CallbackRecord<Callback> record : mCallbackRecords) {
callback.onCannedTextResponsesLoaded(this, cannedTextResponses); final Call call = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onCannedTextResponsesLoaded(call, cannedTextResponses);
}
});
} }
} }
private void fireVideoCallChanged(InCallService.VideoCall videoCall) { private void fireVideoCallChanged(final InCallService.VideoCall videoCall) {
for (Callback callback : mCallbacks) { for (CallbackRecord<Callback> record : mCallbackRecords) {
callback.onVideoCallChanged(this, videoCall); final Call call = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onVideoCallChanged(call, videoCall);
}
});
} }
} }
private void firePostDialWait(String remainingPostDialSequence) { private void firePostDialWait(final String remainingPostDialSequence) {
for (Callback callback : mCallbacks) { for (CallbackRecord<Callback> record : mCallbackRecords) {
callback.onPostDialWait(this, remainingPostDialSequence); final Call call = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onPostDialWait(call, remainingPostDialSequence);
}
});
} }
} }
private void fireCallDestroyed() { private void fireCallDestroyed() {
for (Callback callback : mCallbacks) { for (CallbackRecord<Callback> record: mCallbackRecords) {
callback.onCallDestroyed(this); final Call call = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onCallDestroyed(call);
}
});
} }
} }
private void fireConferenceableCallsChanged() { private void fireConferenceableCallsChanged() {
for (Callback callback : mCallbacks) { for (CallbackRecord<Callback> record : mCallbackRecords) {
callback.onConferenceableCallsChanged(this, mUnmodifiableConferenceableCalls); final Call call = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onConferenceableCallsChanged(call, mUnmodifiableConferenceableCalls);
}
});
} }
} }
} }

View File

@ -0,0 +1,44 @@
/*
* Copyright (C) 2015 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.telecom;
import android.os.Handler;
/**
* This class is used to associate a generic callback of type T with a handler to which commands and
* status updates will be delivered to.
*
* @hide
*/
class CallbackRecord<T> {
private final T mCallback;
private final Handler mHandler;
public CallbackRecord(T callback, Handler handler) {
mCallback = callback;
mHandler = handler;
}
public T getCallback() {
return mCallback;
}
public Handler getHandler() {
return mHandler;
}
}

View File

@ -362,6 +362,9 @@ public abstract class InCallService extends Service {
*/ */
public static abstract class VideoCall { public static abstract class VideoCall {
/** @hide */
public abstract void destroy();
/** /**
* Registers a callback to receive commands and state changes for video calls. * Registers a callback to receive commands and state changes for video calls.
* *
@ -369,10 +372,18 @@ public abstract class InCallService extends Service {
*/ */
public abstract void registerCallback(VideoCall.Callback callback); public abstract void registerCallback(VideoCall.Callback callback);
/**
* Registers a callback to receive commands and state changes for video calls.
*
* @param callback The video call callback.
* @param handler A handler which commands and status changes will be delivered to.
*/
public abstract void registerCallback(VideoCall.Callback callback, Handler handler);
/** /**
* Clears the video call listener set via {@link #registerCallback}. * Clears the video call listener set via {@link #registerCallback}.
*/ */
public abstract void unregisterCallback(); public abstract void unregisterCallback(VideoCall.Callback callback);
/** /**
* Sets the camera to be used for video recording in a video call. * Sets the camera to be used for video recording in a video call.

View File

@ -125,7 +125,7 @@ public final class Phone {
InCallService.VideoCall videoCall = call.getVideoCall(); InCallService.VideoCall videoCall = call.getVideoCall();
if (videoCall != null) { if (videoCall != null) {
videoCall.unregisterCallback(); videoCall.destroy();
} }
fireCallRemoved(call); fireCallRemoved(call);
} }
@ -174,7 +174,7 @@ public final class Phone {
for (Call call : mCalls) { for (Call call : mCalls) {
InCallService.VideoCall videoCall = call.getVideoCall(); InCallService.VideoCall videoCall = call.getVideoCall();
if (videoCall != null) { if (videoCall != null) {
videoCall.unregisterCallback(); videoCall.destroy();
} }
if (call.getState() != Call.STATE_DISCONNECTED) { if (call.getState() != Call.STATE_DISCONNECTED) {
call.internalSetDisconnected(); call.internalSetDisconnected();

View File

@ -18,6 +18,7 @@ package android.telecom;
import com.android.internal.telecom.IConnectionService; import com.android.internal.telecom.IConnectionService;
import android.os.Handler;
import android.os.RemoteException; import android.os.RemoteException;
import java.util.ArrayList; import java.util.ArrayList;
@ -49,7 +50,7 @@ public final class RemoteConference {
private final String mId; private final String mId;
private final IConnectionService mConnectionService; private final IConnectionService mConnectionService;
private final Set<Callback> mCallbacks = new CopyOnWriteArraySet<>(); private final Set<CallbackRecord<Callback>> mCallbackRecords = new CopyOnWriteArraySet<>();
private final List<RemoteConnection> mChildConnections = new CopyOnWriteArrayList<>(); private final List<RemoteConnection> mChildConnections = new CopyOnWriteArrayList<>();
private final List<RemoteConnection> mUnmodifiableChildConnections = private final List<RemoteConnection> mUnmodifiableChildConnections =
Collections.unmodifiableList(mChildConnections); Collections.unmodifiableList(mChildConnections);
@ -77,13 +78,20 @@ public final class RemoteConference {
for (RemoteConnection connection : mChildConnections) { for (RemoteConnection connection : mChildConnections) {
connection.setConference(null); connection.setConference(null);
} }
for (Callback c : mCallbacks) { for (CallbackRecord<Callback> record : mCallbackRecords) {
c.onDestroyed(this); final RemoteConference conference = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onDestroyed(conference);
}
});
} }
} }
/** {@hide} */ /** {@hide} */
void setState(int newState) { void setState(final int newState) {
if (newState != Connection.STATE_ACTIVE && if (newState != Connection.STATE_ACTIVE &&
newState != Connection.STATE_HOLDING && newState != Connection.STATE_HOLDING &&
newState != Connection.STATE_DISCONNECTED) { newState != Connection.STATE_DISCONNECTED) {
@ -93,42 +101,71 @@ public final class RemoteConference {
} }
if (mState != newState) { if (mState != newState) {
int oldState = mState; final int oldState = mState;
mState = newState; mState = newState;
for (Callback c : mCallbacks) { for (CallbackRecord<Callback> record : mCallbackRecords) {
c.onStateChanged(this, oldState, newState); final RemoteConference conference = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onStateChanged(conference, oldState, newState);
}
});
} }
} }
} }
/** {@hide} */ /** {@hide} */
void addConnection(RemoteConnection connection) { void addConnection(final RemoteConnection connection) {
if (!mChildConnections.contains(connection)) { if (!mChildConnections.contains(connection)) {
mChildConnections.add(connection); mChildConnections.add(connection);
connection.setConference(this); connection.setConference(this);
for (Callback c : mCallbacks) { for (CallbackRecord<Callback> record : mCallbackRecords) {
c.onConnectionAdded(this, connection); final RemoteConference conference = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onConnectionAdded(conference, connection);
}
});
} }
} }
} }
/** {@hide} */ /** {@hide} */
void removeConnection(RemoteConnection connection) { void removeConnection(final RemoteConnection connection) {
if (mChildConnections.contains(connection)) { if (mChildConnections.contains(connection)) {
mChildConnections.remove(connection); mChildConnections.remove(connection);
connection.setConference(null); connection.setConference(null);
for (Callback c : mCallbacks) { for (CallbackRecord<Callback> record : mCallbackRecords) {
c.onConnectionRemoved(this, connection); final RemoteConference conference = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onConnectionRemoved(conference, connection);
}
});
} }
} }
} }
/** {@hide} */ /** {@hide} */
void setConnectionCapabilities(int connectionCapabilities) { void setConnectionCapabilities(final int connectionCapabilities) {
if (mConnectionCapabilities != connectionCapabilities) { if (mConnectionCapabilities != connectionCapabilities) {
mConnectionCapabilities = connectionCapabilities; mConnectionCapabilities = connectionCapabilities;
for (Callback c : mCallbacks) { for (CallbackRecord<Callback> record : mCallbackRecords) {
c.onConnectionCapabilitiesChanged(this, mConnectionCapabilities); final RemoteConference conference = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onConnectionCapabilitiesChanged(
conference, mConnectionCapabilities);
}
});
} }
} }
} }
@ -137,18 +174,33 @@ public final class RemoteConference {
void setConferenceableConnections(List<RemoteConnection> conferenceableConnections) { void setConferenceableConnections(List<RemoteConnection> conferenceableConnections) {
mConferenceableConnections.clear(); mConferenceableConnections.clear();
mConferenceableConnections.addAll(conferenceableConnections); mConferenceableConnections.addAll(conferenceableConnections);
for (Callback c : mCallbacks) { for (CallbackRecord<Callback> record : mCallbackRecords) {
c.onConferenceableConnectionsChanged(this, mUnmodifiableConferenceableConnections); final RemoteConference conference = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onConferenceableConnectionsChanged(
conference, mUnmodifiableConferenceableConnections);
}
});
} }
} }
/** {@hide} */ /** {@hide} */
void setDisconnected(DisconnectCause disconnectCause) { void setDisconnected(final DisconnectCause disconnectCause) {
if (mState != Connection.STATE_DISCONNECTED) { if (mState != Connection.STATE_DISCONNECTED) {
mDisconnectCause = disconnectCause; mDisconnectCause = disconnectCause;
setState(Connection.STATE_DISCONNECTED); setState(Connection.STATE_DISCONNECTED);
for (Callback c : mCallbacks) { for (CallbackRecord<Callback> record : mCallbackRecords) {
c.onDisconnected(this, disconnectCause); final RemoteConference conference = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onDisconnected(conference, disconnectCause);
}
});
} }
} }
} }
@ -239,10 +291,24 @@ public final class RemoteConference {
} }
public final void registerCallback(Callback callback) { public final void registerCallback(Callback callback) {
mCallbacks.add(callback); registerCallback(callback, new Handler());
}
public final void registerCallback(Callback callback, Handler handler) {
unregisterCallback(callback);
if (callback != null && handler != null) {
mCallbackRecords.add(new CallbackRecord(callback, handler));
}
} }
public final void unregisterCallback(Callback callback) { public final void unregisterCallback(Callback callback) {
mCallbacks.remove(callback); if (callback != null) {
for (CallbackRecord<Callback> record : mCallbackRecords) {
if (record.getCallback() == callback) {
mCallbackRecords.remove(record);
break;
}
}
}
} }
} }

View File

@ -21,6 +21,7 @@ import com.android.internal.telecom.IVideoCallback;
import com.android.internal.telecom.IVideoProvider; import com.android.internal.telecom.IVideoProvider;
import android.net.Uri; import android.net.Uri;
import android.os.Handler;
import android.os.IBinder; import android.os.IBinder;
import android.os.RemoteException; import android.os.RemoteException;
import android.view.Surface; import android.view.Surface;
@ -392,8 +393,8 @@ public final class RemoteConnection {
* load factor before resizing, 1 means we only expect a single thread to * load factor before resizing, 1 means we only expect a single thread to
* access the map so make only a single shard * access the map so make only a single shard
*/ */
private final Set<Callback> mCallbacks = Collections.newSetFromMap( private final Set<CallbackRecord> mCallbackRecords = Collections.newSetFromMap(
new ConcurrentHashMap<Callback, Boolean>(8, 0.9f, 1)); new ConcurrentHashMap<CallbackRecord, Boolean>(8, 0.9f, 1));
private final List<RemoteConnection> mConferenceableConnections = new ArrayList<>(); private final List<RemoteConnection> mConferenceableConnections = new ArrayList<>();
private final List<RemoteConnection> mUnmodifiableconferenceableConnections = private final List<RemoteConnection> mUnmodifiableconferenceableConnections =
Collections.unmodifiableList(mConferenceableConnections); Collections.unmodifiableList(mConferenceableConnections);
@ -470,7 +471,20 @@ public final class RemoteConnection {
* @param callback A {@code Callback}. * @param callback A {@code Callback}.
*/ */
public void registerCallback(Callback callback) { public void registerCallback(Callback callback) {
mCallbacks.add(callback); registerCallback(callback, new Handler());
}
/**
* Adds a callback to this {@code RemoteConnection}.
*
* @param callback A {@code Callback}.
* @param handler A {@code Handler} which command and status changes will be delivered to.
*/
public void registerCallback(Callback callback, Handler handler) {
unregisterCallback(callback);
if (callback != null && handler != null) {
mCallbackRecords.add(new CallbackRecord(callback, handler));
}
} }
/** /**
@ -480,7 +494,12 @@ public final class RemoteConnection {
*/ */
public void unregisterCallback(Callback callback) { public void unregisterCallback(Callback callback) {
if (callback != null) { if (callback != null) {
mCallbacks.remove(callback); for (CallbackRecord record : mCallbackRecords) {
if (record.getCallback() == callback) {
mCallbackRecords.remove(record);
break;
}
}
} }
} }
@ -800,11 +819,18 @@ public final class RemoteConnection {
/** /**
* @hide * @hide
*/ */
void setState(int state) { void setState(final int state) {
if (mState != state) { if (mState != state) {
mState = state; mState = state;
for (Callback c: mCallbacks) { for (CallbackRecord record: mCallbackRecords) {
c.onStateChanged(this, state); final RemoteConnection connection = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onStateChanged(connection, state);
}
});
} }
} }
} }
@ -812,13 +838,20 @@ public final class RemoteConnection {
/** /**
* @hide * @hide
*/ */
void setDisconnected(DisconnectCause disconnectCause) { void setDisconnected(final DisconnectCause disconnectCause) {
if (mState != Connection.STATE_DISCONNECTED) { if (mState != Connection.STATE_DISCONNECTED) {
mState = Connection.STATE_DISCONNECTED; mState = Connection.STATE_DISCONNECTED;
mDisconnectCause = disconnectCause; mDisconnectCause = disconnectCause;
for (Callback c : mCallbacks) { for (CallbackRecord record : mCallbackRecords) {
c.onDisconnected(this, mDisconnectCause); final RemoteConnection connection = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onDisconnected(connection, disconnectCause);
}
});
} }
} }
} }
@ -826,11 +859,18 @@ public final class RemoteConnection {
/** /**
* @hide * @hide
*/ */
void setRingbackRequested(boolean ringback) { void setRingbackRequested(final boolean ringback) {
if (mRingbackRequested != ringback) { if (mRingbackRequested != ringback) {
mRingbackRequested = ringback; mRingbackRequested = ringback;
for (Callback c : mCallbacks) { for (CallbackRecord record : mCallbackRecords) {
c.onRingbackRequested(this, ringback); final RemoteConnection connection = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onRingbackRequested(connection, ringback);
}
});
} }
} }
} }
@ -838,10 +878,17 @@ public final class RemoteConnection {
/** /**
* @hide * @hide
*/ */
void setConnectionCapabilities(int connectionCapabilities) { void setConnectionCapabilities(final int connectionCapabilities) {
mConnectionCapabilities = connectionCapabilities; mConnectionCapabilities = connectionCapabilities;
for (Callback c : mCallbacks) { for (CallbackRecord record : mCallbackRecords) {
c.onConnectionCapabilitiesChanged(this, connectionCapabilities); final RemoteConnection connection = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onConnectionCapabilitiesChanged(connection, connectionCapabilities);
}
});
} }
} }
@ -849,17 +896,24 @@ public final class RemoteConnection {
* @hide * @hide
*/ */
void setDestroyed() { void setDestroyed() {
if (!mCallbacks.isEmpty()) { if (!mCallbackRecords.isEmpty()) {
// Make sure that the callbacks are notified that the call is destroyed first. // Make sure that the callbacks are notified that the call is destroyed first.
if (mState != Connection.STATE_DISCONNECTED) { if (mState != Connection.STATE_DISCONNECTED) {
setDisconnected( setDisconnected(
new DisconnectCause(DisconnectCause.ERROR, "Connection destroyed.")); new DisconnectCause(DisconnectCause.ERROR, "Connection destroyed."));
} }
for (Callback c : mCallbacks) { for (CallbackRecord record : mCallbackRecords) {
c.onDestroyed(this); final RemoteConnection connection = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onDestroyed(connection);
}
});
} }
mCallbacks.clear(); mCallbackRecords.clear();
mConnected = false; mConnected = false;
} }
@ -868,90 +922,162 @@ public final class RemoteConnection {
/** /**
* @hide * @hide
*/ */
void setPostDialWait(String remainingDigits) { void setPostDialWait(final String remainingDigits) {
for (Callback c : mCallbacks) { for (CallbackRecord record : mCallbackRecords) {
c.onPostDialWait(this, remainingDigits); final RemoteConnection connection = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onPostDialWait(connection, remainingDigits);
}
});
} }
} }
/** /**
* @hide * @hide
*/ */
void onPostDialChar(char nextChar) { void onPostDialChar(final char nextChar) {
for (Callback c : mCallbacks) { for (CallbackRecord record : mCallbackRecords) {
c.onPostDialChar(this, nextChar); final RemoteConnection connection = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onPostDialWait(connection, String.valueOf(nextChar));
}
});
} }
} }
/** /**
* @hide * @hide
*/ */
void setVideoState(int videoState) { void setVideoState(final int videoState) {
mVideoState = videoState; mVideoState = videoState;
for (Callback c : mCallbacks) { for (CallbackRecord record : mCallbackRecords) {
c.onVideoStateChanged(this, videoState); final RemoteConnection connection = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onVideoStateChanged(connection, videoState);
}
});
} }
} }
/** /**
* @hide * @hide
*/ */
void setVideoProvider(VideoProvider videoProvider) { void setVideoProvider(final VideoProvider videoProvider) {
mVideoProvider = videoProvider; mVideoProvider = videoProvider;
for (Callback c : mCallbacks) { for (CallbackRecord record : mCallbackRecords) {
c.onVideoProviderChanged(this, videoProvider); final RemoteConnection connection = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onVideoProviderChanged(connection, videoProvider);
}
});
} }
} }
/** @hide */ /** @hide */
void setIsVoipAudioMode(boolean isVoip) { void setIsVoipAudioMode(final boolean isVoip) {
mIsVoipAudioMode = isVoip; mIsVoipAudioMode = isVoip;
for (Callback c : mCallbacks) { for (CallbackRecord record : mCallbackRecords) {
c.onVoipAudioChanged(this, isVoip); final RemoteConnection connection = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onVoipAudioChanged(connection, isVoip);
}
});
} }
} }
/** @hide */ /** @hide */
void setStatusHints(StatusHints statusHints) { void setStatusHints(final StatusHints statusHints) {
mStatusHints = statusHints; mStatusHints = statusHints;
for (Callback c : mCallbacks) { for (CallbackRecord record : mCallbackRecords) {
c.onStatusHintsChanged(this, statusHints); final RemoteConnection connection = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onStatusHintsChanged(connection, statusHints);
}
});
} }
} }
/** @hide */ /** @hide */
void setAddress(Uri address, int presentation) { void setAddress(final Uri address, final int presentation) {
mAddress = address; mAddress = address;
mAddressPresentation = presentation; mAddressPresentation = presentation;
for (Callback c : mCallbacks) { for (CallbackRecord record : mCallbackRecords) {
c.onAddressChanged(this, address, presentation); final RemoteConnection connection = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onAddressChanged(connection, address, presentation);
}
});
} }
} }
/** @hide */ /** @hide */
void setCallerDisplayName(String callerDisplayName, int presentation) { void setCallerDisplayName(final String callerDisplayName, final int presentation) {
mCallerDisplayName = callerDisplayName; mCallerDisplayName = callerDisplayName;
mCallerDisplayNamePresentation = presentation; mCallerDisplayNamePresentation = presentation;
for (Callback c : mCallbacks) { for (CallbackRecord record : mCallbackRecords) {
c.onCallerDisplayNameChanged(this, callerDisplayName, presentation); final RemoteConnection connection = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onCallerDisplayNameChanged(
connection, callerDisplayName, presentation);
}
});
} }
} }
/** @hide */ /** @hide */
void setConferenceableConnections(List<RemoteConnection> conferenceableConnections) { void setConferenceableConnections(final List<RemoteConnection> conferenceableConnections) {
mConferenceableConnections.clear(); mConferenceableConnections.clear();
mConferenceableConnections.addAll(conferenceableConnections); mConferenceableConnections.addAll(conferenceableConnections);
for (Callback c : mCallbacks) { for (CallbackRecord record : mCallbackRecords) {
c.onConferenceableConnectionsChanged(this, mUnmodifiableconferenceableConnections); final RemoteConnection connection = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onConferenceableConnectionsChanged(
connection, mUnmodifiableconferenceableConnections);
}
});
} }
} }
/** @hide */ /** @hide */
void setConference(RemoteConference conference) { void setConference(final RemoteConference conference) {
if (mConference != conference) { if (mConference != conference) {
mConference = conference; mConference = conference;
for (Callback c : mCallbacks) { for (CallbackRecord record : mCallbackRecords) {
c.onConferenceChanged(this, conference); final RemoteConnection connection = this;
final Callback callback = record.getCallback();
record.getHandler().post(new Runnable() {
@Override
public void run() {
callback.onConferenceChanged(connection, conference);
}
});
} }
} }
} }
@ -968,4 +1094,22 @@ public final class RemoteConnection {
public static RemoteConnection failure(DisconnectCause disconnectCause) { public static RemoteConnection failure(DisconnectCause disconnectCause) {
return new RemoteConnection(disconnectCause); return new RemoteConnection(disconnectCause);
} }
private static final class CallbackRecord extends Callback {
private final Callback mCallback;
private final Handler mHandler;
public CallbackRecord(Callback callback, Handler handler) {
mCallback = callback;
mHandler = handler;
}
public Callback getCallback() {
return mCallback;
}
public Handler getHandler() {
return mHandler;
}
}
} }

View File

@ -36,13 +36,6 @@ import com.android.internal.telecom.IVideoProvider;
* {@hide} * {@hide}
*/ */
public class VideoCallImpl extends VideoCall { public class VideoCallImpl extends VideoCall {
private static final int MSG_RECEIVE_SESSION_MODIFY_REQUEST = 1;
private static final int MSG_RECEIVE_SESSION_MODIFY_RESPONSE = 2;
private static final int MSG_HANDLE_CALL_SESSION_EVENT = 3;
private static final int MSG_CHANGE_PEER_DIMENSIONS = 4;
private static final int MSG_CHANGE_CALL_DATA_USAGE = 5;
private static final int MSG_CHANGE_CAMERA_CAPABILITIES = 6;
private static final int MSG_CHANGE_VIDEO_QUALITY = 7;
private final IVideoProvider mVideoProvider; private final IVideoProvider mVideoProvider;
private final VideoCallListenerBinder mBinder; private final VideoCallListenerBinder mBinder;
@ -61,7 +54,7 @@ public class VideoCallImpl extends VideoCall {
private final class VideoCallListenerBinder extends IVideoCallback.Stub { private final class VideoCallListenerBinder extends IVideoCallback.Stub {
@Override @Override
public void receiveSessionModifyRequest(VideoProfile videoProfile) { public void receiveSessionModifyRequest(VideoProfile videoProfile) {
mHandler.obtainMessage(MSG_RECEIVE_SESSION_MODIFY_REQUEST, mHandler.obtainMessage(MessageHandler.MSG_RECEIVE_SESSION_MODIFY_REQUEST,
videoProfile).sendToTarget(); videoProfile).sendToTarget();
} }
@ -72,12 +65,14 @@ public class VideoCallImpl extends VideoCall {
args.arg1 = status; args.arg1 = status;
args.arg2 = requestProfile; args.arg2 = requestProfile;
args.arg3 = responseProfile; args.arg3 = responseProfile;
mHandler.obtainMessage(MSG_RECEIVE_SESSION_MODIFY_RESPONSE, args).sendToTarget(); mHandler.obtainMessage(MessageHandler.MSG_RECEIVE_SESSION_MODIFY_RESPONSE, args)
.sendToTarget();
} }
@Override @Override
public void handleCallSessionEvent(int event) { public void handleCallSessionEvent(int event) {
mHandler.obtainMessage(MSG_HANDLE_CALL_SESSION_EVENT, event).sendToTarget(); mHandler.obtainMessage(MessageHandler.MSG_HANDLE_CALL_SESSION_EVENT, event)
.sendToTarget();
} }
@Override @Override
@ -85,28 +80,42 @@ public class VideoCallImpl extends VideoCall {
SomeArgs args = SomeArgs.obtain(); SomeArgs args = SomeArgs.obtain();
args.arg1 = width; args.arg1 = width;
args.arg2 = height; args.arg2 = height;
mHandler.obtainMessage(MSG_CHANGE_PEER_DIMENSIONS, args).sendToTarget(); mHandler.obtainMessage(MessageHandler.MSG_CHANGE_PEER_DIMENSIONS, args).sendToTarget();
} }
@Override @Override
public void changeVideoQuality(int videoQuality) { public void changeVideoQuality(int videoQuality) {
mHandler.obtainMessage(MSG_CHANGE_VIDEO_QUALITY, videoQuality, 0).sendToTarget(); mHandler.obtainMessage(MessageHandler.MSG_CHANGE_VIDEO_QUALITY, videoQuality, 0)
.sendToTarget();
} }
@Override @Override
public void changeCallDataUsage(long dataUsage) { public void changeCallDataUsage(long dataUsage) {
mHandler.obtainMessage(MSG_CHANGE_CALL_DATA_USAGE, dataUsage).sendToTarget(); mHandler.obtainMessage(MessageHandler.MSG_CHANGE_CALL_DATA_USAGE, dataUsage)
.sendToTarget();
} }
@Override @Override
public void changeCameraCapabilities(CameraCapabilities cameraCapabilities) { public void changeCameraCapabilities(CameraCapabilities cameraCapabilities) {
mHandler.obtainMessage(MSG_CHANGE_CAMERA_CAPABILITIES, mHandler.obtainMessage(MessageHandler.MSG_CHANGE_CAMERA_CAPABILITIES,
cameraCapabilities).sendToTarget(); cameraCapabilities).sendToTarget();
} }
} }
/** Default handler used to consolidate binder method calls onto a single thread. */ /** Default handler used to consolidate binder method calls onto a single thread. */
private final Handler mHandler = new Handler(Looper.getMainLooper()) { private final class MessageHandler extends Handler {
private static final int MSG_RECEIVE_SESSION_MODIFY_REQUEST = 1;
private static final int MSG_RECEIVE_SESSION_MODIFY_RESPONSE = 2;
private static final int MSG_HANDLE_CALL_SESSION_EVENT = 3;
private static final int MSG_CHANGE_PEER_DIMENSIONS = 4;
private static final int MSG_CHANGE_CALL_DATA_USAGE = 5;
private static final int MSG_CHANGE_CAMERA_CAPABILITIES = 6;
private static final int MSG_CHANGE_VIDEO_QUALITY = 7;
public MessageHandler(Looper looper) {
super(looper);
}
@Override @Override
public void handleMessage(Message msg) { public void handleMessage(Message msg) {
if (mCallback == null) { if (mCallback == null) {
@ -160,7 +169,8 @@ public class VideoCallImpl extends VideoCall {
} }
}; };
/** {@hide} */ private Handler mHandler;
VideoCallImpl(IVideoProvider videoProvider) throws RemoteException { VideoCallImpl(IVideoProvider videoProvider) throws RemoteException {
mVideoProvider = videoProvider; mVideoProvider = videoProvider;
mVideoProvider.asBinder().linkToDeath(mDeathRecipient, 0); mVideoProvider.asBinder().linkToDeath(mDeathRecipient, 0);
@ -169,13 +179,31 @@ public class VideoCallImpl extends VideoCall {
mVideoProvider.addVideoCallback(mBinder); mVideoProvider.addVideoCallback(mBinder);
} }
/** {@inheritDoc} */ public void destroy() {
public void registerCallback(VideoCall.Callback callback) { unregisterCallback(mCallback);
mCallback = callback;
} }
/** {@inheritDoc} */ /** {@inheritDoc} */
public void unregisterCallback() { public void registerCallback(VideoCall.Callback callback) {
registerCallback(callback, null);
}
/** {@inheritDoc} */
public void registerCallback(VideoCall.Callback callback, Handler handler) {
mCallback = callback;
if (handler == null) {
mHandler = new MessageHandler(Looper.getMainLooper());
} else {
mHandler = new MessageHandler(handler.getLooper());
}
}
/** {@inheritDoc} */
public void unregisterCallback(VideoCall.Callback callback) {
if (callback != mCallback) {
return;
}
mCallback = null; mCallback = null;
try { try {
mVideoProvider.removeVideoCallback(mBinder); mVideoProvider.removeVideoCallback(mBinder);