I have written a little programme to retrieve offset value of child element relative to its parent element using TransformToAncestor and VisualTreeHelper.GetOffSet(), but one stunning discovery I made is that both two methods return different value If I have a RenderTransform applied to the child element, here comes the code:
Window1.xaml:
<Window x:Class="TransformToAncestorDemo.Window1"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="TransformationDemo" Height="300" Width="600"
>
<StackPanel Margin="16" Background="Green">
<StackPanel Margin="8" Background="Yellow" Name="container">
<Button Margin="16" Content="Calculate Button Offset" Click="CalculateButtonOffset" Height="30"/>
<Button Margin="16" Content="Calculate Button Offset With RenderTransform" Click="CalculateButtonOffset" Height="30">
<Button.RenderTransform>
<RotateTransform CenterX="0" CenterY="0" Angle="10"/>
</Button.RenderTransform>
</Button>
</StackPanel>
</StackPanel>
</Window>
Window1.xaml.cs:
using System;
using System.Collections.Generic;
using System.Text;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Shapes;
namespace TransformToAncestorDemo
{
/// <summary>
/// Interaction logic for Window1.xaml
/// </summary>
public partial class Window1 : Window
{
public Window1()
{
InitializeComponent();
}
private void CalculateButtonOffset(Object sender, RoutedEventArgs e)
{
Button button = sender as Button;
Point offset = button.TransformToAncestor(container).Transform(new Point(0, 0));
StringBuilder sb = new StringBuilder();
sb.AppendFormat("TransformToAncestor: X={0} Y={0}\n", offset.X, offset.Y);
Vector offsetVector = VisualTreeHelper.GetOffset(button);
sb.AppendFormat("VisualTreeHelper.GetOffSet(): X={0} Y={0}", offsetVector.X, offsetVector.Y);
MessageBox.Show(sb.ToString());
}
}
}
when I calculate offset without RenderTransform applied to the Button element, the result is:
TransformToAncestor: X=16 Y=16
VisualTreeHelper.GetOffSet(): X=16 Y=16
and when the RenderTransform applied, the result is:
TransformToAncestor: X=16 Y=16
VisualTreeHelper.GetOffSet(): X=0 Y=0
So I get confused here, I thought both two methods should return the same value no matter whether or not RenderTransform is applied, or this behaviour is expected, can someone explain this to me?
Thanks in advance
Sheva
Window1.xaml:
<Window x:Class="TransformToAncestorDemo.Window1"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="TransformationDemo" Height="300" Width="600"
>
<StackPanel Margin="16" Background="Green">
<StackPanel Margin="8" Background="Yellow" Name="container">
<Button Margin="16" Content="Calculate Button Offset" Click="CalculateButtonOffset" Height="30"/>
<Button Margin="16" Content="Calculate Button Offset With RenderTransform" Click="CalculateButtonOffset" Height="30">
<Button.RenderTransform>
<RotateTransform CenterX="0" CenterY="0" Angle="10"/>
</Button.RenderTransform>
</Button>
</StackPanel>
</StackPanel>
</Window>
Window1.xaml.cs:
using System;
using System.Collections.Generic;
using System.Text;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Shapes;
namespace TransformToAncestorDemo
{
/// <summary>
/// Interaction logic for Window1.xaml
/// </summary>
public partial class Window1 : Window
{
public Window1()
{
InitializeComponent();
}
private void CalculateButtonOffset(Object sender, RoutedEventArgs e)
{
Button button = sender as Button;
Point offset = button.TransformToAncestor(container).Transform(new Point(0, 0));
StringBuilder sb = new StringBuilder();
sb.AppendFormat("TransformToAncestor: X={0} Y={0}\n", offset.X, offset.Y);
Vector offsetVector = VisualTreeHelper.GetOffset(button);
sb.AppendFormat("VisualTreeHelper.GetOffSet(): X={0} Y={0}", offsetVector.X, offsetVector.Y);
MessageBox.Show(sb.ToString());
}
}
}
when I calculate offset without RenderTransform applied to the Button element, the result is:
TransformToAncestor: X=16 Y=16
VisualTreeHelper.GetOffSet(): X=16 Y=16
and when the RenderTransform applied, the result is:
TransformToAncestor: X=16 Y=16
VisualTreeHelper.GetOffSet(): X=0 Y=0
So I get confused here, I thought both two methods should return the same value no matter whether or not RenderTransform is applied, or this behaviour is expected, can someone explain this to me?
Thanks in advance
Sheva