android - 在自定义 ViewGroup 子类中调整旋转 subview 的大小

标签 android rotation resize subclass mongolian-vertical-script

背景

我有一个自定义的ViewGroup子类,它可以旋转并镜像其 subview 。这样做的目的是正确地display traditional Mongolian text .

我可以在这个ViewGroup中放置任何东西,但对于我当前的项目,我在其中放置一个EditText。 (I was never successful 只是直接旋转和镜像 EditText。但是,将其包装在此自定义 View 组中确实有效。)

问题

我的问题是,当我尝试以编程方式调整 ViewGroup 的大小时,它的 subview 没有随之正确调整大小。我希望 EditText 与父 ViewGroup 的大小相匹配,以便它看起来是一个 View 。

MCVE

我做了一个新项目来展示这个问题。该按钮会增加 ViewGroup 的宽度(以红色显示)。图像显示项目开始(一切正常)和两个宽度增量。 EditText 为白色,即使宽度和高度设置为 match_parent

,也不会调整大小

enter image description here

完整的项目代码如下。

MongolViewGroup.java(旋转和镜像其内容的自定义 ViewGroup)

public class MongolViewGroup extends ViewGroup {

    private int angle = 90;
    private final Matrix rotateMatrix = new Matrix();
    private final Rect viewRectRotated = new Rect();
    private final RectF tempRectF1 = new RectF();
    private final RectF tempRectF2 = new RectF();
    private final float[] viewTouchPoint = new float[2];
    private final float[] childTouchPoint = new float[2];
    private boolean angleChanged = true;

    public MongolViewGroup(Context context) {
        this(context, null);
    }

    public MongolViewGroup(Context context, AttributeSet attrs) {
        super(context, attrs);
        setWillNotDraw(false);
    }

    public View getView() {
        return getChildAt(0);
    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        final View view = getView();
        if (view != null) {
            measureChild(view, heightMeasureSpec, widthMeasureSpec);
            setMeasuredDimension(resolveSize(view.getMeasuredHeight(), widthMeasureSpec),
                    resolveSize(view.getMeasuredWidth(), heightMeasureSpec));
        } else {
            super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        }
    }

    @Override
    protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
        if (angleChanged) {
            final RectF layoutRect = tempRectF1;
            final RectF layoutRectRotated = tempRectF2;
            layoutRect.set(0, 0, right - left, bottom - top);
            rotateMatrix.setRotate(angle, layoutRect.centerX(), layoutRect.centerY());
            rotateMatrix.postScale(-1, 1);
            rotateMatrix.mapRect(layoutRectRotated, layoutRect);
            layoutRectRotated.round(viewRectRotated);
            angleChanged = false;
        }
        final View view = getView();
        if (view != null) {
            view.layout(viewRectRotated.left, viewRectRotated.top, viewRectRotated.right,
                    viewRectRotated.bottom);
        }
    }


    @Override
    protected void dispatchDraw(Canvas canvas) {

        canvas.save();
        canvas.rotate(-angle, getWidth() / 2f, getHeight() / 2f);
        canvas.scale(-1, 1);
        super.dispatchDraw(canvas);
        canvas.restore();
    }

    @Override
    public ViewParent invalidateChildInParent(int[] location, Rect dirty) {
        invalidate();
        return super.invalidateChildInParent(location, dirty);
    }

    @Override
    public boolean dispatchTouchEvent(MotionEvent event) {
        viewTouchPoint[0] = event.getX();
        viewTouchPoint[1] = event.getY();
        rotateMatrix.mapPoints(childTouchPoint, viewTouchPoint);
        event.setLocation(childTouchPoint[0], childTouchPoint[1]);
        boolean result = super.dispatchTouchEvent(event);
        event.setLocation(viewTouchPoint[0], viewTouchPoint[1]);
        return result;
    }

}

MainActivity.java

public class MainActivity extends AppCompatActivity {

    MongolViewGroup viewGroup;
    EditText editText;
    int newWidth = 300;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        viewGroup = (MongolViewGroup) findViewById(R.id.viewGroup);
        editText = (EditText) findViewById(R.id.editText);
    }

    public void buttonClicked(View view) {

        newWidth += 200;
        ViewGroup.LayoutParams params = viewGroup.getLayoutParams();
        params.width=newWidth;
        viewGroup.setLayoutParams(params);
    }
}

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/activity_main"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    tools:context="com.example.mongolviewgrouptest.MainActivity">

    <com.example.mongolviewgrouptest.MongolViewGroup
        android:id="@+id/viewGroup"
        android:layout_width="100dp"
        android:layout_height="200dp"
        android:background="@color/colorAccent">

        <EditText
            android:id="@+id/editText"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:textColor="@android:color/black"
            android:background="@android:color/white"/>

    </com.example.mongolviewgrouptest.MongolViewGroup>

    <Button
        android:text="Button"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:id="@+id/button"
        android:onClick="buttonClicked"
        android:layout_alignParentTop="true"
        android:layout_alignParentRight="true"
        android:layout_alignParentEnd="true"/>


</RelativeLayout>

最佳答案

  • 当 ViewGroup 的 onLayout(...) 方法再次被调用。
  • 由于在 ViewGroups 第一次布局后 angleChanged 设置为 false(并且永远不会更改),因此计算 leftright 的部分、 EditText 的 topbottom 值 在您的 ViewGroup 第一次出现后的任何时间都会被跳过 requestsLayout(当您更改其高度或宽度时)。
  • 因此,您的 EditText 仍采用相同的 leftrighttop 布局 和最初布局的 bottom 值。

去掉angleChanged,它应该可以正常工作。就像这样:

@Override
protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
    final RectF layoutRect = tempRectF1;
    final RectF layoutRectRotated = tempRectF2;
    layoutRect.set(0, 0, right - left, bottom - top);
    rotateMatrix.setRotate(angle, layoutRect.centerX(), layoutRect.centerY());
    rotateMatrix.postScale(-1, 1);
    rotateMatrix.mapRect(layoutRectRotated, layoutRect);
    layoutRectRotated.round(viewRectRotated);
    final View view = getView();
    if (view != null) {
        view.layout(viewRectRotated.left, viewRectRotated.top, viewRectRotated.right,
                viewRectRotated.bottom);
    }
}

我已经测试过了,这样效果很好。

如果您出于任何原因需要 angleChanged,那么只需确保在 ViewGroup 的 onMeasure 方法中将其更改回 true 即可,以便 viewRectRotated 再次重新计算。不过我不建议这样做。

关于android - 在自定义 ViewGroup 子类中调整旋转 subview 的大小,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40484871/

相关文章:

java - 当被测类不是 Activity 时获取用于 AndroidTestCase 的上下文

android - 异常 : 9003: PLACES_API_ACCESS_NOT_CONFIGURED

opencv - `cv2.getRotationMatrix2D`返回的 map 矩阵

javascript - 带有 URL 链接且不重复的旋转横幅

java - 如何随着窗口大小的调整而调整 JLabel 中图像的大小?如何保持适当的规模?

Android ImageView调整父级高度和适配宽度

android - LinearLayout OnClickListener 没有响应

android - 如何判断安卓设备是否有硬键

java - 旋转 3D 人物

jquery - 自动调整内容大小 - 一页网站