Bagaimana cara me-refresh Android ListView
setelah menambahkan / menghapus data dinamis?
Bagaimana cara me-refresh Android ListView
setelah menambahkan / menghapus data dinamis?
Jawaban:
Panggil notifyDataSetChanged()
pada Anda Adapter
objek setelah Anda memodifikasi data dalam adaptor yang.
Beberapa spesifikasi tambahan tentang bagaimana / kapan menelepon notifyDataSetChanged()
dapat dilihat di video Google I / O ini .
Anda juga dapat menggunakan ini:
myListView.invalidateViews();
invalidateViews()
dalam set kode sumber mDataChanged = true;
jadi saya tidak yakin apakah ini menghasilkan kinerja yang lebih baik dibandingkan dengannotifyDataSetChanged()
invalidate()
, invalidateViews()
, requestLayout()
, ... jawaban untuk pertanyaan ini.Hal yang benar untuk dilakukan (dan untungnya juga ditandai sebagai jawaban yang benar) adalah memanggil notifyDataSetChanged()
Adaptor Anda .
Jika panggilan notifyDataSetChanged()
tidak berfungsi, semua metode tata letak tidak akan membantu. Percayalah ListView
itu telah diperbarui dengan benar. Jika Anda gagal menemukan perbedaannya, Anda perlu memeriksa dari mana data di adaptor Anda berasal.
Jika ini hanya koleksi yang Anda simpan di memori, periksa apakah Anda benar-benar menghapus atau menambahkan item ke koleksi sebelum memanggil notifyDataSetChanged()
.
Jika Anda bekerja dengan database atau layanan backend Anda harus memanggil metode untuk mengambil kembali informasi (atau memanipulasi data dalam memori) sebelum memanggil notifyDataSetChanged()
.
Masalahnya adalah ini notifyDataSetChanged
hanya berfungsi jika dataset telah berubah. Jadi itulah tempat untuk mencari jika Anda tidak menemukan perubahan yang terjadi. Debug jika dibutuhkan.
Saya menemukan bahwa bekerja dengan adaptor yang memungkinkan Anda mengelola koleksi, seperti BaseAdapter berfungsi lebih baik. Beberapa adaptor seperti ArrayAdapter sudah mengelola koleksi mereka sendiri sehingga menyulitkan untuk mendapatkan koleksi yang tepat untuk pembaruan. Ini sebenarnya hanya lapisan kesulitan tambahan yang tidak perlu dalam kebanyakan kasus.
Benar bahwa ini harus dipanggil dari utas UI. Jawaban lain memiliki contoh tentang cara mencapai ini. Namun ini hanya diperlukan jika Anda mengerjakan informasi ini dari luar utas UI. Itu dari layanan atau utas non UI. Dalam kasus sederhana, Anda akan memperbarui data dari klik tombol atau aktivitas / fragmen lain. Jadi masih dalam utas UI. Tidak perlu selalu memunculkan runOnUiTrhead itu.
Dapat ditemukan di https://github.com/hanscappelle/so-2250770.git . Cukup klon dan buka proyek di Android Studio (gradle). Proyek ini memiliki MainAcitivity yang membangun ListView
dengan semua data acak. Daftar ini dapat di-refresh menggunakan menu aksi.
Implementasi adaptor yang saya buat untuk contoh ini ModelObject memperlihatkan pengumpulan data
public class MyListAdapter extends BaseAdapter {
/**
* this is our own collection of data, can be anything we
* want it to be as long as we get the abstract methods
* implemented using this data and work on this data
* (see getter) you should be fine
*/
private List<ModelObject> mData;
/**
* our ctor for this adapter, we'll accept all the things
* we need here
*
* @param mData
*/
public MyListAdapter(final Context context, final List<ModelObject> mData) {
this.mData = mData;
this.mContext = context;
}
public List<ModelObject> getData() {
return mData;
}
// implement all abstract methods here
}
Kode dari MainActivity
public class MainActivity extends Activity {
private MyListAdapter mAdapter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
ListView list = (ListView) findViewById(R.id.list);
// create some dummy data here
List<ModelObject> objects = getRandomData();
// and put it into an adapter for the list
mAdapter = new MyListAdapter(this, objects);
list.setAdapter(mAdapter);
// mAdapter is available in the helper methods below and the
// data will be updated based on action menu interactions
// you could also keep the reference to the android ListView
// object instead and use the {@link ListView#getAdapter()}
// method instead. However you would have to cast that adapter
// to your own instance every time
}
/**
* helper to show what happens when all data is new
*/
private void reloadAllData(){
// get new modified random data
List<ModelObject> objects = getRandomData();
// update data in our adapter
mAdapter.getData().clear();
mAdapter.getData().addAll(objects);
// fire the event
mAdapter.notifyDataSetChanged();
}
/**
* helper to show how only changing properties of data
* elements also works
*/
private void scrambleChecked(){
Random random = new Random();
// update data in our adapter, iterate all objects and
// resetting the checked option
for( ModelObject mo : mAdapter.getData()) {
mo.setChecked(random.nextBoolean());
}
// fire the event
mAdapter.notifyDataSetChanged();
}
}
Posting bagus lainnya tentang kekuatan listViews dapat ditemukan di sini: http://www.vogella.com/articles/AndroidListView/article.html
adapter.clear()
, dan adapter.addAll(Array<T>)
sebelum meneleponnotifyDataSetChanged()
Panggil runnable kapan pun Anda mau:
runOnUiThread(run);
OnCreate()
, Anda menetapkan utas runnable Anda:
run = new Runnable() {
public void run() {
//reload content
arraylist.clear();
arraylist.addAll(db.readAll());
adapter.notifyDataSetChanged();
listview.invalidateViews();
listview.refreshDrawableState();
}
};
runOnUiThread()
fungsi.
run = new Runnable()
dan hanya memberi saya fungsi public void refreshUIThread()
metode yang memodelkan run()
metode Anda (yang berfungsi juga).
saya mendapat beberapa masalah dengan penyegaran dinamis dari tampilan daftar saya.
Panggil notifyDataSetChanged () pada Adaptor Anda.
Beberapa spesifik tambahan tentang bagaimana / kapan menelepon notifyDataSetChanged () dapat dilihat di video Google I / O ini.
notifyDataSetChanged () tidak berfungsi dengan baik dalam kasus saya [saya memanggil notifyDataSetChanged dari kelas lain]. Hanya dalam kasus saya mengedit ListView di Activity (Thread) yang sedang berjalan. Video itu berkat Christopher memberi petunjuk terakhir.
Di kelas kedua saya digunakan
Runnable run = new Runnable(){
public void run(){
contactsActivity.update();
}
};
contactsActivity.runOnUiThread(run);
untuk mengakses pembaruan () dari Aktivitas saya. Pembaruan ini termasuk
myAdapter.notifyDataSetChanged();
untuk memberitahu Adaptor untuk menyegarkan tampilan. Bekerja dengan baik sejauh yang bisa saya katakan.
Jika Anda menggunakan SimpleCursorAdapter coba panggil requery () pada objek Cursor.
jika Anda masih tidak puas dengan Refreshment ListView, Anda dapat melihat potongan ini, ini untuk memuat listView dari DB, Sebenarnya yang harus Anda lakukan hanyalah memuat ulang ListView, setelah Anda melakukan Operasi CRUD. Ini bukan cara terbaik untuk kode, tetapi akan menyegarkan ListView seperti yang Anda inginkan ..
Ini bekerja untuk Saya .... jika Anda menemukan solusi yang lebih baik, silakan Bagikan ...
....... ...... lakukan Operasi CRUD Anda .. ...... ..... DBAdapter.open (); DBAdapter.insert_into_SingleList (); // Bawa DB_result itu dan tambahkan ke daftar sebagai isinya .... ls2.setAdapter (ArrayAdapter baru (DynTABSample.this, android.R.layout.simple_list_item_1, DBAdapter.DB_ListView)); DBAdapter.close ();
Solusi yang diajukan oleh orang-orang di pos ini berfungsi atau tidak terutama tergantung pada versi Android perangkat Anda. Sebagai Contoh untuk menggunakan metode AddAll Anda harus meletakkan android: minSdkVersion = "10" di perangkat android Anda.
Untuk mengatasi pertanyaan ini untuk semua perangkat yang saya buat metode saya sendiri di adaptor saya dan gunakan di dalam metode add and remove yang diwarisi dari ArrayAdapter yang memperbarui data Anda tanpa masalah.
Kode Saya: Menggunakan kelas data saya sendiri RaceResult, Anda menggunakan model data Anda sendiri.
ResultGpRowAdapter.java
public class ResultGpRowAdapter extends ArrayAdapter<RaceResult> {
Context context;
int resource;
List<RaceResult> data=null;
public ResultGpRowAdapter(Context context, int resource, List<RaceResult> objects) {
super(context, resource, objects);
this.context = context;
this.resource = resource;
this.data = objects;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
........
}
//my own method to populate data
public void myAddAll(List<RaceResult> items) {
for (RaceResult item:items){
super.add(item);
}
}
ResultsGp.java
public class ResultsGp extends Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
...........
...........
ListView list = (ListView)findViewById(R.id.resultsGpList);
ResultGpRowAdapter adapter = new ResultGpRowAdapter(this, R.layout.activity_result_gp_row, new ArrayList<RaceResult>()); //Empty data
list.setAdapter(adapter);
....
....
....
//LOAD a ArrayList<RaceResult> with data
ArrayList<RaceResult> data = new ArrayList<RaceResult>();
data.add(new RaceResult(....));
data.add(new RaceResult(....));
.......
adapter.myAddAll(data); //Your list will be udpdated!!!
Jika Anda ingin mempertahankan posisi gulir saat menyegarkan, dan Anda dapat melakukan ini:
if (mEventListView.getAdapter() == null) {
EventLogAdapter eventLogAdapter = new EventLogAdapter(mContext, events);
mEventListView.setAdapter(eventLogAdapter);
} else {
((EventLogAdapter)mEventListView.getAdapter()).refill(events);
}
public void refill(List<EventLog> events) {
mEvents.clear();
mEvents.addAll(events);
notifyDataSetChanged();
}
Untuk informasi detail, silakan lihat Android ListView: Pertahankan posisi gulir Anda saat menyegarkan .
Cukup gunakan myArrayList.remove(position);
di dalam pendengar:
myListView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override public void onItemClick(AdapterView<?> parent, android.view.View view, int position, long id) {
myArrayList.remove(position);
myArrayAdapter.notifyDataSetChanged();
}
});
Anda perlu menggunakan satu objek dari daftar itu yang mengumpulkan data yang Anda kembangkan ListView
. Jika referensi berubah maka tidak notifyDataSetChanged()
berfungsi. Kapan pun Anda menghapus elemen dari tampilan daftar, hapus juga dari daftar yang Anda gunakan apakah itu ArrayList <> atau Sesuatu yang lain maka Panggil
notifyDataSetChanged()
objek kelas adaptor Anda.
Jadi di sini lihat bagaimana saya mengaturnya di adaptor saya lihat di bawah
public class CountryCodeListAdapter extends BaseAdapter implements OnItemClickListener{
private Context context;
private ArrayList<CountryDataObject> dObj;
private ViewHolder holder;
private Typeface itemFont;
private int selectedPosition=-1;
private ArrayList<CountryDataObject> completeList;
public CountryCodeListAdapter(Context context, ArrayList<CountryDataObject> dObj) {
this.context = context;
this.dObj=dObj;
completeList=new ArrayList<CountryDataObject>();
completeList.addAll(dObj);
itemFont=Typeface.createFromAsset(context.getAssets(), "CaviarDreams.ttf");
}
@Override
public int getCount() {
return dObj.size();
}
@Override
public Object getItem(int position) {
return dObj.get(position);
}
@Override
public long getItemId(int position) {
return position;
}
@Override
public View getView(int position, View view, ViewGroup parent) {
if(view==null){
holder = new ViewHolder();
LayoutInflater inflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
view = inflater.inflate(R.layout.states_inflator_layout, null);
holder.textView = ((TextView)view.findViewById(R.id.stateNameInflator));
holder.checkImg=(ImageView)view.findViewById(R.id.checkBoxState);
view.setTag(holder);
}else{
holder = (ViewHolder) view.getTag();
}
holder.textView.setText(dObj.get(position).getCountryName());
holder.textView.setTypeface(itemFont);
if(position==selectedPosition)
{
holder.checkImg.setImageResource(R.drawable.check);
}
else
{
holder.checkImg.setImageResource(R.drawable.uncheck);
}
return view;
}
private class ViewHolder{
private TextView textView;
private ImageView checkImg;
}
public void getFilter(String name) {
dObj.clear();
if(!name.equals("")){
for (CountryDataObject item : completeList) {
if(item.getCountryName().toLowerCase().startsWith(name.toLowerCase(),0)){
dObj.add(item);
}
}
}
else {
dObj.addAll(completeList);
}
selectedPosition=-1;
notifyDataSetChanged();
notifyDataSetInvalidated();
}
@Override
public void onItemClick(AdapterView<?> parent, View view, int position,
long id) {
Registration reg=(Registration)context;
selectedPosition=position;
reg.setSelectedCountryCode("+"+dObj.get(position).getCountryCode());
notifyDataSetChanged();
}
}
Setelah menghapus data dari tampilan daftar, Anda harus menelepon refreshDrawableState()
. Berikut ini contohnya:
final DatabaseHelper db = new DatabaseHelper (ActivityName.this);
db.open();
db.deleteContact(arg3);
mListView.refreshDrawableState();
db.close();
dan deleteContact
metode di DatabaseHelper
kelas akan agak mirip
public boolean deleteContact(long rowId) {
return db.delete(TABLE_NAME, BaseColumns._ID + "=" + rowId, null) > 0;
}
Saya tidak bisa mendapatkan notifyDataSetChanged () untuk bekerja memperbarui SimpleAdapter saya, jadi alih-alih saya mencoba menghapus semua tampilan yang dilampirkan ke tata letak induk menggunakan removeAllViews (), kemudian menambahkan ListView, dan yang berfungsi, memungkinkan saya untuk memperbarui UI:
LinearLayout results = (LinearLayout)findViewById(R.id.results);
ListView lv = new ListView(this);
ArrayList<HashMap<String,String>> list = new ArrayList<HashMap<String,String>>();
SimpleAdapter adapter = new SimpleAdapter( this, list, R.layout.directory_row,
new String[] { "name", "dept" }, new int[] { R.id.name, R.id.dept } );
for (...) {
HashMap<String, String> map = new HashMap<String, String>();
map.put("name", name);
map.put("dept", dept);
list.add(map);
}
lv.setAdapter(adapter);
results.removeAllViews();
results.addView(lv);
Bagi saya setelah mengubah informasi dalam basis data sql tidak ada yang bisa menyegarkan tampilan daftar (menjadi tampilan daftar yang dapat diperluas) jadi jika notifyDataSetChanged () tidak membantu, Anda dapat mencoba untuk menghapus daftar Anda terlebih dahulu dan menambahkannya lagi setelah panggilan notifyDataSetChanged (). Sebagai contoh
private List<List<SomeNewArray>> arrayList;
List<SomeNewArray> array1= getArrayList(...);
List<SomeNewArray> array2= getArrayList(...);
arrayList.clear();
arrayList.add(array1);
arrayList.add(array2);
notifyDataSetChanged();
Semoga ini masuk akal untuk Anda.
saat menggunakan SimpleCursorAdapter dapat memanggil changeCursor (newCursor) pada adaptor.
Saya sama ketika, dalam sebuah fragmen, saya ingin mengisi ListView (dalam satu TextView) dengan alamat mac perangkat BLE yang dipindai selama beberapa waktu.
Apa yang saya lakukan adalah ini:
public class Fragment01 extends android.support.v4.app.Fragment implements ...
{
private ListView listView;
private ArrayAdapter<String> arrayAdapter_string;
...
@Override
public void onActivityCreated(Bundle savedInstanceState)
{
...
this.listView= (ListView) super.getActivity().findViewById(R.id.fragment01_listView);
...
this.arrayAdapter_string= new ArrayAdapter<String>(super.getActivity(), R.layout.dispositivo_ble_item, R.id.fragment01_item_textView_titulo);
this.listView.setAdapter(this.arrayAdapter_string);
}
@Override
public void onLeScan(BluetoothDevice device, int rssi, byte[] scanRecord)
{
...
super.getActivity().runOnUiThread(new RefreshListView(device));
}
private class RefreshListView implements Runnable
{
private BluetoothDevice bluetoothDevice;
public RefreshListView(BluetoothDevice bluetoothDevice)
{
this.bluetoothDevice= bluetoothDevice;
}
@Override
public void run()
{
Fragment01.this.arrayAdapter_string.add(new String(bluetoothDevice.toString()));
Fragment01.this.arrayAdapter_string.notifyDataSetChanged();
}
}
Kemudian ListView mulai mengisi secara dinamis dengan alamat mac perangkat yang ditemukan.
Saya pikir itu tergantung pada apa yang Anda maksud dengan penyegaran. Maksud Anda tampilan GUI harus di-refresh, atau apakah maksud Anda bahwa tampilan anak-anak harus di-refresh sedemikian rupa sehingga Anda dapat memanggil getChildAt (int) secara terprogram dan mendapatkan tampilan yang sesuai dengan apa yang ada di Adaptor.
Jika Anda ingin tampilan GUI di-refresh, maka panggil notifyDataSetChanged () pada adaptor. GUI akan disegarkan saat selanjutnya digambar ulang.
Jika Anda ingin dapat memanggil getChildAt (int) dan mendapatkan tampilan yang mencerminkan apa yang ada di adaptor, maka panggil ke layoutChildren (). Ini akan menyebabkan tampilan anak direkonstruksi dari data adaptor.
Saya memiliki ArrayList yang ingin saya tampilkan di listview. ArrayList mengandung elemen dari mysql. Saya mengganti metode onRefresh dan dalam metode itu saya menggunakan tablelayout.removeAllViews (); dan kemudian mengulangi proses untuk mendapatkan data lagi dari database. Tapi sebelum itu pastikan untuk menghapus ArrayList Anda atau apa pun struktur data atau data baru akan ditambahkan ke yang lama ..
Jika Anda ingin memperbarui tampilan daftar tampilan UI dari layanan, maka buat adaptor statis di aktivitas Utama Anda dan lakukan ini:
@Override
public void onDestroy() {
if (MainActivity.isInFront == true) {
if (MainActivity.adapter != null) {
MainActivity.adapter.notifyDataSetChanged();
}
MainActivity.listView.setAdapter(MainActivity.adapter);
}
}
Jika Anda menggunakan garis panduan Android dan Anda menggunakan ContentProviders
untuk mendapatkan data dari Database
dan Anda menampilkannya dalam ListView
menggunakan CursorLoader
danCursorAdapters
, maka Anda semua perubahan pada data terkait akan secara otomatis tercermin dalam ListView.
Anda getContext().getContentResolver().notifyChange(uri, null);
di kursor di jendelaContentProvider
akan cukup untuk mencerminkan perubahan. Tidak perlu untuk bekerja ekstra.
Tetapi ketika Anda tidak menggunakan ini semua maka Anda harus memberi tahu adaptor ketika dataset berubah. Anda juga perlu mengisi ulang / memuat ulang dataset Anda (katakanlah daftar) dan kemudian Anda perlu meneleponnotifyDataSetChanged()
adaptor.
notifyDataSetChanged()
tidak akan berfungsi jika tidak ada perubahan pada datset. Berikut adalah komentar di atas metode dalam dokumen-
/**
* Notifies the attached observers that the underlying data has been changed
* and any View reflecting the data set should refresh itself.
*/
Saya hanya bisa mendapatkan notifyDataSetChanged hanya dengan mendapatkan data adaptor baru, kemudian mengatur ulang adaptor untuk tampilan daftar, lalu membuat panggilan seperti:
expandableAdapter = baseFragmentParent.setupEXLVAdapter();
baseFragmentParent.setAdapter(expandableAdapter);
expandableAdapter.notifyDataSetChanged();
pada pilihan lain adalah onWindowFocusChanged
metode, tetapi yakin itu sensitif dan membutuhkan beberapa pengkodean tambahan untuk siapa yang tertarik
override fun onWindowFocusChanged(hasFocus: Boolean) {
super.onWindowFocusChanged(hasFocus)
// some controls needed
programList = usersDBHelper.readProgram(model.title!!)
notesAdapter = DailyAdapter(this, programList)
notesAdapter.notifyDataSetChanged()
listview_act_daily.adapter = notesAdapter
}
Pertimbangkan Anda telah mengirimkan daftar ke adaptor Anda.
Menggunakan:
list.getAdapter().notifyDataSetChanged()
untuk memperbarui daftar Anda.
Jika saya berbicara tentang skenario saya di sini, jawaban di atas tidak akan berfungsi karena saya memiliki aktivitas yang menampilkan daftar nilai db bersama dengan tombol hapus dan ketika tombol hapus ditekan, saya ingin menghapus item itu dari daftar.
Yang keren adalah, saya tidak menggunakan tampilan pendaur ulang tetapi tampilan daftar sederhana dan tampilan daftar yang diinisialisasi di kelas adaptor. Jadi, memanggil notifyDataSetChanged()
tidak akan melakukan apa pun di dalam kelas adaptor dan bahkan di kelas aktivitas di mana objek adaptor diinisialisasi karena metode hapus berada di kelas adaptor.
Jadi, solusinya adalah menghapus objek dari adaptor dalam getView
metode kelas adaptor (untuk hanya menghapus objek tertentu tetapi jika Anda ingin menghapus semua, panggilclear()
).
Bagi Anda untuk mendapatkan ide, seperti apa kode saya,
public class WordAdapter extends ArrayAdapter<Word> {
Context context;
public WordAdapter(Activity context, ArrayList<Word> words) {}
//.......
@NonNull
@Override
public View getView(final int position, View convertView, ViewGroup group) {
//.......
ImageButton deleteBt = listItemView.findViewById(R.id.word_delete_bt);
deleteBt.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
if (vocabDb.deleteWord(currentWord.id)) {
//.....
} else{
//.....
}
remove(getItem(position)); // <---- here is the trick ---<
//clear() // if you want to clear everything
}
});
//....
Catatan: di sini remove()
dan getItem()
metode diwarisi dari kelas Adapter.
remove()
- untuk menghapus item tertentu yang diklikgetItem(position)
- adalah untuk mendapatkan item (di sini, itu objek Word saya yang telah saya tambahkan ke daftar) dari posisi yang diklik.Ini adalah bagaimana saya mengatur adaptor ke tampilan daftar di kelas aktivitas,
ArrayList<Word> wordList = new ArrayList();
WordAdapter adapter = new WordAdapter(this, wordList);
ListView list_view = (ListView) findViewById(R.id.activity_view_words);
list_view.setAdapter(adapter);
Cara termudah adalah dengan hanya membuat Adaper baru dan drop yang lama:
myListView.setAdapter(new MyListAdapter(...));