如何在多个Windows / Pages / Controls上设置DataContext?

时间:2013-10-12 14:52:21

标签: c# wpf binding datacontext

我有一个关于WPF MVVM如何工作并且有工作代码但不确定它为何起作用的问题。大多数在线教程似乎使用单个窗口提供示例,因此我不确定我是否正确地使用多个窗口/页面/用户控件进行此操作。

如果我有一个名为ViewModel的班级,并且我使用下面的代码在DataContext中设置了MainWindow,那么我设置了DataContext MainWindow只有。

MainWindow.xaml.cs:

public partial class MainWindow

{
    Private ViewModel viewModel = new ViewModel();

    public MainWindow()
    {
       InitializeComponent();
       this.DataContext = this.viewModel;
    }

}

如果我然后创建一个新的usercontrol,然后在没有指定viewModel路径的情况下绑定DataGrid,为什么在我没有设置usercontrol的DataContext时,下面的代码会起作用?

这是WPF的工作原理还是我应该在usercontrol中设置DataContext?这样做的正确方法是什么?

MainSignals.xaml:

<UserControl x:Class="ProjectXYZ.Content.MainSignals"
             xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
             xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
             xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" 
             xmlns:d="http://schemas.microsoft.com/expression/blend/2008" 
             xmlns:core="clr-namespace:System;assembly=mscorlib"
             xmlns:local="clr-namespace:ProjectXYZ.Content"
             xmlns:mui="http://firstfloorsoftware.com/ModernUI"
             mc:Ignorable="d" 
             d:DesignHeight="300" d:DesignWidth="300" >
    <Grid>
        <DockPanel>
            <DataGrid Name="DG1" ItemsSource="{Binding ReceivedSignals}"  >
                <DataGrid.Columns>
                    <mui:DataGridTextColumn Header="SignalID"  Binding="{Binding signalID}"/>
            <mui:DataGridTextColumn Header="SignalType"  Binding="{Binding signalType}"/>
                </DataGrid.Columns>
            </DataGrid>
        </DockPanel>
    </Grid>
</UserControl>

ViewModel.cs:

private ObservableCollection<MainWindow.SignalVar> _receivedSignals;

Public ViewModel()
{
}

public event PropertyChangedEventHandler PropertyChanged;


// Create the OnPropertyChanged method to raise the event
protected void OnPropertyChanged(string name)
{
    PropertyChangedEventHandler handler = PropertyChanged;
    if (handler != null)
    {
        handler(this, new PropertyChangedEventArgs(name));
    }
}

public ObservableCollection<MainWindow.SignalVar> ReceivedSignals
{
    get { return _receivedSignals; }
    set
    {
        if (value != _receivedSignals)
        {
            _receivedSignals = value;
            OnPropertyChanged("ReceivedSignals");
        }
    }
}

UserControl.xaml.cs:

public partial class MainSignals : UserControl
{

    public MainSignals()
    {
        InitializeComponent();
        //this.DataContext = new ViewModel();  //WORKS WITHOUT THIS??
    }

}

1 个答案:

答案 0 :(得分:2)

这是因为如果未明确设置DataContext,则子控件会继承其父级的DataContext。某些DependancyProperties也是如此,例如,如果设置父控件的前景,则所有子控件都继承相同的Foreground属性值。

在您的情况下,由于您没有为子UserControl显式设置DataContext,因此它将使用其父级的DataContext,这是您的Window。