在我的应用程序中,我希望在父视图的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。
提前致谢。
答案 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">
...
如果您需要阻止子视图接收事件,您也可以完全跳过超级电话,但我认为这种情况很少见。如果您需要重新创建一些(但不是全部)默认功能,则需要重写很多内容:
答案 1 :(得分:3)
我尝试过类似的设计(在根FrameLayout上只有一个onTouch Listener)并且它有效。我得到的所有点都是onTouch返回true而不是false。否则,我只是得到第一点。我无法找出这个“回归”问题的原因,因为我期待相反的行为。
但是,根据我的经验,如果您将任何子视图设置为可点击,那么其父视图的onTouch将无效。如果你有另一种解决方案,如果你分享,那就太好了。
答案 2 :(得分:2)
2个解决方案:
在ViewGroup上使用onInterceptTouchEvent,如图所示here
避免布局处理触摸事件,并使视图成为布局的子视图,覆盖其整体大小,以处理触摸事件。
它不如扩展类有效,但它更容易,并且不需要创建新的文件/类。
示例:
只需将触摸事件添加到视图中,它就会处理它们而不是任何其他视图。
答案 3 :(得分:-2)
可以通过使用&#34; onInterceptTouchEvent&#34;来实现。请试试这个,它可能会起作用
@Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
onTouchEvent(ev);
return false;
}