自定义 ViewGroup 中的视图在大小更改后未呈现

Posted

技术标签:

【中文标题】自定义 ViewGroup 中的视图在大小更改后未呈现【英文标题】:Views inside a custom ViewGroup not rendering after a size change 【发布时间】:2011-08-16 16:51:14 【问题描述】:

我遇到了一个让我很困惑的问题,我希望有人能给我一些建议。

我正在开发一个使用自定义 ViewGroup(实际上是包含 RelativeLayout 的 FrameLayout)来呈现事件日历的应用程序。日历中的事件表示为根据事件的持续时间和包含视图的大小调整大小的视图。

我遇到了在调整包含 FrameLayout 大小时出现的问题。当前的实现删除了所有表示事件的视图,并尝试添加新的视图并根据 FrameLayout 的当前大小计算它们的大小。这项工作是通过 View 的 onSizeChanged() 方法触发的,该方法在 FrameLayout 中被覆盖。

当调整视图大小时,会执行此代码并更新视图,但是,它们都不会真正呈现在屏幕上...... FrameLayout 中包含的视图根本不可见。如果我在层次结构查看器工具中加载视图,它们是视图树的一部分,并在概述中它们应该位于的位置上进行了概述,但它们不显示。 (请注意,视图在 FrameLayout 的初始渲染中可见......只有在调整大小后它们才会消失。)

看来resize期间的事件顺序如下:

onMeasure()
onMeasure()
onSizeChanged()
onLayout()

重置视图(在 onSizeChanged() 内部)后调用 requestLayout() 似乎没有效果。但是,如果我在调用 requestLayout() 之前造成一些延迟,视图就会变得可见。我可以通过产生一个线程并休眠,或者通过创建一个简单地调用 requestLayout() 并在自己调整大小后按下它的虚拟按钮,或者甚至是放在 onSizeChanged() 末尾的这个丑陋的黑客来导致这种延迟:

post(new Runnable() 
    public void run() 
        requestLayout();
    
);

当我使用这个 hack 时,包含的视图是可见的,并且事件的顺序如下:

onMeasure()
onMeasure()
onSizeChanged()
onLayout()
onMeasure()
onMeasure()
onLayout()

因此,似乎强制第二次测量通过(在修改视图树之后)会使包含的视图按应有的方式可见。为什么延迟对 requestLayout() 的调用对我来说是个谜。

谁能指出我做错了什么?

我意识到如果不看一些代码就很难理解,所以我创建了一个小示例应用程序来展示我的问题并在 Github 上提供:

https://github.com/MichaelSims/ViewGroupResizeTest

我上面提到的 hack 提交到一个单独的分支:

https://github.com/MichaelSims/ViewGroupResizeTest/tree/post-runnable-hack

如果我可以提供任何其他信息,请提前告诉我并感谢。

【问题讨论】:

【参考方案1】:

这不是 hack,它应该是这样工作的。 onSizeChanged() 作为布局传递的一部分被调用,您不能/不应该在布局传递期间 requestLayout()。在事件队列上发布 requestLayout() 以表明您在布局传递期间更改了 View 层次结构是正确的。

【讨论】:

太棒了!非常感谢您花时间回答。 当他在事件队列上发布 requestLayout() 时,为什么 onMeasure() 会跟随 onLayout() 而不是 onDraw()? 该死,我一直在寻找这个解释几个小时。如果您使用 addViewInLayout 开发类似于 ListView 的自定义布局,请记住此功能 - 您必须通过 Runnalbe 发布 requestLayout 事件以在添加所有视图后刷新度量【参考方案2】:

我遇到了同样的问题。

我的 onMeasure() 是这样的:

@Override    
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) 
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        ...
        //calculate new width and height here
        ...
        setMeasuredDimension(newMeasureSpecWidth, newMeasureSpecHeight);
    

我的错误是我在 onMeasure() 的第一行调用了 super.onMeasure(),所以我的 ViewGroup 的内部子级是根据即将改变的大小来计算的。

所以我的修复方法是这样的:

@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) 
    ...
    //calculate new width and height here
    ...
    int newMeasureSpecWidth = MeasureSpec.makeMeasureSpec(newWidth, MeasureSpec.EXACTLY);
    int newMeasureSpecHeight = MeasureSpec.makeMeasureSpec(newHeight, MeasureSpec.EXACTLY);
    super.onMeasure(newMeasureSpecWidth, newMeasureSpecHeight);

所以我通过调用 super.onMeasure() 为我的 ViewGroup 设置新大小,然后它也将新大小传达给它的孩子。

记住:当你重写 onMeasure() 时的约定是你必须调用 setMeasuredDimension() (你可以通过调用方法本身或调用 super.onMeasure() 来实现)

【讨论】:

以上是关于自定义 ViewGroup 中的视图在大小更改后未呈现的主要内容,如果未能解决你的问题,请参考以下文章

UITableViewCell 在可见路径重新加载行后未调整大小

自定义 UI 视图更改帧大小

自定义 ViewGroup 示例?

Android技术分享| 自定义ViewGroup实现直播间大小屏无缝切换

在使线性布局容器无效后未调用子视图的 ondraw()

Android技术分享| 自定义ViewGroup实现直播间大小屏无缝切换