android 布局实现子View不规则区域点击

2022-09-07  本文已影响0人  黎院根

在学习不规则区域点击中看到了Android中不规则形状View的布局实现
以下是自己的学习总结:

我们知道android中View的不规则区域点击可以通过Path和Region实现,而在布局中依然可以用此方法实现,下面就是实现步骤:

1.android布局中都有drawChild方法,这是绘制每个子View的方法,而canvas可以通过clipPath来裁切不规则区域,我们通过拦截这个方法做操作:

@Override
protected boolean drawChild(Canvas canvas, View child, long drawingTime) {
    beforeChild(canvas,child,drawingTime);
    return super.drawChild(canvas, child, drawingTime);
}

Path path = new Path();
RectF bounds = new RectF();

private void beforeChild(Canvas canvas, View child, long drawingTime) {
    path.addRoundRect(new RectF(child.getLeft(),
                    child.getTop(),
                    child.getRight(),
                    child.getBottom()),
            80,
            80,
            Path.Direction.CW);
    path.computeBounds(bounds, true);
    canvas.clipPath(path);
}

2.android中dispatchTouchEvent()里的isTransformedTouchPointInView()虽然是hide但是protected方法,可以通过重写这个方法来实现View的点击区域判断:

public boolean isTransformedTouchPointInView(float x, float y, View child, PointF outLocalPoint) {
    final float[] point = getTempPoint();
    point[0] = x;
    point[1] = y;
    transformPointToViewLocal(point, child);
    boolean isInView = pointInView(child, point[0], point[1]);
    if (isInView && outLocalPoint != null) {
        outLocalPoint.set(point[0], point[1]);
    }
    return isInView;
}

getTempPoint()复制出来

private float[] mTempPoint;

    private float[] getTempPoint() {
        if (mTempPoint == null) {
            mTempPoint = new float[2];
        }
        return mTempPoint;
    }

child.pointInView这个方法不是public,我们需要去把View中的pointInView()复制出来

public boolean pointInView(View child, float localX, float localY) {
    return localX >= 0 && localY >= 0 && localX < (child.getRight() - child.getLeft()) &&
            localY < (child.getBottom() - child.getTop());
}

transformPointToViewLocal()是hide,同样复制出来

public void transformPointToViewLocal(float[] point, View child) {
    point[0] += getScrollX() - child.getLeft();
    point[1] += getScrollY() - child.getTop();
    //child.hasIdentityMatrix()和child.getInverseMatrix()用下面方法替换
    Matrix matrix = child.getMatrix();
    Matrix inverse = new Matrix();
    if (!matrix.isIdentity()) {
        matrix.invert(inverse);
        inverse.mapPoints(point);
    }
}

最后我们还需要判断是否在不规则的点击区域内

public boolean isTransformedTouchPointInView(float x, float y, View child, PointF outLocalPoint) {
    final float[] point = getTempPoint();
    point[0] = x;
    point[1] = y;
    transformPointToViewLocal(point, child);
    boolean isInView = pointInView(child, point[0], point[1]);
    if (isInView && outLocalPoint != null) {
        outLocalPoint.set(point[0], point[1]);
    }
    if (isInView) {
       //在这里判断是否在规则区域内
        Region region = new Region();
        region.setPath(path, new Region(
                (int) bounds.left,
                (int) bounds.top,
                (int) bounds.right,
                (int) bounds.bottom));
        if (!region.contains((int) x, (int) y)) {
            isInView = false;
        }
    }

    return isInView;
}

完整代码:

public class ClipFrameLayout extends FrameLayout {
    public ClipFrameLayout(@NonNull Context context) {
        super(context);
    }

    public ClipFrameLayout(@NonNull Context context, @Nullable AttributeSet attrs) {
        super(context, attrs);
    }

    @Override
    protected boolean drawChild(Canvas canvas, View child, long drawingTime) {
        beforeChild(canvas, child, drawingTime);
        return super.drawChild(canvas, child, drawingTime);
    }

    Path path = new Path();
    RectF bounds = new RectF();

    private void beforeChild(Canvas canvas, View child, long drawingTime) {
        path.addRoundRect(new RectF(child.getLeft(),
                        child.getTop(),
                        child.getRight(),
                        child.getBottom()),
                80,
                80,
                Path.Direction.CW);
        path.computeBounds(bounds, true);
        canvas.clipPath(path);
    }

    private float[] mTempPoint;

    private float[] getTempPoint() {
        if (mTempPoint == null) {
            mTempPoint = new float[2];
        }
        return mTempPoint;
    }

    public boolean isTransformedTouchPointInView(float x, float y, View child, PointF outLocalPoint) {
        final float[] point = getTempPoint();
        point[0] = x;
        point[1] = y;
        transformPointToViewLocal(point, child);
        boolean isInView = pointInView(child, point[0], point[1]);
        if (isInView && outLocalPoint != null) {
            outLocalPoint.set(point[0], point[1]);
        }
        if (isInView) {
            Region region = new Region();
            region.setPath(path, new Region(
                    (int) bounds.left,
                    (int) bounds.top,
                    (int) bounds.right,
                    (int) bounds.bottom));
            if (!region.contains((int) x, (int) y)) {
                isInView = false;
            }

        }

        return isInView;
    }

    public boolean pointInView(View child, float localX, float localY) {
        return localX >= 0 && localY >= 0 && localX < (child.getRight() - child.getLeft()) &&
                localY < (child.getBottom() - child.getTop());
    }

    public void transformPointToViewLocal(float[] point, View child) {
        point[0] += getScrollX() - child.getLeft();
        point[1] += getScrollY() - child.getTop();

        Matrix matrix = child.getMatrix();
        Matrix inverse = new Matrix();
        if (!matrix.isIdentity()) {
            matrix.invert(inverse);
            inverse.mapPoints(point);
        }
    }

}
上一篇 下一篇

猜你喜欢

热点阅读