|  | /* | 
|  | * Copyright (C) 2006 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.provider; | 
|  |  | 
|  | import android.Manifest; | 
|  | import android.annotation.CallbackExecutor; | 
|  | import android.annotation.FlaggedApi; | 
|  | import android.annotation.IntDef; | 
|  | import android.annotation.LongDef; | 
|  | import android.annotation.NonNull; | 
|  | import android.annotation.RequiresPermission; | 
|  | import android.annotation.SuppressLint; | 
|  | import android.annotation.SystemApi; | 
|  | import android.annotation.UserHandleAware; | 
|  | import android.compat.annotation.UnsupportedAppUsage; | 
|  | import android.content.ContentProvider; | 
|  | import android.content.ContentResolver; | 
|  | import android.content.ContentValues; | 
|  | import android.content.Context; | 
|  | import android.content.Intent; | 
|  | import android.content.pm.UserInfo; | 
|  | import android.database.Cursor; | 
|  | import android.location.Country; | 
|  | import android.location.CountryDetector; | 
|  | import android.net.Uri; | 
|  | import android.os.Build; | 
|  | import android.os.OutcomeReceiver; | 
|  | import android.os.ParcelFileDescriptor; | 
|  | import android.os.ParcelableException; | 
|  | import android.os.UserHandle; | 
|  | import android.os.UserManager; | 
|  | import android.provider.ContactsContract.CommonDataKinds.Callable; | 
|  | import android.provider.ContactsContract.CommonDataKinds.Phone; | 
|  | import android.provider.ContactsContract.Data; | 
|  | import android.provider.ContactsContract.DataUsageFeedback; | 
|  | import android.telecom.CallerInfo; | 
|  | import android.telecom.PhoneAccount; | 
|  | import android.telecom.PhoneAccountHandle; | 
|  | import android.telecom.TelecomManager; | 
|  | import android.telephony.PhoneNumberUtils; | 
|  | import android.text.TextUtils; | 
|  | import android.util.Log; | 
|  |  | 
|  | import com.android.server.telecom.flags.Flags; | 
|  |  | 
|  | import java.io.ByteArrayOutputStream; | 
|  | import java.io.FileNotFoundException; | 
|  | import java.io.FileOutputStream; | 
|  | import java.io.IOException; | 
|  | import java.io.InputStream; | 
|  | import java.lang.annotation.Retention; | 
|  | import java.lang.annotation.RetentionPolicy; | 
|  | import java.util.List; | 
|  | import java.util.Locale; | 
|  | import java.util.Objects; | 
|  | import java.util.concurrent.Executor; | 
|  |  | 
|  | /** | 
|  | * The CallLog provider contains information about placed and received calls. | 
|  | */ | 
|  | public class CallLog { | 
|  | private static final String LOG_TAG = "CallLog"; | 
|  | private static final boolean VERBOSE_LOG = false; // DON'T SUBMIT WITH TRUE. | 
|  |  | 
|  | public static final String AUTHORITY = "call_log"; | 
|  |  | 
|  | /** | 
|  | * The content:// style URL for this provider | 
|  | */ | 
|  | public static final Uri CONTENT_URI = | 
|  | Uri.parse("content://" + AUTHORITY); | 
|  |  | 
|  | /** @hide */ | 
|  | public static final String CALL_COMPOSER_SEGMENT = "call_composer"; | 
|  |  | 
|  | /** @hide */ | 
|  | public static final Uri CALL_COMPOSER_PICTURE_URI = | 
|  | CONTENT_URI.buildUpon().appendPath(CALL_COMPOSER_SEGMENT).build(); | 
|  |  | 
|  | /** | 
|  | * The "shadow" provider stores calllog when the real calllog provider is encrypted.  The | 
|  | * real provider will alter copy from it when it starts, and remove the entries in the shadow. | 
|  | * | 
|  | * <p>See the comment in {@link Calls#addCall} for the details. | 
|  | * | 
|  | * @hide | 
|  | */ | 
|  | public static final String SHADOW_AUTHORITY = "call_log_shadow"; | 
|  |  | 
|  | /** @hide */ | 
|  | public static final Uri SHADOW_CALL_COMPOSER_PICTURE_URI = CALL_COMPOSER_PICTURE_URI.buildUpon() | 
|  | .authority(SHADOW_AUTHORITY).build(); | 
|  |  | 
|  | /** | 
|  | * Describes an error encountered while storing a call composer picture in the call log. | 
|  | * @hide | 
|  | */ | 
|  | @SystemApi | 
|  | public static class CallComposerLoggingException extends Throwable { | 
|  | /** | 
|  | * Indicates an unknown error. | 
|  | */ | 
|  | public static final int ERROR_UNKNOWN = 0; | 
|  |  | 
|  | /** | 
|  | * Indicates that the process hosting the call log died or otherwise encountered an | 
|  | * unrecoverable error while storing the picture. | 
|  | * | 
|  | * The caller should retry if this error is encountered. | 
|  | */ | 
|  | public static final int ERROR_REMOTE_END_CLOSED = 1; | 
|  |  | 
|  | /** | 
|  | * Indicates that the device has insufficient space to store this picture. | 
|  | * | 
|  | * The caller should not retry if this error is encountered. | 
|  | */ | 
|  | public static final int ERROR_STORAGE_FULL = 2; | 
|  |  | 
|  | /** | 
|  | * Indicates that the {@link InputStream} passed to {@link #storeCallComposerPicture} | 
|  | * was closed. | 
|  | * | 
|  | * The caller should retry if this error is encountered, and be sure to not close the stream | 
|  | * before the callback is called this time. | 
|  | */ | 
|  | public static final int ERROR_INPUT_CLOSED = 3; | 
|  |  | 
|  | /** @hide */ | 
|  | @IntDef(prefix = {"ERROR_"}, value = { | 
|  | ERROR_UNKNOWN, | 
|  | ERROR_REMOTE_END_CLOSED, | 
|  | ERROR_STORAGE_FULL, | 
|  | ERROR_INPUT_CLOSED, | 
|  | }) | 
|  | @Retention(RetentionPolicy.SOURCE) | 
|  | public @interface CallComposerLoggingError { } | 
|  |  | 
|  | private final int mErrorCode; | 
|  |  | 
|  | public CallComposerLoggingException(@CallComposerLoggingError int errorCode) { | 
|  | mErrorCode = errorCode; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @return The error code for this exception. | 
|  | */ | 
|  | public @CallComposerLoggingError int getErrorCode() { | 
|  | return mErrorCode; | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public String toString() { | 
|  | String errorString; | 
|  | switch (mErrorCode) { | 
|  | case ERROR_UNKNOWN: | 
|  | errorString = "UNKNOWN"; | 
|  | break; | 
|  | case ERROR_REMOTE_END_CLOSED: | 
|  | errorString = "REMOTE_END_CLOSED"; | 
|  | break; | 
|  | case ERROR_STORAGE_FULL: | 
|  | errorString = "STORAGE_FULL"; | 
|  | break; | 
|  | case ERROR_INPUT_CLOSED: | 
|  | errorString = "INPUT_CLOSED"; | 
|  | break; | 
|  | default: | 
|  | errorString = "[[" + mErrorCode + "]]"; | 
|  | break; | 
|  | } | 
|  | return "CallComposerLoggingException: " + errorString; | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Supplies a call composer picture to the call log for persistent storage. | 
|  | * | 
|  | * This method is used by Telephony to store pictures selected by the user or sent from the | 
|  | * remote party as part of a voice call with call composer. The {@link Uri} supplied in the | 
|  | * callback can be used to retrieve the image via {@link ContentResolver#openFile} or stored in | 
|  | * the {@link Calls} table in the {@link Calls#COMPOSER_PHOTO_URI} column. | 
|  | * | 
|  | * The caller is responsible for closing the {@link InputStream} after the callback indicating | 
|  | * success or failure. | 
|  | * | 
|  | * @param context An instance of {@link Context}. The picture will be stored to the user | 
|  | *                corresponding to {@link Context#getUser()}. | 
|  | * @param input An input stream from which the picture to store should be read. The input data | 
|  | *              must be decodeable as either a JPEG, PNG, or GIF image. | 
|  | * @param executor The {@link Executor} on which to perform the file transfer operation and | 
|  | *                 call the supplied callback. | 
|  | * @param callback Callback that's called after the picture is successfully stored or when an | 
|  | *                 error occurs. | 
|  | * @hide | 
|  | */ | 
|  | @SystemApi | 
|  | @UserHandleAware | 
|  | @RequiresPermission(allOf = { | 
|  | Manifest.permission.WRITE_CALL_LOG, | 
|  | Manifest.permission.INTERACT_ACROSS_USERS | 
|  | }) | 
|  | public static void storeCallComposerPicture(@NonNull Context context, | 
|  | @NonNull InputStream input, | 
|  | @CallbackExecutor @NonNull Executor executor, | 
|  | @NonNull OutcomeReceiver<Uri, CallComposerLoggingException> callback) { | 
|  | Objects.requireNonNull(context); | 
|  | Objects.requireNonNull(input); | 
|  | Objects.requireNonNull(executor); | 
|  | Objects.requireNonNull(callback); | 
|  |  | 
|  | executor.execute(() -> { | 
|  | ByteArrayOutputStream tmpOut = new ByteArrayOutputStream(); | 
|  |  | 
|  | // Read the entire input into memory first in case we have to write multiple times and | 
|  | // the input isn't resettable. | 
|  | byte[] buffer = new byte[1024]; | 
|  | int bytesRead; | 
|  | while (true) { | 
|  | try { | 
|  | bytesRead = input.read(buffer); | 
|  | } catch (IOException e) { | 
|  | Log.e(LOG_TAG, "IOException while reading call composer pic from input: " | 
|  | + e); | 
|  | callback.onError(new CallComposerLoggingException( | 
|  | CallComposerLoggingException.ERROR_INPUT_CLOSED)); | 
|  | return; | 
|  | } | 
|  | if (bytesRead < 0) { | 
|  | break; | 
|  | } | 
|  | tmpOut.write(buffer, 0, bytesRead); | 
|  | } | 
|  | byte[] picData = tmpOut.toByteArray(); | 
|  |  | 
|  | UserManager userManager = context.getSystemService(UserManager.class); | 
|  | UserHandle user = context.getUser(); | 
|  | // Nasty casework for the shadow calllog begins... | 
|  | // First see if we're just inserting for one user. If so, insert into the shadow | 
|  | // based on whether that user is unlocked. | 
|  | UserHandle realUser = UserHandle.CURRENT.equals(user) | 
|  | ? android.os.Process.myUserHandle() : user; | 
|  | if (realUser != UserHandle.ALL) { | 
|  | Uri baseUri = userManager.isUserUnlocked(realUser) ? CALL_COMPOSER_PICTURE_URI | 
|  | : SHADOW_CALL_COMPOSER_PICTURE_URI; | 
|  | Uri pictureInsertionUri = ContentProvider.maybeAddUserId(baseUri, | 
|  | realUser.getIdentifier()); | 
|  | Log.i(LOG_TAG, "Inserting call composer for single user at " | 
|  | + pictureInsertionUri); | 
|  |  | 
|  | try { | 
|  | Uri result = storeCallComposerPictureAtUri( | 
|  | context, pictureInsertionUri, false, picData); | 
|  | callback.onResult(result); | 
|  | } catch (CallComposerLoggingException e) { | 
|  | callback.onError(e); | 
|  | } | 
|  | return; | 
|  | } | 
|  |  | 
|  | // Next, see if the system user is locked. If so, only insert to the system shadow | 
|  | if (!userManager.isUserUnlocked(UserHandle.SYSTEM)) { | 
|  | Uri pictureInsertionUri = ContentProvider.maybeAddUserId( | 
|  | SHADOW_CALL_COMPOSER_PICTURE_URI, | 
|  | UserHandle.SYSTEM.getIdentifier()); | 
|  | Log.i(LOG_TAG, "Inserting call composer for all users, but system locked at " | 
|  | + pictureInsertionUri); | 
|  | try { | 
|  | Uri result = | 
|  | storeCallComposerPictureAtUri(context, pictureInsertionUri, | 
|  | true, picData); | 
|  | callback.onResult(result); | 
|  | } catch (CallComposerLoggingException e) { | 
|  | callback.onError(e); | 
|  | } | 
|  | return; | 
|  | } | 
|  |  | 
|  | // If we're inserting to all users and the system user is unlocked, then insert to all | 
|  | // running users. Non running/still locked users will copy from the system when they | 
|  | // start. | 
|  | // First, insert to the system calllog to get the basename to use for the rest of the | 
|  | // users. | 
|  | Uri systemPictureInsertionUri = ContentProvider.maybeAddUserId( | 
|  | CALL_COMPOSER_PICTURE_URI, | 
|  | UserHandle.SYSTEM.getIdentifier()); | 
|  | Uri systemInsertedPicture; | 
|  | try { | 
|  | systemInsertedPicture = | 
|  | storeCallComposerPictureAtUri(context, systemPictureInsertionUri, | 
|  | true, picData); | 
|  | Log.i(LOG_TAG, "Inserting call composer for all users, succeeded with system," | 
|  | + " result is " + systemInsertedPicture); | 
|  | } catch (CallComposerLoggingException e) { | 
|  | callback.onError(e); | 
|  | return; | 
|  | } | 
|  |  | 
|  | // Next, insert into all users that have call log access AND are running AND are | 
|  | // decrypted. | 
|  | Uri strippedInsertionUri = ContentProvider.getUriWithoutUserId(systemInsertedPicture); | 
|  | for (UserInfo u : userManager.getAliveUsers()) { | 
|  | UserHandle userHandle = u.getUserHandle(); | 
|  | if (userHandle.isSystem()) { | 
|  | // Already written. | 
|  | continue; | 
|  | } | 
|  |  | 
|  | if (!Calls.shouldHaveSharedCallLogEntries( | 
|  | context, userManager, userHandle.getIdentifier())) { | 
|  | // Shouldn't have calllog entries. | 
|  | continue; | 
|  | } | 
|  |  | 
|  | if (userManager.isUserRunning(userHandle) | 
|  | && userManager.isUserUnlocked(userHandle)) { | 
|  | Uri insertionUri = ContentProvider.maybeAddUserId(strippedInsertionUri, | 
|  | userHandle.getIdentifier()); | 
|  | Log.i(LOG_TAG, "Inserting call composer for all users, now on user " | 
|  | + userHandle + " inserting at " + insertionUri); | 
|  | try { | 
|  | storeCallComposerPictureAtUri(context, insertionUri, false, picData); | 
|  | } catch (CallComposerLoggingException e) { | 
|  | Log.e(LOG_TAG, "Error writing for user " + userHandle.getIdentifier() | 
|  | + ": " + e); | 
|  | // If one or more users failed but the system user succeeded, don't return | 
|  | // an error -- the image is still around somewhere, and we'll be able to | 
|  | // find it in the system user's call log if needed. | 
|  | } | 
|  | } | 
|  | } | 
|  | callback.onResult(strippedInsertionUri); | 
|  | }); | 
|  | } | 
|  |  | 
|  | private static Uri storeCallComposerPictureAtUri( | 
|  | Context context, Uri insertionUri, | 
|  | boolean forAllUsers, byte[] picData) throws CallComposerLoggingException { | 
|  | Uri pictureFileUri; | 
|  | try { | 
|  | ContentValues cv = new ContentValues(); | 
|  | cv.put(Calls.ADD_FOR_ALL_USERS, forAllUsers ? 1 : 0); | 
|  | pictureFileUri = context.getContentResolver().insert(insertionUri, cv); | 
|  | } catch (ParcelableException e) { | 
|  | // Most likely an IOException. We don't have a good way of distinguishing them so | 
|  | // just return an unknown error. | 
|  | throw new CallComposerLoggingException(CallComposerLoggingException.ERROR_UNKNOWN); | 
|  | } | 
|  | if (pictureFileUri == null) { | 
|  | // If the call log provider returns null, it means that there's not enough space | 
|  | // left to store the maximum-sized call composer image. | 
|  | throw new CallComposerLoggingException(CallComposerLoggingException.ERROR_STORAGE_FULL); | 
|  | } | 
|  |  | 
|  | try (ParcelFileDescriptor pfd = | 
|  | context.getContentResolver().openFileDescriptor(pictureFileUri, "w")) { | 
|  | FileOutputStream output = new FileOutputStream(pfd.getFileDescriptor()); | 
|  | try { | 
|  | output.write(picData); | 
|  | } catch (IOException e) { | 
|  | Log.e(LOG_TAG, "Got IOException writing to remote end: " + e); | 
|  | // Clean up our mess if we didn't successfully write the file. | 
|  | context.getContentResolver().delete(pictureFileUri, null); | 
|  | throw new CallComposerLoggingException( | 
|  | CallComposerLoggingException.ERROR_REMOTE_END_CLOSED); | 
|  | } | 
|  | } catch (FileNotFoundException e) { | 
|  | throw new CallComposerLoggingException(CallComposerLoggingException.ERROR_UNKNOWN); | 
|  | } catch (IOException e) { | 
|  | // Ignore, this is only thrown upon closing. | 
|  | Log.e(LOG_TAG, "Got IOException closing remote descriptor: " + e); | 
|  | } | 
|  | return pictureFileUri; | 
|  | } | 
|  |  | 
|  | // Only call on the correct executor. | 
|  | private static void sendCallComposerError(OutcomeReceiver<?, CallComposerLoggingException> cb, | 
|  | int error) { | 
|  | cb.onError(new CallComposerLoggingException(error)); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Used as an argument to {@link Calls#addCall(Context, AddCallParams)}. | 
|  | * | 
|  | * Contains details to log about a call. | 
|  | * @hide | 
|  | */ | 
|  | public static class AddCallParams { | 
|  |  | 
|  | /** | 
|  | * Builder for the add-call parameters. | 
|  | */ | 
|  | public static final class AddCallParametersBuilder { | 
|  | public static final int MAX_NUMBER_OF_CHARACTERS = 256; | 
|  | private CallerInfo mCallerInfo; | 
|  | private String mNumber; | 
|  | private String mPostDialDigits; | 
|  | private String mViaNumber; | 
|  | private int mPresentation = TelecomManager.PRESENTATION_UNKNOWN; | 
|  | private int mCallType = Calls.INCOMING_TYPE; | 
|  | private int mFeatures; | 
|  | private PhoneAccountHandle mAccountHandle; | 
|  | private long mStart; | 
|  | private int mDuration; | 
|  | private Long mDataUsage = Long.MIN_VALUE; | 
|  | private boolean mAddForAllUsers; | 
|  | private UserHandle mUserToBeInsertedTo; | 
|  | private boolean mIsRead; | 
|  | private int mCallBlockReason = Calls.BLOCK_REASON_NOT_BLOCKED; | 
|  | private CharSequence mCallScreeningAppName; | 
|  | private String mCallScreeningComponentName; | 
|  | private long mMissedReason = Calls.MISSED_REASON_NOT_MISSED; | 
|  | private int mPriority = Calls.PRIORITY_NORMAL; | 
|  | private String mSubject; | 
|  | private double mLatitude = Double.NaN; | 
|  | private double mLongitude = Double.NaN; | 
|  | private Uri mPictureUri; | 
|  | private int mIsPhoneAccountMigrationPending; | 
|  | private boolean mIsBusinessCall; | 
|  | private String mAssertedDisplayName; | 
|  |  | 
|  | /** | 
|  | * @param callerInfo the CallerInfo object to get the target contact from. | 
|  | */ | 
|  | public @NonNull AddCallParametersBuilder setCallerInfo( | 
|  | @NonNull CallerInfo callerInfo) { | 
|  | mCallerInfo = callerInfo; | 
|  | return this; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @param number the phone number to be added to the calls db | 
|  | */ | 
|  | public @NonNull AddCallParametersBuilder setNumber(@NonNull String number) { | 
|  | mNumber = number; | 
|  | return this; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @param postDialDigits the post-dial digits that were dialed after the number, | 
|  | *        if it was outgoing. Otherwise it is ''. | 
|  | */ | 
|  | public @NonNull AddCallParametersBuilder setPostDialDigits( | 
|  | @NonNull String postDialDigits) { | 
|  | mPostDialDigits = postDialDigits; | 
|  | return this; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @param viaNumber the secondary number that the incoming call received with. If the | 
|  | *        call was received with the SIM assigned number, then this field must be ''. | 
|  | */ | 
|  | public @NonNull AddCallParametersBuilder setViaNumber(@NonNull String viaNumber) { | 
|  | mViaNumber = viaNumber; | 
|  | return this; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @param presentation enum value from TelecomManager.PRESENTATION_xxx, which | 
|  | *        is set by the network and denotes the number presenting rules for | 
|  | *        "allowed", "payphone", "restricted" or "unknown" | 
|  | */ | 
|  | public @NonNull AddCallParametersBuilder setPresentation(int presentation) { | 
|  | mPresentation = presentation; | 
|  | return this; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @param callType enumerated values for "incoming", "outgoing", or "missed" | 
|  | */ | 
|  | public @NonNull AddCallParametersBuilder setCallType(int callType) { | 
|  | mCallType = callType; | 
|  | return this; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @param features features of the call (e.g. Video). | 
|  | */ | 
|  | public @NonNull AddCallParametersBuilder setFeatures(int features) { | 
|  | mFeatures = features; | 
|  | return this; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @param accountHandle The accountHandle object identifying the provider of the call | 
|  | */ | 
|  | public @NonNull AddCallParametersBuilder setAccountHandle( | 
|  | @NonNull PhoneAccountHandle accountHandle) { | 
|  | mAccountHandle = accountHandle; | 
|  | return this; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @param start time stamp for the call in milliseconds | 
|  | */ | 
|  | public @NonNull AddCallParametersBuilder setStart(long start) { | 
|  | mStart = start; | 
|  | return this; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @param duration call duration in seconds | 
|  | */ | 
|  | public @NonNull AddCallParametersBuilder setDuration(int duration) { | 
|  | mDuration = duration; | 
|  | return this; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @param dataUsage data usage for the call in bytes or | 
|  | *                  {@link Long#MIN_VALUE} if data usage was not tracked | 
|  | *                  for the call. | 
|  | */ | 
|  | public @NonNull AddCallParametersBuilder setDataUsage(long dataUsage) { | 
|  | mDataUsage = dataUsage; | 
|  | return this; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @param addForAllUsers If true, the call is added to the call log of all currently | 
|  | *        running users. The caller must have the MANAGE_USERS permission if this is | 
|  | *        true. | 
|  | */ | 
|  | public @NonNull AddCallParametersBuilder setAddForAllUsers( | 
|  | boolean addForAllUsers) { | 
|  | mAddForAllUsers = addForAllUsers; | 
|  | return this; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @param userToBeInsertedTo {@link UserHandle} of user that the call is going to be | 
|  | *                           inserted to. null if it is inserted to the current user. | 
|  | *                           The value is ignored if {@link #setAddForAllUsers} is | 
|  | *                           called with {@code true}. | 
|  | */ | 
|  | @SuppressLint("UserHandleName") | 
|  | public @NonNull AddCallParametersBuilder setUserToBeInsertedTo( | 
|  | @NonNull UserHandle userToBeInsertedTo) { | 
|  | mUserToBeInsertedTo = userToBeInsertedTo; | 
|  | return this; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @param isRead Flag to show if the missed call log has been read by the user or not. | 
|  | *                Used for call log restore of missed calls. | 
|  | */ | 
|  | public @NonNull AddCallParametersBuilder setIsRead(boolean isRead) { | 
|  | mIsRead = isRead; | 
|  | return this; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @param callBlockReason The reason why the call is blocked. | 
|  | */ | 
|  | public @NonNull AddCallParametersBuilder setCallBlockReason(int callBlockReason) { | 
|  | mCallBlockReason = callBlockReason; | 
|  | return this; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @param callScreeningAppName The call screening application name which block the call. | 
|  | */ | 
|  | public @NonNull AddCallParametersBuilder setCallScreeningAppName( | 
|  | @NonNull CharSequence callScreeningAppName) { | 
|  | mCallScreeningAppName = callScreeningAppName; | 
|  | return this; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @param callScreeningComponentName The call screening component name which blocked | 
|  | *                                   the call. | 
|  | */ | 
|  | public @NonNull AddCallParametersBuilder setCallScreeningComponentName( | 
|  | @NonNull String callScreeningComponentName) { | 
|  | mCallScreeningComponentName = callScreeningComponentName; | 
|  | return this; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @param missedReason The encoded missed information of the call. | 
|  | */ | 
|  | public @NonNull AddCallParametersBuilder setMissedReason(long missedReason) { | 
|  | mMissedReason = missedReason; | 
|  | return this; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @param priority The priority of the call, either {@link Calls#PRIORITY_NORMAL} | 
|  | *                 or {@link Calls#PRIORITY_URGENT} as sent via call composer | 
|  | */ | 
|  | public @NonNull AddCallParametersBuilder setPriority(int priority) { | 
|  | mPriority = priority; | 
|  | return this; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @param subject The subject as sent via call composer. | 
|  | */ | 
|  | public @NonNull AddCallParametersBuilder setSubject(@NonNull String subject) { | 
|  | mSubject = subject; | 
|  | return this; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @param latitude Latitude of the location sent via call composer. | 
|  | */ | 
|  | public @NonNull AddCallParametersBuilder setLatitude(double latitude) { | 
|  | mLatitude = latitude; | 
|  | return this; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @param longitude Longitude of the location sent via call composer. | 
|  | */ | 
|  | public @NonNull AddCallParametersBuilder setLongitude(double longitude) { | 
|  | mLongitude = longitude; | 
|  | return this; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @param pictureUri {@link Uri} returned from {@link #storeCallComposerPicture}. | 
|  | *                   Associates that stored picture with this call in the log. | 
|  | */ | 
|  | public @NonNull AddCallParametersBuilder setPictureUri(@NonNull Uri pictureUri) { | 
|  | mPictureUri = pictureUri; | 
|  | return this; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @param isPhoneAccountMigrationPending whether the phone account migration is pending | 
|  | */ | 
|  | public @NonNull AddCallParametersBuilder setIsPhoneAccountMigrationPending( | 
|  | int isPhoneAccountMigrationPending) { | 
|  | mIsPhoneAccountMigrationPending = isPhoneAccountMigrationPending; | 
|  | return this; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @param isBusinessCall should be set if the caller is a business call | 
|  | */ | 
|  | @FlaggedApi(Flags.FLAG_BUSINESS_CALL_COMPOSER) | 
|  | public @NonNull AddCallParametersBuilder setIsBusinessCall(boolean isBusinessCall) { | 
|  | mIsBusinessCall = isBusinessCall; | 
|  | return this; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @param assertedDisplayName the asserted display name associated with the business | 
|  | *                            call | 
|  | * @throws IllegalArgumentException if the assertedDisplayName is over 256 characters | 
|  | */ | 
|  | @FlaggedApi(Flags.FLAG_BUSINESS_CALL_COMPOSER) | 
|  | public @NonNull AddCallParametersBuilder setAssertedDisplayName( | 
|  | String assertedDisplayName) { | 
|  | if (assertedDisplayName != null | 
|  | && assertedDisplayName.length() > MAX_NUMBER_OF_CHARACTERS) { | 
|  | throw new IllegalArgumentException("assertedDisplayName exceeds the character" | 
|  | + " limit of " + MAX_NUMBER_OF_CHARACTERS + "."); | 
|  | } | 
|  | mAssertedDisplayName = assertedDisplayName; | 
|  | return this; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Builds the object | 
|  | */ | 
|  | public @NonNull AddCallParams build() { | 
|  | if (Flags.businessCallComposer()) { | 
|  | return new AddCallParams(mCallerInfo, mNumber, mPostDialDigits, mViaNumber, | 
|  | mPresentation, mCallType, mFeatures, mAccountHandle, mStart, mDuration, | 
|  | mDataUsage, mAddForAllUsers, mUserToBeInsertedTo, mIsRead, | 
|  | mCallBlockReason, | 
|  | mCallScreeningAppName, mCallScreeningComponentName, mMissedReason, | 
|  | mPriority, mSubject, mLatitude, mLongitude, mPictureUri, | 
|  | mIsPhoneAccountMigrationPending, mIsBusinessCall, mAssertedDisplayName); | 
|  | } else { | 
|  | return new AddCallParams(mCallerInfo, mNumber, mPostDialDigits, mViaNumber, | 
|  | mPresentation, mCallType, mFeatures, mAccountHandle, mStart, mDuration, | 
|  | mDataUsage, mAddForAllUsers, mUserToBeInsertedTo, mIsRead, | 
|  | mCallBlockReason, | 
|  | mCallScreeningAppName, mCallScreeningComponentName, mMissedReason, | 
|  | mPriority, mSubject, mLatitude, mLongitude, mPictureUri, | 
|  | mIsPhoneAccountMigrationPending); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | private CallerInfo mCallerInfo; | 
|  | private String mNumber; | 
|  | private String mPostDialDigits; | 
|  | private String mViaNumber; | 
|  | private int mPresentation; | 
|  | private int mCallType; | 
|  | private int mFeatures; | 
|  | private PhoneAccountHandle mAccountHandle; | 
|  | private long mStart; | 
|  | private int mDuration; | 
|  | private long mDataUsage; | 
|  | private boolean mAddForAllUsers; | 
|  | private UserHandle mUserToBeInsertedTo; | 
|  | private boolean mIsRead; | 
|  | private int mCallBlockReason; | 
|  | private CharSequence mCallScreeningAppName; | 
|  | private String mCallScreeningComponentName; | 
|  | private long mMissedReason; | 
|  | private int mPriority; | 
|  | private String mSubject; | 
|  | private double mLatitude = Double.NaN; | 
|  | private double mLongitude = Double.NaN; | 
|  | private Uri mPictureUri; | 
|  | private int mIsPhoneAccountMigrationPending; | 
|  | private boolean mIsBusinessCall; | 
|  | private String mAssertedDisplayName; | 
|  |  | 
|  | private AddCallParams(CallerInfo callerInfo, String number, String postDialDigits, | 
|  | String viaNumber, int presentation, int callType, int features, | 
|  | PhoneAccountHandle accountHandle, long start, int duration, long dataUsage, | 
|  | boolean addForAllUsers, UserHandle userToBeInsertedTo, boolean isRead, | 
|  | int callBlockReason, | 
|  | CharSequence callScreeningAppName, String callScreeningComponentName, | 
|  | long missedReason, | 
|  | int priority, String subject, double latitude, double longitude, Uri pictureUri, | 
|  | int isPhoneAccountMigrationPending) { | 
|  | mCallerInfo = callerInfo; | 
|  | mNumber = number; | 
|  | mPostDialDigits = postDialDigits; | 
|  | mViaNumber = viaNumber; | 
|  | mPresentation = presentation; | 
|  | mCallType = callType; | 
|  | mFeatures = features; | 
|  | mAccountHandle = accountHandle; | 
|  | mStart = start; | 
|  | mDuration = duration; | 
|  | mDataUsage = dataUsage; | 
|  | mAddForAllUsers = addForAllUsers; | 
|  | mUserToBeInsertedTo = userToBeInsertedTo; | 
|  | mIsRead = isRead; | 
|  | mCallBlockReason = callBlockReason; | 
|  | mCallScreeningAppName = callScreeningAppName; | 
|  | mCallScreeningComponentName = callScreeningComponentName; | 
|  | mMissedReason = missedReason; | 
|  | mPriority = priority; | 
|  | mSubject = subject; | 
|  | mLatitude = latitude; | 
|  | mLongitude = longitude; | 
|  | mPictureUri = pictureUri; | 
|  | mIsPhoneAccountMigrationPending = isPhoneAccountMigrationPending; | 
|  | } | 
|  |  | 
|  | private AddCallParams(CallerInfo callerInfo, String number, String postDialDigits, | 
|  | String viaNumber, int presentation, int callType, int features, | 
|  | PhoneAccountHandle accountHandle, long start, int duration, long dataUsage, | 
|  | boolean addForAllUsers, UserHandle userToBeInsertedTo, boolean isRead, | 
|  | int callBlockReason, | 
|  | CharSequence callScreeningAppName, String callScreeningComponentName, | 
|  | long missedReason, | 
|  | int priority, String subject, double latitude, double longitude, Uri pictureUri, | 
|  | int isPhoneAccountMigrationPending, boolean isBusinessCall, | 
|  | String assertedDisplayName) { | 
|  | mCallerInfo = callerInfo; | 
|  | mNumber = number; | 
|  | mPostDialDigits = postDialDigits; | 
|  | mViaNumber = viaNumber; | 
|  | mPresentation = presentation; | 
|  | mCallType = callType; | 
|  | mFeatures = features; | 
|  | mAccountHandle = accountHandle; | 
|  | mStart = start; | 
|  | mDuration = duration; | 
|  | mDataUsage = dataUsage; | 
|  | mAddForAllUsers = addForAllUsers; | 
|  | mUserToBeInsertedTo = userToBeInsertedTo; | 
|  | mIsRead = isRead; | 
|  | mCallBlockReason = callBlockReason; | 
|  | mCallScreeningAppName = callScreeningAppName; | 
|  | mCallScreeningComponentName = callScreeningComponentName; | 
|  | mMissedReason = missedReason; | 
|  | mPriority = priority; | 
|  | mSubject = subject; | 
|  | mLatitude = latitude; | 
|  | mLongitude = longitude; | 
|  | mPictureUri = pictureUri; | 
|  | mIsPhoneAccountMigrationPending = isPhoneAccountMigrationPending; | 
|  | mIsBusinessCall = isBusinessCall; | 
|  | mAssertedDisplayName = assertedDisplayName; | 
|  | } | 
|  |  | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Contains the recent calls. | 
|  | * <p> | 
|  | * Note: If you want to query the call log and limit the results to a single value, you should | 
|  | * append the {@link #LIMIT_PARAM_KEY} parameter to the content URI.  For example: | 
|  | * <pre> | 
|  | * {@code | 
|  | * getContentResolver().query( | 
|  | *                 Calls.CONTENT_URI.buildUpon().appendQueryParameter(LIMIT_PARAM_KEY, "1") | 
|  | *                 .build(), | 
|  | *                 null, null, null, null); | 
|  | * } | 
|  | * </pre> | 
|  | * <p> | 
|  | * The call log provider enforces strict SQL grammar, so you CANNOT append "LIMIT" to the SQL | 
|  | * query as below: | 
|  | * <pre> | 
|  | * {@code | 
|  | * getContentResolver().query(Calls.CONTENT_URI, null, "LIMIT 1", null, null); | 
|  | * } | 
|  | * </pre> | 
|  | */ | 
|  | public static class Calls implements BaseColumns { | 
|  | /** | 
|  | * The content:// style URL for this table | 
|  | */ | 
|  | public static final Uri CONTENT_URI = | 
|  | Uri.parse("content://call_log/calls"); | 
|  |  | 
|  | /** @hide */ | 
|  | public static final Uri SHADOW_CONTENT_URI = | 
|  | Uri.parse("content://call_log_shadow/calls"); | 
|  |  | 
|  | /** | 
|  | * The content:// style URL for filtering this table on phone numbers | 
|  | */ | 
|  | public static final Uri CONTENT_FILTER_URI = | 
|  | Uri.parse("content://call_log/calls/filter"); | 
|  |  | 
|  | /** | 
|  | * Query parameter used to limit the number of call logs returned. | 
|  | * <p> | 
|  | * TYPE: integer | 
|  | */ | 
|  | public static final String LIMIT_PARAM_KEY = "limit"; | 
|  |  | 
|  | /** | 
|  | * Form of {@link #CONTENT_URI} which limits the query results to a single result. | 
|  | */ | 
|  | private static final Uri CONTENT_URI_LIMIT_1 = CONTENT_URI.buildUpon() | 
|  | .appendQueryParameter(LIMIT_PARAM_KEY, "1") | 
|  | .build(); | 
|  |  | 
|  | /** | 
|  | * Query parameter used to specify the starting record to return. | 
|  | * <p> | 
|  | * TYPE: integer | 
|  | */ | 
|  | public static final String OFFSET_PARAM_KEY = "offset"; | 
|  |  | 
|  | /** | 
|  | * An optional URI parameter which instructs the provider to allow the operation to be | 
|  | * applied to voicemail records as well. | 
|  | * <p> | 
|  | * TYPE: Boolean | 
|  | * <p> | 
|  | * Using this parameter with a value of {@code true} will result in a security error if the | 
|  | * calling package does not have appropriate permissions to access voicemails. | 
|  | * | 
|  | * @hide | 
|  | */ | 
|  | public static final String ALLOW_VOICEMAILS_PARAM_KEY = "allow_voicemails"; | 
|  |  | 
|  | /** | 
|  | * An optional extra used with {@link #CONTENT_TYPE Calls.CONTENT_TYPE} and | 
|  | * {@link Intent#ACTION_VIEW} to specify that the presented list of calls should be | 
|  | * filtered for a particular call type. | 
|  | * | 
|  | * Applications implementing a call log UI should check for this extra, and display a | 
|  | * filtered list of calls based on the specified call type. If not applicable within the | 
|  | * application's UI, it should be silently ignored. | 
|  | * | 
|  | * <p> | 
|  | * The following example brings up the call log, showing only missed calls. | 
|  | * <pre> | 
|  | * Intent intent = new Intent(Intent.ACTION_VIEW); | 
|  | * intent.setType(CallLog.Calls.CONTENT_TYPE); | 
|  | * intent.putExtra(CallLog.Calls.EXTRA_CALL_TYPE_FILTER, CallLog.Calls.MISSED_TYPE); | 
|  | * startActivity(intent); | 
|  | * </pre> | 
|  | * </p> | 
|  | */ | 
|  | public static final String EXTRA_CALL_TYPE_FILTER = | 
|  | "android.provider.extra.CALL_TYPE_FILTER"; | 
|  |  | 
|  | /** | 
|  | * Content uri used to access call log entries, including voicemail records. You must have | 
|  | * the READ_CALL_LOG and WRITE_CALL_LOG permissions to read and write to the call log, as | 
|  | * well as READ_VOICEMAIL and WRITE_VOICEMAIL permissions to read and write voicemails. | 
|  | */ | 
|  | public static final Uri CONTENT_URI_WITH_VOICEMAIL = CONTENT_URI.buildUpon() | 
|  | .appendQueryParameter(ALLOW_VOICEMAILS_PARAM_KEY, "true") | 
|  | .build(); | 
|  |  | 
|  | /** | 
|  | * The default sort order for this table | 
|  | */ | 
|  | public static final String DEFAULT_SORT_ORDER = "date DESC"; | 
|  |  | 
|  | /** | 
|  | * The MIME type of {@link #CONTENT_URI} and {@link #CONTENT_FILTER_URI} | 
|  | * providing a directory of calls. | 
|  | */ | 
|  | public static final String CONTENT_TYPE = "vnd.android.cursor.dir/calls"; | 
|  |  | 
|  | /** | 
|  | * The MIME type of a {@link #CONTENT_URI} sub-directory of a single | 
|  | * call. | 
|  | */ | 
|  | public static final String CONTENT_ITEM_TYPE = "vnd.android.cursor.item/calls"; | 
|  |  | 
|  | /** | 
|  | * The type of the call (incoming, outgoing or missed). | 
|  | * <P>Type: INTEGER (int)</P> | 
|  | * | 
|  | * <p> | 
|  | * Allowed values: | 
|  | * <ul> | 
|  | * <li>{@link #INCOMING_TYPE}</li> | 
|  | * <li>{@link #OUTGOING_TYPE}</li> | 
|  | * <li>{@link #MISSED_TYPE}</li> | 
|  | * <li>{@link #VOICEMAIL_TYPE}</li> | 
|  | * <li>{@link #REJECTED_TYPE}</li> | 
|  | * <li>{@link #BLOCKED_TYPE}</li> | 
|  | * <li>{@link #ANSWERED_EXTERNALLY_TYPE}</li> | 
|  | * </ul> | 
|  | * </p> | 
|  | */ | 
|  | public static final String TYPE = "type"; | 
|  |  | 
|  | /** Call log type for incoming calls. */ | 
|  | public static final int INCOMING_TYPE = 1; | 
|  | /** Call log type for outgoing calls. */ | 
|  | public static final int OUTGOING_TYPE = 2; | 
|  | /** Call log type for missed calls. */ | 
|  | public static final int MISSED_TYPE = 3; | 
|  | /** Call log type for voicemails. */ | 
|  | public static final int VOICEMAIL_TYPE = 4; | 
|  | /** Call log type for calls rejected by direct user action. */ | 
|  | public static final int REJECTED_TYPE = 5; | 
|  | /** Call log type for calls blocked automatically. */ | 
|  | public static final int BLOCKED_TYPE = 6; | 
|  | /** | 
|  | * Call log type for a call which was answered on another device.  Used in situations where | 
|  | * a call rings on multiple devices simultaneously and it ended up being answered on a | 
|  | * device other than the current one. | 
|  | */ | 
|  | public static final int ANSWERED_EXTERNALLY_TYPE = 7; | 
|  |  | 
|  | /** | 
|  | * Bit-mask describing features of the call (e.g. video). | 
|  | * | 
|  | * <P>Type: INTEGER (int)</P> | 
|  | */ | 
|  | public static final String FEATURES = "features"; | 
|  |  | 
|  | /** Call had video. */ | 
|  | public static final int FEATURES_VIDEO = 1 << 0; | 
|  |  | 
|  | /** Call was pulled externally. */ | 
|  | public static final int FEATURES_PULLED_EXTERNALLY = 1 << 1; | 
|  |  | 
|  | /** Call was HD. */ | 
|  | public static final int FEATURES_HD_CALL = 1 << 2; | 
|  |  | 
|  | /** Call was WIFI call. */ | 
|  | public static final int FEATURES_WIFI = 1 << 3; | 
|  |  | 
|  | /** | 
|  | * Indicates the call underwent Assisted Dialing. | 
|  | * @see TelecomManager#EXTRA_USE_ASSISTED_DIALING | 
|  | */ | 
|  | public static final int FEATURES_ASSISTED_DIALING_USED = 1 << 4; | 
|  |  | 
|  | /** Call was on RTT at some point */ | 
|  | public static final int FEATURES_RTT = 1 << 5; | 
|  |  | 
|  | /** Call was VoLTE */ | 
|  | public static final int FEATURES_VOLTE = 1 << 6; | 
|  |  | 
|  | /** | 
|  | * The phone number as the user entered it. | 
|  | * <P>Type: TEXT</P> | 
|  | */ | 
|  | public static final String NUMBER = "number"; | 
|  |  | 
|  |  | 
|  | /** | 
|  | * Boolean indicating whether the call is a business call. | 
|  | */ | 
|  | @FlaggedApi(Flags.FLAG_BUSINESS_CALL_COMPOSER) | 
|  | public static final String IS_BUSINESS_CALL = "is_business_call"; | 
|  |  | 
|  | /** | 
|  | * String that stores the asserted display name associated with business call. | 
|  | */ | 
|  | @FlaggedApi(Flags.FLAG_BUSINESS_CALL_COMPOSER) | 
|  | public static final String ASSERTED_DISPLAY_NAME = "asserted_display_name"; | 
|  |  | 
|  | /** | 
|  | * The number presenting rules set by the network. | 
|  | * | 
|  | * <p> | 
|  | * Allowed values: | 
|  | * <ul> | 
|  | * <li>{@link #PRESENTATION_ALLOWED}</li> | 
|  | * <li>{@link #PRESENTATION_RESTRICTED}</li> | 
|  | * <li>{@link #PRESENTATION_UNKNOWN}</li> | 
|  | * <li>{@link #PRESENTATION_PAYPHONE}</li> | 
|  | * <li>{@link #PRESENTATION_UNAVAILABLE}</li> | 
|  | * </ul> | 
|  | * </p> | 
|  | * | 
|  | * <P>Type: INTEGER</P> | 
|  | */ | 
|  | public static final String NUMBER_PRESENTATION = "presentation"; | 
|  |  | 
|  | /** Number is allowed to display for caller id. */ | 
|  | public static final int PRESENTATION_ALLOWED = 1; | 
|  | /** Number is blocked by user. */ | 
|  | public static final int PRESENTATION_RESTRICTED = 2; | 
|  | /** Number is not specified or unknown by network. */ | 
|  | public static final int PRESENTATION_UNKNOWN = 3; | 
|  | /** Number is a pay phone. */ | 
|  | public static final int PRESENTATION_PAYPHONE = 4; | 
|  | /** Number is unavailable. */ | 
|  | public static final int PRESENTATION_UNAVAILABLE = 5; | 
|  |  | 
|  | /** | 
|  | * The ISO 3166-1 two letters country code of the country where the | 
|  | * user received or made the call. | 
|  | * <P> | 
|  | * Type: TEXT | 
|  | * </P> | 
|  | */ | 
|  | public static final String COUNTRY_ISO = "countryiso"; | 
|  |  | 
|  | /** | 
|  | * The date the call occured, in milliseconds since the epoch | 
|  | * <P>Type: INTEGER (long)</P> | 
|  | */ | 
|  | public static final String DATE = "date"; | 
|  |  | 
|  | /** | 
|  | * The duration of the call in seconds | 
|  | * <P>Type: INTEGER (long)</P> | 
|  | */ | 
|  | public static final String DURATION = "duration"; | 
|  |  | 
|  | /** | 
|  | * The data usage of the call in bytes. | 
|  | * <P>Type: INTEGER (long)</P> | 
|  | */ | 
|  | public static final String DATA_USAGE = "data_usage"; | 
|  |  | 
|  | /** | 
|  | * Whether or not the call has been acknowledged | 
|  | * <P>Type: INTEGER (boolean)</P> | 
|  | */ | 
|  | public static final String NEW = "new"; | 
|  |  | 
|  | /** | 
|  | * The cached name associated with the phone number, if it exists. | 
|  | * | 
|  | * <p>This value is typically filled in by the dialer app for the caching purpose, | 
|  | * so it's not guaranteed to be present, and may not be current if the contact | 
|  | * information associated with this number has changed. | 
|  | * <P>Type: TEXT</P> | 
|  | */ | 
|  | public static final String CACHED_NAME = "name"; | 
|  |  | 
|  | /** | 
|  | * The cached number type (Home, Work, etc) associated with the | 
|  | * phone number, if it exists. | 
|  | * | 
|  | * <p>This value is typically filled in by the dialer app for the caching purpose, | 
|  | * so it's not guaranteed to be present, and may not be current if the contact | 
|  | * information associated with this number has changed. | 
|  | * <P>Type: INTEGER</P> | 
|  | */ | 
|  | public static final String CACHED_NUMBER_TYPE = "numbertype"; | 
|  |  | 
|  | /** | 
|  | * The cached number label, for a custom number type, associated with the | 
|  | * phone number, if it exists. | 
|  | * | 
|  | * <p>This value is typically filled in by the dialer app for the caching purpose, | 
|  | * so it's not guaranteed to be present, and may not be current if the contact | 
|  | * information associated with this number has changed. | 
|  | * <P>Type: TEXT</P> | 
|  | */ | 
|  | public static final String CACHED_NUMBER_LABEL = "numberlabel"; | 
|  |  | 
|  | /** | 
|  | * URI of the voicemail entry. Populated only for {@link #VOICEMAIL_TYPE}. | 
|  | * <P>Type: TEXT</P> | 
|  | */ | 
|  | public static final String VOICEMAIL_URI = "voicemail_uri"; | 
|  |  | 
|  | /** | 
|  | * Transcription of the call or voicemail entry. This will only be populated for call log | 
|  | * entries of type {@link #VOICEMAIL_TYPE} that have valid transcriptions. | 
|  | */ | 
|  | public static final String TRANSCRIPTION = "transcription"; | 
|  |  | 
|  | /** | 
|  | * State of voicemail transcription entry. This will only be populated for call log | 
|  | * entries of type {@link #VOICEMAIL_TYPE}. | 
|  | * @hide | 
|  | */ | 
|  | public static final String TRANSCRIPTION_STATE = "transcription_state"; | 
|  |  | 
|  | /** | 
|  | * Whether this item has been read or otherwise consumed by the user. | 
|  | * <p> | 
|  | * Unlike the {@link #NEW} field, which requires the user to have acknowledged the | 
|  | * existence of the entry, this implies the user has interacted with the entry. | 
|  | * <P>Type: INTEGER (boolean)</P> | 
|  | */ | 
|  | public static final String IS_READ = "is_read"; | 
|  |  | 
|  | /** | 
|  | * A geocoded location for the number associated with this call. | 
|  | * <p> | 
|  | * The string represents a city, state, or country associated with the number. | 
|  | * <P>Type: TEXT</P> | 
|  | */ | 
|  | public static final String GEOCODED_LOCATION = "geocoded_location"; | 
|  |  | 
|  | /** | 
|  | * The cached URI to look up the contact associated with the phone number, if it exists. | 
|  | * | 
|  | * <p>This value is typically filled in by the dialer app for the caching purpose, | 
|  | * so it's not guaranteed to be present, and may not be current if the contact | 
|  | * information associated with this number has changed. | 
|  | * <P>Type: TEXT</P> | 
|  | */ | 
|  | public static final String CACHED_LOOKUP_URI = "lookup_uri"; | 
|  |  | 
|  | /** | 
|  | * The cached phone number of the contact which matches this entry, if it exists. | 
|  | * | 
|  | * <p>This value is typically filled in by the dialer app for the caching purpose, | 
|  | * so it's not guaranteed to be present, and may not be current if the contact | 
|  | * information associated with this number has changed. | 
|  | * <P>Type: TEXT</P> | 
|  | */ | 
|  | public static final String CACHED_MATCHED_NUMBER = "matched_number"; | 
|  |  | 
|  | /** | 
|  | * The cached normalized(E164) version of the phone number, if it exists. | 
|  | * | 
|  | * <p>This value is typically filled in by the dialer app for the caching purpose, | 
|  | * so it's not guaranteed to be present, and may not be current if the contact | 
|  | * information associated with this number has changed. | 
|  | * <P>Type: TEXT</P> | 
|  | */ | 
|  | public static final String CACHED_NORMALIZED_NUMBER = "normalized_number"; | 
|  |  | 
|  | /** | 
|  | * The cached photo id of the picture associated with the phone number, if it exists. | 
|  | * | 
|  | * <p>This value is typically filled in by the dialer app for the caching purpose, | 
|  | * so it's not guaranteed to be present, and may not be current if the contact | 
|  | * information associated with this number has changed. | 
|  | * <P>Type: INTEGER (long)</P> | 
|  | */ | 
|  | public static final String CACHED_PHOTO_ID = "photo_id"; | 
|  |  | 
|  | /** | 
|  | * The cached photo URI of the picture associated with the phone number, if it exists. | 
|  | * | 
|  | * <p>This value is typically filled in by the dialer app for the caching purpose, | 
|  | * so it's not guaranteed to be present, and may not be current if the contact | 
|  | * information associated with this number has changed. | 
|  | * <P>Type: TEXT (URI)</P> | 
|  | */ | 
|  | public static final String CACHED_PHOTO_URI = "photo_uri"; | 
|  |  | 
|  | /** | 
|  | * The cached phone number, formatted with formatting rules based on the country the | 
|  | * user was in when the call was made or received. | 
|  | * | 
|  | * <p>This value is typically filled in by the dialer app for the caching purpose, | 
|  | * so it's not guaranteed to be present, and may not be current if the contact | 
|  | * information associated with this number has changed. | 
|  | * <P>Type: TEXT</P> | 
|  | */ | 
|  | public static final String CACHED_FORMATTED_NUMBER = "formatted_number"; | 
|  |  | 
|  | // Note: PHONE_ACCOUNT_* constant values are "subscription_*" due to a historic naming | 
|  | // that was encoded into call log databases. | 
|  |  | 
|  | /** | 
|  | * The component name of the account used to place or receive the call; in string form. | 
|  | * <P>Type: TEXT</P> | 
|  | */ | 
|  | public static final String PHONE_ACCOUNT_COMPONENT_NAME = "subscription_component_name"; | 
|  |  | 
|  | /** | 
|  | * The identifier for the account used to place or receive the call. | 
|  | * <P>Type: TEXT</P> | 
|  | */ | 
|  | public static final String PHONE_ACCOUNT_ID = "subscription_id"; | 
|  |  | 
|  | /** | 
|  | * The address associated with the account used to place or receive the call; in string | 
|  | * form. For SIM-based calls, this is the user's own phone number. | 
|  | * <P>Type: TEXT</P> | 
|  | * | 
|  | * @hide | 
|  | */ | 
|  | public static final String PHONE_ACCOUNT_ADDRESS = "phone_account_address"; | 
|  |  | 
|  | /** | 
|  | * Indicates that the entry will be hidden from all queries until the associated | 
|  | * {@link android.telecom.PhoneAccount} is registered with the system. | 
|  | * <P>Type: INTEGER</P> | 
|  | * | 
|  | * @hide | 
|  | */ | 
|  | public static final String PHONE_ACCOUNT_HIDDEN = "phone_account_hidden"; | 
|  |  | 
|  | /** | 
|  | * The subscription ID used to place this call.  This is no longer used and has been | 
|  | * replaced with PHONE_ACCOUNT_COMPONENT_NAME/PHONE_ACCOUNT_ID. | 
|  | * For ContactsProvider internal use only. | 
|  | * <P>Type: INTEGER</P> | 
|  | * | 
|  | * @Deprecated | 
|  | * @hide | 
|  | */ | 
|  | public static final String SUB_ID = "sub_id"; | 
|  |  | 
|  | /** | 
|  | * The post-dial portion of a dialed number, including any digits dialed after a | 
|  | * {@link TelecomManager#DTMF_CHARACTER_PAUSE} or a {@link | 
|  | * TelecomManager#DTMF_CHARACTER_WAIT} and these characters themselves. | 
|  | * <P>Type: TEXT</P> | 
|  | */ | 
|  | public static final String POST_DIAL_DIGITS = "post_dial_digits"; | 
|  |  | 
|  | /** | 
|  | * For an incoming call, the secondary line number the call was received via. | 
|  | * When a SIM card has multiple phone numbers associated with it, the via number indicates | 
|  | * which of the numbers associated with the SIM was called. | 
|  | */ | 
|  | public static final String VIA_NUMBER = "via_number"; | 
|  |  | 
|  | /** | 
|  | * Indicates that the entry will be copied from primary user to other users. | 
|  | * <P>Type: INTEGER</P> | 
|  | * | 
|  | * @hide | 
|  | */ | 
|  | public static final String ADD_FOR_ALL_USERS = "add_for_all_users"; | 
|  |  | 
|  | /** | 
|  | * The date the row is last inserted, updated, or marked as deleted, in milliseconds | 
|  | * since the epoch. Read only. | 
|  | * <P>Type: INTEGER (long)</P> | 
|  | */ | 
|  | public static final String LAST_MODIFIED = "last_modified"; | 
|  |  | 
|  | /** | 
|  | * If a successful call is made that is longer than this duration, update the phone number | 
|  | * in the ContactsProvider with the normalized version of the number, based on the user's | 
|  | * current country code. | 
|  | */ | 
|  | private static final int MIN_DURATION_FOR_NORMALIZED_NUMBER_UPDATE_MS = 1000 * 10; | 
|  |  | 
|  | /** | 
|  | * Value for {@link CallLog.Calls#BLOCK_REASON}, set as the default value when a call was | 
|  | * not blocked by a CallScreeningService or any other system call blocking method. | 
|  | */ | 
|  | public static final int BLOCK_REASON_NOT_BLOCKED = 0; | 
|  |  | 
|  | /** | 
|  | * Value for {@link CallLog.Calls#BLOCK_REASON}, set when {@link CallLog.Calls#TYPE} is | 
|  | * {@link CallLog.Calls#BLOCKED_TYPE} to indicate that a call was blocked by a | 
|  | * CallScreeningService. The {@link CallLog.Calls#CALL_SCREENING_COMPONENT_NAME} and | 
|  | * {@link CallLog.Calls#CALL_SCREENING_APP_NAME} columns will indicate which call screening | 
|  | * service was responsible for blocking the call. | 
|  | */ | 
|  | public static final int BLOCK_REASON_CALL_SCREENING_SERVICE = 1; | 
|  |  | 
|  | /** | 
|  | * Value for {@link CallLog.Calls#BLOCK_REASON}, set when {@link CallLog.Calls#TYPE} is | 
|  | * {@link CallLog.Calls#BLOCKED_TYPE} to indicate that a call was blocked because the user | 
|  | * configured a contact to be sent directly to voicemail. | 
|  | */ | 
|  | public static final int BLOCK_REASON_DIRECT_TO_VOICEMAIL = 2; | 
|  |  | 
|  | /** | 
|  | * Value for {@link CallLog.Calls#BLOCK_REASON}, set when {@link CallLog.Calls#TYPE} is | 
|  | * {@link CallLog.Calls#BLOCKED_TYPE} to indicate that a call was blocked because it is | 
|  | * in the BlockedNumbers provider. | 
|  | */ | 
|  | public static final int BLOCK_REASON_BLOCKED_NUMBER = 3; | 
|  |  | 
|  | /** | 
|  | * Value for {@link CallLog.Calls#BLOCK_REASON}, set when {@link CallLog.Calls#TYPE} is | 
|  | * {@link CallLog.Calls#BLOCKED_TYPE} to indicate that a call was blocked because the user | 
|  | * has chosen to block all calls from unknown numbers. | 
|  | */ | 
|  | public static final int BLOCK_REASON_UNKNOWN_NUMBER = 4; | 
|  |  | 
|  | /** | 
|  | * Value for {@link CallLog.Calls#BLOCK_REASON}, set when {@link CallLog.Calls#TYPE} is | 
|  | * {@link CallLog.Calls#BLOCKED_TYPE} to indicate that a call was blocked because the user | 
|  | * has chosen to block all calls from restricted numbers. | 
|  | */ | 
|  | public static final int BLOCK_REASON_RESTRICTED_NUMBER = 5; | 
|  |  | 
|  | /** | 
|  | * Value for {@link CallLog.Calls#BLOCK_REASON}, set when {@link CallLog.Calls#TYPE} is | 
|  | * {@link CallLog.Calls#BLOCKED_TYPE} to indicate that a call was blocked because the user | 
|  | * has chosen to block all calls from pay phones. | 
|  | */ | 
|  | public static final int BLOCK_REASON_PAY_PHONE = 6; | 
|  |  | 
|  | /** | 
|  | * Value for {@link CallLog.Calls#BLOCK_REASON}, set when {@link CallLog.Calls#TYPE} is | 
|  | * {@link CallLog.Calls#BLOCKED_TYPE} to indicate that a call was blocked because the user | 
|  | * has chosen to block all calls from numbers not in their contacts. | 
|  | */ | 
|  | public static final int BLOCK_REASON_NOT_IN_CONTACTS = 7; | 
|  |  | 
|  | /** | 
|  | * The ComponentName of the CallScreeningService which blocked this call. Will be | 
|  | * populated when the {@link CallLog.Calls#TYPE} is {@link CallLog.Calls#BLOCKED_TYPE}. | 
|  | * <P>Type: TEXT</P> | 
|  | */ | 
|  | public static final String CALL_SCREENING_COMPONENT_NAME = "call_screening_component_name"; | 
|  |  | 
|  | /** | 
|  | * The name of the app which blocked a call. Will be populated when the | 
|  | * {@link CallLog.Calls#TYPE} is {@link CallLog.Calls#BLOCKED_TYPE}. Provided as a | 
|  | * convenience so that the call log can still indicate which app blocked a call, even if | 
|  | * that app is no longer installed. | 
|  | * <P>Type: TEXT</P> | 
|  | */ | 
|  | public static final String CALL_SCREENING_APP_NAME = "call_screening_app_name"; | 
|  |  | 
|  | /** | 
|  | * Where the {@link CallLog.Calls#TYPE} is {@link CallLog.Calls#BLOCKED_TYPE}, | 
|  | * indicates the reason why a call is blocked. | 
|  | * <P>Type: INTEGER</P> | 
|  | * | 
|  | * <p> | 
|  | * Allowed values: | 
|  | * <ul> | 
|  | * <li>{@link CallLog.Calls#BLOCK_REASON_NOT_BLOCKED}</li> | 
|  | * <li>{@link CallLog.Calls#BLOCK_REASON_CALL_SCREENING_SERVICE}</li> | 
|  | * <li>{@link CallLog.Calls#BLOCK_REASON_DIRECT_TO_VOICEMAIL}</li> | 
|  | * <li>{@link CallLog.Calls#BLOCK_REASON_BLOCKED_NUMBER}</li> | 
|  | * <li>{@link CallLog.Calls#BLOCK_REASON_UNKNOWN_NUMBER}</li> | 
|  | * <li>{@link CallLog.Calls#BLOCK_REASON_RESTRICTED_NUMBER}</li> | 
|  | * <li>{@link CallLog.Calls#BLOCK_REASON_PAY_PHONE}</li> | 
|  | * <li>{@link CallLog.Calls#BLOCK_REASON_NOT_IN_CONTACTS}</li> | 
|  | * </ul> | 
|  | * </p> | 
|  | */ | 
|  | public static final String BLOCK_REASON = "block_reason"; | 
|  |  | 
|  | /** @hide */ | 
|  | @LongDef(flag = true, value = { | 
|  | MISSED_REASON_NOT_MISSED, | 
|  | AUTO_MISSED_EMERGENCY_CALL, | 
|  | AUTO_MISSED_MAXIMUM_RINGING, | 
|  | AUTO_MISSED_MAXIMUM_DIALING, | 
|  | USER_MISSED_NO_ANSWER, | 
|  | USER_MISSED_SHORT_RING, | 
|  | USER_MISSED_DND_MODE, | 
|  | USER_MISSED_LOW_RING_VOLUME, | 
|  | USER_MISSED_NO_VIBRATE, | 
|  | USER_MISSED_CALL_SCREENING_SERVICE_SILENCED, | 
|  | USER_MISSED_CALL_FILTERS_TIMEOUT, | 
|  | USER_MISSED_NEVER_RANG, | 
|  | USER_MISSED_NOT_RUNNING | 
|  | }) | 
|  | @Retention(RetentionPolicy.SOURCE) | 
|  | public @interface MissedReason {} | 
|  |  | 
|  | /** | 
|  | * Value for {@link CallLog.Calls#MISSED_REASON}, set as the default value when a call was | 
|  | * not missed. | 
|  | */ | 
|  | public static final long MISSED_REASON_NOT_MISSED = 0; | 
|  |  | 
|  | /** | 
|  | * Value for {@link CallLog.Calls#MISSED_REASON}, set when {@link CallLog.Calls#TYPE} is | 
|  | * {@link CallLog.Calls#MISSED_TYPE} to indicate that a call was automatically rejected by | 
|  | * system because an ongoing emergency call. | 
|  | */ | 
|  | public static final long AUTO_MISSED_EMERGENCY_CALL = 1 << 0; | 
|  |  | 
|  | /** | 
|  | * Value for {@link CallLog.Calls#MISSED_REASON}, set when {@link CallLog.Calls#TYPE} is | 
|  | * {@link CallLog.Calls#MISSED_TYPE} to indicate that a call was automatically rejected by | 
|  | * system because the system cannot support any more ringing calls. | 
|  | */ | 
|  | public static final long AUTO_MISSED_MAXIMUM_RINGING = 1 << 1; | 
|  |  | 
|  | /** | 
|  | * Value for {@link CallLog.Calls#MISSED_REASON}, set when {@link CallLog.Calls#TYPE} is | 
|  | * {@link CallLog.Calls#MISSED_TYPE} to indicate that a call was automatically rejected by | 
|  | * system because the system cannot support any more dialing calls. | 
|  | */ | 
|  | public static final long AUTO_MISSED_MAXIMUM_DIALING = 1 << 2; | 
|  |  | 
|  | /** | 
|  | * When {@link CallLog.Calls#TYPE} is {@link CallLog.Calls#MISSED_TYPE}, set this bit when | 
|  | * the call was missed just because user didn't answer it. | 
|  | */ | 
|  | public static final long USER_MISSED_NO_ANSWER = 1 << 16; | 
|  |  | 
|  | /** | 
|  | * When {@link CallLog.Calls#TYPE} is {@link CallLog.Calls#MISSED_TYPE}, set this bit when | 
|  | * this call rang for a short period of time. | 
|  | */ | 
|  | public static final long USER_MISSED_SHORT_RING = 1 << 17; | 
|  |  | 
|  | /** | 
|  | * When {@link CallLog.Calls#TYPE} is {@link CallLog.Calls#MISSED_TYPE}, when this call | 
|  | * rings less than this defined time in millisecond, set | 
|  | * {@link CallLog.Calls#USER_MISSED_SHORT_RING} bit. | 
|  | * @hide | 
|  | */ | 
|  | public static final long SHORT_RING_THRESHOLD = 5000L; | 
|  |  | 
|  | /** | 
|  | * When {@link CallLog.Calls#TYPE} is {@link CallLog.Calls#MISSED_TYPE}, set this bit when | 
|  | * this call is silenced because the phone is in 'do not disturb mode'. | 
|  | */ | 
|  | public static final long USER_MISSED_DND_MODE = 1 << 18; | 
|  |  | 
|  | /** | 
|  | * When {@link CallLog.Calls#TYPE} is {@link CallLog.Calls#MISSED_TYPE}, set this bit when | 
|  | * this call rings with a low ring volume. | 
|  | */ | 
|  | public static final long USER_MISSED_LOW_RING_VOLUME = 1 << 19; | 
|  |  | 
|  | /** | 
|  | * When {@link CallLog.Calls#TYPE} is {@link CallLog.Calls#MISSED_TYPE}, when this call | 
|  | * rings in volume less than this defined volume threshold, set | 
|  | * {@link CallLog.Calls#USER_MISSED_LOW_RING_VOLUME} bit. | 
|  | * @hide | 
|  | */ | 
|  | public static final int LOW_RING_VOLUME = 0; | 
|  |  | 
|  | /** | 
|  | * When {@link CallLog.Calls#TYPE} is {@link CallLog.Calls#MISSED_TYPE} set this bit when | 
|  | * this call rings without vibration. | 
|  | */ | 
|  | public static final long USER_MISSED_NO_VIBRATE = 1 << 20; | 
|  |  | 
|  | /** | 
|  | * When {@link CallLog.Calls#TYPE} is {@link CallLog.Calls#MISSED_TYPE}, set this bit when | 
|  | * this call is silenced by the call screening service. | 
|  | */ | 
|  | public static final long USER_MISSED_CALL_SCREENING_SERVICE_SILENCED = 1 << 21; | 
|  |  | 
|  | /** | 
|  | * When {@link CallLog.Calls#TYPE} is {@link CallLog.Calls#MISSED_TYPE}, set this bit when | 
|  | * the call filters timed out. | 
|  | */ | 
|  | public static final long USER_MISSED_CALL_FILTERS_TIMEOUT = 1 << 22; | 
|  |  | 
|  | /** | 
|  | * When {@link CallLog.Calls#TYPE} is {@link CallLog.Calls#MISSED_TYPE}, set this bit when | 
|  | * the call ended before ringing. | 
|  | * @hide | 
|  | */ | 
|  | public static final long USER_MISSED_NEVER_RANG = 1 << 23; | 
|  |  | 
|  | /** | 
|  | * When {@link CallLog.Calls#TYPE} is {@link CallLog.Calls#MISSED_TYPE}, set this bit when | 
|  | * the user receiving the call is not running (i.e. work profile paused). | 
|  | * @hide | 
|  | */ | 
|  | public static final long USER_MISSED_NOT_RUNNING = 1 << 24; | 
|  |  | 
|  | /** | 
|  | * Where the {@link CallLog.Calls#TYPE} is {@link CallLog.Calls#MISSED_TYPE}, | 
|  | * indicates factors which may have lead the user to miss the call. | 
|  | * <P>Type: INTEGER</P> | 
|  | * | 
|  | * <p> | 
|  | * There are two main cases. Auto missed cases and user missed cases. Default value is: | 
|  | * <ul> | 
|  | * <li>{@link CallLog.Calls#MISSED_REASON_NOT_MISSED}</li> | 
|  | * </ul> | 
|  | * </p> | 
|  | * <P> | 
|  | * Auto missed cases are those where a call was missed because it was not possible for the | 
|  | * incoming call to be presented to the user at all. Possible values are: | 
|  | * <ul> | 
|  | * <li>{@link CallLog.Calls#AUTO_MISSED_EMERGENCY_CALL}</li> | 
|  | * <li>{@link CallLog.Calls#AUTO_MISSED_MAXIMUM_RINGING}</li> | 
|  | * <li>{@link CallLog.Calls#AUTO_MISSED_MAXIMUM_DIALING}</li> | 
|  | * </ul> | 
|  | * </P> | 
|  | * <P> | 
|  | * User missed cases are those where the incoming call was presented to the user, but | 
|  | * factors such as a low ringing volume may have contributed to the call being missed. | 
|  | * Following bits can be set to indicate possible reasons for this: | 
|  | * <ul> | 
|  | * <li>{@link CallLog.Calls#USER_MISSED_SHORT_RING}</li> | 
|  | * <li>{@link CallLog.Calls#USER_MISSED_DND_MODE}</li> | 
|  | * <li>{@link CallLog.Calls#USER_MISSED_LOW_RING_VOLUME}</li> | 
|  | * <li>{@link CallLog.Calls#USER_MISSED_NO_VIBRATE}</li> | 
|  | * <li>{@link CallLog.Calls#USER_MISSED_CALL_SCREENING_SERVICE_SILENCED}</li> | 
|  | * <li>{@link CallLog.Calls#USER_MISSED_CALL_FILTERS_TIMEOUT}</li> | 
|  | * </ul> | 
|  | * </P> | 
|  | */ | 
|  | public static final String MISSED_REASON = "missed_reason"; | 
|  |  | 
|  | /** | 
|  | * The subject of the call, as delivered via call composer. | 
|  | * | 
|  | * For outgoing calls, contains the subject set by the local user. For incoming calls, | 
|  | * contains the subject set by the remote caller. May be null if no subject was set. | 
|  | * <p>Type: TEXT</p> | 
|  | */ | 
|  | public static final String SUBJECT = "subject"; | 
|  |  | 
|  | /** | 
|  | * Used as a value in the {@link #PRIORITY} column. | 
|  | * | 
|  | * Indicates that the call is of normal priority. This is also the default value for calls | 
|  | * that did not include call composer elements. | 
|  | */ | 
|  | public static final int PRIORITY_NORMAL = 0; | 
|  |  | 
|  | /** | 
|  | * Used as a value in the {@link #PRIORITY} column. | 
|  | * | 
|  | * Indicates that the call is of urgent priority. | 
|  | */ | 
|  | public static final int PRIORITY_URGENT = 1; | 
|  |  | 
|  | /** | 
|  | * The priority of the call, as delivered via call composer. | 
|  | * | 
|  | * For outgoing calls, contains the priority set by the local user. For incoming calls, | 
|  | * contains the priority set by the remote caller. If no priority was set or the call | 
|  | * did not include call composer elements, defaults to {@link #PRIORITY_NORMAL}. | 
|  | * Valid values are {@link #PRIORITY_NORMAL} and {@link #PRIORITY_URGENT}. | 
|  | * <p>Type: INTEGER</p> | 
|  | */ | 
|  | public static final String PRIORITY = "priority"; | 
|  |  | 
|  | /** | 
|  | * A reference to the picture that was sent via call composer. | 
|  | * | 
|  | * The string contained in this field should be converted to an {@link Uri} via | 
|  | * {@link Uri#parse(String)}, then passed to {@link ContentResolver#openFileDescriptor} | 
|  | * in order to obtain a file descriptor to access the picture data. | 
|  | * | 
|  | * The user may choose to delete the picture associated with a call independently of the | 
|  | * call log entry, in which case {@link ContentResolver#openFileDescriptor} may throw a | 
|  | * {@link FileNotFoundException}. | 
|  | * | 
|  | * Note that pictures sent or received via call composer will not be included in any | 
|  | * backups of the call log. | 
|  | * | 
|  | * <p>Type: TEXT</p> | 
|  | */ | 
|  | public static final String COMPOSER_PHOTO_URI = "composer_photo_uri"; | 
|  |  | 
|  | /** | 
|  | * A reference to the location that was sent via call composer. | 
|  | * | 
|  | * This column contains the content URI of the corresponding entry in {@link Locations} | 
|  | * table, which contains the actual location data. The | 
|  | * {@link Manifest.permission#ACCESS_FINE_LOCATION} permission is required to access that | 
|  | * table. | 
|  | * | 
|  | * If your app has the appropriate permissions, the location data may be obtained by | 
|  | * converting the value of this column to an {@link Uri} via {@link Uri#parse}, then passing | 
|  | * the result to {@link ContentResolver#query}. | 
|  | * | 
|  | * The user may choose to delete the location associated with a call independently of the | 
|  | * call log entry, in which case the {@link Cursor} returned from | 
|  | * {@link ContentResolver#query} will either be {@code null} or empty, with | 
|  | * {@link Cursor#getCount()} returning {@code 0}. | 
|  | * | 
|  | * This column will not be populated when a call is received while the device is locked, and | 
|  | * it will not be part of any backups. | 
|  | * | 
|  | * <p>Type: TEXT</p> | 
|  | */ | 
|  | public static final String LOCATION = "location"; | 
|  |  | 
|  | /** | 
|  | * A reference to indicate whether phone account migration process is pending. | 
|  | * | 
|  | * Before Android 13, {@link PhoneAccountHandle#getId()} returns the ICCID for Telephony | 
|  | * PhoneAccountHandle. Starting from Android 13, {@link PhoneAccountHandle#getId()} returns | 
|  | * the Subscription ID for Telephony PhoneAccountHandle. A phone account migration process | 
|  | * is to ensure this PhoneAccountHandle migration process cross the Android versions in | 
|  | * the CallLog database. | 
|  | * | 
|  | * <p>Type: INTEGER</p> | 
|  | * @hide | 
|  | */ | 
|  | public static final String IS_PHONE_ACCOUNT_MIGRATION_PENDING = | 
|  | "is_call_log_phone_account_migration_pending"; | 
|  |  | 
|  | /** | 
|  | * Adds a call to the call log. | 
|  | * | 
|  | * @param ci the CallerInfo object to get the target contact from.  Can be null | 
|  | * if the contact is unknown. | 
|  | * @param context the context used to get the ContentResolver | 
|  | * @param number the phone number to be added to the calls db | 
|  | * @param presentation enum value from TelecomManager.PRESENTATION_xxx, which | 
|  | *        is set by the network and denotes the number presenting rules for | 
|  | *        "allowed", "payphone", "restricted" or "unknown" | 
|  | * @param callType enumerated values for "incoming", "outgoing", or "missed" | 
|  | * @param features features of the call (e.g. Video). | 
|  | * @param accountHandle The accountHandle object identifying the provider of the call | 
|  | * @param start time stamp for the call in milliseconds | 
|  | * @param duration call duration in seconds | 
|  | * @param dataUsage data usage for the call in bytes, null if data usage was not tracked for | 
|  | *                  the call. | 
|  | * @param isPhoneAccountMigrationPending whether the PhoneAccountHandle ID need to migrate | 
|  | * @result The URI of the call log entry belonging to the user that made or received this | 
|  | *        call. | 
|  | * {@hide} | 
|  | */ | 
|  | public static Uri addCall(CallerInfo ci, Context context, String number, | 
|  | int presentation, int callType, int features, | 
|  | PhoneAccountHandle accountHandle, | 
|  | long start, int duration, Long dataUsage, long missedReason, | 
|  | int isPhoneAccountMigrationPending) { | 
|  | return addCall(ci, context, number, "" /* postDialDigits */, "" /* viaNumber */, | 
|  | presentation, callType, features, accountHandle, start, duration, | 
|  | dataUsage, false /* addForAllUsers */, null /* userToBeInsertedTo */, | 
|  | false /* isRead */, Calls.BLOCK_REASON_NOT_BLOCKED /* callBlockReason */, | 
|  | null /* callScreeningAppName */, null /* callScreeningComponentName */, | 
|  | missedReason, isPhoneAccountMigrationPending); | 
|  | } | 
|  |  | 
|  |  | 
|  | /** | 
|  | * Adds a call to the call log. | 
|  | * | 
|  | * @param ci the CallerInfo object to get the target contact from.  Can be null | 
|  | * if the contact is unknown. | 
|  | * @param context the context used to get the ContentResolver | 
|  | * @param number the phone number to be added to the calls db | 
|  | * @param viaNumber the secondary number that the incoming call received with. If the | 
|  | *       call was received with the SIM assigned number, then this field must be ''. | 
|  | * @param presentation enum value from TelecomManager.PRESENTATION_xxx, which | 
|  | *        is set by the network and denotes the number presenting rules for | 
|  | *        "allowed", "payphone", "restricted" or "unknown" | 
|  | * @param callType enumerated values for "incoming", "outgoing", or "missed" | 
|  | * @param features features of the call (e.g. Video). | 
|  | * @param accountHandle The accountHandle object identifying the provider of the call | 
|  | * @param start time stamp for the call in milliseconds | 
|  | * @param duration call duration in seconds | 
|  | * @param dataUsage data usage for the call in bytes, null if data usage was not tracked for | 
|  | *                  the call. | 
|  | * @param addForAllUsers If true, the call is added to the call log of all currently | 
|  | *        running users. The caller must have the MANAGE_USERS permission if this is true. | 
|  | * @param userToBeInsertedTo {@link UserHandle} of user that the call is going to be | 
|  | *                           inserted to. null if it is inserted to the current user. The | 
|  | *                           value is ignored if @{link addForAllUsers} is true. | 
|  | * @param isPhoneAccountMigrationPending whether the PhoneAccountHandle ID need to migrate | 
|  | * @result The URI of the call log entry belonging to the user that made or received this | 
|  | *        call. | 
|  | * {@hide} | 
|  | */ | 
|  | public static Uri addCall(CallerInfo ci, Context context, String number, | 
|  | String postDialDigits, String viaNumber, int presentation, int callType, | 
|  | int features, PhoneAccountHandle accountHandle, long start, int duration, | 
|  | Long dataUsage, boolean addForAllUsers, UserHandle userToBeInsertedTo, | 
|  | long missedReason, int isPhoneAccountMigrationPending) { | 
|  | return addCall(ci, context, number, postDialDigits, viaNumber, presentation, callType, | 
|  | features, accountHandle, start, duration, dataUsage, addForAllUsers, | 
|  | userToBeInsertedTo, false /* isRead */ , Calls.BLOCK_REASON_NOT_BLOCKED | 
|  | /* callBlockReason */, null /* callScreeningAppName */, | 
|  | null /* callScreeningComponentName */, missedReason, | 
|  | isPhoneAccountMigrationPending); | 
|  | } | 
|  |  | 
|  |  | 
|  | /** | 
|  | * Adds a call to the call log. | 
|  | * | 
|  | * @param ci the CallerInfo object to get the target contact from.  Can be null | 
|  | * if the contact is unknown. | 
|  | * @param context the context used to get the ContentResolver | 
|  | * @param number the phone number to be added to the calls db | 
|  | * @param postDialDigits the post-dial digits that were dialed after the number, | 
|  | *        if it was outgoing. Otherwise it is ''. | 
|  | * @param viaNumber the secondary number that the incoming call received with. If the | 
|  | *        call was received with the SIM assigned number, then this field must be ''. | 
|  | * @param presentation enum value from TelecomManager.PRESENTATION_xxx, which | 
|  | *        is set by the network and denotes the number presenting rules for | 
|  | *        "allowed", "payphone", "restricted" or "unknown" | 
|  | * @param callType enumerated values for "incoming", "outgoing", or "missed" | 
|  | * @param features features of the call (e.g. Video). | 
|  | * @param accountHandle The accountHandle object identifying the provider of the call | 
|  | * @param start time stamp for the call in milliseconds | 
|  | * @param duration call duration in seconds | 
|  | * @param dataUsage data usage for the call in bytes, null if data usage was not tracked for | 
|  | *                  the call. | 
|  | * @param addForAllUsers If true, the call is added to the call log of all currently | 
|  | *        running users. The caller must have the MANAGE_USERS permission if this is true. | 
|  | * @param userToBeInsertedTo {@link UserHandle} of user that the call is going to be | 
|  | *                           inserted to. null if it is inserted to the current user. The | 
|  | *                           value is ignored if @{link addForAllUsers} is true. | 
|  | * @param isRead Flag to show if the missed call log has been read by the user or not. | 
|  | *                Used for call log restore of missed calls. | 
|  | * @param callBlockReason The reason why the call is blocked. | 
|  | * @param callScreeningAppName The call screening application name which block the call. | 
|  | * @param callScreeningComponentName The call screening component name which block the call. | 
|  | * @param missedReason The encoded missed information of the call. | 
|  | * @param isPhoneAccountMigrationPending whether the PhoneAccountHandle ID need to migrate | 
|  | * | 
|  | * @result The URI of the call log entry belonging to the user that made or received this | 
|  | *        call.  This could be of the shadow provider.  Do not return it to non-system apps, | 
|  | *        as they don't have permissions. | 
|  | * {@hide} | 
|  | */ | 
|  | @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023) | 
|  | public static Uri addCall(CallerInfo ci, Context context, String number, | 
|  | String postDialDigits, String viaNumber, int presentation, int callType, | 
|  | int features, PhoneAccountHandle accountHandle, long start, int duration, | 
|  | Long dataUsage, boolean addForAllUsers, UserHandle userToBeInsertedTo, | 
|  | boolean isRead, int callBlockReason, CharSequence callScreeningAppName, | 
|  | String callScreeningComponentName, long missedReason, | 
|  | int isPhoneAccountMigrationPending) { | 
|  | AddCallParams.AddCallParametersBuilder builder = | 
|  | new AddCallParams.AddCallParametersBuilder(); | 
|  | builder.setCallerInfo(ci); | 
|  | builder.setNumber(number); | 
|  | builder.setPostDialDigits(postDialDigits); | 
|  | builder.setViaNumber(viaNumber); | 
|  | builder.setPresentation(presentation); | 
|  | builder.setCallType(callType); | 
|  | builder.setFeatures(features); | 
|  | builder.setAccountHandle(accountHandle); | 
|  | builder.setStart(start); | 
|  | builder.setDuration(duration); | 
|  | builder.setDataUsage(dataUsage == null ? Long.MIN_VALUE : dataUsage); | 
|  | builder.setAddForAllUsers(addForAllUsers); | 
|  | builder.setUserToBeInsertedTo(userToBeInsertedTo); | 
|  | builder.setIsRead(isRead); | 
|  | builder.setCallBlockReason(callBlockReason); | 
|  | builder.setCallScreeningAppName(callScreeningAppName); | 
|  | builder.setCallScreeningComponentName(callScreeningComponentName); | 
|  | builder.setMissedReason(missedReason); | 
|  | builder.setIsPhoneAccountMigrationPending(isPhoneAccountMigrationPending); | 
|  |  | 
|  | return addCall(context, builder.build()); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Adds a call to the call log, using the provided parameters | 
|  | * @result The URI of the call log entry belonging to the user that made or received this | 
|  | *        call.  This could be of the shadow provider.  Do not return it to non-system apps, | 
|  | *        as they don't have permissions. | 
|  | * @hide | 
|  | */ | 
|  | public static @NonNull Uri addCall( | 
|  | @NonNull Context context, @NonNull AddCallParams params) { | 
|  | if (VERBOSE_LOG) { | 
|  | Log.v(LOG_TAG, String.format("Add call: number=%s, user=%s, for all=%s", | 
|  | params.mNumber, params.mUserToBeInsertedTo, params.mAddForAllUsers)); | 
|  | } | 
|  | final ContentResolver resolver = context.getContentResolver(); | 
|  |  | 
|  | String accountAddress = getLogAccountAddress(context, params.mAccountHandle); | 
|  |  | 
|  | int numberPresentation = getLogNumberPresentation(params.mNumber, params.mPresentation); | 
|  | String name = (params.mCallerInfo != null) ? params.mCallerInfo.getName() : ""; | 
|  | if (numberPresentation != PRESENTATION_ALLOWED) { | 
|  | params.mNumber = ""; | 
|  | if (params.mCallerInfo != null) { | 
|  | name = ""; | 
|  | } | 
|  | } | 
|  |  | 
|  | // accountHandle information | 
|  | String accountComponentString = null; | 
|  | String accountId = null; | 
|  | if (params.mAccountHandle != null) { | 
|  | accountComponentString = params.mAccountHandle.getComponentName().flattenToString(); | 
|  | accountId = params.mAccountHandle.getId(); | 
|  | } | 
|  |  | 
|  | ContentValues values = new ContentValues(14); | 
|  | values.put(NUMBER, params.mNumber); | 
|  | values.put(POST_DIAL_DIGITS, params.mPostDialDigits); | 
|  | values.put(VIA_NUMBER, params.mViaNumber); | 
|  | values.put(NUMBER_PRESENTATION, Integer.valueOf(numberPresentation)); | 
|  | values.put(TYPE, Integer.valueOf(params.mCallType)); | 
|  | values.put(FEATURES, params.mFeatures); | 
|  | values.put(DATE, Long.valueOf(params.mStart)); | 
|  | values.put(DURATION, Long.valueOf(params.mDuration)); | 
|  | if (params.mDataUsage != Long.MIN_VALUE) { | 
|  | values.put(DATA_USAGE, params.mDataUsage); | 
|  | } | 
|  | values.put(PHONE_ACCOUNT_COMPONENT_NAME, accountComponentString); | 
|  | values.put(PHONE_ACCOUNT_ID, accountId); | 
|  | values.put(PHONE_ACCOUNT_ADDRESS, accountAddress); | 
|  | values.put(NEW, Integer.valueOf(1)); | 
|  | values.put(CACHED_NAME, name); | 
|  | values.put(ADD_FOR_ALL_USERS, params.mAddForAllUsers ? 1 : 0); | 
|  |  | 
|  | if (params.mCallType == MISSED_TYPE) { | 
|  | values.put(IS_READ, Integer.valueOf(params.mIsRead ? 1 : 0)); | 
|  | } | 
|  |  | 
|  | values.put(BLOCK_REASON, params.mCallBlockReason); | 
|  | values.put(CALL_SCREENING_APP_NAME, charSequenceToString(params.mCallScreeningAppName)); | 
|  | values.put(CALL_SCREENING_COMPONENT_NAME, params.mCallScreeningComponentName); | 
|  | values.put(MISSED_REASON, Long.valueOf(params.mMissedReason)); | 
|  | values.put(PRIORITY, params.mPriority); | 
|  | values.put(SUBJECT, params.mSubject); | 
|  | if (params.mPictureUri != null) { | 
|  | values.put(COMPOSER_PHOTO_URI, params.mPictureUri.toString()); | 
|  | } | 
|  | values.put(IS_PHONE_ACCOUNT_MIGRATION_PENDING, params.mIsPhoneAccountMigrationPending); | 
|  | if (Flags.businessCallComposer()) { | 
|  | values.put(IS_BUSINESS_CALL, Integer.valueOf(params.mIsBusinessCall ? 1 : 0)); | 
|  | values.put(ASSERTED_DISPLAY_NAME, params.mAssertedDisplayName); | 
|  | } | 
|  | if ((params.mCallerInfo != null) && (params.mCallerInfo.getContactId() > 0)) { | 
|  | // Update usage information for the number associated with the contact ID. | 
|  | // We need to use both the number and the ID for obtaining a data ID since other | 
|  | // contacts may have the same number. | 
|  |  | 
|  | final Cursor cursor; | 
|  |  | 
|  | // We should prefer normalized one (probably coming from | 
|  | // Phone.NORMALIZED_NUMBER column) first. If it isn't available try others. | 
|  | if (params.mCallerInfo.normalizedNumber != null) { | 
|  | final String normalizedPhoneNumber = params.mCallerInfo.normalizedNumber; | 
|  | cursor = resolver.query(Phone.CONTENT_URI, | 
|  | new String[] { Phone._ID }, | 
|  | Phone.CONTACT_ID + " =? AND " + Phone.NORMALIZED_NUMBER + " =?", | 
|  | new String[] { String.valueOf(params.mCallerInfo.getContactId()), | 
|  | normalizedPhoneNumber}, | 
|  | null); | 
|  | } else { | 
|  | final String phoneNumber = params.mCallerInfo.getPhoneNumber() != null | 
|  | ? params.mCallerInfo.getPhoneNumber() : params.mNumber; | 
|  | cursor = resolver.query( | 
|  | Uri.withAppendedPath(Callable.CONTENT_FILTER_URI, | 
|  | Uri.encode(phoneNumber)), | 
|  | new String[] { Phone._ID }, | 
|  | Phone.CONTACT_ID + " =?", | 
|  | new String[] { String.valueOf(params.mCallerInfo.getContactId()) }, | 
|  | null); | 
|  | } | 
|  |  | 
|  | if (cursor != null) { | 
|  | try { | 
|  | if (cursor.getCount() > 0 && cursor.moveToFirst()) { | 
|  | final String dataId = cursor.getString(0); | 
|  | updateDataUsageStatForData(resolver, dataId); | 
|  | if (params.mDuration >= MIN_DURATION_FOR_NORMALIZED_NUMBER_UPDATE_MS | 
|  | && params.mCallType == Calls.OUTGOING_TYPE | 
|  | && TextUtils.isEmpty(params.mCallerInfo.normalizedNumber)) { | 
|  | updateNormalizedNumber(context, resolver, dataId, params.mNumber); | 
|  | } | 
|  | } | 
|  | } finally { | 
|  | cursor.close(); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | /* | 
|  | Writing the calllog works in the following way: | 
|  | - All user entries | 
|  | - if user-0 is encrypted, insert to user-0's shadow only. | 
|  | (other users should also be encrypted, so nothing to do for other users.) | 
|  | - if user-0 is decrypted, insert to user-0's real provider, as well as | 
|  | all other users that are running and decrypted and should have calllog. | 
|  |  | 
|  | - Single user entry. | 
|  | - If the target user is encryted, insert to its shadow. | 
|  | - Otherwise insert to its real provider. | 
|  |  | 
|  | When the (real) calllog provider starts, it copies entries that it missed from | 
|  | elsewhere. | 
|  | - When user-0's (real) provider starts, it copies from user-0's shadow, and clears | 
|  | the shadow. | 
|  |  | 
|  | - When other users (real) providers start, unless it shouldn't have calllog entries, | 
|  | - Copy from the user's shadow, and clears the shadow. | 
|  | - Copy from user-0's entries that are FOR_ALL_USERS = 1.  (and don't clear it.) | 
|  | */ | 
|  |  | 
|  | Uri result = null; | 
|  |  | 
|  | final UserManager userManager = context.getSystemService(UserManager.class); | 
|  | final int currentUserId = userManager.getProcessUserId(); | 
|  |  | 
|  | if (params.mAddForAllUsers) { | 
|  | if (userManager.isUserUnlocked(UserHandle.SYSTEM)) { | 
|  | // If the user is unlocked, insert to the location provider if a location is | 
|  | // provided. Do not store location if the device is still locked -- this | 
|  | // puts it into device-encrypted storage instead of credential-encrypted | 
|  | // storage. | 
|  | Uri locationUri = maybeInsertLocation(params, resolver, UserHandle.SYSTEM); | 
|  | if (locationUri != null) { | 
|  | values.put(Calls.LOCATION, locationUri.toString()); | 
|  | } | 
|  | } | 
|  |  | 
|  | // First, insert to the system user. | 
|  | final Uri uriForSystem = addEntryAndRemoveExpiredEntries( | 
|  | context, userManager, UserHandle.SYSTEM, values); | 
|  | if (uriForSystem == null | 
|  | || SHADOW_AUTHORITY.equals(uriForSystem.getAuthority())) { | 
|  | // This means the system user is still encrypted and the entry has inserted | 
|  | // into the shadow.  This means other users are still all encrypted. | 
|  | // Nothing further to do; just return null. | 
|  | return null; | 
|  | } | 
|  | if (UserHandle.USER_SYSTEM == currentUserId) { | 
|  | result = uriForSystem; | 
|  | } | 
|  |  | 
|  | // Otherwise, insert to all other users that are running and unlocked. | 
|  |  | 
|  | final List<UserInfo> users = userManager.getAliveUsers(); | 
|  |  | 
|  | final int count = users.size(); | 
|  | for (int i = 0; i < count; i++) { | 
|  | final UserInfo userInfo = users.get(i); | 
|  | final UserHandle userHandle = userInfo.getUserHandle(); | 
|  | final int userId = userHandle.getIdentifier(); | 
|  |  | 
|  | if (userHandle.isSystem()) { | 
|  | // Already written. | 
|  | continue; | 
|  | } | 
|  |  | 
|  | if (!shouldHaveSharedCallLogEntries(context, userManager, userId)) { | 
|  | // Shouldn't have calllog entries. | 
|  | continue; | 
|  | } | 
|  |  | 
|  | // For other users, we write only when they're running *and* decrypted. | 
|  | // Other providers will copy from the system user's real provider, when they | 
|  | // start. | 
|  | if (userManager.isUserRunning(userHandle) | 
|  | && userManager.isUserUnlocked(userHandle)) { | 
|  | Uri locationUri = maybeInsertLocation(params, resolver, userHandle); | 
|  | if (locationUri != null) { | 
|  | values.put(Calls.LOCATION, locationUri.toString()); | 
|  | } else { | 
|  | values.put(Calls.LOCATION, (String) null); | 
|  | } | 
|  | final Uri uri = addEntryAndRemoveExpiredEntries(context, userManager, | 
|  | userHandle, values); | 
|  | if (userId == currentUserId) { | 
|  | result = uri; | 
|  | } | 
|  | } | 
|  | } | 
|  | } else { | 
|  | // Single-user entry. Just write to that user, assuming it's running.  If the | 
|  | // user is encrypted, we write to the shadow calllog. | 
|  | final UserHandle targetUserHandle = params.mUserToBeInsertedTo != null | 
|  | ? params.mUserToBeInsertedTo | 
|  | : UserHandle.of(currentUserId); | 
|  |  | 
|  | if (userManager.isUserRunning(targetUserHandle) | 
|  | && userManager.isUserUnlocked(targetUserHandle)) { | 
|  | Uri locationUri = maybeInsertLocation(params, resolver, targetUserHandle); | 
|  | if (locationUri != null) { | 
|  | values.put(Calls.LOCATION, locationUri.toString()); | 
|  | } else { | 
|  | values.put(Calls.LOCATION, (String) null); | 
|  | } | 
|  | } | 
|  |  | 
|  | result = addEntryAndRemoveExpiredEntries(context, userManager, targetUserHandle, | 
|  | values); | 
|  | } | 
|  | return result; | 
|  | } | 
|  |  | 
|  | private static String charSequenceToString(CharSequence sequence) { | 
|  | return sequence == null ? null : sequence.toString(); | 
|  | } | 
|  |  | 
|  | /** @hide */ | 
|  | public static boolean shouldHaveSharedCallLogEntries(Context context, | 
|  | UserManager userManager, int userId) { | 
|  | if (userManager.hasUserRestriction(UserManager.DISALLOW_OUTGOING_CALLS, | 
|  | UserHandle.of(userId))) { | 
|  | return false; | 
|  | } | 
|  | final UserInfo userInfo = userManager.getUserInfo(userId); | 
|  | return userInfo != null && !userInfo.isManagedProfile(); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Query the call log database for the last dialed number. | 
|  | * @param context Used to get the content resolver. | 
|  | * @return The last phone number dialed (outgoing) or an empty | 
|  | * string if none exist yet. | 
|  | */ | 
|  | public static String getLastOutgoingCall(Context context) { | 
|  | final ContentResolver resolver = context.getContentResolver(); | 
|  | Cursor c = null; | 
|  | try { | 
|  | c = resolver.query( | 
|  | CONTENT_URI_LIMIT_1, | 
|  | new String[] {NUMBER}, | 
|  | TYPE + " = " + OUTGOING_TYPE, | 
|  | null, | 
|  | DEFAULT_SORT_ORDER); | 
|  | if (c == null || !c.moveToFirst()) { | 
|  | return ""; | 
|  | } | 
|  | return c.getString(0); | 
|  | } finally { | 
|  | if (c != null) c.close(); | 
|  | } | 
|  | } | 
|  |  | 
|  | private static Uri addEntryAndRemoveExpiredEntries(Context context, UserManager userManager, | 
|  | UserHandle user, ContentValues values) { | 
|  | final ContentResolver resolver = context.getContentResolver(); | 
|  |  | 
|  | // Since we're doing this operation on behalf of an app, we only | 
|  | // want to use the actual "unlocked" state. | 
|  | final Uri uri = ContentProvider.maybeAddUserId( | 
|  | userManager.isUserUnlocked(user) ? CONTENT_URI : SHADOW_CONTENT_URI, | 
|  | user.getIdentifier()); | 
|  |  | 
|  | Log.i(LOG_TAG, String.format(Locale.getDefault(), | 
|  | "addEntryAndRemoveExpiredEntries: provider uri=%s", uri)); | 
|  |  | 
|  | try { | 
|  | // When cleaning up the call log, try to delete older call long entries on a per | 
|  | // PhoneAccount basis first.  There can be multiple ConnectionServices causing | 
|  | // the addition of entries in the call log.  With the introduction of Self-Managed | 
|  | // ConnectionServices, we want to ensure that a misbehaving self-managed CS cannot | 
|  | // spam the call log with its own entries, causing entries from Telephony to be | 
|  | // removed. | 
|  | final Uri result = resolver.insert(uri, values); | 
|  | if (result != null) { | 
|  | String lastPathSegment = result.getLastPathSegment(); | 
|  | // When inserting into the call log, if ContentProvider#insert detect an appops | 
|  | // denial a non-null "silent rejection" URI is returned which ends in 0. | 
|  | // Example: content://call_log/calls/0 | 
|  | // The 0 in the last part of the path indicates a fake call id of 0. | 
|  | // A denial when logging calls from the platform is bad; there is no other | 
|  | // logging to indicate that this has happened so we will check for that scenario | 
|  | // here and log a warning so we have a hint as to what is going on. | 
|  | if (lastPathSegment != null && lastPathSegment.equals("0")) { | 
|  | Log.w(LOG_TAG, "Failed to insert into call log due to appops denial;" | 
|  | + " resultUri=" + result); | 
|  | } | 
|  | } else { | 
|  | Log.w(LOG_TAG, "Failed to insert into call log; null result uri."); | 
|  | } | 
|  |  | 
|  | int numDeleted; | 
|  | if (values.containsKey(PHONE_ACCOUNT_ID) | 
|  | && !TextUtils.isEmpty(values.getAsString(PHONE_ACCOUNT_ID)) | 
|  | && values.containsKey(PHONE_ACCOUNT_COMPONENT_NAME) | 
|  | && !TextUtils.isEmpty(values.getAsString(PHONE_ACCOUNT_COMPONENT_NAME))) { | 
|  | // Only purge entries for the same phone account. | 
|  | numDeleted = resolver.delete(uri, "_id IN " | 
|  | + "(SELECT _id FROM calls" | 
|  | + " WHERE " + PHONE_ACCOUNT_COMPONENT_NAME + " = ?" | 
|  | + " AND " + PHONE_ACCOUNT_ID + " = ?" | 
|  | + " ORDER BY " + DEFAULT_SORT_ORDER | 
|  | + " LIMIT -1 OFFSET 500)", new String[] { | 
|  | values.getAsString(PHONE_ACCOUNT_COMPONENT_NAME), | 
|  | values.getAsString(PHONE_ACCOUNT_ID) | 
|  | }); | 
|  | } else { | 
|  | // No valid phone account specified, so default to the old behavior. | 
|  | numDeleted = resolver.delete(uri, "_id IN " | 
|  | + "(SELECT _id FROM calls ORDER BY " + DEFAULT_SORT_ORDER | 
|  | + " LIMIT -1 OFFSET 500)", null); | 
|  | } | 
|  | Log.i(LOG_TAG, "addEntry: cleaned up " + numDeleted + " old entries"); | 
|  |  | 
|  | return result; | 
|  | } catch (IllegalArgumentException e) { | 
|  | Log.e(LOG_TAG, "Failed to insert calllog", e); | 
|  | // Even though we make sure the target user is running and decrypted before calling | 
|  | // this method, there's a chance that the user just got shut down, in which case | 
|  | // we'll still get "IllegalArgumentException: Unknown URL content://call_log/calls". | 
|  | return null; | 
|  | } | 
|  | } | 
|  |  | 
|  | private static Uri maybeInsertLocation(AddCallParams params, ContentResolver resolver, | 
|  | UserHandle user) { | 
|  | if (Double.isNaN(params.mLatitude) || Double.isNaN(params.mLongitude)) { | 
|  | return null; | 
|  | } | 
|  | ContentValues locationValues = new ContentValues(); | 
|  | locationValues.put(Locations.LATITUDE, params.mLatitude); | 
|  | locationValues.put(Locations.LONGITUDE, params.mLongitude); | 
|  | Uri locationUri = ContentProvider.maybeAddUserId(Locations.CONTENT_URI, | 
|  | user.getIdentifier()); | 
|  | try { | 
|  | return resolver.insert(locationUri, locationValues); | 
|  | } catch (SecurityException e) { | 
|  | // This can happen if the caller doesn't have location permissions. If that's the | 
|  | // case just skip the insertion. | 
|  | Log.w(LOG_TAG, "Skipping inserting location because caller lacks" | 
|  | + " ACCESS_FINE_LOCATION."); | 
|  | return null; | 
|  | } | 
|  | } | 
|  |  | 
|  | private static void updateDataUsageStatForData(ContentResolver resolver, String dataId) { | 
|  | final Uri feedbackUri = DataUsageFeedback.FEEDBACK_URI.buildUpon() | 
|  | .appendPath(dataId) | 
|  | .appendQueryParameter(DataUsageFeedback.USAGE_TYPE, | 
|  | DataUsageFeedback.USAGE_TYPE_CALL) | 
|  | .build(); | 
|  | resolver.update(feedbackUri, new ContentValues(), null, null); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Update the normalized phone number for the given dataId in the ContactsProvider, based | 
|  | * on the user's current country. | 
|  | */ | 
|  | private static void updateNormalizedNumber(Context context, ContentResolver resolver, | 
|  | String dataId, String number) { | 
|  | if (TextUtils.isEmpty(number) || TextUtils.isEmpty(dataId)) { | 
|  | return; | 
|  | } | 
|  | final String countryIso = getCurrentCountryIso(context); | 
|  | if (TextUtils.isEmpty(countryIso)) { | 
|  | return; | 
|  | } | 
|  | final String normalizedNumber = PhoneNumberUtils.formatNumberToE164(number, countryIso); | 
|  | if (TextUtils.isEmpty(normalizedNumber)) { | 
|  | return; | 
|  | } | 
|  | final ContentValues values = new ContentValues(); | 
|  | values.put(Phone.NORMALIZED_NUMBER, normalizedNumber); | 
|  | resolver.update(Data.CONTENT_URI, values, Data._ID + "=?", new String[] {dataId}); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Remap network specified number presentation types | 
|  | * TelecomManager.PRESENTATION_xxx to calllog number presentation types | 
|  | * Calls.PRESENTATION_xxx, in order to insulate the persistent calllog | 
|  | * from any future radio changes. | 
|  | * If the number field is empty set the presentation type to Unknown. | 
|  | */ | 
|  | private static int getLogNumberPresentation(String number, int presentation) { | 
|  | if (presentation == TelecomManager.PRESENTATION_RESTRICTED) { | 
|  | return presentation; | 
|  | } | 
|  |  | 
|  | if (presentation == TelecomManager.PRESENTATION_PAYPHONE) { | 
|  | return presentation; | 
|  | } | 
|  |  | 
|  | if (presentation == TelecomManager.PRESENTATION_UNAVAILABLE) { | 
|  | return PRESENTATION_UNAVAILABLE; | 
|  | } | 
|  |  | 
|  | if (TextUtils.isEmpty(number) | 
|  | || presentation == TelecomManager.PRESENTATION_UNKNOWN) { | 
|  | return PRESENTATION_UNKNOWN; | 
|  | } | 
|  |  | 
|  | return PRESENTATION_ALLOWED; | 
|  | } | 
|  |  | 
|  | private static String getLogAccountAddress(Context context, | 
|  | PhoneAccountHandle accountHandle) { | 
|  | TelecomManager tm = null; | 
|  | try { | 
|  | tm = context.getSystemService(TelecomManager.class); | 
|  | } catch (UnsupportedOperationException e) { | 
|  | if (VERBOSE_LOG) { | 
|  | Log.v(LOG_TAG, "No TelecomManager found to get account address."); | 
|  | } | 
|  | } | 
|  |  | 
|  | String accountAddress = null; | 
|  | if (tm != null && accountHandle != null) { | 
|  | PhoneAccount account = tm.getPhoneAccount(accountHandle); | 
|  | if (account != null) { | 
|  | Uri address = account.getSubscriptionAddress(); | 
|  | if (address != null) { | 
|  | accountAddress = address.getSchemeSpecificPart(); | 
|  | } | 
|  | } | 
|  | } | 
|  | return accountAddress; | 
|  | } | 
|  |  | 
|  | private static String getCurrentCountryIso(Context context) { | 
|  | String countryIso = null; | 
|  | final CountryDetector detector = (CountryDetector) context.getSystemService( | 
|  | Context.COUNTRY_DETECTOR); | 
|  | if (detector != null) { | 
|  | final Country country = detector.detectCountry(); | 
|  | if (country != null) { | 
|  | countryIso = country.getCountryIso(); | 
|  | } | 
|  | } | 
|  | return countryIso; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Check if the missedReason code indicate that the call was user missed or automatically | 
|  | * rejected by system. | 
|  | * | 
|  | * @param missedReason | 
|  | * The result is true if the call was user missed, false if the call was automatically | 
|  | * rejected by system. | 
|  | * | 
|  | * @hide | 
|  | */ | 
|  | public static boolean isUserMissed(long missedReason) { | 
|  | return missedReason >= (USER_MISSED_NO_ANSWER); | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Table that contains information on location data sent via call composer. | 
|  | * | 
|  | * All fields in this table require the {@link Manifest.permission#ACCESS_FINE_LOCATION} | 
|  | * permission for access. | 
|  | */ | 
|  | public static class Locations implements BaseColumns { | 
|  | private Locations() {} | 
|  | /** | 
|  | * Authority for the locations content provider. | 
|  | */ | 
|  | public static final String AUTHORITY = "call_composer_locations"; | 
|  |  | 
|  | /** | 
|  | * Content type for the location table. | 
|  | */ | 
|  | public static final String CONTENT_TYPE = "vnd.android.cursor.dir/call_composer_location"; | 
|  |  | 
|  | /** | 
|  | * Content type for the location entries. | 
|  | */ | 
|  | public static final String CONTENT_ITEM_TYPE = | 
|  | "vnd.android.cursor.item/call_composer_location"; | 
|  |  | 
|  | /** | 
|  | * The content URI for this table | 
|  | */ | 
|  | @NonNull | 
|  | public static final Uri CONTENT_URI = Uri.parse("content://" + AUTHORITY); | 
|  |  | 
|  | /** | 
|  | * Latitude in degrees. See {@link android.location.Location#setLatitude(double)}. | 
|  | * <p>Type: REAL</p> | 
|  | */ | 
|  | public static final String LATITUDE = "latitude"; | 
|  |  | 
|  | /** | 
|  | * Longitude in degrees. See {@link android.location.Location#setLongitude(double)}. | 
|  | * <p>Type: REAL</p> | 
|  | */ | 
|  | public static final String LONGITUDE = "longitude"; | 
|  | } | 
|  | } |