blob: 15c64c927e330978ca87082e079cd4b10a68e29a [file] [log] [blame]
/*
* Copyright (C) 2017 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 com.android.dialer.calldetails;
import android.content.Context;
import android.content.CursorLoader;
import android.database.Cursor;
import com.android.dialer.CoalescedIds;
import com.android.dialer.calldetails.CallDetailsEntries.CallDetailsEntry;
import com.android.dialer.calllog.database.contract.AnnotatedCallLogContract.AnnotatedCallLog;
import com.android.dialer.common.Assert;
import com.android.dialer.duo.DuoConstants;
/**
* A {@link CursorLoader} that loads call detail entries from {@link AnnotatedCallLog} for {@link
* CallDetailsActivity}.
*/
public final class CallDetailsCursorLoader extends CursorLoader {
// Columns in AnnotatedCallLog that are needed to build a CallDetailsEntry proto.
// Be sure to update (1) constants that store indexes of the elements and (2) method
// toCallDetailsEntry(Cursor) when updating this array.
public static final String[] COLUMNS_FOR_CALL_DETAILS =
new String[] {
AnnotatedCallLog._ID,
AnnotatedCallLog.CALL_TYPE,
AnnotatedCallLog.FEATURES,
AnnotatedCallLog.TIMESTAMP,
AnnotatedCallLog.DURATION,
AnnotatedCallLog.DATA_USAGE,
AnnotatedCallLog.PHONE_ACCOUNT_COMPONENT_NAME
};
// Indexes for COLUMNS_FOR_CALL_DETAILS
private static final int ID = 0;
private static final int CALL_TYPE = 1;
private static final int FEATURES = 2;
private static final int TIMESTAMP = 3;
private static final int DURATION = 4;
private static final int DATA_USAGE = 5;
private static final int PHONE_ACCOUNT_COMPONENT_NAME = 6;
CallDetailsCursorLoader(Context context, CoalescedIds coalescedIds) {
super(
context,
AnnotatedCallLog.CONTENT_URI,
COLUMNS_FOR_CALL_DETAILS,
annotatedCallLogIdsSelection(coalescedIds),
annotatedCallLogIdsSelectionArgs(coalescedIds),
AnnotatedCallLog.TIMESTAMP + " DESC");
}
@Override
public void onContentChanged() {
// Do nothing here.
// This is to prevent the loader to reload data when Loader.ForceLoadContentObserver detects a
// change.
// Without this, the app will crash when the user deletes call details as the deletion triggers
// the data loading but no data can be fetched and we want to ensure the data set is not empty
// when building CallDetailsEntries proto (see toCallDetailsEntries(Cursor)).
//
// CallDetailsActivity doesn't respond to underlying data changes when launched from the old
// call log and we decided to keep it that way when launched from the new call log.
}
/**
* Build a string of the form "COLUMN_NAME IN (?, ?, ..., ?)", where COLUMN_NAME is the name of
* the ID column in {@link AnnotatedCallLog}.
*
* <p>This string will be used as the {@code selection} parameter to initialize the loader.
*/
private static String annotatedCallLogIdsSelection(CoalescedIds coalescedIds) {
// First, build a string of question marks ('?') separated by commas (',').
StringBuilder questionMarks = new StringBuilder();
for (int i = 0; i < coalescedIds.getCoalescedIdCount(); i++) {
if (i != 0) {
questionMarks.append(", ");
}
questionMarks.append("?");
}
return AnnotatedCallLog._ID + " IN (" + questionMarks + ")";
}
/**
* Returns a string that will be used as the {@code selectionArgs} parameter to initialize the
* loader.
*/
private static String[] annotatedCallLogIdsSelectionArgs(CoalescedIds coalescedIds) {
String[] args = new String[coalescedIds.getCoalescedIdCount()];
for (int i = 0; i < coalescedIds.getCoalescedIdCount(); i++) {
args[i] = String.valueOf(coalescedIds.getCoalescedId(i));
}
return args;
}
/**
* Creates a new {@link CallDetailsEntries} from the entire data set loaded by this loader.
*
* @param cursor A cursor pointing to the data set loaded by this loader. The caller must ensure
* the cursor is not null and the data set it points to is not empty.
* @return A {@link CallDetailsEntries} proto.
*/
static CallDetailsEntries toCallDetailsEntries(Cursor cursor) {
Assert.isNotNull(cursor);
Assert.checkArgument(cursor.moveToFirst());
CallDetailsEntries.Builder entries = CallDetailsEntries.newBuilder();
do {
entries.addEntries(toCallDetailsEntry(cursor));
} while (cursor.moveToNext());
return entries.build();
}
/** Creates a new {@link CallDetailsEntry} from the provided cursor using its current position. */
private static CallDetailsEntry toCallDetailsEntry(Cursor cursor) {
CallDetailsEntry.Builder entry = CallDetailsEntry.newBuilder();
entry
.setCallId(cursor.getLong(ID))
.setCallType(cursor.getInt(CALL_TYPE))
.setFeatures(cursor.getInt(FEATURES))
.setDate(cursor.getLong(TIMESTAMP))
.setDuration(cursor.getLong(DURATION))
.setDataUsage(cursor.getLong(DATA_USAGE));
String phoneAccountComponentName = cursor.getString(PHONE_ACCOUNT_COMPONENT_NAME);
entry.setIsDuoCall(
DuoConstants.PHONE_ACCOUNT_COMPONENT_NAME
.flattenToString()
.equals(phoneAccountComponentName));
return entry.build();
}
}