Search code examples
c#wpfactualwidthactualheight

Unable to retrieve the ActualWidth / ActualHeight of ContentPresenter


In my application, we have a view model object from which we're retrieving its attached FrameworkElement using our custom function GetVisualChildWithDataContext (see below).

What's happening is that this function returns a ContentPresenter object with its ActualWidth and ActualHeight properties being 0. I would be expecting to get a value of 24 for the 2 properties since my content consists of a 24x24 rectangle.

My question is, how can I retrieve the values I'm expecting (ActualWidth and ActualHeight of 24) by only having a view model? If there's a better way of doing this, I'm all ears.

Thanks for your time. (see sample below)

xaml:

<Window x:Class="WpfApplication2.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:l="clr-namespace:WpfApplication2"
        Title="MainWindow"
        Width="525"
        Height="350">
  <Canvas x:Name="_RootCanvas">
  <ItemsControl Margin="-5,0,0,0" ItemsSource="{Binding MyItems}">
    <ItemsControl.ItemsPanel>
      <ItemsPanelTemplate>
        <Canvas />
      </ItemsPanelTemplate>
    </ItemsControl.ItemsPanel>
    <ItemsControl.ItemTemplate>
      <DataTemplate DataType="{x:Type l:MyItem}">
        <Canvas>
          <Rectangle
              Canvas.Left="{Binding ActualLeft}"
              IsHitTestVisible="True"
              Margin="0,0,0,0"
              Width="24"
              Height="24"
              Fill="Black" />
        </Canvas>
      </DataTemplate>
    </ItemsControl.ItemTemplate>
  </ItemsControl>

    <Button Canvas.Top="50" Width="100" Height="30" Click="Button_Clicked">
      <TextBlock Text="Button" />
    </Button>
  </Canvas>
</Window>

Code behind:

private MyItem _MyItem;
public MainWindow()
{
    InitializeComponent();

    _MyItem = new MyItem(0);
    MyItems = new ObservableCollection<MyItem>();
    MyItems.Add(_MyItem);

    DataContext = this;
}

public ObservableCollection<MyItem> MyItems { get; set; }

public void Button_Clicked(object sender, RoutedEventArgs e)
{
    FrameworkElement fe = GetVisualChildWithDataContext(_RootCanvas, _MyItem);
}

public static FrameworkElement GetVisualChildWithDataContext(DependencyObject parent, object dataContext)
{
    int count = VisualTreeHelper.GetChildrenCount(parent);
    for (int i = 0; i < count; i++)
    {
        DependencyObject d = VisualTreeHelper.GetChild(parent, i);
        FrameworkElement element = d as FrameworkElement;
        if (element != null)
        {
            if (element.DataContext == dataContext)
                return element;

            var f = GetVisualChildWithDataContext(element, dataContext);
            if (f != null)
                return f;
        }
    }

    return null;
}

public class MyItem
{
    public MyItem(double left)
    {
        ActualLeft = left;
    }
    public double ActualLeft { get; set; }
}

Solution

  • To fix my issue I created the following extension method:

    public static Size GetActualSize(this FrameworkElement parent)
    {
        var contentPresenter = parent as ContentPresenter;
        if (contentPresenter != null)
        {
            Rect descendantBounds = VisualTreeHelper.GetDescendantBounds(contentPresenter);
            if (!descendantBounds.Size.IsEmpty)
                return descendantBounds.Size;
        }
    
        return new Size(parent.ActualWidth, parent.ActualHeight);
    }
    

    Not pretty, but it gets the job done.