Java自学者论坛

 找回密码
 立即注册

手机号码,快捷登录

恭喜Java自学者论坛(https://www.javazxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,会员资料板块,购买链接:点击进入购买VIP会员

JAVA高级面试进阶训练营视频教程

Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程Go语言视频零基础入门到精通Java架构师3期(课件+源码)
Java开发全终端实战租房项目视频教程SpringBoot2.X入门到高级使用教程大数据培训第六期全套视频教程深度学习(CNN RNN GAN)算法原理Java亿级流量电商系统视频教程
互联网架构师视频教程年薪50万Spark2.0从入门到精通年薪50万!人工智能学习路线教程年薪50万大数据入门到精通学习路线年薪50万机器学习入门到精通教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程MySQL入门到精通教程
查看: 722|回复: 0

关于ScrollView中嵌套listview焦点滑动问题 解决

[复制链接]
  • TA的每日心情
    奋斗
    2024-11-24 15:47
  • 签到天数: 804 天

    [LV.10]以坛为家III

    2053

    主题

    2111

    帖子

    72万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    726782
    发表于 2021-4-23 12:23:44 | 显示全部楼层 |阅读模式

    (第三种,第四种简单推荐使用)

    在这里我要提出的是,listview能滚动的前提是:当listview本身的高度小于listview里的子view。

    第一种方法

    只需在MainActivity中 找到listview 和 scrollview

    然后给listview设置监听事件

    listView.setOnTouchListener(new OnTouchListener() {
                
                public boolean onTouch(View v, MotionEvent event) {
                    // TODO Auto-generated method stub
                  
                    if(event.getAction() == MotionEvent.ACTION_UP){  
                        scrollView.requestDisallowInterceptTouchEvent(false);  
                    }else{  
                        scrollView.requestDisallowInterceptTouchEvent(true);  
                    }  
                    return false;
                }
            });

    第二种方法

    只需重写listview即可

    package com.bawei.day06;
    
    import android.content.Context;
    import android.util.AttributeSet;
    import android.view.MotionEvent;
    import android.view.View;
    import android.widget.ListView;
    
    public class ListViewForScrollView extends ListView {
        int mLastMotionY;
        boolean bottomFlag;
        public ListViewForScrollView(Context context, AttributeSet attrs) {
            super(context, attrs);
            // TODO Auto-generated constructor stub
        }
        
        @Override
        public boolean onInterceptTouchEvent(MotionEvent ev) {
            
            if (bottomFlag) {
                getParent().requestDisallowInterceptTouchEvent(true);
            }
            return super.onInterceptTouchEvent(ev);
        }
    
        @Override
        public boolean onTouchEvent(MotionEvent ev) {
            // TODO Auto-generated method stub
            int y = (int) ev.getRawY();
            switch (ev.getAction()) {
            case MotionEvent.ACTION_DOWN:
                mLastMotionY = y;
                break;
            case MotionEvent.ACTION_MOVE:
                int deltaY = y - mLastMotionY;
                if (deltaY < 0) {
                    View child = getChildAt(0);
                    if (child != null) {
                        if (getLastVisiblePosition() == (getChildCount()-1) && child.getBottom() == (getChildCount()-1)) {
                            bottomFlag = true;
                            getParent().requestDisallowInterceptTouchEvent(true);
                        }
    
                        int bottom = child.getBottom();
                        int padding = getPaddingTop();
                        if (getLastVisiblePosition() == (getChildCount()-1)
                                && Math.abs(bottom - padding) >= 20) {
                            bottomFlag = true;
                            getParent().requestDisallowInterceptTouchEvent(true);
                        }
                    }
                }
                break;
            case MotionEvent.ACTION_UP:
            case MotionEvent.ACTION_CANCEL:
                break;
            }
            return super.onTouchEvent(ev);
        }
    
        public void setBottomFlag(boolean flag) {
            bottomFlag = flag;
        }
    }

     

    第三种方法

    只需重写listview即可

    package com.bawei.day06;
    
    import android.content.Context;
    import android.util.AttributeSet;
    import android.view.MotionEvent;
    import android.view.View;
    import android.widget.ListView;
    
    public class ListViewForScrollView extends ListView {
        int mLastMotionY;
        boolean bottomFlag;
        public ListViewForScrollView(Context context, AttributeSet attrs) {
            super(context, attrs);
            // TODO Auto-generated constructor stub
        }
    
        @Override
        public boolean dispatchTouchEvent(MotionEvent ev) {
            // TODO Auto-generated method stub
            getParent().requestDisallowInterceptTouchEvent(true);
            
            return  super.dispatchTouchEvent(ev);
        }
    
    }

     

    第四种方法

    只需在MainActivity中 找到listview 

    然后给listview设置监听事件

    listView.setOnTouchListener(new OnTouchListener() {
                
                public boolean onTouch(View v, MotionEvent event) {
                    // TODO Auto-generated method stub
                    listView.getParent().requestDisallowInterceptTouchEvent(true);
                    return false;
                }
            });

     解释原理:

    当手指触摸到屏幕时,系统就会调用相应View的onTouchEvent,并传入一系列的action。当有多个层级的View时,在父层级允许的情况下,这个action会一直向下传递直到遇到最深层的View。所以touch事件最先调用的是最底层View的onTouchEent,如果View的onTouchEvent接收到某个touch action并作了相应处理,最后有两种返回方式return true和return falsereturn true会告诉系统当前的View需要处理这次的touch事件,以后的系统发出的ACTION_MOVE,ACTION_UP还是需要继续监听并接收的,而且这次的action已经被处理掉了,父层的View是不可能出发onTouchEvent了。所以每一个action最多只能有一个onTouchEvent接口返回true。如果return false,便会通知系统,当前View不关心这一次的touch事件,此时这个action会传向父级,调用父级View的onTouchEvent。但是这一次的touch事件之后发出的任何action,该View都不会再接受,onTouchEvent在这一次的touch事件中再也不会触发,也就是说一旦View返回false,那么之后的ACTION_MOVE,ACTION_UP等ACTION就不会在传入这个View,但是下一次touch事件的action还是会传进来的。
        前面说了底层的View能够接收到这次的事件有一个前提条件:在父层级允许的情况下。假设不改变父层级的dispatch方法,在系统调用底层onTouchEvent之前会先调用父View的onInterceptTouchEvent方法判断,父层View是不是要截获本次touch事件之后的action。如果onInterceptTouchEvent返回了true,那么本次touch事件之后的所有action都不会再向深层的View传递,统统都会传给父层View的onTouchEvent,就是说父层已经截获了这次touch事件,之后的action也不必询问onInterceptTouchEvent,在这次的touch事件之后发出的action时onInterceptTouchEvent不会再次调用,直到下一次touch事件的来临。如果onInterceptTouchEvent返回false,那么本次action将发送给更深层的View,并且之后的每一次action都会询问父层的onInterceptTouchEvent需不需要截获本次touch事件。只有ViewGroup才有onInterceptTouchEvent方法,因为一个普通的View肯定是位于最深层的View,touch事件能够传到这里已经是最后一站了,肯定会调用View的onTouchEvent。
    对于底层的View来说,有一种方法可以阻止父层的View截获touch事件,就是调用getParent().requestDisallowInterceptTouchEvent(true);方法。一旦底层View收到touch的action后调用这个方法那么父层View就不会再调用onInterceptTouchEvent了,也无法截获以后的action。
    用例子总结一下onInterceptTouchEvent和onTouchEvent的调用顺序:
    假设最高层View叫OuterLayout,中间层View叫InnerLayout,最底层View叫MyVIew。调用顺序是这样的(假设各个函数返回的都是false)
    OuterLayout.onInterceptTouchEvent->InnerLayout.onInterceptTouchEvent->MyView.onTouchEvent->InnerLayout.onTouchEvent->OuterLayout.onTouchEvent。
    
    [java] view plaincopy
    
    @Override    
        public boolean dispatchTouchEvent(MotionEvent ev) {   
            getParent().requestDisallowInterceptTouchEvent(true);  
            return super.dispatchTouchEvent(ev);    
        }  
    
    这句话是告诉父view,我的事件自己处理
    
     
    
    [java] view plaincopy
    
    public boolean onTouch(View v, MotionEvent event) {  
         switch (event.getAction()) {  
         case MotionEvent.ACTION_MOVE:   
             pager.requestDisallowInterceptTouchEvent(true);  
             break;  
         case MotionEvent.ACTION_UP:  
         case MotionEvent.ACTION_CANCEL:  
             pager.requestDisallowInterceptTouchEvent(false);  
             break;  
         }  
    }  
     
    
    也可以写成类似于上面那样,当用户按下的时候,我们告诉父组件,不要拦截我的事件(这个时候子组件是可以正常响应事件的),拿起之后就会告诉父组件可以阻止。
    
     
    
    还有一个关于子控件和父控件的事件响应问题 
    当父控件中有子控件的时候,并且父控件和子空间都有事件处理(比如单击事件)。这时,点击子控件,父控件的单击事件就无效了。
    
    比如一个LinearLayout里面有一个子控件TextView,但是TextView的大小没有LinearLayout大
    
    ①如果LinearLayout和TextView都设置了单击事件,那么
    
    点击TextView区域的时候,触发的是TextView的事件,
    点击TextView以外的区域的时候,还是触发的LinearLayout的事件。
    ②如果LinearLayout设置了单击事件,而TextView没有设置单击事件的话,那么
    
    不管单击的是TextView区域,还是TextView以外的区域,都是触发的LinearLayout的单击事件
    如果LinearLayout的大小和TextView一样的话,那么
    
    ①如果LinearLayout和TextView都设置了单击事件,那么
    
    只有TextView的单击事件有效
    ②如果LinearLayout设置了单击事件,而TextView没有设置单击事件的话,那么
    
    触发的是LinearLayout的单击事件

     

    哎...今天够累的,签到来了1...
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|小黑屋|Java自学者论坛 ( 声明:本站文章及资料整理自互联网,用于Java自学者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2025-1-12 15:46 , Processed in 0.071401 second(s), 30 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

    快速回复 返回顶部 返回列表