이 과정에서는 데이터가 전체 또는 일부와 일치하는 연락처 목록을 검색하는 방법을 다음과 같은 기법을 사용해 검색 문자열을 검색합니다.
- 연락처 이름 일치
- 검색 문자열을 연락처 이름 데이터의 전체 또는 일부와 일치시켜서 연락처 목록을 검색합니다. 연락처 제공자를 사용하면 동일한 이름의 여러 가지 인스턴스가 허용되므로 이 기법을 통해 일치 항목 목록이 반환될 수 있습니다.
- 특정 데이터 유형 일치(예: 전화번호)
- 검색 문자열을 이메일 주소와 같은 세부적인 특정 데이터 유형과 일치시켜서 연락처 목록을 검색합니다. 예를 들어 이 기법을 사용하면 이메일 주소가 검색 문자열과 일치하는 모든 연락처 목록을 표시할 수 있습니다.
- 모든 데이터 유형 일치
- 검색 문자열을 이름, 전화번호, 상세 주소, 이메일 주소 등을 포함한 모든 세부적인 데이터 유형과 일치시켜서 연락처 목록을 검색합니다. 예를 들어 이 기법을 사용하면 검색 문자열에 모든 유형의 데이터를 수용한 다음 데이터가 문자열과 일치하는 연락처입니다.
참고: 이 과정의 모든 예에서는 CursorLoader
를 사용하여 연락처 제공자의 데이터를 검색합니다. CursorLoader
는
UI 스레드와는 별개의 스레드입니다. 이렇게 하면 쿼리로 인해 UI 속도가 느려지지 않습니다.
사용자 경험이 저하될 수 있습니다. 자세한 내용은 Android
교육 과정
백그라운드에서 데이터 로드를 참조하세요.
제공자 읽기 권한 요청
연락처 제공자에서 검색을 수행하려면 앱에 다음 항목이 있어야 합니다.
READ_CONTACTS
권한
이를 요청하려면 다음을 추가하세요.
<uses-permission>
요소를 매니페스트 파일에
<manifest>
:
<uses-permission android:name="android.permission.READ_CONTACTS" />
이름으로 연락처를 일치시킨 후 결과 표시
이 기법에서는 검색 문자열을 연락처 제공자의 ContactsContract.Contacts
표에 있는 연락처 이름과 일치시키려고 합니다. 일반적으로
ListView
에 결과를 표시하여 사용자가 선택할 수 있도록 합니다.
확인할 수 있습니다.
ListView 및 항목 레이아웃 정의
검색 결과를 ListView
에 표시하려면 기본 레이아웃 파일이 필요합니다.
ListView
를 포함한 전체 UI와 항목 레이아웃을 정의하는
ListView
의 한 줄을 정의하는 파일입니다. 예를 들어 다음 XML을 사용하여 기본 레이아웃 파일 res/layout/contacts_list_view.xml
을 만들 수 있습니다.
<?xml version="1.0" encoding="utf-8"?> <ListView xmlns:android="http://schemas.android.com/apk/res/android" android:id="@android:id/list" android:layout_width="match_parent" android:layout_height="match_parent"/>
이 XML은 내장된 Android ListView
위젯 android:id/list
을 사용합니다.
다음 XML을 사용하여 항목 레이아웃 파일 contacts_list_item.xml
를 정의합니다.
<?xml version="1.0" encoding="utf-8"?> <TextView xmlns:android="http://schemas.android.com/apk/res/android" android:id="@android:id/text1" android:layout_width="match_parent" android:layout_height="wrap_content" android:clickable="true"/>
이 XML은 내장된 Android TextView
위젯을 사용합니다.
android:text1
입니다.
참고: 문자열을 간접적으로 가져오고 싶어 할 수 있으므로 이 과정에서는 사용자로부터 검색 문자열을 가져오는 것과 관련된 UI는 설명하지 않습니다. 예를 들어 사용자에게 수신 문자 메시지의 문자열과 이름이 일치하는 연락처를 검색하는 옵션
작성한 두 개의 레이아웃 파일은 ListView
를 표시하는 사용자 인터페이스를 정의합니다. 다음 단계에서는 이 UI를 사용하는 코드를 작성하여 연락처 목록을 표시합니다.
연락처 목록을 표시하는 프래그먼트 정의
연락처 목록을 표시하려면 먼저 Activity
에서 로드되는 Fragment
를 정의하세요. Fragment
사용은 더 유연한 기법입니다. 목록을 표시하는 데 Fragment
하나를 사용하고, 사용자가 목록에서 선택한 연락처의 세부정보를 표시하는 데 두 번째 Fragment
를 사용할 수 있기 때문입니다. 이 접근 방식을 사용하면 이 과정에서 소개된 기법 중 한 가지를
연락처 세부정보 검색 과정의 기법 중 하나와 조합할 수 있습니다.
다음에서 하나 이상의 Fragment
객체를 사용하는 방법을 알아보려면
Activity
, 교육 과정 읽기
프래그먼트로 동적 UI 빌드
연락처 제공자에 대한 쿼리를 작성하는 데 도움이 되도록 Android 프레임워크는
ContactsContract
라는 계약 클래스로, 유용한 API를 정의합니다.
제공자에 액세스하기 위한 상수와 메서드. 이 클래스를 사용하면 콘텐츠 URI, 표 이름, 열에 관한 고유 상수를 정의하지 않아도 됩니다. 이 클래스를 사용하려면
다음 진술을 포함해야 합니다.
Kotlin
import android.provider.ContactsContract
자바
import android.provider.ContactsContract;
이 코드는 CursorLoader
를 사용하여 데이터를 가져오므로
로더 인터페이스를 구현하도록 지정해야 합니다.
LoaderManager.LoaderCallbacks
입니다. 또한 사용자가 검색 결과 목록에서 어떤 연락처를 선택하는지 감지하는 데 도움이 되도록 어댑터 인터페이스 AdapterView.OnItemClickListener
를 구현하세요. 예를 들면 다음과 같습니다.
Kotlin
... import android.support.v4.app.Fragment import android.support.v4.app.LoaderManager import android.widget.AdapterView ... class ContactsFragment : Fragment(), LoaderManager.LoaderCallbacks<Cursor>, AdapterView.OnItemClickListener {
자바
... import android.support.v4.app.Fragment; import android.support.v4.app.LoaderManager.LoaderCallbacks; import android.widget.AdapterView; ... public class ContactsFragment extends Fragment implements LoaderManager.LoaderCallbacks<Cursor>, AdapterView.OnItemClickListener {
전역 변수 정의
코드의 다른 부분에서 사용되는 전역 변수를 정의합니다.
Kotlin
... /* * Defines an array that contains column names to move from * the Cursor to the ListView. */ @SuppressLint("InlinedApi") private val FROM_COLUMNS: Array<String> = arrayOf( if ((Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB)) { ContactsContract.Contacts.DISPLAY_NAME_PRIMARY } else { ContactsContract.Contacts.DISPLAY_NAME } ) /* * Defines an array that contains resource ids for the layout views * that get the Cursor column contents. The id is pre-defined in * the Android framework, so it is prefaced with "android.R.id" */ private val TO_IDS: IntArray = intArrayOf(android.R.id.text1) ... class ContactsFragment : Fragment(), LoaderManager.LoaderCallbacks<Cursor>, AdapterView.OnItemClickListener { ... // Define global mutable variables // Define a ListView object lateinit var contactsList: ListView // Define variables for the contact the user selects // The contact's _ID value var contactId: Long = 0 // The contact's LOOKUP_KEY var contactKey: String? = null // A content URI for the selected contact var contactUri: Uri? = null // An adapter that binds the result Cursor to the ListView private val cursorAdapter: SimpleCursorAdapter? = null
자바
... /* * Defines an array that contains column names to move from * the Cursor to the ListView. */ @SuppressLint("InlinedApi") private final static String[] FROM_COLUMNS = { Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB ? ContactsContract.Contacts.DISPLAY_NAME_PRIMARY : ContactsContract.Contacts.DISPLAY_NAME }; /* * Defines an array that contains resource ids for the layout views * that get the Cursor column contents. The id is pre-defined in * the Android framework, so it is prefaced with "android.R.id" */ private final static int[] TO_IDS = { android.R.id.text1 }; // Define global mutable variables // Define a ListView object ListView contactsList; // Define variables for the contact the user selects // The contact's _ID value long contactId; // The contact's LOOKUP_KEY String contactKey; // A content URI for the selected contact Uri contactUri; // An adapter that binds the result Cursor to the ListView private SimpleCursorAdapter cursorAdapter; ...
참고: Contacts.DISPLAY_NAME_PRIMARY
에는 Android 3.0(API 버전 11) 이상이 필요하므로 앱의 minSdkVersion
을 10 이하로 설정하면 Android 스튜디오에서 Android 린트 경고가 생성됩니다. 이 경고를 사용 중지하려면 FROM_COLUMNS
정의 앞에 @SuppressLint("InlinedApi")
주석을 추가하세요.
프래그먼트 초기화
Fragment
를 초기화합니다. Android 시스템에 필요한 빈 공개 생성자를 추가하고 콜백 메서드 onCreateView()
에서 Fragment
객체의 UI를 확장합니다.
예를 들면 다음과 같습니다.
Kotlin
// A UI Fragment must inflate its View override fun onCreateView( inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle? ): View? { // Inflate the fragment layout return inflater.inflate(R.layout.contact_list_fragment, container, false) }
자바
// Empty public constructor, required by the system public ContactsFragment() {} // A UI Fragment must inflate its View @Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { // Inflate the fragment layout return inflater.inflate(R.layout.contact_list_fragment, container, false); }
ListView의 CursorAdapter 설정
검색 결과를 ListView
에 바인딩하는 SimpleCursorAdapter
를 설정합니다. 연락처를 표시하는 ListView
객체를 가져오려면 Fragment
의 상위 활동을 사용하여 Activity.findViewById()
를 호출해야 합니다. setAdapter()
를 호출할 때 상위 활동의 Context
를 사용하세요.
예를 들면 다음과 같습니다.
Kotlin
override fun onActivityCreated(savedInstanceState: Bundle?) { super.onActivityCreated(savedInstanceState) ... // Gets the ListView from the View list of the parent activity activity?.also { contactsList = it.findViewById<ListView>(R.id.contact_list_view) // Gets a CursorAdapter cursorAdapter = SimpleCursorAdapter( it, R.layout.contact_list_item, null, FROM_COLUMNS, TO_IDS, 0 ) // Sets the adapter for the ListView contactsList.adapter = cursorAdapter } }
자바
public void onActivityCreated(Bundle savedInstanceState) { super.onActivityCreated(savedInstanceState); ... // Gets the ListView from the View list of the parent activity contactsList = (ListView) getActivity().findViewById(R.layout.contact_list_view); // Gets a CursorAdapter cursorAdapter = new SimpleCursorAdapter( getActivity(), R.layout.contact_list_item, null, FROM_COLUMNS, TO_IDS, 0); // Sets the adapter for the ListView contactsList.setAdapter(cursorAdapter); }
특정 연락처 리스너 설정
일반적으로 개발자가 검색 결과를 표시할 때는 사용자가 추가로 처리할 연락처 한 개를 선택하도록 허용합니다. 예를 들어 사용자가 연락처를 클릭하면 연락처의 주소를 지도에 표시할 수 있습니다. 이 기능을 제공하기 위해 먼저
클래스를 클릭 리스너로 지정하여 Fragment
를 클릭 리스너로 지정할 수 있습니다.
섹션에 표시된 대로 AdapterView.OnItemClickListener
를 구현합니다.
연락처 목록을 표시하는 Fragment 정의.
리스너 설정을 계속하려면 다음과 같이 ListView
에 바인딩하세요.
onActivityCreated()
에서 setOnItemClickListener()
메서드를 호출합니다. 예를 들면 다음과 같습니다.
Kotlin
fun onActivityCreated(savedInstanceState:Bundle) { ... // Set the item click listener to be the current fragment. contactsList.onItemClickListener = this ... }
자바
public void onActivityCreated(Bundle savedInstanceState) { ... // Set the item click listener to be the current fragment. contactsList.setOnItemClickListener(this); ... }
현재 Fragment
가 ListView
의 OnItemClickListener
라고 지정했으므로 이제 클릭 이벤트를 처리하는 필수 메서드 onItemClick()
을 구현해야 합니다. 이는 이후 섹션에 설명되어 있습니다.
프로젝션 정의
쿼리에서 반환하려는 열을 포함한 상수를 정의합니다. ListView
에 있는 각 항목은 연락처 이름의 기본 양식이 포함된 연락처의 표시 이름을 보여줍니다. Android 3.0 (API 버전 11) 이상에서
이 열의 이름은
Contacts.DISPLAY_NAME_PRIMARY
; 해당 이름은
Contacts.DISPLAY_NAME
입니다.
Contacts._ID
열은
SimpleCursorAdapter
바인딩 프로세스
Contacts._ID
및
LOOKUP_KEY
를 함께 사용하면
사용자가 선택하는 연락처에 대한 콘텐츠 URI를 구성합니다.
Kotlin
... @SuppressLint("InlinedApi") private val PROJECTION: Array<out String> = arrayOf( ContactsContract.Contacts._ID, ContactsContract.Contacts.LOOKUP_KEY, if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) ContactsContract.Contacts.DISPLAY_NAME_PRIMARY else ContactsContract.Contacts.DISPLAY_NAME )
자바
... @SuppressLint("InlinedApi") private static final String[] PROJECTION = { Contacts._ID, Contacts.LOOKUP_KEY, Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB ? ContactsContract.Contacts.DISPLAY_NAME_PRIMARY : ContactsContract.Contacts.DISPLAY_NAME };
Cursor 열 색인의 상수 정의
Cursor
의 개별 열에서 데이터를 가져오려면 다음이 필요합니다.
열 색인의 Cursor
상수를 정의할 수 있습니다.
Cursor
열의 색인
프로젝션의 열 이름 순서와 동일합니다. 예를 들면 다음과 같습니다.
Kotlin
// The column index for the _ID column private const val CONTACT_ID_INDEX: Int = 0 // The column index for the CONTACT_KEY column private const val CONTACT_KEY_INDEX: Int = 1
자바
// The column index for the _ID column private static final int CONTACT_ID_INDEX = 0; // The column index for the CONTACT_KEY column private static final int CONTACT_KEY_INDEX = 1;
선택 기준 지정
원하는 데이터를 지정하려면 텍스트 표현식과 변수의 조합을 만드세요. 는 제공자에게 검색할 데이터 열과 찾을 값을 알려줍니다.
텍스트 표현식의 경우 검색 열을 표시하는 상수를 정의합니다. 이 표현식에는 값도 포함될 수 있지만 '?' 자리표시자로 값을 표시하는 것이 좋습니다. 검색 중에 자리표시자는 배열 값으로 교체됩니다. '?'를 자리표시자로 사용하면 검색 사양이 SQL 컴파일이 아닌 바인딩으로 생성됩니다. 이렇게 하면 악의적인 SQL을 사용할 가능성이 없어지고 인코더-디코더입니다. 예를 들면 다음과 같습니다.
Kotlin
// Defines the text expression @SuppressLint("InlinedApi") private val SELECTION: String = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) "${ContactsContract.Contacts.DISPLAY_NAME_PRIMARY} LIKE ?" else "${ContactsContract.Contacts.DISPLAY_NAME} LIKE ?" ... // Defines a variable for the search string private val searchString: String = ... // Defines the array to hold values that replace the ? private val selectionArgs = arrayOf<String>(searchString)
자바
// Defines the text expression @SuppressLint("InlinedApi") private static final String SELECTION = Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB ? Contacts.DISPLAY_NAME_PRIMARY + " LIKE ?" : Contacts.DISPLAY_NAME + " LIKE ?"; // Defines a variable for the search string private String searchString; // Defines the array to hold values that replace the ? private String[] selectionArgs = { searchString };
onItemClick() 메서드 정의
이전 섹션에서는 ListView
의 항목 클릭 리스너를 설정했습니다.
이제 AdapterView.OnItemClickListener.onItemClick()
메서드를 정의하여 리스너의 작업을 구현합니다.
Kotlin
override fun onItemClick(parent: AdapterView<*>, view: View?, position: Int, id: Long) { // Get the Cursor val cursor: Cursor? = (parent.adapter as? CursorAdapter)?.cursor?.apply { // Move to the selected contact moveToPosition(position) // Get the _ID value contactId = getLong(CONTACT_ID_INDEX) // Get the selected LOOKUP KEY contactKey = getString(CONTACT_KEY_INDEX) // Create the contact's content Uri contactUri = ContactsContract.Contacts.getLookupUri(contactId, mContactKey) /* * You can use contactUri as the content URI for retrieving * the details for a contact. */ } }
자바
@Override public void onItemClick( AdapterView<?> parent, View item, int position, long rowID) { // Get the Cursor Cursor cursor = parent.getAdapter().getCursor(); // Move to the selected contact cursor.moveToPosition(position); // Get the _ID value contactId = cursor.getLong(CONTACT_ID_INDEX); // Get the selected LOOKUP KEY contactKey = cursor.getString(CONTACT_KEY_INDEX); // Create the contact's content Uri contactUri = Contacts.getLookupUri(contactId, mContactKey); /* * You can use contactUri as the content URI for retrieving * the details for a contact. */ }
로더 초기화
CursorLoader
를 사용하여 데이터를 검색하므로
백그라운드 스레드와 비동기식 코드를 제어하는 기타 변수를 초기화해야 함
가져올 수 있습니다. 다음에서 초기화 수행
다음으로 onCreate()
다음 예에 나와 있습니다.
Kotlin
class ContactsFragment : Fragment(), LoaderManager.LoaderCallbacks<Cursor> { ... override fun onCreate(savedInstanceState: Bundle?) { // Always call the super method first super.onCreate(savedInstanceState) ... // Initializes the loader loaderManager.initLoader(0, null, this)
자바
public class ContactsFragment extends Fragment implements LoaderManager.LoaderCallbacks<Cursor> { ... // Called just before the Fragment displays its UI @Override public void onCreate(Bundle savedInstanceState) { // Always call the super method first super.onCreate(savedInstanceState); ... // Initializes the loader getLoaderManager().initLoader(0, null, this);
onCreateLoader() 구현
onCreateLoader()
메서드를 구현합니다. 이 메서드는 initLoader()
를 호출한 직후 로더 프레임워크에서 호출됩니다.
onCreateLoader()
후
검색 문자열 패턴을 설정합니다. 문자열을 패턴으로 만들려면 '%'(퍼센트) 문자를 삽입하여 0개 이상의 문자 시퀀스를 표시하거나 '_'(밑줄) 문자를 사용하여 단일 문자를 표시하거나 또는 두 가지를 모두 표시할 수 있습니다. 예를 들어 패턴 '%Jefferson%'는
'토마스 제퍼슨' 모두와 일치 그리고 'Jefferson Davis'가 있습니다.
메서드에서 새 CursorLoader
를 반환합니다. 콘텐츠 URI의 경우 Contacts.CONTENT_URI
를 사용합니다.
이 URI는 다음 예와 같이 전체 표를 참조합니다.
Kotlin
... override fun onCreateLoader(loaderId: Int, args: Bundle?): Loader<Cursor> { /* * Makes search string into pattern and * stores it in the selection array */ selectionArgs[0] = "%$mSearchString%" // Starts the query return activity?.let { return CursorLoader( it, ContactsContract.Contacts.CONTENT_URI, PROJECTION, SELECTION, selectionArgs, null ) } ?: throw IllegalStateException() }
자바
... @Override public Loader<Cursor> onCreateLoader(int loaderId, Bundle args) { /* * Makes search string into pattern and * stores it in the selection array */ selectionArgs[0] = "%" + searchString + "%"; // Starts the query return new CursorLoader( getActivity(), ContactsContract.Contacts.CONTENT_URI, PROJECTION, SELECTION, selectionArgs, null ); }
onLoadFinished() 및 onLoaderReset() 구현
onLoadFinished()
메서드를 구현합니다. 연락처 제공자가 쿼리 결과를 반환하면 로더 프레임워크는 onLoadFinished()
를 호출합니다. 이 메서드에서
검색결과 Cursor
개
SimpleCursorAdapter
입니다. 그러면 ListView
가 검색 결과로 자동으로 업데이트됩니다.
Kotlin
override fun onLoadFinished(loader: Loader<Cursor>, cursor: Cursor) { // Put the result Cursor in the adapter for the ListView cursorAdapter?.swapCursor(cursor) }
자바
@Override public void onLoadFinished(Loader<Cursor> loader, Cursor cursor) { // Put the result Cursor in the adapter for the ListView cursorAdapter.swapCursor(cursor); }
onLoaderReset()
메서드는 로더 프레임워크가
결과 Cursor
에 오래된 데이터가 포함되어 있습니다. 기존 Cursor
의 SimpleCursorAdapter
참조를 삭제합니다. 삭제하지 않으면 로더 프레임워크에서 Cursor
를 재활용하지 않아서 메모리 누수가 발생합니다. 예를 들면 다음과 같습니다.
Kotlin
override fun onLoaderReset(loader: Loader<Cursor>) { // Delete the reference to the existing Cursor cursorAdapter?.swapCursor(null) }
자바
@Override public void onLoaderReset(Loader<Cursor> loader) { // Delete the reference to the existing Cursor cursorAdapter.swapCursor(null); }
이제 검색 문자열을 연락처 이름과 일치시키고
결과는 ListView
입니다. 사용자는 연락처 이름을 클릭하여 선택할 수 있습니다.
그러면 리스너가 실행되어 연락처 데이터로 추가 작업을 할 수 있습니다. 예를 들어 연락처 세부정보를 검색할 수 있습니다. 방법을 알아보려면
연락처 세부정보 검색을 참조하세요.
검색 사용자 인터페이스에 관해 자세히 알아보려면 API 가이드를 참고하세요. 검색 인터페이스를 만듭니다.
이 강의의 나머지 섹션에서는 연락처 제공자.
특정 데이터 유형별로 연락처 일치시키기
이 기술을 사용하면 일치시키려는 데이터 유형을 지정할 수 있습니다. 검색 중 이름은 이러한 유형의 쿼리의 구체적인 예이지만 모든 유형에 대해 할 수도 있습니다. 정보를 제공합니다. 예를 들어 특정 우편번호 이 경우 검색 문자열은 우편번호에 저장된 데이터와 일치해야 하며 행
이러한 유형의 검색을 구현하려면 먼저 이전 섹션:
- 제공자 읽기 권한 요청
- ListView 및 항목 레이아웃 정의
- 연락처 목록을 표시하는 Fragment 정의
- 전역 변수 정의
- Fragment 초기화
- ListView의 CursorAdapter 설정
- 특정 연락처 리스너 설정
-
Cursor 열 색인의 상수 정의
다른 테이블에서 데이터를 검색하더라도 프로젝션은 동일하므로 Cursor의 경우 동일한 색인을 사용할 수 있습니다.
- onItemClick() 메서드 정의
- 로더 초기화
- onLoadFinished() 및 onLoaderReset() 구현
다음 단계에서는 검색 문자열을 특정 세부 데이터 유형과 일치시키고 결과를 표시하기 위해 필요한 추가 코드를 보여줍니다.
데이터 유형 및 표 선택
특정 유형의 세부정보 데이터를 검색하려면 맞춤 MIME 유형 값을 알아야 합니다.
데이터 유형일 수도 있습니다. 각 데이터 유형에는 데이터 유형과 연결된 ContactsContract.CommonDataKinds
의 서브클래스에서 상수 CONTENT_ITEM_TYPE
으로 정의된 고유한 MIME 유형 값이 있습니다.
서브클래스에는 데이터 유형을 나타내는 이름이 있습니다. 예를 들어 이메일 데이터의 서브클래스는 ContactsContract.CommonDataKinds.Email
이고 이메일 데이터의 맞춤 MIME 유형은 상수 Email.CONTENT_ITEM_TYPE
으로 정의됩니다.
검색에 ContactsContract.Data
표를 사용합니다. 모든
프로젝션, 선택 절, 정렬 순서에 필요한 상수는
이 테이블에 상속됩니다.
프로젝션 정의
투영을 정의하려면 다음에 정의된 열을 하나 이상 선택합니다.
ContactsContract.Data
또는 상속되는 클래스. 이
연락처 제공자가 ContactsContract.Data
간에 암시적 조인 실행
다른 테이블의 행을 반환할 수 있습니다. 예를 들면 다음과 같습니다.
Kotlin
@SuppressLint("InlinedApi") private val PROJECTION: Array<out String> = arrayOf( /* * The detail data row ID. To make a ListView work, * this column is required. */ ContactsContract.Data._ID, // The primary display name if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) ContactsContract.Data.DISPLAY_NAME_PRIMARY else ContactsContract.Data.DISPLAY_NAME, // The contact's _ID, to construct a content URI ContactsContract.Data.CONTACT_ID, // The contact's LOOKUP_KEY, to construct a content URI ContactsContract.Data.LOOKUP_KEY )
자바
@SuppressLint("InlinedApi") private static final String[] PROJECTION = { /* * The detail data row ID. To make a ListView work, * this column is required. */ ContactsContract.Data._ID, // The primary display name Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB ? ContactsContract.Data.DISPLAY_NAME_PRIMARY : ContactsContract.Data.DISPLAY_NAME, // The contact's _ID, to construct a content URI ContactsContract.Data.CONTACT_ID, // The contact's LOOKUP_KEY, to construct a content URI ContactsContract.Data.LOOKUP_KEY // A permanent link to the contact };
검색 기준 정의
특정 데이터 유형 내에서 문자열을 검색하려면 다음 중에서 선택 절을 구성하세요.
-
검색 문자열을 포함한 열 이름: 이 이름은 데이터 유형에 따라 다르므로 데이터 유형에 상응하는
ContactsContract.CommonDataKinds
의 서브클래스를 찾은 다음 서브클래스에서 열 이름을 선택해야 합니다. 예를 들어 사용하려면 열 사용Email.ADDRESS
- 검색 문자열 자체: 선택 절에서 '?' 문자로 표시됩니다.
-
맞춤 MIME 유형 값을 포함한 열의 이름: 이 이름은 항상
Data.MIMETYPE
-
데이터 유형의 맞춤 MIME 유형 값: 앞서 설명한 것처럼 이 값은
CONTENT_ITEM_TYPE
ContactsContract.CommonDataKinds
서브클래스. 예를 들어 MIME는 이메일 데이터의 유형 값이Email.CONTENT_ITEM_TYPE
상수의 시작과 끝에 ''
'(작은따옴표) 문자를 연결하여 값을 작은따옴표로 묶습니다. 작은따옴표로 묶지 않으면 제공자가 값을 문자열 값이 아닌 변수 이름으로 해석합니다. 사용자가 제공한 값 대신 상수를 사용하므로 이 값에 자리표시자를 사용하지 않아도 됩니다.
예를 들면 다음과 같습니다.
Kotlin
/* * Constructs search criteria from the search string * and email MIME type */ private val SELECTION: String = /* * Searches for an email address * that matches the search string */ "${Email.ADDRESS} LIKE ? AND " + /* * Searches for a MIME type that matches * the value of the constant * Email.CONTENT_ITEM_TYPE. Note the * single quotes surrounding Email.CONTENT_ITEM_TYPE. */ "${ContactsContract.Data.MIMETYPE } = '${Email.CONTENT_ITEM_TYPE}'"
자바
/* * Constructs search criteria from the search string * and email MIME type */ private static final String SELECTION = /* * Searches for an email address * that matches the search string */ Email.ADDRESS + " LIKE ? " + "AND " + /* * Searches for a MIME type that matches * the value of the constant * Email.CONTENT_ITEM_TYPE. Note the * single quotes surrounding Email.CONTENT_ITEM_TYPE. */ ContactsContract.Data.MIMETYPE + " = '" + Email.CONTENT_ITEM_TYPE + "'";
다음으로 선택 인수를 포함하도록 변수를 정의합니다.
Kotlin
private var searchString: String? = null private val selectionArgs: Array<String> = arrayOf("")
자바
String searchString; String[] selectionArgs = { "" };
onCreateLoader() 구현
이제 원하는 데이터와 이 데이터를 찾는 방법을 지정했으므로 onCreateLoader()
구현에서 쿼리를 정의합니다.
이 항목에서 새 CursorLoader
반환
메서드를 사용하여 프로젝션, 선택 텍스트 표현식 및 선택 배열을
인수입니다. 콘텐츠 URI의 경우 다음을 사용합니다.
Data.CONTENT_URI
예를 들면 다음과 같습니다.
Kotlin
override fun onCreateLoader(id: Int, args: Bundle?): Loader<Cursor> { // OPTIONAL: Makes search string into pattern searchString = "%$mSearchString%" searchString?.also { // Puts the search string into the selection criteria selectionArgs[0] = it } // Starts the query return activity?.let { CursorLoader( it, ContactsContract.Data.CONTENT_URI, PROJECTION, SELECTION, selectionArgs, null ) } ?: throw IllegalStateException() }
자바
@Override public Loader<Cursor> onCreateLoader(int loaderId, Bundle args) { // OPTIONAL: Makes search string into pattern searchString = "%" + searchString + "%"; // Puts the search string into the selection criteria selectionArgs[0] = searchString; // Starts the query return new CursorLoader( getActivity(), Data.CONTENT_URI, PROJECTION, SELECTION, selectionArgs, null ); }
이러한 코드 스니펫은 특정 세부 데이터 유형을 기반으로 한 간단한 역순 검색의 기반이 됩니다. 앱이 이메일과 같은 특정 데이터 유형에 초점을 맞추고 사용자가 일부 데이터와 관련된 이름을 가져오도록 허용하려고 할 때 유용한 기술입니다.
모든 데이터 유형별로 연락처 일치시키기
모든 데이터 유형을 기반으로 연락처를 검색하면 사용자의 데이터가 이름, 이메일 주소, 우편 주소, 전화번호 등을 포함한 검색 문자열을 검색합니다. 이에 따라 광범위한 검색결과가 표시됩니다. 예를 들어 검색 문자열이 'Doe'라면 모든 데이터 유형 검색을 실행할 경우 'John Doe'라는 연락처를 반환하며 'Doe Street'에 거주 중인 연락처도 반환합니다.
이러한 검색 유형을 구현하려면 먼저 이전 섹션에서 설명한 대로 다음 코드를 구현하세요.
- 제공자 읽기 권한 요청
- ListView 및 항목 레이아웃 정의
- 연락처 목록을 표시하는 Fragment 정의
- 전역 변수 정의
- Fragment 초기화
- ListView의 CursorAdapter 설정
- 특정 연락처 리스너 설정
- 프로젝션 정의
-
Cursor 열 색인의 상수 정의
이 유형의 검색에서는 이름으로 연락처를 일치시킨 후 결과 표시 섹션에 사용된 것과 동일한 표를 사용하게 됩니다. 열 색인도 동일한 것으로 사용합니다.
- onItemClick() 메서드 정의
- 로더 초기화
- onLoadFinished() 및 onLoaderReset() 구현
다음 단계에서는 검색 문자열을 모든 데이터 유형과 일치시키고 결과를 표시하기 위해 필요한 추가 코드를 보여줍니다.
선택 기준 삭제
SELECTION
상수 또는 mSelectionArgs
변수를 정의하지 마세요.
이러한 검색 유형에서는 사용되지 않습니다.
onCreateLoader() 구현
onCreateLoader()
구현
새 CursorLoader
를 반환합니다.
검색 문자열을 패턴으로 변환할 필요는 없습니다. 연락처 제공자가
자동으로 수행됩니다 Contacts.CONTENT_FILTER_URI
를 기본 URI로 사용하고 Uri.withAppendedPath()
를 호출하여 검색 문자열을 이 URI에 추가합니다. 이 URI를 사용하면 모든 데이터 유형과 관련해 자동으로 검색이 실행됩니다. 다음 예를 확인하세요.
Kotlin
override fun onCreateLoader(loaderId: Int, args: Bundle?): Loader<Cursor> { /* * Appends the search string to the base URI. Always * encode search strings to ensure they're in proper * format. */ val contentUri: Uri = Uri.withAppendedPath( ContactsContract.Contacts.CONTENT_FILTER_URI, Uri.encode(searchString) ) // Starts the query return activity?.let { CursorLoader( it, contentUri, PROJECTION2, null, null, null ) } ?: throw IllegalStateException() }
자바
@Override public Loader<Cursor> onCreateLoader(int loaderId, Bundle args) { /* * Appends the search string to the base URI. Always * encode search strings to ensure they're in proper * format. */ Uri contentUri = Uri.withAppendedPath( Contacts.CONTENT_FILTER_URI, Uri.encode(searchString)); // Starts the query return new CursorLoader( getActivity(), contentUri, PROJECTION, null, null, null ); }
이러한 코드 스니펫은 연락처 제공자를 광범위하게 검색하는 앱의 기반이 됩니다. 이 기법은 피플 앱의 연락처 목록 화면과 유사한 기능을 구현하려는 앱에 유용합니다.