How do I use WPF bindings with RelativeSource?

前端 未结 13 1974
执念已碎
执念已碎 2020-11-22 03:00

How do I use RelativeSource with WPF bindings and what are the different use-cases?

相关标签:
13条回答
  • 2020-11-22 03:31

    Here's a more visual explanation in the context of a MVVM architecture:

    enter image description here

    0 讨论(0)
  • 2020-11-22 03:32

    In WPF RelativeSource binding exposes three properties to set:

    1. Mode: This is an enum that could have four values:

    a. PreviousData(value=0): It assigns the previous value of the property to the bound one

    b. TemplatedParent(value=1): This is used when defining the templates of any control and want to bind to a value/Property of the control.

    For example, define ControlTemplate:

      <ControlTemplate>
            <CheckBox IsChecked="{Binding RelativeSource={RelativeSource TemplatedParent}, Path=Value, Mode=TwoWay, UpdateSourceTrigger=PropertyChanged}" />
     </ControlTemplate>
    

    c. Self(value=2): When we want to bind from a self or a property of self.

    For example: Send checked state of checkbox as CommandParameter while setting the Command on CheckBox

    <CheckBox ...... CommandParameter="{Binding RelativeSource={RelativeSource Self},Path=IsChecked}" />
    

    d. FindAncestor(value=3): When want to bind from a parent control in Visual Tree.

    For example: Bind a checkbox in records if a grid,if header checkbox is checked

    <CheckBox IsChecked="{Binding RelativeSource={RelativeSource Mode=FindAncestor, AncestorType={x:Type iDP:XamDataGrid}}, Path=DataContext.IsHeaderChecked, Mode=TwoWay}" />
    

    2. AncestorType: when mode is FindAncestor then define what type of ancestor

    RelativeSource={RelativeSource Mode=FindAncestor, AncestorType={x:Type iDP:XamDataGrid}}
    

    3. AncestorLevel: when mode is FindAncestor then what level of ancestor (if there are two same type of parent in visual tree)

    RelativeSource={RelativeSource Mode=FindAncestor, AncestorType={x:Type iDP:XamDataGrid, AncestorLevel=1}}
    

    Above are all use-cases for RelativeSource binding.

    Here is a reference link.

    0 讨论(0)
  • 2020-11-22 03:32

    Don't forget TemplatedParent:

    <Binding RelativeSource="{RelativeSource TemplatedParent}"/>
    

    or

    {Binding RelativeSource={RelativeSource TemplatedParent}}
    
    0 讨论(0)
  • 2020-11-22 03:34

    Bechir Bejaoui exposes the use cases of the RelativeSources in WPF in his article here:

    The RelativeSource is a markup extension that is used in particular binding cases when we try to bind a property of a given object to another property of the object itself, when we try to bind a property of a object to another one of its relative parents, when binding a dependency property value to a piece of XAML in case of custom control development and finally in case of using a differential of a series of a bound data. All of those situations are expressed as relative source modes. I will expose all of those cases one by one.

    1. Mode Self:

    Imagine this case, a rectangle that we want that its height is always equal to its width, a square let's say. We can do this using the element name

    <Rectangle Fill="Red" Name="rectangle" 
                    Height="100" Stroke="Black" 
                    Canvas.Top="100" Canvas.Left="100"
                    Width="{Binding ElementName=rectangle,
                    Path=Height}"/>
    

    But in this above case we are obliged to indicate the name of the binding object, namely the rectangle. We can reach the same purpose differently using the RelativeSource

    <Rectangle Fill="Red" Height="100" 
                   Stroke="Black" 
                   Width="{Binding RelativeSource={RelativeSource Self},
                   Path=Height}"/>
    

    For that case we are not obliged to mention the name of the binding object and the Width will be always equal to the Height whenever the height is changed.

    If you want to parameter the Width to be the half of the height then you can do this by adding a converter to the Binding markup extension. Let's imagine another case now:

     <TextBlock Width="{Binding RelativeSource={RelativeSource Self},
                   Path=Parent.ActualWidth}"/>
    

    The above case is used to tie a given property of a given element to one of its direct parent ones as this element holds a property that is called Parent. This leads us to another relative source mode which is the FindAncestor one.

    1. Mode FindAncestor

    In this case, a property of a given element will be tied to one of its parents, Of Corse. The main difference with the above case is the fact that, it's up to you to determine the ancestor type and the ancestor rank in the hierarchy to tie the property. By the way try to play with this piece of XAML

    <Canvas Name="Parent0">
        <Border Name="Parent1"
                 Width="{Binding RelativeSource={RelativeSource Self},
                 Path=Parent.ActualWidth}"
                 Height="{Binding RelativeSource={RelativeSource Self},
                 Path=Parent.ActualHeight}">
            <Canvas Name="Parent2">
                <Border Name="Parent3"
                Width="{Binding RelativeSource={RelativeSource Self},
               Path=Parent.ActualWidth}"
               Height="{Binding RelativeSource={RelativeSource Self},
                  Path=Parent.ActualHeight}">
                   <Canvas Name="Parent4">
                   <TextBlock FontSize="16" 
                   Margin="5" Text="Display the name of the ancestor"/>
                   <TextBlock FontSize="16" 
                     Margin="50" 
                Text="{Binding RelativeSource={RelativeSource  
                           FindAncestor,
                           AncestorType={x:Type Border}, 
                           AncestorLevel=2},Path=Name}" 
                           Width="200"/>
                    </Canvas>
                </Border>
            </Canvas>
         </Border>
       </Canvas>
    

    The above situation is of two TextBlock elements those are embedded within a series of borders and canvas elements those represent their hierarchical parents. The second TextBlock will display the name of the given parent at the relative source level.

    So try to change AncestorLevel=2 to AncestorLevel=1 and see what happens. Then try to change the type of the ancestor from AncestorType=Border to AncestorType=Canvas and see what's happens.

    The displayed text will change according to the Ancestor type and level. Then what's happen if the ancestor level is not suitable to the ancestor type? This is a good question, I know that you're about to ask it. The response is no exceptions will be thrown and nothings will be displayed at the TextBlock level.

    1. TemplatedParent

    This mode enables tie a given ControlTemplate property to a property of the control that the ControlTemplate is applied to. To well understand the issue here is an example bellow

    <Window.Resources>
    <ControlTemplate x:Key="template">
            <Canvas>
                <Canvas.RenderTransform>
                    <RotateTransform Angle="20"/>
                    </Canvas.RenderTransform>
                <Ellipse Height="100" Width="150" 
                     Fill="{Binding 
                RelativeSource={RelativeSource TemplatedParent},
                Path=Background}">
    
                  </Ellipse>
                <ContentPresenter Margin="35" 
                      Content="{Binding RelativeSource={RelativeSource  
                      TemplatedParent},Path=Content}"/>
            </Canvas>
        </ControlTemplate>
    </Window.Resources>
        <Canvas Name="Parent0">
        <Button   Margin="50" 
                  Template="{StaticResource template}" Height="0" 
                  Canvas.Left="0" Canvas.Top="0" Width="0">
            <TextBlock FontSize="22">Click me</TextBlock>
        </Button>
     </Canvas>
    

    If I want to apply the properties of a given control to its control template then I can use the TemplatedParent mode. There is also a similar one to this markup extension which is the TemplateBinding which is a kind of short hand of the first one, but the TemplateBinding is evaluated at compile time at the contrast of the TemplatedParent which is evaluated just after the first run time. As you can remark in the bellow figure, the background and the content are applied from within the button to the control template.

    0 讨论(0)
  • 2020-11-22 03:34

    I didn't read every answer, but I just want to add this information in case of relative source command binding of a button.

    When you use a relative source with Mode=FindAncestor, the binding must be like:

    Command="{Binding Path=DataContext.CommandProperty, RelativeSource={...}}"
    

    If you don't add DataContext in your path, at execution time it can't retrieve the property.

    0 讨论(0)
  • 2020-11-22 03:35

    If an element is not part of the visual tree, then RelativeSource will never work.

    In this case, you need to try a different technique, pioneered by Thomas Levesque.

    He has the solution on his blog under [WPF] How to bind to data when the DataContext is not inherited. And it works absolutely brilliantly!

    In the unlikely event that his blog is down, Appendix A contains a mirror copy of his article.

    Please do not comment here, please comment directly on his blog post.

    Appendix A: Mirror of blog post

    The DataContext property in WPF is extremely handy, because it is automatically inherited by all children of the element where you assign it; therefore you don’t need to set it again on each element you want to bind. However, in some cases the DataContext is not accessible: it happens for elements that are not part of the visual or logical tree. It can be very difficult then to bind a property on those elements…

    Let’s illustrate with a simple example: we want to display a list of products in a DataGrid. In the grid, we want to be able to show or hide the Price column, based on the value of a ShowPrice property exposed by the ViewModel. The obvious approach is to bind the Visibility of the column to the ShowPrice property:

    <DataGridTextColumn Header="Price" Binding="{Binding Price}" IsReadOnly="False"
                    Visibility="{Binding ShowPrice,
                    Converter={StaticResource visibilityConverter}}"/>
    

    Unfortunately, changing the value of ShowPrice has no effect, and the column is always visible… why? If we look at the Output window in Visual Studio, we notice the following line:

    System.Windows.Data Error: 2 : Cannot find governing FrameworkElement or FrameworkContentElement for target element. BindingExpression:Path=ShowPrice; DataItem=null; target element is ‘DataGridTextColumn’ (HashCode=32685253); target property is ‘Visibility’ (type ‘Visibility’)

    The message is rather cryptic, but the meaning is actually quite simple: WPF doesn’t know which FrameworkElement to use to get the DataContext, because the column doesn’t belong to the visual or logical tree of the DataGrid.

    We can try to tweak the binding to get the desired result, for instance by setting the RelativeSource to the DataGrid itself:

    <DataGridTextColumn Header="Price" Binding="{Binding Price}" IsReadOnly="False"
                    Visibility="{Binding DataContext.ShowPrice,
                    Converter={StaticResource visibilityConverter},
                    RelativeSource={RelativeSource FindAncestor, AncestorType=DataGrid}}"/>
    

    Or we can add a CheckBox bound to ShowPrice, and try to bind the column visibility to the IsChecked property by specifying the element name:

    <DataGridTextColumn Header="Price" Binding="{Binding Price}" IsReadOnly="False"
                    Visibility="{Binding IsChecked,
                    Converter={StaticResource visibilityConverter},
                    ElementName=chkShowPrice}"/>
    

    But none of these workarounds seems to work, we always get the same result…

    At this point, it seems that the only viable approach would be to change the column visibility in code-behind, which we usually prefer to avoid when using the MVVM pattern… But I’m not going to give up so soon, at least not while there are other options to consider

    0 讨论(0)
提交回复
热议问题