2012-07-21 05:23:25 +00:00
|
|
|
/**
|
2011-12-20 18:20:44 +00:00
|
|
|
* Copyright (C) 2011 Whisper Systems
|
2012-07-21 05:23:25 +00:00
|
|
|
*
|
2011-12-20 18:20:44 +00:00
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
2012-07-21 05:23:25 +00:00
|
|
|
*
|
2011-12-20 18:20:44 +00:00
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
package org.thoughtcrime.securesms.contacts;
|
|
|
|
|
|
|
|
import android.content.ContentResolver;
|
|
|
|
import android.content.Context;
|
|
|
|
import android.database.Cursor;
|
2012-10-23 02:17:08 +00:00
|
|
|
import android.database.MergeCursor;
|
2011-12-20 18:20:44 +00:00
|
|
|
import android.net.Uri;
|
|
|
|
import android.os.Parcel;
|
|
|
|
import android.os.Parcelable;
|
2012-10-23 02:17:08 +00:00
|
|
|
import android.provider.ContactsContract;
|
|
|
|
import android.provider.ContactsContract.CommonDataKinds.Phone;
|
|
|
|
import android.provider.ContactsContract.Contacts;
|
|
|
|
import android.provider.ContactsContract.Data;
|
|
|
|
import android.provider.ContactsContract.PhoneLookup;
|
2012-07-21 05:23:25 +00:00
|
|
|
import android.support.v4.content.CursorLoader;
|
2014-03-18 06:25:09 +00:00
|
|
|
import android.support.v4.content.Loader;
|
2012-10-23 02:17:08 +00:00
|
|
|
import android.telephony.PhoneNumberUtils;
|
2012-07-21 05:23:25 +00:00
|
|
|
|
2014-11-10 04:35:08 +00:00
|
|
|
import org.thoughtcrime.securesms.database.TextSecureDirectory;
|
2012-07-21 05:23:25 +00:00
|
|
|
|
2012-10-23 02:17:08 +00:00
|
|
|
import java.util.ArrayList;
|
2014-03-18 06:25:09 +00:00
|
|
|
import java.util.Collection;
|
2012-07-21 05:23:25 +00:00
|
|
|
import java.util.LinkedList;
|
|
|
|
import java.util.List;
|
2011-12-20 18:20:44 +00:00
|
|
|
|
|
|
|
/**
|
2012-10-23 02:17:08 +00:00
|
|
|
* This class was originally a layer of indirection between
|
|
|
|
* ContactAccessorNewApi and ContactAccesorOldApi, which corresponded
|
|
|
|
* to the API changes between 1.x and 2.x.
|
|
|
|
*
|
|
|
|
* Now that we no longer support 1.x, this class mostly serves as a place
|
|
|
|
* to encapsulate Contact-related logic. It's still a singleton, mostly
|
|
|
|
* just because that's how it's currently called from everywhere.
|
2012-07-21 05:23:25 +00:00
|
|
|
*
|
2011-12-20 18:20:44 +00:00
|
|
|
* @author Moxie Marlinspike
|
|
|
|
*/
|
|
|
|
|
2012-10-23 02:17:08 +00:00
|
|
|
public class ContactAccessor {
|
2011-12-20 18:20:44 +00:00
|
|
|
|
2014-02-07 02:06:23 +00:00
|
|
|
public static final String PUSH_COLUMN = "push";
|
|
|
|
|
2012-10-23 02:17:08 +00:00
|
|
|
private static final ContactAccessor instance = new ContactAccessor();
|
2012-07-21 05:23:25 +00:00
|
|
|
|
2011-12-20 18:20:44 +00:00
|
|
|
public static synchronized ContactAccessor getInstance() {
|
2012-10-23 02:17:08 +00:00
|
|
|
return instance;
|
|
|
|
}
|
|
|
|
|
|
|
|
public CursorLoader getCursorLoaderForContactsWithNumbers(Context context) {
|
|
|
|
Uri uri = ContactsContract.Contacts.CONTENT_URI;
|
|
|
|
String selection = ContactsContract.Contacts.HAS_PHONE_NUMBER + " = 1";
|
|
|
|
|
|
|
|
return new CursorLoader(context, uri, null, selection, null,
|
|
|
|
ContactsContract.Contacts.DISPLAY_NAME + " ASC");
|
|
|
|
}
|
|
|
|
|
|
|
|
public CursorLoader getCursorLoaderForContactGroups(Context context) {
|
|
|
|
return new CursorLoader(context, ContactsContract.Groups.CONTENT_URI,
|
|
|
|
null, null, null, ContactsContract.Groups.TITLE + " ASC");
|
|
|
|
}
|
|
|
|
|
2014-04-01 23:40:16 +00:00
|
|
|
public Loader<Cursor> getCursorLoaderForContacts(Context context, String filter) {
|
|
|
|
return new ContactsCursorLoader(context, filter, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
public Loader<Cursor> getCursorLoaderForPushContacts(Context context, String filter) {
|
|
|
|
return new ContactsCursorLoader(context, filter, true);
|
2014-03-18 06:25:09 +00:00
|
|
|
}
|
|
|
|
|
2012-10-23 02:17:08 +00:00
|
|
|
public Cursor getCursorForContactsWithNumbers(Context context) {
|
|
|
|
Uri uri = ContactsContract.Contacts.CONTENT_URI;
|
|
|
|
String selection = ContactsContract.Contacts.HAS_PHONE_NUMBER + " = 1";
|
|
|
|
|
|
|
|
return context.getContentResolver().query(uri, null, selection, null,
|
|
|
|
ContactsContract.Contacts.DISPLAY_NAME + " ASC");
|
|
|
|
}
|
|
|
|
|
2014-03-18 06:25:09 +00:00
|
|
|
public Collection<ContactData> getContactsWithPush(Context context) {
|
2014-02-07 02:06:23 +00:00
|
|
|
final ContentResolver resolver = context.getContentResolver();
|
2014-03-18 06:25:09 +00:00
|
|
|
final String[] inProjection = new String[]{PhoneLookup._ID, PhoneLookup.DISPLAY_NAME};
|
|
|
|
|
2014-11-10 04:35:08 +00:00
|
|
|
List<String> pushNumbers = TextSecureDirectory.getInstance(context).getActiveNumbers();
|
2014-03-18 06:25:09 +00:00
|
|
|
final Collection<ContactData> lookupData = new ArrayList<ContactData>(pushNumbers.size());
|
|
|
|
|
2014-02-07 02:06:23 +00:00
|
|
|
for (String pushNumber : pushNumbers) {
|
|
|
|
Uri uri = Uri.withAppendedPath(PhoneLookup.CONTENT_FILTER_URI, Uri.encode(pushNumber));
|
|
|
|
Cursor lookupCursor = resolver.query(uri, inProjection, null, null, null);
|
|
|
|
try {
|
|
|
|
if (lookupCursor != null && lookupCursor.moveToFirst()) {
|
2014-03-18 06:25:09 +00:00
|
|
|
final ContactData contactData = new ContactData(lookupCursor.getLong(0), lookupCursor.getString(1));
|
|
|
|
contactData.numbers.add(new NumberData("TextSecure", pushNumber));
|
|
|
|
lookupData.add(contactData);
|
2014-02-07 02:06:23 +00:00
|
|
|
}
|
|
|
|
} finally {
|
|
|
|
if (lookupCursor != null)
|
|
|
|
lookupCursor.close();
|
|
|
|
}
|
|
|
|
}
|
2014-03-18 06:25:09 +00:00
|
|
|
return lookupData;
|
2014-02-07 02:06:23 +00:00
|
|
|
}
|
|
|
|
|
2012-10-23 02:17:08 +00:00
|
|
|
public String getNameFromContact(Context context, Uri uri) {
|
|
|
|
Cursor cursor = null;
|
|
|
|
|
|
|
|
try {
|
|
|
|
cursor = context.getContentResolver().query(uri, new String[] {Contacts.DISPLAY_NAME},
|
|
|
|
null, null, null);
|
|
|
|
|
|
|
|
if (cursor != null && cursor.moveToFirst())
|
|
|
|
return cursor.getString(0);
|
|
|
|
|
|
|
|
} finally {
|
|
|
|
if (cursor != null)
|
|
|
|
cursor.close();
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
public String getNameForNumber(Context context, String number) {
|
|
|
|
Uri uri = Uri.withAppendedPath(PhoneLookup.CONTENT_FILTER_URI, Uri.encode(number));
|
|
|
|
Cursor cursor = context.getContentResolver().query(uri, null, null, null, null);
|
|
|
|
|
|
|
|
try {
|
|
|
|
if (cursor != null && cursor.moveToFirst())
|
|
|
|
return cursor.getString(cursor.getColumnIndexOrThrow(PhoneLookup.DISPLAY_NAME));
|
|
|
|
} finally {
|
|
|
|
if (cursor != null)
|
|
|
|
cursor.close();
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
public GroupData getGroupData(Context context, Cursor cursor) {
|
|
|
|
long id = cursor.getLong(cursor.getColumnIndexOrThrow(ContactsContract.Groups._ID));
|
|
|
|
String title = cursor.getString(cursor.getColumnIndexOrThrow(ContactsContract.Groups.TITLE));
|
|
|
|
|
|
|
|
return new GroupData(id, title);
|
|
|
|
}
|
|
|
|
|
|
|
|
public ContactData getContactData(Context context, Cursor cursor) {
|
|
|
|
return getContactData(context,
|
|
|
|
cursor.getString(cursor.getColumnIndexOrThrow(Contacts.DISPLAY_NAME)),
|
|
|
|
cursor.getLong(cursor.getColumnIndexOrThrow(Contacts._ID)));
|
2013-10-17 00:28:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public ContactData getContactData(Context context, Uri uri) {
|
|
|
|
return getContactData(context, getNameFromContact(context, uri), Long.parseLong(uri.getLastPathSegment()));
|
|
|
|
}
|
2012-10-23 02:17:08 +00:00
|
|
|
|
|
|
|
private ContactData getContactData(Context context, String displayName, long id) {
|
|
|
|
ContactData contactData = new ContactData(id, displayName);
|
|
|
|
Cursor numberCursor = null;
|
|
|
|
|
|
|
|
try {
|
|
|
|
numberCursor = context.getContentResolver().query(Phone.CONTENT_URI, null,
|
|
|
|
Phone.CONTACT_ID + " = ?",
|
|
|
|
new String[] {contactData.id + ""}, null);
|
|
|
|
|
|
|
|
while (numberCursor != null && numberCursor.moveToNext()) {
|
|
|
|
int type = numberCursor.getInt(numberCursor.getColumnIndexOrThrow(Phone.TYPE));
|
|
|
|
String label = numberCursor.getString(numberCursor.getColumnIndexOrThrow(Phone.LABEL));
|
|
|
|
String number = numberCursor.getString(numberCursor.getColumnIndexOrThrow(Phone.NUMBER));
|
|
|
|
String typeLabel = Phone.getTypeLabel(context.getResources(), type, label).toString();
|
|
|
|
|
|
|
|
contactData.numbers.add(new NumberData(typeLabel, number));
|
|
|
|
}
|
|
|
|
} finally {
|
|
|
|
if (numberCursor != null)
|
|
|
|
numberCursor.close();
|
|
|
|
}
|
|
|
|
|
|
|
|
return contactData;
|
|
|
|
}
|
|
|
|
|
|
|
|
public List<ContactData> getGroupMembership(Context context, long groupId) {
|
|
|
|
LinkedList<ContactData> contacts = new LinkedList<ContactData>();
|
|
|
|
Cursor groupMembership = null;
|
|
|
|
|
|
|
|
try {
|
|
|
|
String selection = ContactsContract.CommonDataKinds.GroupMembership.GROUP_ROW_ID + " = ? AND " +
|
|
|
|
ContactsContract.CommonDataKinds.GroupMembership.MIMETYPE + " = ?";
|
|
|
|
String[] args = new String[] {groupId+"",
|
|
|
|
ContactsContract.CommonDataKinds.GroupMembership.CONTENT_ITEM_TYPE};
|
|
|
|
|
|
|
|
groupMembership = context.getContentResolver().query(Data.CONTENT_URI, null, selection, args, null);
|
|
|
|
|
|
|
|
while (groupMembership != null && groupMembership.moveToNext()) {
|
|
|
|
String displayName = groupMembership.getString(groupMembership.getColumnIndexOrThrow(Data.DISPLAY_NAME));
|
|
|
|
long contactId = groupMembership.getLong(groupMembership.getColumnIndexOrThrow(Data.CONTACT_ID));
|
|
|
|
|
|
|
|
contacts.add(getContactData(context, displayName, contactId));
|
|
|
|
}
|
|
|
|
} finally {
|
|
|
|
if (groupMembership != null)
|
|
|
|
groupMembership.close();
|
|
|
|
}
|
|
|
|
|
|
|
|
return contacts;
|
|
|
|
}
|
|
|
|
|
|
|
|
public List<String> getNumbersForThreadSearchFilter(String constraint, ContentResolver contentResolver) {
|
|
|
|
LinkedList<String> numberList = new LinkedList<String>();
|
|
|
|
Cursor cursor = null;
|
|
|
|
|
|
|
|
try {
|
|
|
|
cursor = contentResolver.query(Uri.withAppendedPath(Phone.CONTENT_FILTER_URI,
|
|
|
|
Uri.encode(constraint)),
|
|
|
|
null, null, null, null);
|
|
|
|
|
|
|
|
while (cursor != null && cursor.moveToNext()) {
|
|
|
|
numberList.add(cursor.getString(cursor.getColumnIndexOrThrow(Phone.NUMBER)));
|
|
|
|
}
|
|
|
|
|
|
|
|
} finally {
|
|
|
|
if (cursor != null)
|
|
|
|
cursor.close();
|
|
|
|
}
|
|
|
|
|
|
|
|
return numberList;
|
|
|
|
}
|
|
|
|
|
|
|
|
public CharSequence phoneTypeToString(Context mContext, int type, CharSequence label) {
|
|
|
|
return Phone.getTypeLabel(mContext.getResources(), type, label);
|
|
|
|
}
|
|
|
|
|
|
|
|
private long getContactIdFromLookupUri(Context context, Uri uri) {
|
|
|
|
Cursor cursor = null;
|
|
|
|
|
|
|
|
try {
|
|
|
|
cursor = context.getContentResolver().query(uri,
|
|
|
|
new String[] {ContactsContract.Contacts._ID},
|
|
|
|
null, null, null);
|
|
|
|
|
|
|
|
if (cursor != null && cursor.moveToFirst()) {
|
|
|
|
return cursor.getLong(0);
|
|
|
|
} else {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
} finally {
|
|
|
|
if (cursor != null)
|
|
|
|
cursor.close();
|
|
|
|
}
|
|
|
|
}
|
2012-07-21 05:23:25 +00:00
|
|
|
|
2011-12-20 18:20:44 +00:00
|
|
|
public static class NumberData implements Parcelable {
|
2012-07-21 05:23:25 +00:00
|
|
|
|
2011-12-20 18:20:44 +00:00
|
|
|
public static final Parcelable.Creator<NumberData> CREATOR = new Parcelable.Creator<NumberData>() {
|
|
|
|
public NumberData createFromParcel(Parcel in) {
|
|
|
|
return new NumberData(in);
|
|
|
|
}
|
|
|
|
|
|
|
|
public NumberData[] newArray(int size) {
|
|
|
|
return new NumberData[size];
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2012-10-23 02:17:08 +00:00
|
|
|
public final String number;
|
|
|
|
public final String type;
|
2011-12-20 18:20:44 +00:00
|
|
|
|
|
|
|
public NumberData(String type, String number) {
|
|
|
|
this.type = type;
|
|
|
|
this.number = number;
|
|
|
|
}
|
2012-07-21 05:23:25 +00:00
|
|
|
|
2011-12-20 18:20:44 +00:00
|
|
|
public NumberData(Parcel in) {
|
|
|
|
number = in.readString();
|
|
|
|
type = in.readString();
|
|
|
|
}
|
2012-07-21 05:23:25 +00:00
|
|
|
|
2011-12-20 18:20:44 +00:00
|
|
|
public int describeContents() {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void writeToParcel(Parcel dest, int flags) {
|
|
|
|
dest.writeString(number);
|
|
|
|
dest.writeString(type);
|
|
|
|
}
|
|
|
|
}
|
2012-07-21 05:23:25 +00:00
|
|
|
|
2011-12-20 18:20:44 +00:00
|
|
|
public static class GroupData {
|
2012-10-23 02:17:08 +00:00
|
|
|
public final long id;
|
|
|
|
public final String name;
|
|
|
|
|
|
|
|
public GroupData(long id, String name) {
|
|
|
|
this.id = id;
|
|
|
|
this.name = name;
|
|
|
|
}
|
2011-12-20 18:20:44 +00:00
|
|
|
}
|
2012-07-21 05:23:25 +00:00
|
|
|
|
2011-12-20 18:20:44 +00:00
|
|
|
public static class ContactData implements Parcelable {
|
2012-07-21 05:23:25 +00:00
|
|
|
|
2011-12-20 18:20:44 +00:00
|
|
|
public static final Parcelable.Creator<ContactData> CREATOR = new Parcelable.Creator<ContactData>() {
|
|
|
|
public ContactData createFromParcel(Parcel in) {
|
|
|
|
return new ContactData(in);
|
|
|
|
}
|
|
|
|
|
|
|
|
public ContactData[] newArray(int size) {
|
|
|
|
return new ContactData[size];
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2012-10-23 02:17:08 +00:00
|
|
|
public final long id;
|
|
|
|
public final String name;
|
|
|
|
public final List<NumberData> numbers;
|
2011-12-20 18:20:44 +00:00
|
|
|
|
2012-10-23 02:17:08 +00:00
|
|
|
public ContactData(long id, String name) {
|
|
|
|
this.id = id;
|
|
|
|
this.name = name;
|
|
|
|
this.numbers = new LinkedList<NumberData>();
|
|
|
|
}
|
2012-07-21 05:23:25 +00:00
|
|
|
|
2014-02-22 01:11:52 +00:00
|
|
|
public ContactData(long id, String name, List<NumberData> numbers) {
|
|
|
|
this.id = id;
|
|
|
|
this.name = name;
|
|
|
|
this.numbers = numbers;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-12-20 18:20:44 +00:00
|
|
|
public ContactData(Parcel in) {
|
|
|
|
id = in.readLong();
|
|
|
|
name = in.readString();
|
|
|
|
numbers = new LinkedList<NumberData>();
|
|
|
|
in.readTypedList(numbers, NumberData.CREATOR);
|
|
|
|
}
|
2012-07-21 05:23:25 +00:00
|
|
|
|
2011-12-20 18:20:44 +00:00
|
|
|
public int describeContents() {
|
|
|
|
return 0;
|
|
|
|
}
|
2012-07-21 05:23:25 +00:00
|
|
|
|
2011-12-20 18:20:44 +00:00
|
|
|
public void writeToParcel(Parcel dest, int flags) {
|
|
|
|
dest.writeLong(id);
|
|
|
|
dest.writeString(name);
|
|
|
|
dest.writeTypedList(numbers);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-23 02:17:08 +00:00
|
|
|
/***
|
|
|
|
* If the code below looks shitty to you, that's because it was taken
|
|
|
|
* directly from the Android source, where shitty code is all you get.
|
|
|
|
*/
|
|
|
|
|
|
|
|
public Cursor getCursorForRecipientFilter(CharSequence constraint,
|
|
|
|
ContentResolver mContentResolver)
|
|
|
|
{
|
|
|
|
final String SORT_ORDER = Contacts.TIMES_CONTACTED + " DESC," +
|
2014-02-09 22:28:10 +00:00
|
|
|
Contacts.DISPLAY_NAME + "," +
|
|
|
|
Contacts.Data.IS_SUPER_PRIMARY + " DESC," +
|
|
|
|
Phone.TYPE;
|
2012-10-23 02:17:08 +00:00
|
|
|
|
|
|
|
final String[] PROJECTION_PHONE = {
|
|
|
|
Phone._ID, // 0
|
|
|
|
Phone.CONTACT_ID, // 1
|
|
|
|
Phone.TYPE, // 2
|
|
|
|
Phone.NUMBER, // 3
|
|
|
|
Phone.LABEL, // 4
|
|
|
|
Phone.DISPLAY_NAME, // 5
|
|
|
|
};
|
|
|
|
|
|
|
|
String phone = "";
|
|
|
|
String cons = null;
|
|
|
|
|
|
|
|
if (constraint != null) {
|
|
|
|
cons = constraint.toString();
|
|
|
|
|
|
|
|
if (RecipientsAdapter.usefulAsDigits(cons)) {
|
|
|
|
phone = PhoneNumberUtils.convertKeypadLettersToDigits(cons);
|
2014-02-17 01:44:51 +00:00
|
|
|
if (phone.equals(cons) && !PhoneNumberUtils.isWellFormedSmsAddress(phone)) {
|
2012-10-23 02:17:08 +00:00
|
|
|
phone = "";
|
|
|
|
} else {
|
|
|
|
phone = phone.trim();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Uri uri = Uri.withAppendedPath(Phone.CONTENT_FILTER_URI, Uri.encode(cons));
|
|
|
|
String selection = String.format("%s=%s OR %s=%s OR %s=%s",
|
|
|
|
Phone.TYPE,
|
|
|
|
Phone.TYPE_MOBILE,
|
|
|
|
Phone.TYPE,
|
|
|
|
Phone.TYPE_WORK_MOBILE,
|
|
|
|
Phone.TYPE,
|
|
|
|
Phone.TYPE_MMS);
|
|
|
|
|
|
|
|
Cursor phoneCursor = mContentResolver.query(uri,
|
|
|
|
PROJECTION_PHONE,
|
|
|
|
null,
|
|
|
|
null,
|
|
|
|
SORT_ORDER);
|
|
|
|
|
|
|
|
if (phone.length() > 0) {
|
|
|
|
ArrayList result = new ArrayList();
|
|
|
|
result.add(Integer.valueOf(-1)); // ID
|
|
|
|
result.add(Long.valueOf(-1)); // CONTACT_ID
|
|
|
|
result.add(Integer.valueOf(Phone.TYPE_CUSTOM)); // TYPE
|
|
|
|
result.add(phone); // NUMBER
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The "\u00A0" keeps Phone.getDisplayLabel() from deciding
|
|
|
|
* to display the default label ("Home") next to the transformation
|
|
|
|
* of the letters into numbers.
|
|
|
|
*/
|
|
|
|
result.add("\u00A0"); // LABEL
|
|
|
|
result.add(cons); // NAME
|
|
|
|
|
|
|
|
ArrayList<ArrayList> wrap = new ArrayList<ArrayList>();
|
|
|
|
wrap.add(result);
|
|
|
|
|
|
|
|
ArrayListCursor translated = new ArrayListCursor(PROJECTION_PHONE, wrap);
|
|
|
|
|
|
|
|
return new MergeCursor(new Cursor[] { translated, phoneCursor });
|
|
|
|
} else {
|
|
|
|
return phoneCursor;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-12-20 18:20:44 +00:00
|
|
|
}
|