Android CursorLoader не отвечает на уведомления ContentProvider - PullRequest
5 голосов
/ 19 января 2012

Я обновляю приложение Android 2.2 для использования CursorLoader (используя библиотеку совместимости v4) и пытаюсь понять, почему метод onLoadFinished не вызывается, когда поставщик контента уведомляет об изменении в содержимое, связанное с запросом CursorLoader.

CursorLoader запрашивает контент-провайдера клиента. Мой провайдер устанавливает URI уведомления в своем методе запроса:

cursor.setNotificationUri(getContext().getContentResolver(), uri);

и уведомляет об изменениях в методах вставки / обновления / удаления:

getContext().getContentResolver().notifyChange(uri, null);

Я проверил, что URI идентичен в обоих случаях. Ранее я использовал ManagedQuery с тем же поставщиком контента, и запрашиваемый контент был обновлен нормально, что заставляет меня думать, что поставщик контента, вероятно, в порядке.

Я рассмотрел пример LoaderCursorSupport и, что интересно, когда я запускаю его на своем Nexus One, я не вижу, чтобы он отражал изменения, которые я вносил в имена контактов (переключение между примером приложения и приложение контактов). Должно ли это? Если да, есть ли какая-то основная проблема, о которой я не знаю?

Ответы [ 2 ]

4 голосов
/ 19 января 2012

Я наконец дошел до сути, и, как обычно, это была глупая ошибка с моей стороны. Я вызывал cursor.close () в моем методе onLoadFinished () - я использую возвращенный курсор для создания ArrayAdapter (мне нужно вручную вставить элемент в верхней части списка), и закрытие курсора осталось после использования ManagedQuery перед миграцией использовать CursorLoader.

В процессе нахождения этого я создал простой тестовый класс для отображения списка закладок и добавления случайной закладки (используя меню параметров). Это сработало так, как должно, т. Е. OnLoadFinished () вызывается после добавления элемента. Вот код на случай, если он пригодится кому-либо еще:

package com.test;

import android.content.ContentValues;
import android.database.Cursor;
import android.os.Bundle;
import android.provider.Browser;
import android.support.v4.app.FragmentActivity;
import android.support.v4.app.FragmentManager;
import android.support.v4.app.ListFragment;
import android.support.v4.app.LoaderManager;
import android.support.v4.content.CursorLoader;
import android.support.v4.content.Loader;
import android.support.v4.view.MenuItemCompat;
import android.support.v4.widget.SimpleCursorAdapter;
import android.util.Log;
import android.view.Menu;
import android.view.MenuInflater;
import android.view.MenuItem;

public class CursorLoaderTestActivity extends FragmentActivity 
{
    private static final String TAG = CursorLoaderTestActivity.class.getSimpleName();

    @Override
    protected void onCreate(Bundle savedInstanceState) 
    {
        super.onCreate(savedInstanceState);

        FragmentManager fm = getSupportFragmentManager();

        // Create the list fragment and add it as our sole content.
        if (fm.findFragmentById(android.R.id.content) == null) 
        {
            CursorLoaderListFragment list = new CursorLoaderListFragment();
            fm.beginTransaction().add(android.R.id.content, list).commit();
        }
    }


    public static class CursorLoaderListFragment extends ListFragment implements LoaderManager.LoaderCallbacks<Cursor> 
    {

        // This is the Adapter being used to display the list's data.
        SimpleCursorAdapter mAdapter;

        // If non-null, this is the current filter the user has provided.
        String mCurFilter;

        @Override public void onActivityCreated(Bundle savedInstanceState) 
        {
            super.onActivityCreated(savedInstanceState);

            // Give some text to display if there is no data.  In a real
            // application this would come from a resource.
            setEmptyText("No data");

            // We have a menu item to show in action bar.
            setHasOptionsMenu(true);

            // Create an empty adapter we will use to display the loaded data.
            mAdapter = new SimpleCursorAdapter(getActivity(),
                    android.R.layout.simple_list_item_1, null,
                    new String[] { Browser.BookmarkColumns.TITLE },
                    new int[] { android.R.id.text1}, 0);

            setListAdapter(mAdapter);

            // Start out with a progress indicator.
            setListShown(false);

            // Prepare the loader.  Either re-connect with an existing one,
            // or start a new one.
            getLoaderManager().initLoader(0, null, this);
        }

        //@Override 
        public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) 
        {
            // Place an action bar item for searching.
            MenuItem item = menu.add("Add Item");
            //item.setIcon(android.R.drawable.ic_menu_search);
            MenuItemCompat.setShowAsAction(item, MenuItemCompat.SHOW_AS_ACTION_ALWAYS);
        }

        @Override
        public boolean onOptionsItemSelected (MenuItem item)
        {
            ContentValues cv=new ContentValues();
            cv.put(Browser.BookmarkColumns.TITLE, "!AA " + System.currentTimeMillis());
            cv.put(Browser.BookmarkColumns.URL, "http://test/");
            cv.put(Browser.BookmarkColumns.BOOKMARK, 1);
            getActivity().getContentResolver().insert(Browser.BOOKMARKS_URI, cv);
            return true;
        }

       //columns to query
        static final String[] PROJECTION = new String[] { Browser.BookmarkColumns.TITLE };


        public Loader<Cursor> onCreateLoader(int id, Bundle args) 
        {
            Log.i(TAG, "onCreateLoader");

            return new CursorLoader(getActivity(), Browser.BOOKMARKS_URI,
                    PROJECTION, null, null,
                    Browser.BookmarkColumns.TITLE + " ASC");
        }

        public void onLoadFinished(Loader<Cursor> loader, Cursor data) 
        {
            Log.i(TAG, "onLoadFinished");

            // Swap the new cursor in.  (The framework will take care of closing the
            // old cursor once we return.)
            mAdapter.swapCursor(data);

            // The list should now be shown.
            if (isResumed()) 
                setListShown(true);
            else 
                setListShownNoAnimation(true);
        }

        public void onLoaderReset(Loader<Cursor> loader) 
        {
            // This is called when the last Cursor provided to onLoadFinished()
            // above is about to be closed.  We need to make sure we are no
            // longer using it.
            mAdapter.swapCursor(null);
        }
    }

}
0 голосов
/ 19 января 2012

Может быть, вы должны использовать контекст приложения:

getContext().getApplicationContext().getContentResolver().notifyChange(uri, null);
...