gpt4 book ai didi

android - 在重叠 View 的情况下控制触摸事件传播

转载 作者:塔克拉玛干 更新时间:2023-11-02 21:15:36 24 4
gpt4 key购买 nike

我在尝试控制 RecycleView 中的触摸事件传播时遇到了一些问题。所以我有一个 RecyclerView 填充一组模仿卡片堆的 CardView(因此它们以一定的位移相互重叠,尽管它们具有不同的高程值)。我目前的问题是每张卡片都有一个按钮,并且由于卡片的相对位移小于按钮的高度,因此会导致按钮重叠的情况,并且无论何时调度触摸事件,它都会从 View 层次结构的底部开始传播(来自子级 child 数量最多)。

根据我阅读的文章(thisthisthis video),触摸传播取决于父 View 中 View 的顺序,因此触摸将首先传递给索引最高的 child ,而我希望触摸事件仅由最顶层 View 和 RecyclerView 的可触摸对象处理(它还必须处理拖放手势)。目前我正在使用卡片回退,这些卡片知道它们在父 View 层次结构中的位置,以防止错误的 child 处理触摸,但这确实是一种丑陋的方式。我的假设是我必须覆盖 dispatchTouchEvent RecyclerView 的方法,并正确地将触摸事件仅分派(dispatch)给最顶层的 child 。但是,当我尝试这样做时(这也有点笨拙):

    @Override
public boolean dispatchTouchEvent(MotionEvent ev) {
View topChild = getChildAt(0);
for (View touchable : topChild.getTouchables()) {
int[] location = new int[2];
touchable.getLocationOnScreen(location);
RectF touchableRect = new RectF(location[0],
location[1],
location[0] + touchable.getWidth(),
location[1] + touchable.getHeight());
if (touchableRect.contains(ev.getRawX(), ev.getRawY())) {
return touchable.dispatchTouchEvent(ev);
}
}
return onTouchEvent(ev);
}

只有 DOWN 事件被传送到卡片内的按钮(没有触发点击事件)。对于反转触摸事件传播顺序的方式或将触摸事件委托(delegate)给特定 View 的任何建议,我将不胜感激。非常感谢您。

编辑:这是示例卡组的截图 Screen

示例适配器代码:

public class TestAdapter extends RecyclerView.Adapter {

List<Integer> items;

public TestAdapter(List<Integer> items) {
this.items = items;
}

@Override
public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
View v = LayoutInflater.from(parent.getContext())
.inflate(R.layout.test_layout, parent, false);
return new TestHolder(v);
}

@Override
public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {
TestHolder currentHolder = (TestHolder) holder;
currentHolder.number.setText(Integer.toString(position));
currentHolder.tv.setTag(position);
currentHolder.tv.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
int pos = (int) v.getTag();
Toast.makeText(v.getContext(), Integer.toString(pos) + "clicked", Toast.LENGTH_SHORT).show();
}
});
}

@Override
public int getItemCount() {
return items.size();
}

private class TestHolder extends RecyclerView.ViewHolder {

private TextView tv;
private TextView number;

public TestHolder(View itemView) {
super(itemView);
tv = (TextView) itemView.findViewById(R.id.click);
number = (TextView) itemView.findViewById(R.id.number);
}

}
}

以及卡片布局示例:

<android.support.v7.widget.CardView xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="match_parent">
<RelativeLayout android:layout_width="match_parent"
android:layout_height="match_parent">

<TextView android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerInParent="true"
android:layout_margin="16dp"
android:textSize="24sp"
android:id="@+id/number"/>

<TextView android:layout_width="wrap_content"
android:layout_height="64dp"
android:gravity="center"
android:layout_alignParentBottom="true"
android:layout_alignParentLeft="true"
android:layout_margin="16dp"
android:textSize="24sp"
android:text="CLICK ME"
android:id="@+id/click"/>
</RelativeLayout>

</android.support.v7.widget.CardView>

这是我现在用来解决问题的代码(我不喜欢这种方法,我想找到更好的方法)

public class PositionAwareCardView extends CardView {

private int mChildPosition;

public PositionAwareCardView(Context context) {
super(context);
}

public PositionAwareCardView(Context context, AttributeSet attrs) {
super(context, attrs);
}

public PositionAwareCardView(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
}

public void setChildPosition(int pos) {
mChildPosition = pos;
}

@Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
// if it's not the topmost view in view hierarchy then block touch events
return mChildPosition != 0 || super.onInterceptTouchEvent(ev);
}

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

编辑 2:我忘了说,这个问题只出现在 Lolipop 之前的设备上,似乎从 Lolipop 开始,ViewGroups 在调度触摸事件时也考虑了高度

编辑 3:这是我当前的子绘图顺序:

@Override
protected int getChildDrawingOrder(int childCount, int i) {
return childCount - i - 1;
}

编辑 4:感谢用户 random 和 this answer,最终我能够解决问题,解决方案非常简单:

@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
if (!onInterceptTouchEvent(ev)) {
if (getChildCount() > 0) {
final float offsetX = -getChildAt(0).getLeft();
final float offsetY = -getChildAt(0).getTop();
ev.offsetLocation(offsetX, offsetY);
if (getChildAt(0).dispatchTouchEvent(ev)) {
// if touch event is consumed by the child - then just record it's coordinates
x = ev.getRawX();
y = ev.getRawY();
return true;
}
ev.offsetLocation(-offsetX, -offsetY);
}
}
return super.dispatchTouchEvent(ev);
}

最佳答案

CardView在 L 上和 L 之前使用高程 API,它会更改阴影大小。 L 中的 dispatchTouchEvent 在遍历子级时遵循 Z 顺序,这在 L 之前不会发生。

查看源代码:

前 Lollipop

ViewGroup#dispatchTouchEvent

public boolean dispatchTouchEvent(MotionEvent ev) {
final boolean customOrder = isChildrenDrawingOrderEnabled();
for (int i = childrenCount - 1; i >= 0; i--) {
final int childIndex = customOrder ?
getChildDrawingOrder(childrenCount, i) : i;
final View child = children[childIndex];

...

Lollipop

ViewGroup#dispatchTouchEvent

public boolean dispatchTouchEvent(MotionEvent ev) {
// Check whether any clickable siblings cover the child
// view and if so keep track of the intersections. Also
// respect Z ordering when iterating over children.

ArrayList<View> orderedList = buildOrderedChildList();

final boolean useCustomOrder = orderedList == null
&& isChildrenDrawingOrderEnabled();
final int childCount = mChildrenCount;
for (int i = childCount - 1; i >= 0; i--) {
final int childIndex = useCustomOrder
? getChildDrawingOrder(childCount, i) : i;
final View sibling = (orderedList == null)
? mChildren[childIndex] : orderedList.get(childIndex);

// We care only about siblings over the child
if (sibling == child) {
break;
}
...

可以用 custom child drawing order in a ViewGroup 覆盖子绘图顺序, 和 setZ(float)在 View 上设置的自定义 Z 值。

您可能想查看 custom child drawing order in a ViewGroup但我认为您目前对问题的解决方案已经足够好了。

关于android - 在重叠 View 的情况下控制触摸事件传播,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31029660/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com