ListView学习(二)

本文介绍了Android中ListView的两种滑动监听方式:OnTouchListener和OnScrollListener,并提供了实现弹性滑动的方法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

### ListView 的滑动监听事件

OnTouchListener

OnTouchListener 是 View 中的监听事件,通过 ACTION_DOWN、ACTION_MOVE、ACTION_UP 这三个事件发生时的坐标,就可以根据坐标判断用户滑动的方向,并在不同的事件中进行相应的逻辑处理:

“`java

mListView.setOnTouchListener(new View.OnTouchListener(){

    @Override
    public boolean onTouch(View v, MotionEvent event){
        switch(event.getAction()){
            case MotionEvent.ACTION_DOWN : 
                //触摸操作
                break;
            case MotionEvent.ACTION_MOVE :
                //滑动操作
                break;
            case MotionEvent.ACTION_UP :
                //离开时操作
                break;
        }
    }
});

“`

OnScrollListener

OnScrollListener 是 AbsListView 中的监听事件,它封装了很多与ListView 相关的信息,使用起来也更加的灵活。


 mListView.setOnScrollLinstener(new OnScrollListener(){
     @Override
     public void onScrollStateChanged(AbsListView view, int scrollState){
        switch(scrollState){
            case OnScrollListener.SCROLL_STATE_IDLE : 
                //滑动停止时
                Log.d("TAG","SCROLL_STATE_IDLE");
                break;
            case OnScrollListener.SCROLL_STATE_TOUCH_SCROLL :
                //正在滑动
                Log.d("TAG","SCROLL_STATE_TOUCH_SCROLL");
                break;
            case OnScrollListener.SCROLL_STATE_FLING : 
                //手指抛动的时候,就是快速滑动
                //在手指离开屏幕后 ListView 由于惯性继续滑动
                Log.d("TAG","SCROLL_STATE_FLING");
                break;
        }

        @Override
        public void onScroll(AbsListView view,
                            int firstVisibleItem,//当前能看到的第一个 item 的 ID(从 0 开始)
                            int visibleItemConut,//当前能看到的 item 的总数
                            int totalItemCount){//整个 ListView 的 item 总数
            //滚动的时候一直调用
            Log.d("TAG","onScroll");
        }
     }
 });

判断是否滚动到了最后一行:


//这就说明 firstVisibleItem 表示的在所有的 Item 中的 id 号
if(firstVisibleItem + visibleItemConut == totalItemCount && totalItemCount > 0){
    //滚动到最后一行的时候
}

判断滚动的方向:


if(firstVisibleItem > lastVisibleItemPosition){
    //上滑
}else if (firstVisibleItem < lastVisibleItemPosition){
    //下滑
}
lastVisibleItemPosition = firstVisibleItem;

通过一个成员变量 lastVisibleItemPosition 来记录上次第一个可视的 item 的 ID 并与当前的可视 Item 的 ID 进行比较,即可知当前滚动的方向。

当然,ListView 为我们封装好了获取当前可视的 Item 的位置等信息:


    mListView.getFirstVisiblePosition();
    mListView.getLastVisiblePosition();

ListView 常用扩展

具有弹性的 ListView

Android 默认的 ListView 在滚动到底端或者顶端的时候,并没有很好的显示。在 Android 5X 中,Google 为这样的行为添加了一个半月形的阴影效果。

在 ListView 的源码中可以发现,ListView 中有一个控制滑动到边缘的处理方法:


    protected boolean overScrollBy(int detalX, int detalY,
                        int scrollX, int scrollY,
                        int scrollRangeX, int scrollRangeY,
                        int maxOverScrollX, int maxOverScrollY,
                        boolean isTouchEvent){

    }

其中的参数 maxOverScrollY ,注释是这样解释的 umber of Pixels to overscroll by in either direction along the Y axis ,虽然这个参数的值默认值是 0 ,但只要修改这个参数的值,就可以让 ListView 具有弹性了。 我们来对这个方法进行复写:我们重新定义一个变量 mMaxOverDistance ,通过对这个变量的修改我们就达到的弹性滑动的效果。


    @Override
    protected boolean overScrollBy(int detalX, int detalY,
                        int scrollX, int scrollY,
                        int scrollRangeX, int scrollRangeY,
                        int maxOverScrollX, int maxOverScrollY,
                        boolean isTouchEvent){
            super.overScrollBy(detalX, detalY, 
                        scrollX, scrollY,
                        scrollRangeX, scrollRangeY,
                        maxOverScrollX, mMaxOverDistance,
                        isTouchEvent);
    }

为了在不同的屏幕分辨率下滑动的距离基本保持一致,我们通过计算屏幕的 density 来计算具体的值。

    private void initView(){
        DisplayMetrics metrics = mContext.getResources().getDisplayMetrics();
        float density = metrics.density;
        mMaxOverDistance = (int) (density * mMaxOverDistance);
    }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值