Please note that the contents of this offline web site may be out of date. To access the most recent documentation visit the online version .
Note that links that point to online resources are green in color and will open in a new window.
We would love it if you could give us feedback about this material by filling this form (You have to be online to fill it)
BasicSyncAdapter / src / com.example.android.basicsyncadapter /

EntryListFragment.java

       
        1
       
       
        /*
       
       
        2
       
       
        * Copyright 2013 The Android Open Source Project
       
       
        3
       
       
        *
       
       
        4
       
       
        * Licensed under the Apache License, Version 2.0 (the "License");
       
       
        5
       
       
        * you may not use this file except in compliance with the License.
       
       
        6
       
       
        * You may obtain a copy of the License at
       
       
        7
       
       
        *
       
       
        8
       
       
        *      http://www.apache.org/licenses/LICENSE-2.0
       
       
        9
       
       
        *
       
       
        10
       
       
        * Unless required by applicable law or agreed to in writing, software
       
       
        11
       
       
        * distributed under the License is distributed on an "AS IS" BASIS,
       
       
        12
       
       
        * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
       
       
        13
       
       
        * See the License for the specific language governing permissions and
       
       
        14
       
       
        * limitations under the License.
       
       
        15
       
       
        */
       
       
        16
       
       
       
       
        17
       
       
        package com.example.android.basicsyncadapter;
       
       
        18
       
       
       
       
        19
       
       
        import android.accounts.Account;
       
       
        20
       
       
        import android.annotation.TargetApi;
       
       
        21
       
       
        import android.app.Activity;
       
       
        22
       
       
        import android.content.ContentResolver;
       
       
        23
       
       
        import android.content.Intent;
       
       
        24
       
       
        import android.content.SyncStatusObserver;
       
       
        25
       
       
        import android.database.Cursor;
       
       
        26
       
       
        import android.net.Uri;
       
       
        27
       
       
        import android.os.Build;
       
       
        28
       
       
        import android.os.Bundle;
       
       
        29
       
       
        import android.support.v4.app.ListFragment;
       
       
        30
       
       
        import android.support.v4.app.LoaderManager;
       
       
        31
       
       
        import android.support.v4.content.CursorLoader;
       
       
        32
       
       
        import android.support.v4.content.Loader;
       
       
        33
       
       
        import android.support.v4.widget.SimpleCursorAdapter;
       
       
        34
       
       
        import android.text.format.Time;
       
       
        35
       
       
        import android.util.Log;
       
       
        36
       
       
        import android.view.Menu;
       
       
        37
       
       
        import android.view.MenuInflater;
       
       
        38
       
       
        import android.view.MenuItem;
       
       
        39
       
       
        import android.view.View;
       
       
        40
       
       
        import android.widget.ListView;
       
       
        41
       
       
        import android.widget.TextView;
       
       
        42
       
       
       
       
        43
       
       
        import com.example.android.common.accounts.GenericAccountService;
       
       
        44
       
       
        import com.example.android.basicsyncadapter.provider.FeedContract;
       
       
        45
       
       
       
       
        46
       
       
        /**
       
       
        47
       
       
        * List fragment containing a list of Atom entry objects (articles) stored in the local database.
       
       
        48
       
       
        *
       
       
        49
       
       
        * <p>Database access is mediated by a content provider, specified in
       
       
        50
       
       
        * {@link com.example.android.basicsyncadapter.provider.FeedProvider}. This content
       
       
        51
       
       
        * provider is
       
       
        52
       
       
        * automatically populated by  {@link SyncService}.
       
       
        53
       
       
        *
       
       
        54
       
       
        * <p>Selecting an item from the displayed list displays the article in the default browser.
       
       
        55
       
       
        *
       
       
        56
       
       
        * <p>If the content provider doesn't return any data, then the first sync hasn't run yet. This sync
       
       
        57
       
       
        * adapter assumes data exists in the provider once a sync has run. If your app doesn't work like
       
       
        58
       
       
        * this, you should add a flag that notes if a sync has run, so you can differentiate between "no
       
       
        59
       
       
        * available data" and "no initial sync", and display this in the UI.
       
       
        60
       
       
        *
       
       
        61
       
       
        * <p>The ActionBar displays a "Refresh" button. When the user clicks "Refresh", the sync adapter
       
       
        62
       
       
        * runs immediately. An indeterminate ProgressBar element is displayed, showing that the sync is
       
       
        63
       
       
        * occurring.
       
       
        64
       
       
        */
       
       
        65
       
       
        public class EntryListFragment extends ListFragment
       
       
        66
       
       
        implements LoaderManager.LoaderCallbacks<Cursor> {
       
       
        67
       
       
       
       
        68
       
       
        private static final String TAG = "EntryListFragment";
       
       
        69
       
       
       
       
        70
       
       
        /**
       
       
        71
       
       
        * Cursor adapter for controlling ListView results.
       
       
        72
       
       
        */
       
       
        73
       
       
        private SimpleCursorAdapter mAdapter;
       
       
        74
       
       
       
       
        75
       
       
        /**
       
       
        76
       
       
        * Handle to a SyncObserver. The ProgressBar element is visible until the SyncObserver reports
       
       
        77
       
       
        * that the sync is complete.
       
       
        78
       
       
        *
       
       
        79
       
       
        * <p>This allows us to delete our SyncObserver once the application is no longer in the
       
       
        80
       
       
        * foreground.
       
       
        81
       
       
        */
       
       
        82
       
       
        private Object mSyncObserverHandle;
       
       
        83
       
       
       
       
        84
       
       
        /**
       
       
        85
       
       
        * Options menu used to populate ActionBar.
       
       
        86
       
       
        */
       
       
        87
       
       
        private Menu mOptionsMenu;
       
       
        88
       
       
       
       
        89
       
       
        /**
       
       
        90
       
       
        * Projection for querying the content provider.
       
       
        91
       
       
        */
       
       
        92
       
       
        private static final String[] PROJECTION = new String[]{
       
       
        93
       
       
        FeedContract.Entry._ID,
       
       
        94
       
       
        FeedContract.Entry.COLUMN_NAME_TITLE,
       
       
        95
       
       
        FeedContract.Entry.COLUMN_NAME_LINK,
       
       
        96
       
       
        FeedContract.Entry.COLUMN_NAME_PUBLISHED
       
       
        97
       
       
        };
       
       
        98
       
       
       
       
        99
       
       
        // Column indexes. The index of a column in the Cursor is the same as its relative position in
       
       
        100
       
       
        // the projection.
       
       
        101
       
       
        /** Column index for _ID */
       
       
        102
       
       
        private static final int COLUMN_ID = 0;
       
       
        103
       
       
        /** Column index for title */
       
       
        104
       
       
        private static final int COLUMN_TITLE = 1;
       
       
        105
       
       
        /** Column index for link */
       
       
        106
       
       
        private static final int COLUMN_URL_STRING = 2;
       
       
        107
       
       
        /** Column index for published */
       
       
        108
       
       
        private static final int COLUMN_PUBLISHED = 3;
       
       
        109
       
       
       
       
        110
       
       
        /**
       
       
        111
       
       
        * List of Cursor columns to read from when preparing an adapter to populate the ListView.
       
       
        112
       
       
        */
       
       
        113
       
       
        private static final String[] FROM_COLUMNS = new String[]{
       
       
        114
       
       
        FeedContract.Entry.COLUMN_NAME_TITLE,
       
       
        115
       
       
        FeedContract.Entry.COLUMN_NAME_PUBLISHED
       
       
        116
       
       
        };
       
       
        117
       
       
       
       
        118
       
       
        /**
       
       
        119
       
       
        * List of Views which will be populated by Cursor data.
       
       
        120
       
       
        */
       
       
        121
       
       
        private static final int[] TO_FIELDS = new int[]{
       
       
        122
       
       
        android.R.id.text1,
       
       
        123
       
       
        android.R.id.text2};
       
       
        124
       
       
       
       
        125
       
       
        /**
       
       
        126
       
       
        * Mandatory empty constructor for the fragment manager to instantiate the
       
       
        127
       
       
        * fragment (e.g. upon screen orientation changes).
       
       
        128
       
       
        */
       
       
        129
       
       
        public EntryListFragment() {}
       
       
        130
       
       
       
       
        131
       
       
        @Override
       
       
        132
       
       
        public void onCreate(Bundle savedInstanceState) {
       
       
        133
       
       
        super.onCreate(savedInstanceState);
       
       
        134
       
       
        setHasOptionsMenu(true);
       
       
        135
       
       
        }
       
       
        136
       
       
       
       
        137
       
       
        /**
       
       
        138
       
       
        * Create SyncAccount at launch, if needed.
       
       
        139
       
       
        *
       
       
        140
       
       
        * <p>This will create a new account with the system for our application, register our
       
       
        141
       
       
        * {@link SyncService} with it, and establish a sync schedule.
       
       
        142
       
       
        */
       
       
        143
       
       
        @Override
       
       
        144
       
       
        public void onAttach(Activity activity) {
       
       
        145
       
       
        super.onAttach(activity);
       
       
        146
       
       
       
       
        147
       
       
        // Create account, if needed
       
       
        148
       
       
        SyncUtils.CreateSyncAccount(activity);
       
       
        149
       
       
        }
       
       
        150
       
       
       
       
        151
       
       
        @Override
       
       
        152
       
       
        public void onViewCreated(View view, Bundle savedInstanceState) {
       
       
        153
       
       
        super.onViewCreated(view, savedInstanceState);
       
       
        154
       
       
       
       
        155
       
       
        mAdapter = new SimpleCursorAdapter(
       
       
        156
       
       
        getActivity(),       // Current context
       
       
        157
       
       
        android.R.layout.simple_list_item_activated_2,  // Layout for individual rows
       
       
        158
       
       
        null,                // Cursor
       
       
        159
       
       
        FROM_COLUMNS,        // Cursor columns to use
       
       
        160
       
       
        TO_FIELDS,           // Layout fields to use
       
       
        161
       
       
        0                    // No flags
       
       
        162
       
       
        );
       
       
        163
       
       
        mAdapter.setViewBinder(new SimpleCursorAdapter.ViewBinder() {
       
       
        164
       
       
        @Override
       
       
        165
       
       
        public boolean setViewValue(View view, Cursor cursor, int i) {
       
       
        166
       
       
        if (i == COLUMN_PUBLISHED) {
       
       
        167
       
       
        // Convert timestamp to human-readable date
       
       
        168
       
       
        Time t = new Time();
       
       
        169
       
       
        t.set(cursor.getLong(i));
       
       
        170
       
       
        ((TextView) view).setText(t.format("%Y-%m-%d %H:%M"));
       
       
        171
       
       
        return true;
       
       
        172
       
       
        } else {
       
       
        173
       
       
        // Let SimpleCursorAdapter handle other fields automatically
       
       
        174
       
       
        return false;
       
       
        175
       
       
        }
       
       
        176
       
       
        }
       
       
        177
       
       
        });
       
       
        178
       
       
        setListAdapter(mAdapter);
       
       
        179
       
       
        setEmptyText(getText(R.string.loading));
       
       
        180
       
       
        getLoaderManager().initLoader(0, null, this);
       
       
        181
       
       
        }
       
       
        182
       
       
       
       
        183
       
       
        @Override
       
       
        184
       
       
        public void onResume() {
       
       
        185
       
       
        super.onResume();
       
       
        186
       
       
        mSyncStatusObserver.onStatusChanged(0);
       
       
        187
       
       
       
       
        188
       
       
        // Watch for sync state changes
       
       
        189
       
       
        final int mask = ContentResolver.SYNC_OBSERVER_TYPE_PENDING |
       
       
        190
       
       
        ContentResolver.SYNC_OBSERVER_TYPE_ACTIVE;
       
       
        191
       
       
        mSyncObserverHandle = ContentResolver.addStatusChangeListener(mask, mSyncStatusObserver);
       
       
        192
       
       
        }
       
       
        193
       
       
       
       
        194
       
       
        @Override
       
       
        195
       
       
        public void onPause() {
       
       
        196
       
       
        super.onPause();
       
       
        197
       
       
        if (mSyncObserverHandle != null) {
       
       
        198
       
       
        ContentResolver.removeStatusChangeListener(mSyncObserverHandle);
       
       
        199
       
       
        mSyncObserverHandle = null;
       
       
        200
       
       
        }
       
       
        201
       
       
        }
       
       
        202
       
       
       
       
        203
       
       
        /**
       
       
        204
       
       
        * Query the content provider for data.
       
       
        205
       
       
        *
       
       
        206
       
       
        * <p>Loaders do queries in a background thread. They also provide a ContentObserver that is
       
       
        207
       
       
        * triggered when data in the content provider changes. When the sync adapter updates the
       
       
        208
       
       
        * content provider, the ContentObserver responds by resetting the loader and then reloading
       
       
        209
       
       
        * it.
       
       
        210
       
       
        */
       
       
        211
       
       
        @Override
       
       
        212
       
       
        public Loader<Cursor> onCreateLoader(int i, Bundle bundle) {
       
       
        213
       
       
        // We only have one loader, so we can ignore the value of i.
       
       
        214
       
       
        // (It'll be '0', as set in onCreate().)
       
       
        215
       
       
        return new CursorLoader(getActivity(),  // Context
       
       
        216
       
       
        FeedContract.Entry.CONTENT_URI, // URI
       
       
        217
       
       
        PROJECTION,                // Projection
       
       
        218
       
       
        null,                           // Selection
       
       
        219
       
       
        null,                           // Selection args
       
       
        220
       
       
        FeedContract.Entry.COLUMN_NAME_PUBLISHED + " desc"); // Sort
       
       
        221
       
       
        }
       
       
        222
       
       
       
       
        223
       
       
        /**
       
       
        224
       
       
        * Move the Cursor returned by the query into the ListView adapter. This refreshes the existing
       
       
        225
       
       
        * UI with the data in the Cursor.
       
       
        226
       
       
        */
       
       
        227
       
       
        @Override
       
       
        228
       
       
        public void onLoadFinished(Loader<Cursor> cursorLoader, Cursor cursor) {
       
       
        229
       
       
        mAdapter.changeCursor(cursor);
       
       
        230
       
       
        }
       
       
        231
       
       
       
       
        232
       
       
        /**
       
       
        233
       
       
        * Called when the ContentObserver defined for the content provider detects that data has
       
       
        234
       
       
        * changed. The ContentObserver resets the loader, and then re-runs the loader. In the adapter,
       
       
        235
       
       
        * set the Cursor value to null. This removes the reference to the Cursor, allowing it to be
       
       
        236
       
       
        * garbage-collected.
       
       
        237
       
       
        */
       
       
        238
       
       
        @Override
       
       
        239
       
       
        public void onLoaderReset(Loader<Cursor> cursorLoader) {
       
       
        240
       
       
        mAdapter.changeCursor(null);
       
       
        241
       
       
        }
       
       
        242
       
       
       
       
        243
       
       
        /**
       
       
        244
       
       
        * Create the ActionBar.
       
       
        245
       
       
        */
       
       
        246
       
       
        @Override
       
       
        247
       
       
        public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) {
       
       
        248
       
       
        super.onCreateOptionsMenu(menu, inflater);
       
       
        249
       
       
        mOptionsMenu = menu;
       
       
        250
       
       
        inflater.inflate(R.menu.main, menu);
       
       
        251
       
       
        }
       
       
        252
       
       
       
       
        253
       
       
        /**
       
       
        254
       
       
        * Respond to user gestures on the ActionBar.
       
       
        255
       
       
        */
       
       
        256
       
       
        @Override
       
       
        257
       
       
        public boolean onOptionsItemSelected(MenuItem item) {
       
       
        258
       
       
        switch (item.getItemId()) {
       
       
        259
       
       
        // If the user clicks the "Refresh" button.
       
       
        260
       
       
        case R.id.menu_refresh:
       
       
        261
       
       
        SyncUtils.TriggerRefresh();
       
       
        262
       
       
        return true;
       
       
        263
       
       
        }
       
       
        264
       
       
        return super.onOptionsItemSelected(item);
       
       
        265
       
       
        }
       
       
        266
       
       
       
       
        267
       
       
        /**
       
       
        268
       
       
        * Load an article in the default browser when selected by the user.
       
       
        269
       
       
        */
       
       
        270
       
       
        @Override
       
       
        271
       
       
        public void onListItemClick(ListView listView, View view, int position, long id) {
       
       
        272
       
       
        super.onListItemClick(listView, view, position, id);
       
       
        273
       
       
       
       
        274
       
       
        // Get a URI for the selected item, then start an Activity that displays the URI. Any
       
       
        275
       
       
        // Activity that filters for ACTION_VIEW and a URI can accept this. In most cases, this will
       
       
        276
       
       
        // be a browser.
       
       
        277
       
       
       
       
        278
       
       
        // Get the item at the selected position, in the form of a Cursor.
       
       
        279
       
       
        Cursor c = (Cursor) mAdapter.getItem(position);
       
       
        280
       
       
        // Get the link to the article represented by the item.
       
       
        281
       
       
        String articleUrlString = c.getString(COLUMN_URL_STRING);
       
       
        282
       
       
        if (articleUrlString == null) {
       
       
        283
       
       
        Log.e(TAG, "Attempt to launch entry with null link");
       
       
        284
       
       
        return;
       
       
        285
       
       
        }
       
       
        286
       
       
       
       
        287
       
       
        Log.i(TAG, "Opening URL: " + articleUrlString);
       
       
        288
       
       
        // Get a Uri object for the URL string
       
       
        289
       
       
        Uri articleURL = Uri.parse(articleUrlString);
       
       
        290
       
       
        Intent i = new Intent(Intent.ACTION_VIEW, articleURL);
       
       
        291
       
       
        startActivity(i);
       
       
        292
       
       
        }
       
       
        293
       
       
       
       
        294
       
       
        /**
       
       
        295
       
       
        * Set the state of the Refresh button. If a sync is active, turn on the ProgressBar widget.
       
       
        296
       
       
        * Otherwise, turn it off.
       
       
        297
       
       
        *
       
       
        298
       
       
        * @param refreshing True if an active sync is occuring, false otherwise
       
       
        299
       
       
        */
       
       
        300
       
       
        @TargetApi(Build.VERSION_CODES.HONEYCOMB)
       
       
        301
       
       
        public void setRefreshActionButtonState(boolean refreshing) {
       
       
        302
       
       
        if (mOptionsMenu == null || Build.VERSION.SDK_INT < Build.VERSION_CODES.HONEYCOMB) {
       
       
        303
       
       
        return;
       
       
        304
       
       
        }
       
       
        305
       
       
       
       
        306
       
       
        final MenuItem refreshItem = mOptionsMenu.findItem(R.id.menu_refresh);
       
       
        307
       
       
        if (refreshItem != null) {
       
       
        308
       
       
        if (refreshing) {
       
       
        309
       
       
        refreshItem.setActionView(R.layout.actionbar_indeterminate_progress);
       
       
        310
       
       
        } else {
       
       
        311
       
       
        refreshItem.setActionView(null);
       
       
        312
       
       
        }
       
       
        313
       
       
        }
       
       
        314
       
       
        }
       
       
        315
       
       
       
       
        316
       
       
        /**
       
       
        317
       
       
        * Crfate a new anonymous SyncStatusObserver. It's attached to the app's ContentResolver in
       
       
        318
       
       
        * onResume(), and removed in onPause(). If status changes, it sets the state of the Refresh
       
       
        319
       
       
        * button. If a sync is active or pending, the Refresh button is replaced by an indeterminate
       
       
        320
       
       
        * ProgressBar; otherwise, the button itself is displayed.
       
       
        321
       
       
        */
       
       
        322
       
       
        private SyncStatusObserver mSyncStatusObserver = new SyncStatusObserver() {
       
       
        323
       
       
        /** Callback invoked with the sync adapter status changes. */
       
       
        324
       
       
        @Override
       
       
        325
       
       
        public void onStatusChanged(int which) {
       
       
        326
       
       
        getActivity().runOnUiThread(new Runnable() {
       
       
        327
       
       
        /**
       
       
        328
       
       
        * The SyncAdapter runs on a background thread. To update the UI, onStatusChanged()
       
       
        329
       
       
        * runs on the UI thread.
       
       
        330
       
       
        */
       
       
        331
       
       
        @Override
       
       
        332
       
       
        public void run() {
       
       
        333
       
       
        // Create a handle to the account that was created by
       
       
        334
       
       
        // SyncService.CreateSyncAccount(). This will be used to query the system to
       
       
        335
       
       
        // see how the sync status has changed.
       
       
        336
       
       
        Account account = GenericAccountService.GetAccount(SyncUtils.ACCOUNT_TYPE);
       
       
        337
       
       
        if (account == null) {
       
       
        338
       
       
        // GetAccount() returned an invalid value. This shouldn't happen, but
       
       
        339
       
       
        // we'll set the status to "not refreshing".
       
       
        340
       
       
        setRefreshActionButtonState(false);
       
       
        341
       
       
        return;
       
       
        342
       
       
        }
       
       
        343
       
       
       
       
        344
       
       
        // Test the ContentResolver to see if the sync adapter is active or pending.
       
       
        345
       
       
        // Set the state of the refresh button accordingly.
       
       
        346
       
       
        boolean syncActive = ContentResolver.isSyncActive(
       
       
        347
       
       
        account, FeedContract.CONTENT_AUTHORITY);
       
       
        348
       
       
        boolean syncPending = ContentResolver.isSyncPending(
       
       
        349
       
       
        account, FeedContract.CONTENT_AUTHORITY);
       
       
        350
       
       
        setRefreshActionButtonState(syncActive || syncPending);
       
       
        351
       
       
        }
       
       
        352
       
       
        });
       
       
        353
       
       
        }
       
       
        354
       
       
        };
       
       
        355
       
       
       
       
        356
       
       
        }