Data Storage: Part 4 (Content Providers)
Content Providers Content providers allow the sharing of data between applications. Inter-process communication (IPC) and content providers are the two main ways to share application data. (Technically applications can also share data via external storage.) ©SoftMoore Consulting
Content Providers (continued) Content providers are one of the primary building blocks of Android applications (along with activities, services, and broadcast receivers). A content provider is implemented as a subclass of class ContentProvider (in package android.content). Android ships with a number of content providers for common data types (audio, video, images, contacts, etc.) See documentation for package android.provider. Most of these providers require permission to read their data. ©SoftMoore Consulting
Selected Android Content Providers Browser – bookmarks, browser history, etc. Contacts – names, phone numbers, etc. Media Store – audio (music), video, images, etc. Calendar – user’s calendar events Settings – Bluetooth settings, ringtones, Wi-Fi settings, locale, etc. User Dictionary – user-defined words for use with predictive text input. ©SoftMoore Consulting
Overview of Content Providers Applications can extend the abstract class ContentProvider (in package android.content) and override the appropriate methods to allow shared access to their data. Regardless of how a content provider actually stores its data, it exposes that data as though it were a simple database table. Note: Every record must include a numeric _id field that uniquely identifies the record within the table. Content providers encapsulate data and make it available to other applications though the ContentResolver interface. ©SoftMoore Consulting
Overview of Content Providers (continued) Accessing the data exposed by a content provider is very similar to accessing the data in an SQLite database via the convenience methods except that the content provider methods have a URI as their first parameter rather than a database. Content providers are useful for building applications with databases even if the data is not shared outside of the application. A content provider must be declared in the application manifest (AndroidManifest.xml) using a <provider> element, similar to the way that each activity must be declared. ©SoftMoore Consulting
Example: Querying a SQLite Database Versus Querying a Content Provider Querying a SQLite database using convenience methods SQLiteOpenHelper openHelper = ... ; SQLiteDatabase db = openHelper.getReadableDatabase(); String tableName = ... ; String[] columns = ... ; String orderBy = ... ; Cursor cursor = db.query(tableName, columns, null, null, null, null, orderBy); Querying a content provider ContentResolver cr = getContentResolver(); Uri uri = ... ; Cursor cursor = cr.query(uri, columns, null, null, ©SoftMoore Consulting
Example: Inserting into a SQLite Database Versus Inserting into a Content Provider Inserting into a SQLite database using convenience methods SQLiteOpenHelper openHelper = ... ; SQLiteDatabase db = openHelper.getWritableDatabase(); String tableName = " ... "; ContentValues values = new ContentValues(); values.put(" ... ", " ... "); ... // additional values.put() statements db.insert(tableName, null, values); Inserting into a content provider ContentResolver cr = getContentResolver(); Uri uri = ... ; cr.insert(uri, null, values); ©SoftMoore Consulting
URIs Each content provider exposes public URIs (Uniform Resource Identifiers) that uniquely identify its data sets. A content provider that controls multiple data sets (multiple tables) exposes a separate URI for each one plus possibly additional URIs for joins of the tables. Classes Uri and Uri.Builder (in package android.net) provide support for creating and accessing URIs for content providers. ©SoftMoore Consulting
URI Format The URI for a content provider is a string of the form content://authority/path/id content:// – the standard required prefix that identifies the data as being controlled by a content provider. authority – a string that uniquely identifies the content provider; e.g., the provider package name with table name. path – zero or more segments, separated by a forward slash (/), that identify some subset of the provider’s data. Most providers use the path part to identify individual tables. id – the identifier of the specific record being requested. If the request is not limited to a single record, this segment and the trailing slash are omitted. ©SoftMoore Consulting
URI Constants A content provider usually defines constants in a “contract” class for its URIs to simplify client code and make future updates cleaner. public static final Uri CONTENT_URI = ...; The URI constant is used in all interactions with the content provider. Android defines constants for all the providers that come with the platform. Examples include Media Store classes (e.g., MediaStore.Audio.Albums) define two constants for their data, INTERNAL_CONTENT_URI and EXTERNAL_CONTENT_URI (for different storage locations) Class Browser defines two constants for its data, BOOKMARKS_URI and SEARCHES_URI. ©SoftMoore Consulting
Content URI Examples Applications usually just use content URI content://user_dictionary/words UserDictionary.Words.CONTENT_URI content://user_dictionary/words/4 row from user dictionary with _id = 4 can be constructed as ContentUris.withAppendedId(UserDictionary.Words.CONTENT_URI, 4); content://edu.citadel.android.emergency/emergency_contacts EmergencyProvider.CONTENT_URI (sample application) Applications usually just use content URI constants declared in a contract class. ©SoftMoore Consulting
Class ContentResolver Content providers encapsulate data and make it available to other applications though the ContentResolver interface. When a request is made via a ContentResolver the system inspects the authority of the given URI and passes the request to the content provider registered with the authority. The content provider can interpret the rest of the URI however it wants. Most ContentResolver methods take the URI as their first parameter. The URI identifies which provider the ContentResolver should talk to and which table of the provider is being targeted. ©SoftMoore Consulting
Class ContentProvider Class ContentProvider is an abstract class in package android.content. Class ContentProvider contains several abstract methods that resemble the convenience methods use to access a SQLite database. To create a content provider, extend this abstract class and provide implementations for the abstract methods (shown on next slide). ©SoftMoore Consulting
Abstract Methods in Class ContentProvider onCreate() used to initialize the provider query(Uri, String[], String, String[], String) returns data to the caller insert(Uri, ContentValues) inserts new data into the content provider update(Uri, ContentValues, String, String[]) updates existing data in the content provider delete(Uri, String, String[]) deletes data from the content provider getType(Uri) returns the MIME type of data in the content provider ©SoftMoore Consulting
Content Provider Permissions A content provider application can specify permissions (as part of the <provider> element in the application manifest) that other applications must have in order to access the provider’s data. Other applications request the permissions they need in order to access the provider. End users see the requested permissions when they install the application. If a content provider application doesn’t specify any permissions, then other applications have no access to the provider's data. However, components within the application always have full read and write access, regardless of the specified permissions. ©SoftMoore Consulting
Querying a Content Provider Querying a content provider requires three pieces of information: The URI that identifies the provider The names of the data fields you want to receive The data types for those fields It is also possible to query a particular record if you know the ID for that record. Primary method used to query a content provider: ContentResolver.query() ©SoftMoore Consulting
Example: Querying Contacts for the Display Name Uri uri = ContactsContract.Contacts.CONTENT_URI; String[] columns = new String[] { ContactsContract.Contacts._ID, ContactsContract.Contacts.DISPLAY_NAME }; String orderBy = ContactsContract.Contacts.DISPLAY_NAME + " COLLATE LOCALIZED ASC"; ContentResolver cr = getContentResolver(); Cursor cursor = cr.query(uri, columns, null, null, orderBy); ©SoftMoore Consulting
Creating a Content Provider To create a content provider you must: Set up a system for storing the data. (e.g., a SQLite database) Extend the ContentProvider class to provide access. Override the abstract methods onCreate(), insert(), query(), delete(), etc. Note: Some methods can have trivial implementations. For example, if other applications are only allowed to read content, only methods onCreate() and query() need nontrivial implementations. Declare the content provider in the application’s manifest file (AndroidManifest.xml). ©SoftMoore Consulting
Creating a Content Provider (continued) To create a content provider you should: Define a public static final Uri named CONTENT_URI that represents the full “content:” URI managed by the content provider. To guarantee that the string has a unique value, use the fully-qualified class name of the content provider converted to lowercase. Define the column names that the content provider will return to clients; e.g., the column names of the underlying database. Also define public static String constants that clients can use to specify the columns in queries and other instructions. Be sure to include an integer column named _id. ©SoftMoore Consulting
Creating a Content Provider (continued) Carefully document the data type of each column. If you are handling a new data type, define a new MIME type to return in your implementation of ContentProvider.getType(). There is one form of the MIME type for a single record and another for multiple records. (See the link for “Content Providers” at the end of this section). Use the Uri methods to help determine what is being requested. ©SoftMoore Consulting
Example: Content Provider for Emergency Contacts in Previous Section public class EmergencyProvider extends ContentProvider { private static final String AUTHORITY = EmergencyDbContract.AUTHORITY; private static final String TABLE_NAME = EmergencyDbContract.TABLE_NAME; public static final Uri CONTENT_URI = EmergencyDbContract.CONTENT_URI; public static final String[] COLUMNS = EmergencyDbContract.COLUMNS; public static final int CONTACTS = 1; // contacts table public static final int CONTACTS_ID = 2; // single contact ... } See handout EmergencyProvider.java. Pay special attention to use of UriMatcher. ©SoftMoore Consulting
Example: Emergency Contacts Version 4 This example implements a ListView for emergency contacts using a content provider and a SimpleCursorAdapter. Same layouts and screen shots as with other versions of Emergency Contacts. Special considerations for this version: no entity class for emergency contacts uses the content provider to query and update the underlying database uses a SimpleCursorAdapter rather than an array adapter does not need a custom CursorLoader ©SoftMoore Consulting
Example: Emergency Contacts Version 4 (Main Activity) ... private static final String[] COLUMNS = { "name", "phone_num" }; private static final int[] VIEWS = { R.id.name, R.id.phoneNum }; private SimpleCursorAdapter adapter; private ListView lv; @Override protected void onCreate(Bundle savedInstanceState) { lv = (ListView) findViewById(R.id.contactListView); lv.setEmptyView(findViewById(R.id.empty_list)); (continued on next slide) ©SoftMoore Consulting
Example: Emergency Contacts Version 4 (Main Activity – continued) // Create an empty adapter for displaying the loaded data adapter = new SimpleCursorAdapter(this, R.layout.contact_layout, null, COLUMNS, VIEWS, 0); lv.setAdapter(adapter); lv.setOnItemClickListener(new AdapterView.OnItemClickListener() { ... }); lv.setOnItemLongClickListener( new AdapterView.OnItemLongClickListener() // in the onClick listener for dialog's positive button Uri itemUri = ContentUris.withAppendedId( EmergencyProvider.CONTENT_URI, id); getContentResolver().delete(itemUri, null, null); }) (continued on next slide) ©SoftMoore Consulting
Example: Emergency Contacts Version 4 (Main Activity – continued) Button addButton = (Button)findViewById(R.id.addButton); addButton.setOnClickListener(new View.OnClickListener() { ... }); // Initialize the loader manager. getLoaderManager().initLoader(0, null, this); } @Override public Loader<Cursor> onCreateLoader(int id, Bundle args) Uri uri = EmergencyProvider.CONTENT_URI; return new CursorLoader(this, uri, EmergencyProvider.COLUMNS, null, null, null); (continued on next slide) ©SoftMoore Consulting
Example: Emergency Contacts Version 4 (Main Activity – continued) @Override public void onLoadFinished(Loader<Cursor> loader, Cursor data) { // swap in the cursor adapter.swapCursor(data); } public void onLoaderReset(Loader<Cursor> loader) adapter.swapCursor(null); (continued on next slide) ©SoftMoore Consulting
Example: Emergency Contacts Version 4 (Main Activity – continued) @Override protected void onActivityResult(int requestCode, int resultCode, Intent intent) { if (requestCode == ADD_CONTACT_ACTIVITY && resultCode == RESULT_OK) adapter.notifyDataSetChanged(); } ©SoftMoore Consulting
Example: Emergency Contacts Version 4 (Second “Add Contact” Activity) // in the onClick listener for the submit button ... if (name.length() > 0 && phoneNum.length() > 0) { ContentValues contentValues = new ContentValues(); contentValues.put("name", name); contentValues.put("phone_num", phoneNum); getContentResolver().insert( EmergencyProvider.CONTENT_URI, contentValues); Intent intent = new Intent(); setResult(RESULT_OK, intent); finish(); } else This part differs from Version 2. (continued on next slide) ©SoftMoore Consulting
Relevant Links Content Providers Android Content Providers https://developer.android.com/guide/topics/providers/content-providers.html Android Content Providers http://www.tutorialspoint.com/android/android_content_providers.htm An Android Content Provider Tutorial http://www.techotopia.com/index.php/An_Android_Content_Provider_Tutorial Retrieving a List of Contacts https://developer.android.com/training/contacts-provider/retrieve-names.html Loading Contacts with Content Providers https://github.com/codepath/android_guides/wiki/Loading-Contacts-with-Content-Providers ©SoftMoore Consulting