如何在父视图的touchlistener中使用子视图触摸事件?

时间:2012-04-06 10:19:01

标签: android view touch

在我的应用程序中,我希望在父视图的onTouchListener中获取所有子视图的触摸事件,但我无法得到它。

示例:

在我的活动视图中,我有一个框架布局,里面有一些按钮和edittexts。因此,每当我触摸按钮或编辑文本时,我都希望在 framlayout onTouchListeners中获得此触摸事件。

这是我的代码..

活动:

private FrameLayout rootView;
    @Override
    public void onCreate(Bundle savedInstanceState)
    {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);

        rootView = (FrameLayout) findViewById(R.id.rootview);
        rootView.setOnTouchListener(new OnTouchListener()
        {
            @Override
            public boolean onTouch(View v, MotionEvent event)
            {
                Log.e("Touch Event: ", " X: " + event.getX() + " Y: " + event.getY());
                return false;
            }
        });
         }

main.xml中:

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:padding="10dip"
    android:id="@+id/rootview">

<LinearLayout 
    android:layout_width="fill_parent"
    android:layout_height="wrap_content"
    android:background="@android:color/white"
    android:orientation="vertical"
    android:layout_gravity="center" 
    android:padding="10dip">

     <TextView
                    android:layout_width="wrap_content"
                    android:layout_height="wrap_content"
                    android:layout_gravity="center"
                    android:padding="4dp"
                    android:text="Login"
                    android:textColor="@android:color/black"
                    android:textSize="16dp"
                    android:textStyle="bold" />

                <TextView
                    android:layout_width="fill_parent"
                    android:layout_height="wrap_content"
                    android:layout_marginTop="10dp"
                    android:text="Username"
                    android:textColor="@android:color/black"
                    android:textSize="14dp"
                    android:textStyle="bold" />

                <EditText
                    android:id="@+id/edttextusername"
                    android:layout_width="fill_parent"
                    android:layout_height="wrap_content"
                    android:imeOptions="flagNoExtractUi"
                    android:singleLine="true" />

                <TextView
                    android:layout_width="fill_parent"
                    android:layout_height="wrap_content"
                    android:layout_marginTop="10dp"
                    android:text="Password"
                    android:textColor="@android:color/black"
                    android:textSize="14dp"
                    android:textStyle="bold" />

                <EditText
                    android:id="@+id/edttextpassword"
                    android:layout_width="fill_parent"
                    android:layout_height="wrap_content"
                    android:imeOptions="flagNoExtractUi"
                    android:password="true" />

                <LinearLayout
                    android:layout_width="fill_parent"
                    android:layout_height="wrap_content"
                    android:layout_gravity="center"
                    android:gravity="center"
                    android:orientation="horizontal"
                    android:padding="10dp" >

                    <Button
                        android:id="@+id/btnlogin"
                        android:layout_width="0dip"
                        android:layout_height="40dip"
                        android:layout_marginRight="5dp"
                        android:layout_weight="1"
                        android:enabled="false"
                        android:text="Login"
                        android:padding="5dip"
                        android:textColor="@android:color/black"
                        android:textStyle="bold" 
                        />

                    <Button
                        android:id="@+id/btncall"
                        android:layout_width="0dip"
                        android:layout_height="40dip"
                        android:layout_marginLeft="5dp"
                        android:layout_weight="1"
                        android:padding="5dip"
                        android:text="Cancel"
                        android:textColor="@android:color/black"
                        android:textStyle="bold" />
                </LinearLayout>


</LinearLayout>

    </FrameLayout> 

问题:因此,当我触摸按钮或编辑文本时,我无法在我的框架布局的onTouchListener(它未被调用)中进行触摸协作。

注意:我不想为所有childView添加单独的onTouchListener。

提前致谢。

4 个答案:

答案 0 :(得分:23)

您可以通过覆盖布局中的dispatchTouchEvent来实现这一目标。

public class MyFrameLayout extends FrameLayout {
    @Override
    public boolean dispatchTouchEvent(MotionEvent e) {
        // do what you need to with the event, and then...
        return super.dispatchTouchEvent(e);
    }
}

然后使用该布局代替通常的FrameLayout:

<com.example.android.MyFrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:padding="10dip"
    android:id="@+id/rootview">
  ...

如果您需要阻止子视图接收事件,您也可以完全跳过超级电话,但我认为这种情况很少见。如果您需要重新创建一些(但不是全部)默认功能,则需要重写很多内容:

http://grepcode.com/file/repository.grepcode.com/java/ext/com.google.android/android/4.1.1_r1/android/view/ViewGroup.java#ViewGroup.dispatchTouchEvent%28android.view.MotionEvent%29

答案 1 :(得分:3)

我尝试过类似的设计(在根FrameLayout上只有一个onTouch Listener)并且它有效。我得到的所有点都是onTouch返回true而不是false。否则,我只是得到第一点。我无法找出这个“回归”问题的原因,因为我期待相反的行为。

但是,根据我的经验,如果您将任何子视图设置为可点击,那么其父视图的onTouch将无效。如果你有另一种解决方案,如果你分享,那就太好了。

答案 2 :(得分:2)

2个解决方案:

  1. 在ViewGroup上使用onInterceptTouchEvent,如图所示here

  2. 避免布局处理触摸事件,并使视图成为布局的子视图,覆盖其整体大小,以处理触摸事件。

    它不如扩展类有效,但它更容易,并且不需要创建新的文件/类。

    示例:

         

    只需将触摸事件添加到视图中,它就会处理它们而不是任何其他视图。

答案 3 :(得分:-2)

可以通过使用&#34; onInterceptTouchEvent&#34;来实现。请试试这个,它可能会起作用

@Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
     onTouchEvent(ev);
     return  false;
}