从onScrollListener添加ListView

时间:2013-10-31 19:15:58

标签: android listview arraylist

我正在尝试更新我的列表视图以制作“无限滚动”。会发生的是前40个结果加载正常,当我到达滚动的底部时,接下来40个结果替换前40个...

我想要的是第二组40个结果添加到前40个,所以我有一个无穷无尽的列表和能够滚动回到列表的开头。

我在下面发布我的代码。谢谢!

public class SearchResults extends Activity implements BannerAdListener, OnScrollListener{

    private LinearLayout bottomNav;
    private ListView ringtoneList;
    private int start = 0, num = 40, curPage = 1;
    private Handler handler = new Handler();
    private ProgressDialog progressDialog = null;
    private ArrayList<Ringtone> ringtones;
    private MoPubView moPubView;  
    private String searchString;

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

        Bundle extras = getIntent().getExtras();
        if (extras == null) {
            // no search string defined
            finish();
        } else {
            searchString = extras.getString("search_string");
        }

        setContentView(R.layout.search_results);    

        ringtoneList = (ListView)findViewById(R.id.ringtone_list);
        ringtoneList.setOnItemClickListener(new AdapterView.OnItemClickListener() {
            @Override
            public void onItemClick(AdapterView<?> av, View view, int position, long id) {
                Intent i = new Intent(SearchResults.this, RingtoneView.class);
                i.putExtra("ringtone", ringtones.get(position));
                startActivity(i);
            }
        });

        performSearch();

        moPubView = (MoPubView) findViewById(R.id.adview);
        moPubView.setAdUnitId(Utils.MoPubBannerId);
        moPubView.loadAd();
        moPubView.setBannerAdListener(this);

        ringtoneList.setOnScrollListener(this);

    }

    public void onScroll(AbsListView view, int firstVisible, final int visibleCount, int totalCount) {

            Log.i("List", "firstVisible="+firstVisible+" visibleCount="+visibleCount+" totalCount="+totalCount);

            boolean loadMore = firstVisible + visibleCount >= totalCount;

            if(loadMore) {

                Log.i("List", "Loading More Results");                

                curPage++;
                start = num * (curPage-1);

                new Thread() {

                    public void run() {

                        ringtones = Utils.search(start, num, searchString);

                        if (ringtones != null && ringtones.size() > 0) {

                            handler.post(new Runnable() {
                                public void run() {
                                    ringtoneList.setAdapter(new RingtoneRowAdapter(SearchResults.this, ringtones));
                                }
                            });

                        } else {

                            handler.post(new Runnable() {
                                public void run() {
                                    new AlertDialog.Builder(SearchResults.this)
                                    .setTitle(getResources().getString(R.string.context_info)).setMessage(getResources().getString(R.string.context_noresult))
                                    .setPositiveButton(getResources().getString(R.string.context_ok), null).show();
                                }
                            });

                        }

                    }
                }

                .start();
         }
    }

    public void onScrollStateChanged(AbsListView v, int s) { } 

    private void performSearch() {
        progressDialog = ProgressDialog.show(SearchResults.this, getResources().getString(R.string.loading_message), getResources().getString(R.string.loading_search), true); 
        new Thread() {
            public void run() {
                ringtones = Utils.search(start, num, searchString);
                if (ringtones != null && ringtones.size() > 0) {
                    updateList();
                } else {
                    handler.post(new Runnable() {
                        public void run() {
                            new AlertDialog.Builder(SearchResults.this)
                            .setTitle(getResources().getString(R.string.context_info)).setMessage(getResources().getString(R.string.context_noresult))
                            .setPositiveButton(getResources().getString(R.string.context_ok), null).show();
                            ringtoneList.setVisibility(View.INVISIBLE);
                            bottomNav.setVisibility(View.INVISIBLE);
                        }
                    });
                }
                progressDialog.dismiss();
            }
        }.start();
    }

    private void updateList() {
        handler.post(new Runnable() {
            public void run() {
                //Log.d("search", "ringtones.size() " + ringtones.size());
                ringtoneList.setVisibility(View.VISIBLE);
                ringtoneList.setAdapter(new RingtoneRowAdapter(SearchResults.this, ringtones));
            }
        });
    }

}

请帮忙!谢谢!

2 个答案:

答案 0 :(得分:0)

虽然我无法100%确定,但我认为你的问题与你正在设置一个只有加载铃声部分的新适配器这一事实有关。它可能与此片段有关:

ringtones = Utils.search(start, num, searchString);

if (ringtones != null && ringtones.size() > 0) {

    handler.post(new Runnable() {
        public void run() {
            ringtoneList.setAdapter(new RingtoneRowAdapter(SearchResults.this, ringtones));
        }
    });

}

您应该添加已有的铃声,而不是添加全新的铃声阵列。您的ringtones变量已经是一个实例变量,所以我确定您是否更改了此行:

ringtones = Utils.search(start, num, searchString);

以下内容:

ringtones.addAll(Utils.search(start, num, searchString));

它可能会解决您的问题。

答案 1 :(得分:0)

您的代码有点乱,但您的updateList方法正在创建一个新的RingtoneRowAdapter。您应该将项目添加到列表并调用

mRingtoneRowAdapter.notifyDatasetChanged();

这将告诉适配器获取新视图(如果需要)以及在Adapter级别发生的大量内部事务。所以它有点:

   private void updateList() {
         handler.post(new Runnable() {
            public void run() {
                if ( mAdapter == null ) { 
                   mAdapter = new RingtoneRowAdapter(SearchResults.this, ringtones);
                   ringtoneList.setAdapter(mAdapter);
                } else {
                   mAdapter.notifyDataSetChanged();
                }
                ringtoneList.setVisibility(View.VISIBLE);
            }
        });
    }

当然,正如已经建议的那样,不要一直初始化你的铃声阵列。只需向其添加数据即可。

您不需要(大多数时候)隐藏列表,您可以添加您的布局,任何视图(包括完整的布局!)ID为:

android:id="@android:id/empty"

如果您的ListView的id为android:id =“@ id / android:list”,并且您在ListFragment或ListActivity中,那么当列表为空时,将显示“空”布局(如果您不想看到它,可以是虚拟视图。)

只是建议:)

UPDATE :对于你的null,我看到逻辑有点奇怪,因为我们不知道你的应用程序的要求(即如果没有结果怎么办,什么如果结果无效等等,我会假设你只是想确保它有效并稍后处理。

因此,考虑到这一点,我发现了两个问题。

  1. 我假设您的Utils.search方法可以返回null,因为您正在检查它。对我来说感觉很奇怪,我宁愿返回一个空数组,表明搜索没有产生结果;抛开那个小小的评论,你没有检查(或者我们不知道,因为我们没有看到你的搜索方法的来源),如果searchString为null。
  2. 您还没有提供堆栈跟踪,因此我们无法判断null的位置(在搜索内部还是?)
  3. 快速解决方案是在搜索之前检查null ...我会亲自在搜索功能中执行此操作。

    类似的东西:

    public ArrayList<Ringtone> search(final int start, final int num, final String searchString) {
    
       if ( searchString == null ) {
           //DECIDE WHAT YOU WANT TO DO, EITHER:
           return null;
           // OR YOU CAN RETURN AN EMPTY ARRAY
           return new ArrayList<Ringtone>(); 
        }
       // You should check for these (change according to your rules)
       if (start < 0 ) {
          start = 0; // protect yourself from bad data.
       }
       if ( num < 0 ) {
           num = 0; 
       } 
        /// THE REST OF YOUR search FUNCTION
    
       return <your array>
    }
    

    现在另一件事是你可能希望搜索返回增量结果(这样你就可以将它们添加到你的数组中(而不是每次都返回一个新数组)。为此,正如已经建议的那样,使用addAll技巧,但是然后,不要返回null,返回一个新的空数组,所以如果没有别的东西可以添加就没有坏处。