Android ListFragment данные не обновляются через SimpleCursorAdapter/LoaderManager/notifyChange()

У меня есть ListFragment, который показывает имена всех списков покупок, хранящихся в моей таблице базы данных.

Проблема заключается в том, что когда я добавляю новую таблицу списка покупок в таблицу, ListFragment в пользовательском интерфейсе не обновляется автоматически. (Изменение можно увидеть только в том случае, если я закрою и перезагрузите приложение.)

Во-первых, это код, который выполняется в моем классе DbContentProvider когда я добавляю список покупок:

'

// Insert the values into the table
 id = db.insert(SHOPPING_LISTS_META_TABLE, null, values);

 if (id > -1) {
 // Construct and return the URI of the newly inserted row.
 Uri insertedId = ContentUris.withAppendedId(CONTENT_URI_SHOPPING_LISTS_META, id);

 // Notify any observers of the change in the data set.
 Log.d(LOG_TAG, "................. notifyChange(\"" + insertedId + "\", null)");
 getContext().getContentResolver().notifyChange(insertedId, null);
 Log.d(LOG_TAG, "................. notifyChange() done");
 return insertedId;
 }
 else {
 return null;
 }

'

... и вот для него выход LogCat...

10-28 12:29:41.133: D/SQLiteOpenHelper(19401):................. notifyChange("content://org.example.myapp.DbContentProvider/shopping_lists_meta/12", null) 10-28 12:29:41.143: D/SQLiteOpenHelper(19401):................. notifyChange() done 10-28 12:29:41.153: D/HomeActivity(19401): Shopping list, "My Test Shopping List" created: content://org.example.myapp.DbContentProvider/shopping_lists_meta/12 10-28 12:29:41.183: D/AbsListView(19401): unregisterIRListener() is called 10-28 12:29:41.193: E/ViewRootImpl(19401): sendUserActionEvent() mView == null 10-28 12:29:41.503: D/AbsListView(19401): unregisterIRListener() is called

В LogCat нет никакого вывода из моего класса ListFragment, когда я добавляю новую строку списка покупок. Вот мой класс ListFragment...

'

public class ShoppingListNamesListFragment extends ListFragment реализует LoaderManager.LoaderCallbacks {

private final static String LOG_TAG = ShoppingListNamesListFragment.class.getSimpleName(); 

// This is the Adapter being used to display the list data
public static SimpleCursorAdapter mAdapter;

// These are the Contacts rows that we will retrieve
static final String[] PROJECTION = {DbContentProvider.KEY_ID,
 DbContentProvider.KEY_SHOPPING_LIST_NAME,
 DbContentProvider.KEY_IS_SHOPPING_LIST_SELECTED};

// This is the select criteria
static final String SELECTION = null;

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

 Log.d(LOG_TAG, "................. onCreate()");

}

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

 Log.d(LOG_TAG, "................. onActivityCreated()");

 // For the cursor adapter, specify which columns go into which views
 String[] fromColumns = {DbContentProvider.KEY_SHOPPING_LIST_NAME};
 int[] toViews = {android.R.id.text1}; // The TextView in simple_list_item_1

 // Create an empty adapter we will use to display the loaded data.
 // We pass null for the cursor, then update it in onLoadFinished()
 mAdapter = new SimpleCursorAdapter(this.getActivity(), 
 android.R.layout.simple_list_item_1, null,
 fromColumns, toViews, 0);
 setListAdapter(mAdapter);

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

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
 Log.d(LOG_TAG, "................. onCreateView()");
 return super.onCreateView(inflater, container, savedInstanceState);
}

// Called when a new Loader needs to be created
@Override
public Loader<cursor> onCreateLoader(int id, Bundle args) {
 Log.d(LOG_TAG, "................. onCreateLoader()");
 // Now create and return a CursorLoader that will take care of
 // creating a Cursor for the data being displayed.
 return new CursorLoader(getActivity(), DbContentProvider.CONTENT_URI_SHOPPING_LISTS_META,
 PROJECTION, SELECTION, null, null);

}

// Called when a previously created loader has finished loading
@Override
public void onLoadFinished(Loader<cursor> loader, Cursor data) {
 Log.d(LOG_TAG, "................. onLoaderFinished()");
 // Swap the new cursor in. (The framework will take care of closing the
 // old cursor once we return.)
 mAdapter.swapCursor(data); 
}

// Called when a previously created loader is reset, making the data unavailable
@Override
public void onLoaderReset(Loader<cursor> loader) {
 Log.d(LOG_TAG, "................. onLoaderReset()");
 // 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); 
}

@Override 
public void onListItemClick(ListView l, View v, int position, long id) {
 makeToast("shopping list clicked: " + position);
 Log.d(LOG_TAG, "shopping list clicked: " + position);
}

private void makeToast(String msg) {
 Toast.makeText(getActivity(), msg, Toast.LENGTH_SHORT).show();
}
</cursor></cursor></cursor>

}

NB - В моем классе DbContentProvider меня есть...

public static final Uri CONTENT_URI_SHOPPING_LISTS_META = Uri.parse("content://org.example.myapp.DbContentProvider/shopping_lists_meta"); public static final String SHOPPING_LISTS_META_TABLE = "shopping_lists_meta";

Я основал свой код на этом примере Android ListFragment/LoaderManager. И ни один из подобных вопросов к этому, который я нашел, не предлагает решение, которое устраняет мою проблему.

Я новичок в Android, так что это может быть простая ошибка, которую я сделал. Но, по сути, мне кажется, что когда notifyChange() вызывается в моем классе DbContentProvider, мой ListFragment не уведомляется (или есть какая-то другая ошибка в этой области). Может ли кто-нибудь помочь?

1 ответ

Потратив часы и часы на это, я создал TestListActivity, который подключился к собственному поставщику контента для контактов - и все это работало/обновлялось, так как я знал, что проблема, вероятно, была в моем собственном провайдере контента, который я написал.

Я нашел ответ здесь. Оказывается, я не вызывал setNotificationUri(ContentResolver cr, Uri uri) в cursor возвращаемом методом query() моего поставщика контента. (Я уверен, что это никогда не упоминалось в книге Рето Мауэра, с которой я работал...):/

В любом случае, все отсортировано прямо сейчас! :)

licensed under cc by-sa 3.0 with attribution.