WPF: MeasureOverride with side effects?

344 Views Asked by At

The Situation:

I am trying to create a panel class StickyInfoPanel for two child items laid out vertically. The lower of the two items shall be 100px in height. The upper item shall consume the rest of the total available height.

The upper item is a ScrollView with a stackpanel and two children inside (upper1 and upper2). So whenever there is not enough room for the stackpanel, the vertical scrollbar should appear, like in the image below:

enter image description here

The Problem:

Although the correct height is passed to the upper item during the Arrange phase, its resulting height is higher, and in consequence the scrollbar is not displayed. (See 2nd screenshot)

Only when the window will be decreased further, so that only upper1 can be displayed, the scrollbar appears. But its down-button is still gone (see 3rd screenshot)

Strangely, when passing the correct desired height to the item from MeasureOverride, everything works as expected.

From my understanding, MeasureOverride should be side-effect-free, which it obviously is not. Can anyone explain, what I am missing here?

enter image description here

enter image description here

The XAML:

<Window x:Class="GridTest.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:local="clr-namespace:GridTest"
        Title="MainWindow" Height="190.57" Width="800">
    <local:StickyInfoPanel VerticalAlignment="Stretch" HorizontalAlignment="Stretch" >
        <ScrollViewer Background="LightYellow" VerticalScrollBarVisibility="Auto" CanContentScroll="false">
            <StackPanel Margin="30 0" CanVerticallyScroll="False">
                <TextBlock Background="AliceBlue" Height="100">upper1</TextBlock>
                <TextBlock Background="Azure" Height="100">upper2</TextBlock>
            </StackPanel>
        </ScrollViewer>
        <TextBlock Background="Gainsboro" Height="100">Lower</TextBlock>
    </local:StickyInfoPanel>
</Window>

The panel class:

class StickyInfoPanel : Panel
{
    public StickyInfoPanel()
        : base()
    {
    }
    protected override Size MeasureOverride(Size availableSize)
    {
        InternalChildren[0].Measure(availableSize);
        InternalChildren[1].Measure(availableSize);
        //this works:
        //InternalChildren[0].Measure(new Size(availableSize.Width, availableSize.Height - 100));
        return availableSize;
    }
    protected override Size ArrangeOverride(Size finalSize)
    {
        InternalChildren[0].Arrange(new Rect(new Point(0, 0),                      new Size(finalSize.Width, finalSize.Height - 100)));
        InternalChildren[1].Arrange(new Rect(new Point(0, finalSize.Height - 100), new Size(finalSize.Width, 100)));
        return finalSize; 
    }
}
1

There are 1 best solutions below

1
On

So after some measurements, here is the final formula for resulting child element sizes inside custom panels (min and max operators work element-wise here):

actualSize = max( min( availableSize , desiredSizeTheoretical), finalSize )

Where

  • actualSize is the resulting size of the child element
  • availableSize is the size that the panel passed to the element's Measure method from its own MeasureOverride
  • desiredSizeTheoretical is the size the child element would want to occupy if there was infinite space available
  • finalSize is the size that the panel passed to the child element's Arrange method from its own ArrangeOverride

Or put in words: The resulting size will always be finalSize, unless the child element wanted more space and was promised more space in MeasureOverride. In that case the resulting size would be the smaller of those two values.

This has been found for a ScrollViewer as child element. I hope, other classes will behave the same.