| /* | 
 |  * Copyright (C) 2016 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.Manifest; | 
 | import android.annotation.IntDef; | 
 | import android.annotation.NonNull; | 
 | import android.annotation.RequiresPermission; | 
 | import android.annotation.SdkConstant; | 
 | import android.annotation.SystemApi; | 
 | import android.app.Service; | 
 | import android.content.ComponentName; | 
 | import android.content.Intent; | 
 | import android.content.pm.ServiceInfo; | 
 | import android.net.Uri; | 
 | import android.os.Handler; | 
 | import android.os.IBinder; | 
 | import android.os.Looper; | 
 | import android.os.Message; | 
 | import android.os.Parcel; | 
 | import android.os.Parcelable; | 
 | import android.os.RemoteException; | 
 |  | 
 | import com.android.internal.os.SomeArgs; | 
 | import com.android.internal.telecom.ICallScreeningAdapter; | 
 | import com.android.internal.telecom.ICallScreeningService; | 
 |  | 
 | import java.lang.annotation.Retention; | 
 | import java.lang.annotation.RetentionPolicy; | 
 | import java.util.Objects; | 
 |  | 
 | /** | 
 |  * This service can be implemented by the default dialer (see | 
 |  * {@link TelecomManager#getDefaultDialerPackage()}) or a third party app to allow or disallow | 
 |  * incoming calls before they are shown to a user. A {@link CallScreeningService} can also see | 
 |  * outgoing calls for the purpose of providing caller ID services for those calls. | 
 |  * <p> | 
 |  * Below is an example manifest registration for a {@code CallScreeningService}. | 
 |  * <pre> | 
 |  * {@code | 
 |  * <service android:name="your.package.YourCallScreeningServiceImplementation" | 
 |  *          android:permission="android.permission.BIND_SCREENING_SERVICE"> | 
 |  *      <intent-filter> | 
 |  *          <action android:name="android.telecom.CallScreeningService"/> | 
 |  *      </intent-filter> | 
 |  * </service> | 
 |  * } | 
 |  * </pre> | 
 |  * <p> | 
 |  * A CallScreeningService performs two functions: | 
 |  * <ol> | 
 |  *     <li>Call blocking/screening - the service can choose which calls will ring on the user's | 
 |  *     device, and which will be silently sent to voicemail.</li> | 
 |  *     <li>Call identification - services which provide call identification functionality can | 
 |  *     display a user-interface of their choosing which contains identifying information for a call. | 
 |  *     </li> | 
 |  * </ol> | 
 |  * <p> | 
 |  * <h2>Becoming the CallScreeningService</h2> | 
 |  * Telecom will bind to a single app chosen by the user which implements the | 
 |  * {@link CallScreeningService} API when there are new incoming and outgoing calls. | 
 |  * <p> | 
 |  * The code snippet below illustrates how your app can request that it fills the call screening | 
 |  * role. | 
 |  * <pre> | 
 |  * {@code | 
 |  * private static final int REQUEST_ID = 1; | 
 |  * | 
 |  * public void requestRole() { | 
 |  *     RoleManager roleManager = (RoleManager) getSystemService(ROLE_SERVICE); | 
 |  *     Intent intent = roleManager.createRequestRoleIntent(RoleManager.ROLE_CALL_SCREENING); | 
 |  *     startActivityForResult(intent, REQUEST_ID); | 
 |  * } | 
 |  * | 
 |  * @Override | 
 |  * public void onActivityResult(int requestCode, int resultCode, Intent data) { | 
 |  *     if (requestCode == REQUEST_ID) { | 
 |  *         if (resultCode == android.app.Activity.RESULT_OK) { | 
 |  *             // Your app is now the call screening app | 
 |  *         } else { | 
 |  *             // Your app is not the call screening app | 
 |  *         } | 
 |  *     } | 
 |  * } | 
 |  * } | 
 |  * </pre> | 
 |  * | 
 |  * <h2>CallScreeningService Lifecycle</h2> | 
 |  * | 
 |  * The framework binds to the {@link CallScreeningService} implemented by the user-chosen app | 
 |  * filling the {@link android.app.role.RoleManager#ROLE_CALL_SCREENING} role when incoming calls are | 
 |  * received (prior to ringing) and when outgoing calls are placed.  The platform calls the | 
 |  * {@link #onScreenCall(Call.Details)} method to provide your service with details about the call. | 
 |  * <p> | 
 |  * For incoming calls, the {@link CallScreeningService} must call | 
 |  * {@link #respondToCall(Call.Details, CallResponse)} within 5 seconds of being bound to indicate to | 
 |  * the platform whether the call should be blocked or not.  Your app must do this even if it is | 
 |  * primarily performing caller ID operations and not screening calls.  It is important to perform | 
 |  * screening operations in a timely matter as the user's device will not begin ringing until the | 
 |  * response is received (or the timeout is hit).  A {@link CallScreeningService} may choose to | 
 |  * perform local database lookups to help determine if a call should be screened or not; care should | 
 |  * be taken to ensure the timeout is not repeatedly hit, causing delays in the incoming call flow. | 
 |  * <p> | 
 |  * If your app provides a caller ID experience, it should launch an activity to show the caller ID | 
 |  * information from {@link #onScreenCall(Call.Details)}. | 
 |  */ | 
 | public abstract class CallScreeningService extends Service { | 
 |     /** | 
 |      * The {@link Intent} that must be declared as handled by the service. | 
 |      */ | 
 |     @SdkConstant(SdkConstant.SdkConstantType.SERVICE_ACTION) | 
 |     public static final String SERVICE_INTERFACE = "android.telecom.CallScreeningService"; | 
 |  | 
 |     private static final int MSG_SCREEN_CALL = 1; | 
 |  | 
 |     private final Handler mHandler = new Handler(Looper.getMainLooper()) { | 
 |         @Override | 
 |         public void handleMessage(Message msg) { | 
 |             switch (msg.what) { | 
 |                 case MSG_SCREEN_CALL: | 
 |                     SomeArgs args = (SomeArgs) msg.obj; | 
 |                     try { | 
 |                         mCallScreeningAdapter = (ICallScreeningAdapter) args.arg1; | 
 |                         Call.Details callDetails = Call.Details | 
 |                                 .createFromParcelableCall((ParcelableCall) args.arg2); | 
 |                         onScreenCall(callDetails); | 
 |                         if (callDetails.getCallDirection() == Call.Details.DIRECTION_OUTGOING) { | 
 |                             mCallScreeningAdapter.onScreeningResponse( | 
 |                                     callDetails.getTelecomCallId(), | 
 |                                     new ComponentName(getPackageName(), getClass().getName()), | 
 |                                     null); | 
 |                         } | 
 |                     } catch (RemoteException e) { | 
 |                         Log.w(this, "Exception when screening call: " + e); | 
 |                     } finally { | 
 |                         args.recycle(); | 
 |                     } | 
 |                     break; | 
 |             } | 
 |         } | 
 |     }; | 
 |  | 
 |     private final class CallScreeningBinder extends ICallScreeningService.Stub { | 
 |         @Override | 
 |         public void screenCall(ICallScreeningAdapter adapter, ParcelableCall call) { | 
 |             Log.v(this, "screenCall"); | 
 |             SomeArgs args = SomeArgs.obtain(); | 
 |             args.arg1 = adapter; | 
 |             args.arg2 = call; | 
 |             mHandler.obtainMessage(MSG_SCREEN_CALL, args).sendToTarget(); | 
 |         } | 
 |     } | 
 |  | 
 |     private ICallScreeningAdapter mCallScreeningAdapter; | 
 |  | 
 |     /** | 
 |      * Parcelable version of {@link CallResponse} used to do IPC. | 
 |      * @hide | 
 |      */ | 
 |     public static class ParcelableCallResponse implements Parcelable { | 
 |         private final boolean mShouldDisallowCall; | 
 |         private final boolean mShouldRejectCall; | 
 |         private final boolean mShouldSilenceCall; | 
 |         private final boolean mShouldSkipCallLog; | 
 |         private final boolean mShouldSkipNotification; | 
 |         private final boolean mShouldScreenCallViaAudioProcessing; | 
 |  | 
 |         private final int mCallComposerAttachmentsToShow; | 
 |  | 
 |         private ParcelableCallResponse( | 
 |                 boolean shouldDisallowCall, | 
 |                 boolean shouldRejectCall, | 
 |                 boolean shouldSilenceCall, | 
 |                 boolean shouldSkipCallLog, | 
 |                 boolean shouldSkipNotification, | 
 |                 boolean shouldScreenCallViaAudioProcessing, | 
 |                 int callComposerAttachmentsToShow) { | 
 |             mShouldDisallowCall = shouldDisallowCall; | 
 |             mShouldRejectCall = shouldRejectCall; | 
 |             mShouldSilenceCall = shouldSilenceCall; | 
 |             mShouldSkipCallLog = shouldSkipCallLog; | 
 |             mShouldSkipNotification = shouldSkipNotification; | 
 |             mShouldScreenCallViaAudioProcessing = shouldScreenCallViaAudioProcessing; | 
 |             mCallComposerAttachmentsToShow = callComposerAttachmentsToShow; | 
 |         } | 
 |  | 
 |         protected ParcelableCallResponse(Parcel in) { | 
 |             mShouldDisallowCall = in.readBoolean(); | 
 |             mShouldRejectCall = in.readBoolean(); | 
 |             mShouldSilenceCall = in.readBoolean(); | 
 |             mShouldSkipCallLog = in.readBoolean(); | 
 |             mShouldSkipNotification = in.readBoolean(); | 
 |             mShouldScreenCallViaAudioProcessing = in.readBoolean(); | 
 |             mCallComposerAttachmentsToShow = in.readInt(); | 
 |         } | 
 |  | 
 |         public CallResponse toCallResponse() { | 
 |             return new CallResponse.Builder() | 
 |                     .setDisallowCall(mShouldDisallowCall) | 
 |                     .setRejectCall(mShouldRejectCall) | 
 |                     .setSilenceCall(mShouldSilenceCall) | 
 |                     .setSkipCallLog(mShouldSkipCallLog) | 
 |                     .setSkipNotification(mShouldSkipNotification) | 
 |                     .setShouldScreenCallViaAudioProcessing(mShouldScreenCallViaAudioProcessing) | 
 |                     .setCallComposerAttachmentsToShow(mCallComposerAttachmentsToShow) | 
 |                     .build(); | 
 |         } | 
 |  | 
 |         public boolean shouldDisallowCall() { | 
 |             return mShouldDisallowCall; | 
 |         } | 
 |  | 
 |         public boolean shouldRejectCall() { | 
 |             return mShouldRejectCall; | 
 |         } | 
 |  | 
 |         public boolean shouldSilenceCall() { | 
 |             return mShouldSilenceCall; | 
 |         } | 
 |  | 
 |         public boolean shouldSkipCallLog() { | 
 |             return mShouldSkipCallLog; | 
 |         } | 
 |  | 
 |         public boolean shouldSkipNotification() { | 
 |             return mShouldSkipNotification; | 
 |         } | 
 |  | 
 |         public boolean shouldScreenCallViaAudioProcessing() { | 
 |             return mShouldScreenCallViaAudioProcessing; | 
 |         } | 
 |  | 
 |         public int getCallComposerAttachmentsToShow() { | 
 |             return mCallComposerAttachmentsToShow; | 
 |         } | 
 |  | 
 |         public static final Creator<ParcelableCallResponse> CREATOR = | 
 |                 new Creator<ParcelableCallResponse>() { | 
 |                     @Override | 
 |                     public ParcelableCallResponse createFromParcel(Parcel in) { | 
 |                         return new ParcelableCallResponse(in); | 
 |                     } | 
 |  | 
 |                     @Override | 
 |                     public ParcelableCallResponse[] newArray(int size) { | 
 |                         return new ParcelableCallResponse[size]; | 
 |                     } | 
 |                 }; | 
 |  | 
 |         @Override | 
 |         public int describeContents() { | 
 |             return 0; | 
 |         } | 
 |  | 
 |         @Override | 
 |         public void writeToParcel(Parcel dest, int flags) { | 
 |             dest.writeBoolean(mShouldDisallowCall); | 
 |             dest.writeBoolean(mShouldRejectCall); | 
 |             dest.writeBoolean(mShouldSilenceCall); | 
 |             dest.writeBoolean(mShouldSkipCallLog); | 
 |             dest.writeBoolean(mShouldSkipNotification); | 
 |             dest.writeBoolean(mShouldScreenCallViaAudioProcessing); | 
 |             dest.writeInt(mCallComposerAttachmentsToShow); | 
 |         } | 
 |     } | 
 |  | 
 |     /** | 
 |      * Information about how to respond to an incoming call. Call screening apps can construct an | 
 |      * instance of this class using {@link CallResponse.Builder}. | 
 |      */ | 
 |     public static class CallResponse { | 
 |         /** | 
 |          * Bit flag indicating whether to show the picture attachment for call composer. | 
 |          * | 
 |          * Used with {@link Builder#setCallComposerAttachmentsToShow(int)}. | 
 |          */ | 
 |         public static final int CALL_COMPOSER_ATTACHMENT_PICTURE = 1; | 
 |  | 
 |         /** | 
 |          * Bit flag indicating whether to show the location attachment for call composer. | 
 |          * | 
 |          * Used with {@link Builder#setCallComposerAttachmentsToShow(int)}. | 
 |          */ | 
 |         public static final int CALL_COMPOSER_ATTACHMENT_LOCATION = 1 << 1; | 
 |  | 
 |         /** | 
 |          * Bit flag indicating whether to show the subject attachment for call composer. | 
 |          * | 
 |          * Used with {@link Builder#setCallComposerAttachmentsToShow(int)}. | 
 |          */ | 
 |         public static final int CALL_COMPOSER_ATTACHMENT_SUBJECT = 1 << 2; | 
 |  | 
 |         /** | 
 |          * Bit flag indicating whether to show the priority attachment for call composer. | 
 |          * | 
 |          * Used with {@link Builder#setCallComposerAttachmentsToShow(int)}. | 
 |          */ | 
 |         public static final int CALL_COMPOSER_ATTACHMENT_PRIORITY = 1 << 3; | 
 |  | 
 |         /** @hide */ | 
 |         @Retention(RetentionPolicy.SOURCE) | 
 |         @IntDef(prefix = "CALL_COMPOSER_ATTACHMENT_", flag = true, | 
 |                 value = { | 
 |                         CALL_COMPOSER_ATTACHMENT_PICTURE, | 
 |                         CALL_COMPOSER_ATTACHMENT_LOCATION, | 
 |                         CALL_COMPOSER_ATTACHMENT_SUBJECT, | 
 |                         CALL_COMPOSER_ATTACHMENT_PRIORITY | 
 |                 } | 
 |         ) | 
 |         public @interface CallComposerAttachmentType {} | 
 |  | 
 |         private static final int NUM_CALL_COMPOSER_ATTACHMENT_TYPES = 4; | 
 |  | 
 |         private final boolean mShouldDisallowCall; | 
 |         private final boolean mShouldRejectCall; | 
 |         private final boolean mShouldSilenceCall; | 
 |         private final boolean mShouldSkipCallLog; | 
 |         private final boolean mShouldSkipNotification; | 
 |         private final boolean mShouldScreenCallViaAudioProcessing; | 
 |         private final int mCallComposerAttachmentsToShow; | 
 |  | 
 |         private CallResponse( | 
 |                 boolean shouldDisallowCall, | 
 |                 boolean shouldRejectCall, | 
 |                 boolean shouldSilenceCall, | 
 |                 boolean shouldSkipCallLog, | 
 |                 boolean shouldSkipNotification, | 
 |                 boolean shouldScreenCallViaAudioProcessing, | 
 |                 int callComposerAttachmentsToShow) { | 
 |             if (!shouldDisallowCall | 
 |                     && (shouldRejectCall || shouldSkipCallLog || shouldSkipNotification)) { | 
 |                 throw new IllegalStateException("Invalid response state for allowed call."); | 
 |             } | 
 |  | 
 |             if (shouldDisallowCall && shouldScreenCallViaAudioProcessing) { | 
 |                 throw new IllegalStateException("Invalid response state for allowed call."); | 
 |             } | 
 |  | 
 |             mShouldDisallowCall = shouldDisallowCall; | 
 |             mShouldRejectCall = shouldRejectCall; | 
 |             mShouldSkipCallLog = shouldSkipCallLog; | 
 |             mShouldSkipNotification = shouldSkipNotification; | 
 |             mShouldSilenceCall = shouldSilenceCall; | 
 |             mShouldScreenCallViaAudioProcessing = shouldScreenCallViaAudioProcessing; | 
 |             mCallComposerAttachmentsToShow = callComposerAttachmentsToShow; | 
 |         } | 
 |  | 
 |         /* | 
 |          * @return Whether the incoming call should be blocked. | 
 |          */ | 
 |         public boolean getDisallowCall() { | 
 |             return mShouldDisallowCall; | 
 |         } | 
 |  | 
 |         /* | 
 |          * @return Whether the incoming call should be disconnected as if the user had manually | 
 |          * rejected it. | 
 |          */ | 
 |         public boolean getRejectCall() { | 
 |             return mShouldRejectCall; | 
 |         } | 
 |  | 
 |         /* | 
 |          * @return Whether the ringtone should be silenced for the incoming call. | 
 |          */ | 
 |         public boolean getSilenceCall() { | 
 |             return mShouldSilenceCall; | 
 |         } | 
 |  | 
 |         /* | 
 |          * @return Whether the incoming call should not be displayed in the call log. | 
 |          */ | 
 |         public boolean getSkipCallLog() { | 
 |             return mShouldSkipCallLog; | 
 |         } | 
 |  | 
 |         /* | 
 |          * @return Whether a missed call notification should not be shown for the incoming call. | 
 |          */ | 
 |         public boolean getSkipNotification() { | 
 |             return mShouldSkipNotification; | 
 |         } | 
 |  | 
 |         /** | 
 |          * @return Whether we should enter the {@link Call#STATE_AUDIO_PROCESSING} state to allow | 
 |          * for further screening of the call. | 
 |          * @hide | 
 |          */ | 
 |         public boolean getShouldScreenCallViaAudioProcessing() { | 
 |             return mShouldScreenCallViaAudioProcessing; | 
 |         } | 
 |  | 
 |         /** | 
 |          * @return A bitmask of call composer attachments that should be shown to the user. | 
 |          */ | 
 |         public @CallComposerAttachmentType int getCallComposerAttachmentsToShow() { | 
 |             return mCallComposerAttachmentsToShow; | 
 |         } | 
 |  | 
 |         /** @hide */ | 
 |         public ParcelableCallResponse toParcelable() { | 
 |             return new ParcelableCallResponse( | 
 |                     mShouldDisallowCall, | 
 |                     mShouldRejectCall, | 
 |                     mShouldSilenceCall, | 
 |                     mShouldSkipCallLog, | 
 |                     mShouldSkipNotification, | 
 |                     mShouldScreenCallViaAudioProcessing, | 
 |                     mCallComposerAttachmentsToShow | 
 |             ); | 
 |         } | 
 |  | 
 |         @Override | 
 |         public boolean equals(Object o) { | 
 |             if (this == o) return true; | 
 |             if (o == null || getClass() != o.getClass()) return false; | 
 |             CallResponse that = (CallResponse) o; | 
 |             return mShouldDisallowCall == that.mShouldDisallowCall && | 
 |                     mShouldRejectCall == that.mShouldRejectCall && | 
 |                     mShouldSilenceCall == that.mShouldSilenceCall && | 
 |                     mShouldSkipCallLog == that.mShouldSkipCallLog && | 
 |                     mShouldSkipNotification == that.mShouldSkipNotification && | 
 |                     mShouldScreenCallViaAudioProcessing | 
 |                             == that.mShouldScreenCallViaAudioProcessing && | 
 |                     mCallComposerAttachmentsToShow == that.mCallComposerAttachmentsToShow; | 
 |         } | 
 |  | 
 |         @Override | 
 |         public int hashCode() { | 
 |             return Objects.hash(mShouldDisallowCall, mShouldRejectCall, mShouldSilenceCall, | 
 |                     mShouldSkipCallLog, mShouldSkipNotification, | 
 |                     mShouldScreenCallViaAudioProcessing, | 
 |                     mCallComposerAttachmentsToShow); | 
 |         } | 
 |  | 
 |         public static class Builder { | 
 |             private boolean mShouldDisallowCall; | 
 |             private boolean mShouldRejectCall; | 
 |             private boolean mShouldSilenceCall; | 
 |             private boolean mShouldSkipCallLog; | 
 |             private boolean mShouldSkipNotification; | 
 |             private boolean mShouldScreenCallViaAudioProcessing; | 
 |             private int mCallComposerAttachmentsToShow = -1; | 
 |  | 
 |             /** | 
 |              * Sets whether the incoming call should be blocked. | 
 |              */ | 
 |             public Builder setDisallowCall(boolean shouldDisallowCall) { | 
 |                 mShouldDisallowCall = shouldDisallowCall; | 
 |                 return this; | 
 |             } | 
 |  | 
 |             /** | 
 |              * Sets whether the incoming call should be disconnected as if the user had manually | 
 |              * rejected it. This property should only be set to true if the call is disallowed. | 
 |              */ | 
 |             public Builder setRejectCall(boolean shouldRejectCall) { | 
 |                 mShouldRejectCall = shouldRejectCall; | 
 |                 return this; | 
 |             } | 
 |  | 
 |             /** | 
 |              * Sets whether ringing should be silenced for the incoming call.  When set | 
 |              * to {@code true}, the Telecom framework will not play a ringtone for the call. | 
 |              * The call will, however, still be sent to the default dialer app if it is not blocked. | 
 |              * A {@link CallScreeningService} can use this to ensure a potential nuisance call is | 
 |              * still surfaced to the user, but in a less intrusive manner. | 
 |              * | 
 |              * Setting this to true only makes sense when the call has not been disallowed | 
 |              * using {@link #setDisallowCall(boolean)}. | 
 |              */ | 
 |             public @NonNull Builder setSilenceCall(boolean shouldSilenceCall) { | 
 |                 mShouldSilenceCall = shouldSilenceCall; | 
 |                 return this; | 
 |             } | 
 |  | 
 |             /** | 
 |              * Sets whether the incoming call should not be displayed in the call log. This property | 
 |              * should only be set to true if the call is disallowed. | 
 |              * <p> | 
 |              * Note: Calls will still be logged with type | 
 |              * {@link android.provider.CallLog.Calls#BLOCKED_TYPE}, regardless of how this property | 
 |              * is set. | 
 |              * <p> | 
 |              * Note: Only the carrier and system call screening apps can use this parameter; | 
 |              * this parameter is ignored otherwise. | 
 |              */ | 
 |             public Builder setSkipCallLog(boolean shouldSkipCallLog) { | 
 |                 mShouldSkipCallLog = shouldSkipCallLog; | 
 |                 return this; | 
 |             } | 
 |  | 
 |             /** | 
 |              * Sets whether a missed call notification should not be shown for the incoming call. | 
 |              * This property should only be set to true if the call is disallowed. | 
 |              */ | 
 |             public Builder setSkipNotification(boolean shouldSkipNotification) { | 
 |                 mShouldSkipNotification = shouldSkipNotification; | 
 |                 return this; | 
 |             } | 
 |  | 
 |             /** | 
 |              * Sets whether to request background audio processing so that the in-call service can | 
 |              * screen the call further. If set to {@code true}, {@link #setDisallowCall} should be | 
 |              * called with {@code false}, and all other parameters in this builder will be ignored. | 
 |              * <p> | 
 |              * This request will only be honored if the {@link CallScreeningService} shares the same | 
 |              * uid as the system dialer app. Otherwise, the call will go through as usual. | 
 |              * <p> | 
 |              * Apps built with SDK version {@link android.os.Build.VERSION_CODES#R} or later which | 
 |              * are using the microphone as part of audio processing should specify the | 
 |              * foreground service type using the attribute | 
 |              * {@link android.R.attr#foregroundServiceType} in the {@link CallScreeningService} | 
 |              * service element of the app's manifest file. | 
 |              * The {@link ServiceInfo#FOREGROUND_SERVICE_TYPE_MICROPHONE} attribute should be | 
 |              * specified. | 
 |              * @see | 
 |              * <a href="https://developer.android.com/preview/privacy/foreground-service-types"> | 
 |              *     the Android Developer Site</a> for more information. | 
 |              * | 
 |              * @param shouldScreenCallViaAudioProcessing Whether to request further call screening. | 
 |              * @hide | 
 |              */ | 
 |             @SystemApi | 
 |             @RequiresPermission(Manifest.permission.CAPTURE_AUDIO_OUTPUT) | 
 |             public @NonNull Builder setShouldScreenCallViaAudioProcessing( | 
 |                     boolean shouldScreenCallViaAudioProcessing) { | 
 |                 mShouldScreenCallViaAudioProcessing = shouldScreenCallViaAudioProcessing; | 
 |                 return this; | 
 |             } | 
 |  | 
 |             /** | 
 |              * Sets the call composer attachments that should be shown to the user. | 
 |              * | 
 |              * Attachments that are not shown will not be passed to the in-call UI responsible for | 
 |              * displaying the call to the user. | 
 |              * | 
 |              * If this method is not called on a {@link Builder}, all attachments will be shown, | 
 |              * except pictures, which will only be shown to users if the call is from a contact. | 
 |              * | 
 |              * Setting attachments to show will have no effect if the call screening service does | 
 |              * not belong to the same package as the system dialer (as returned by | 
 |              * {@link TelecomManager#getSystemDialerPackage()}). | 
 |              * | 
 |              * @param callComposerAttachmentsToShow A bitmask of call composer attachments to show. | 
 |              */ | 
 |             public @NonNull Builder setCallComposerAttachmentsToShow( | 
 |                     @CallComposerAttachmentType int callComposerAttachmentsToShow) { | 
 |                 // If the argument is less than zero (meaning unset), no-op since the conversion | 
 |                 // to/from the parcelable version may call with that value. | 
 |                 if (callComposerAttachmentsToShow < 0) { | 
 |                     return this; | 
 |                 } | 
 |  | 
 |                 if ((callComposerAttachmentsToShow | 
 |                         & (1 << NUM_CALL_COMPOSER_ATTACHMENT_TYPES)) != 0) { | 
 |                     throw new IllegalArgumentException("Attachment types must match the ones" | 
 |                             + " defined in CallResponse"); | 
 |                 } | 
 |                 mCallComposerAttachmentsToShow = callComposerAttachmentsToShow; | 
 |                 return this; | 
 |             } | 
 |  | 
 |             public CallResponse build() { | 
 |                 return new CallResponse( | 
 |                         mShouldDisallowCall, | 
 |                         mShouldRejectCall, | 
 |                         mShouldSilenceCall, | 
 |                         mShouldSkipCallLog, | 
 |                         mShouldSkipNotification, | 
 |                         mShouldScreenCallViaAudioProcessing, | 
 |                         mCallComposerAttachmentsToShow); | 
 |             } | 
 |        } | 
 |     } | 
 |  | 
 |     public CallScreeningService() { | 
 |     } | 
 |  | 
 |     @Override | 
 |     public IBinder onBind(Intent intent) { | 
 |         Log.v(this, "onBind"); | 
 |         return new CallScreeningBinder(); | 
 |     } | 
 |  | 
 |     @Override | 
 |     public boolean onUnbind(Intent intent) { | 
 |         Log.v(this, "onUnbind"); | 
 |         return false; | 
 |     } | 
 |  | 
 |     /** | 
 |      * Called when a new incoming or outgoing call is added. | 
 |      * <p> | 
 |      * A {@link CallScreeningService} must indicate whether an incoming call is allowed or not by | 
 |      * calling | 
 |      * {@link CallScreeningService#respondToCall(Call.Details, CallScreeningService.CallResponse)}. | 
 |      * Your app can tell if a call is an incoming call by checking to see if | 
 |      * {@link Call.Details#getCallDirection()} is {@link Call.Details#DIRECTION_INCOMING}. | 
 |      * <p> | 
 |      * <em>Note:</em> A {@link CallScreeningService} must respond to a call within 5 seconds.  After | 
 |      * this time, the framework will unbind from the {@link CallScreeningService} and ignore its | 
 |      * response. | 
 |      * <p> | 
 |      * <em>Note:</em> The {@link Call.Details} instance provided to a call screening service will | 
 |      * only have the following properties set.  The rest of the {@link Call.Details} properties will | 
 |      * be set to their default value or {@code null}. | 
 |      * <ul> | 
 |      *     <li>{@link Call.Details#getCallDirection()}</li> | 
 |      *     <li>{@link Call.Details#getCallerNumberVerificationStatus()}</li> | 
 |      *     <li>{@link Call.Details#getConnectTimeMillis()}</li> | 
 |      *     <li>{@link Call.Details#getCreationTimeMillis()}</li> | 
 |      *     <li>{@link Call.Details#getHandle()}</li> | 
 |      * </ul> | 
 |      * <p> | 
 |      * Only calls where the {@link Call.Details#getHandle() handle} {@link Uri#getScheme() scheme} | 
 |      * is {@link PhoneAccount#SCHEME_TEL} are passed for call | 
 |      * screening.  Further, only calls which are not in the user's contacts are passed for | 
 |      * screening, unless the {@link CallScreeningService} has been granted | 
 |      * {@link Manifest.permission#READ_CONTACTS} permission by the user.  For outgoing calls, no | 
 |      * post-dial digits are passed. | 
 |      * <p> | 
 |      * Calls with a {@link Call.Details#getHandlePresentation()} of | 
 |      * {@link TelecomManager#PRESENTATION_RESTRICTED}, {@link TelecomManager#PRESENTATION_UNKNOWN}, | 
 |      * {@link TelecomManager#PRESENTATION_UNAVAILABLE} or | 
 |      * {@link TelecomManager#PRESENTATION_PAYPHONE} presentation are not provided to the | 
 |      * {@link CallScreeningService}. | 
 |      * | 
 |      * @param callDetails Information about a new call, see {@link Call.Details}. | 
 |      */ | 
 |     public abstract void onScreenCall(@NonNull Call.Details callDetails); | 
 |  | 
 |     /** | 
 |      * Responds to the given incoming call, either allowing it, silencing it or disallowing it. | 
 |      * <p> | 
 |      * The {@link CallScreeningService} calls this method to inform the system whether the call | 
 |      * should be silently blocked or not. In the event that it should not be blocked, it may | 
 |      * also be requested to ring silently. | 
 |      * <p> | 
 |      * Calls to this method are ignored unless the {@link Call.Details#getCallDirection()} is | 
 |      * {@link Call.Details#DIRECTION_INCOMING}. | 
 |      * <p> | 
 |      * For incoming calls, a {@link CallScreeningService} MUST call this method within 5 seconds of | 
 |      * {@link #onScreenCall(Call.Details)} being invoked by the platform. | 
 |      * <p> | 
 |      * Calls which are blocked/rejected will be logged to the system call log with a call type of | 
 |      * {@link android.provider.CallLog.Calls#BLOCKED_TYPE} and | 
 |      * {@link android.provider.CallLog.Calls#BLOCK_REASON_CALL_SCREENING_SERVICE} block reason. | 
 |      * | 
 |      * @param callDetails The call to allow. | 
 |      *                    <p> | 
 |      *                    Must be the same {@link Call.Details call} which was provided to the | 
 |      *                    {@link CallScreeningService} via {@link #onScreenCall(Call.Details)}. | 
 |      * @param response The {@link CallScreeningService.CallResponse} which contains information | 
 |      * about how to respond to a call. | 
 |      */ | 
 |     public final void respondToCall(@NonNull Call.Details callDetails, | 
 |             @NonNull CallResponse response) { | 
 |         try { | 
 |             mCallScreeningAdapter.onScreeningResponse( | 
 |                     callDetails.getTelecomCallId(), | 
 |                     new ComponentName(getPackageName(), getClass().getName()), | 
 |                     response.toParcelable()); | 
 |         } catch (RemoteException e) { | 
 |             Log.e(this, e, "Got remote exception when returning response"); | 
 |         } | 
 |     } | 
 | } |