Android - Ajout et éléments à ListView montrant un à la fois en utilisant un ArrayAdapter
-
22-09-2019 - |
Question
J'utilise un ArrayAdapter pour ajouter des éléments à un ListView personnalisé et montrant les résultats dans mon application Android. Le problème que je vais avoir est que le ArrayAdapter semble attendre jusqu'à ce que tous les articles sont en avant qu'il montre la vue. C'est-à-dire, lorsque j'ajoute les articles à la ArrayAdapter et j'appeler notifyDataSetChanged, il ne met pas à jour le ListView pour afficher l'élément ajouté. Il attend que tous les éléments sont ajoutés et GetView est appelé avant de montrer les éléments.
Ce que je voudrais à faire est de montrer l'élément immédiatement après l'avoir ajouté à la ListView. Est-ce possible?
Je crois que le code correspondant est le suivant:
r_adapter = new ReminderAdapater(Activity_ContentSearch.this, R.layout.search_listitem, myList);
listView.setAdapter(r_adapter);
...
r_adapter.notifyDataSetChanged();
r_adapter.clear();
for(int i = 0; i < myList.size(); i++)
{
r_adapter.add(myList.get(i));
r_adapter.notifyDataSetChanged();
}
Comme vous pouvez le voir, même si je fais appel notifyDataSetChanged après la méthode d'ajout, il ne met pas à jour en fait le point de vue. Une fois qu'il a terminé la boucle au-dessus de la vue est finalement mis à jour (basé sur ce que je sais, c'est parce que GetView n'est pas appelé jusqu'à ce que cette partie du code est fait).
J'ai essayé de remplacer la méthode d'ajout de mon ArrayAdapter personnalisé avec pas de chance, puisque je n'ai pas accès à la vue dans cette méthode.
Toute aide serait la bienvenue:)
Bara
La solution
interface utilisateur Android est mono-thread. Vous ne donnez pas le contrôle à Android de l'application thread principal chaque fois que vous ajoutez une entrée à l'adaptateur. Par conséquent, Android ne reçoit pas une chance d'afficher vos entrées jusqu'à ce que vous revenez le contrôle, et vous n'êtes pas le faire jusqu'à ce que vous avez rempli votre carte dans son intégralité.
Voici un exemple montrant l'utilisation d'un AsyncTask
pour remplir un ArrayAdapter
progressivement par l'intermédiaire d'un fil de fond.
/***
Copyright (c) 2008-2012 CommonsWare, LLC
Licensed under the Apache License, Version 2.0 (the "License"); you may not
use this file except in compliance with the License. You may obtain a copy
of the License at http://www.apache.org/licenses/LICENSE-2.0. Unless required
by applicable law or agreed to in writing, software distributed under the
License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS
OF ANY KIND, either express or implied. See the License for the specific
language governing permissions and limitations under the License.
From _The Busy Coder's Guide to Android Development_
http://commonsware.com/Android
*/
package com.commonsware.android.async;
import android.app.ListActivity;
import android.os.AsyncTask;
import android.os.Bundle;
import android.os.SystemClock;
import android.widget.ArrayAdapter;
import android.widget.Toast;
import java.util.ArrayList;
public class AsyncDemo extends ListActivity {
private static final String[] items={"lorem", "ipsum", "dolor",
"sit", "amet", "consectetuer",
"adipiscing", "elit", "morbi",
"vel", "ligula", "vitae",
"arcu", "aliquet", "mollis",
"etiam", "vel", "erat",
"placerat", "ante",
"porttitor", "sodales",
"pellentesque", "augue",
"purus"};
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
setListAdapter(new ArrayAdapter<String>(this,
android.R.layout.simple_list_item_1,
new ArrayList<String>()));
new AddStringTask().execute();
}
class AddStringTask extends AsyncTask<Void, String, Void> {
@Override
protected Void doInBackground(Void... unused) {
for (String item : items) {
publishProgress(item);
SystemClock.sleep(200);
}
return(null);
}
@SuppressWarnings("unchecked")
@Override
protected void onProgressUpdate(String... item) {
((ArrayAdapter<String>)getListAdapter()).add(item[0]);
}
@Override
protected void onPostExecute(Void unused) {
Toast
.makeText(AsyncDemo.this, "Done!", Toast.LENGTH_SHORT)
.show();
}
}
}