Heim > Fragen und Antworten > Hauptteil
我当前有个需求是做类似于饿了么外卖App的选择商品页面的两个ListView联动的效果:
1.左边的分类ListView点击后,右边的商品ListView能滚动到指定的分类板块。
2.滑动右边的商品ListView,左边的分类ListView能切换选中标记。
我的做法是:
private boolean mIsClickTrigger;
....
mCategoryListView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
int productPos = ...;
isClickTrigger = true;
mProductListView.setSelection(productPos);
}
});
mProductListView.setOnScrollListener(new AbsListView.OnScrollListener() {
@Override
public void onScrollStateChanged(AbsListView view, int scrollState) {}
@Override
public void onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount, int totalItemCount) {
if (isClickTrigger) {
// 消耗掉点击触发
isClickTrigger = false;
} else {
int categoryPos = ...;
mCategoryListView.setItemChecked(categoryPos, true);
}
}
});
isClickTrigger这个标志位的作用是:当点击左边的分类ListView的某一项时,使滚动监听里的代码不执行。只有当用手指滑动右边的商品ListView时,才执行滚动监听里的代码。
但这样有个问题,当我使用listview.setSection(10)后发现滚动很突兀,为了让它能平滑地滚动,我用了listview.smoothScrollToPosition(10),滚动是很平滑了,但是由于滚动监听被多次回调了,标志位就没效果了。
所以想请问大家有没有什么好的办法,先谢谢了!!!
伊谢尔伦2017-04-17 15:02:26
所以你应该是想解决:
当mProductListView
滑动结束时,根据标志位判断是用户手动滑动的,就改变mCategoryListView
相应条目的check状态。
因为若是mCategoryListView.onItemClick()
导致mProductListView滑动,就没必要这样做。
onScroll(...)
会在滚动过程中一直调用,所以无法处理标志位。应该做的是:『如何判断ListView滚动结束了?』onScrollStateChanged(...)
的参数scrollState
,它有3个状态:其中一个是SCROLL_STATE_IDLE
,它就可以用来表示“滚动结束”的状态。所以应该在onScrollStateChanged()
判断标志位,而不是在onScroll
中:
listView.setOnScrollListener(new OnScrollListener() {
@Override
public void onScrollStateChanged(AbsListView view, int scrollState) {
Log.d(TAG, "Scroll state " + scrollState);
// 所以应该在这里判断标志位,而不是在onScroll中:
if (scrollState == SCROLL_STATE_IDLE) {
if (isClickTrigger == true) {
isClickTrigger = false;
} else {
//...
mCategoryListView.setItemChecked(categoryPos, true);
}
}
}
});
但测试了十几次,发现了一个问题,在一半的测试中,SCROLL_STATE_IDLE出现了1次,另一半测试中出现了2次,像下面log显示的这样:
11-30 14:29:16.092: D/CrimeListFragment(9638): setSelection(18)
11-30 14:29:16.095: D/CrimeListFragment(9638): onScroll: 5, 13, 18
11-30 14:29:16.819: D/CrimeListFragment(9638): smoothScrollToPosition(1)
11-30 14:29:16.824: D/CrimeListFragment(9638): onScrollStateChanged: 2
11-30 14:29:16.841: D/CrimeListFragment(9638): onScroll: 5, 13, 18
11-30 14:29:16.860: D/CrimeListFragment(9638): onScroll: 4, 14, 18
11-30 14:29:16.875: D/CrimeListFragment(9638): onScroll: 4, 13, 18
......省略若干onScroll
11-30 14:29:16.943: D/CrimeListFragment(9638): onScroll: 2, 14, 18
11-30 14:29:16.944: D/CrimeListFragment(9638): onScroll: 2, 14, 18
11-30 14:29:16.944: D/CrimeListFragment(9638): onScrollStateChanged: 0
11-30 14:29:16.944: D/CrimeListFragment(9638): onScrollStateChanged: 2
11-30 14:29:16.959: D/CrimeListFragment(9638): onScroll: 2, 13, 18
11-30 14:29:16.977: D/CrimeListFragment(9638): onScroll: 1, 14, 18
......省略若干onScroll
11-30 14:29:17.028: D/CrimeListFragment(9638): onScroll: 0, 14, 18
11-30 14:29:17.028: D/CrimeListFragment(9638): onScroll: 0, 14, 18
11-30 14:29:17.028: D/CrimeListFragment(9638): onScrollStateChanged: 0
你先看这样效果怎样,如果不行,还需要找workaround来判断『ListView滚动结束』。。。
ringa_lee2017-04-17 15:02:26
在Adapter上实现View.OnTouchListener的监听,实现onTouch()方法.
@Override
public boolean onTouch(View v, MotionEvent event) {
Log.i("Tag","被触摸了");
return false;
}