Android View的onTouchEvent和OnTouch区别

可以通过重写onTouchEvent方法来处理诸如down move up的消息:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
public class TestButton extends Button {
 
     public TestButton(Context context) {
         super (context);
         // TODO Auto-generated constructor stub
     }
     
     public TestButton(Context context, AttributeSet attributeSet) {
         super (context, attributeSet);
         // TODO Auto-generated constructor stub
     }
     
     @Override
     public boolean onTouchEvent(MotionEvent event) {
         boolean value = super .onTouchEvent(event);
         System.out.println( "super.onTouchEvent: " + value+ " event: " + event.getAction());
         return value;
     }
也可以通过实现OnTouchListener的接口,然后设置TestButton的onTouchListener可以达到同样的目的
?
1
2
3
4
5
6
7
class OnTouchListenerTest implements View.OnTouchListener{
     @Override
     public boolean onTouch(View v, MotionEvent event) {
         return false ;
     }
     
}
?
1
2
3
TestButton b = (TestButton)findViewById(R.id.button);
OnTouchListenerTest listener = new OnTouchListenerTest();
b.setOnTouchListener(listener);
但上述两种监听有什么区别呢?

先看一下Android源码中对于View中dispatchTouchEvent的实现:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
public boolean dispatchTouchEvent(MotionEvent event){
     ... ...
     if (onFilterTouchEventForSecurity(event)){
         ListenerInfo li = mListenerInfo;
         if (li != null && li.mOnTouchListener != null && (mViewFlags & ENABLED_MASK) == ENABLED
             && li.mOnTouchListener.onTouch( this , event)) {
             return true ;
         }
         if (onTouchEvent(event)){
             return true ;
         }
     }
     ... ...
     return false ;
}

可以看到onTouchListener的接口的优先级是要高于onTouchEvent的,假若onTouchListener中的onTouch方法返回true,

表示此次事件已经被消费了,那onTouchEvent是接收不到消息的。

因为Button的performClick是利用onTouchEvent实现,假若onTouchEvent没有被调用到,那么Button的Click事件也无法响应。


综合来讲:

onTouchListener的onTouch方法优先级比onTouchEvent高,会先触发。

假如onTouch方法返回false会接着触发onTouchEvent,反之onTouchEvent方法不会被调用。

内置诸如click事件的实现等等都基于onTouchEvent,假如onTouch返回true,这些事件将不会被触发


调用布局View的performClick()方法

修改之前的xml片段。

            <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" 
                android:id="@+id/layout_deletesms" 
                android:clickable="true" 
                android:padding="0dip" 
                android:background="@drawable/sms_dialog_button" 
                android:layout_width="50dip" 
                android:layout_height="match_parent" > 
                <ImageView 
                    android:id="@+id/deleteSms" 
                    android:layout_centerInParent="true" 
                    android:src="@drawable/sms_delete" 
                    android:scaleType="centerInside" 
                    android:focusable="false" // 仅仅这样设置无法实现所需效果。 
                    android:background="#00000000" 
                    android:layout_width="30dip" 
                    android:layout_height="30dip"/> 
             </RelativeLayout> 
实现效果:点击ImageView时,ImageView的容器对象RelativeLayout,也有被点击的效果出现。

这里,布局被点击时的效果是,切换不同的drawable图片。

在xml布局文件中,无论怎么改都达不到效果。

最后采取在代码中实现。当点击ImageView时,调用布局View的performClick()方法。

            <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" 
                android:id="@+id/layout_deletesms" 
                android:clickable="true" 
                android:padding="0dip" 
                android:background="@drawable/sms_dialog_button" 
                android:layout_width="50dip" 
                android:layout_height="match_parent" > 
                <ImageView 
                    android:id="@+id/deleteSms" 
                    android:layout_centerInParent="true" 
                    android:src="@drawable/sms_delete" 
                    android:scaleType="centerInside" 
                    android:background="#00000000" 
                    android:layout_width="30dip" 
                    android:layout_height="30dip"/> 
             </RelativeLayout>

mLayoutDelsms = (RelativeLayout) findViewById(R.id.layout_deletesms);

case R.id.deleteSms: 
    mLayoutDelsms.performClick();


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值