2

У меня есть ListFragment, который показывает имена всех списков покупок, хранящихся в моей таблице базы данных.Android ListFragment данные не обновляются через SimpleCursorAdapter/LoaderManager/notifyChange()

Проблема заключается в том, что когда я добавляю новую таблицу списка покупок в таблицу, 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 ...

`

общественный класс ShoppingListNamesListFragment расширяет ListFragment реализует LoaderManager.LoaderCallbacks {

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

// This is the Adapter being used to display the list's 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(); 
} 

}

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";

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

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

ответ

1

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

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

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