7

メイン ウィンドウで にバインドしようとしましたが、代わりにboolカスタム コントロールを参照しています。DataContextユーザー コントロールで割り当てない場合DataContext、メイン ウィンドウのバインドは機能しますが、(明らかに) これにより、ユーザー コントロールのバインドが中断されます。

エラーは次のとおりです。

System.Windows.Data エラー: 40: BindingExpression パス エラー: 'MyControlVisible' プロパティが 'object' ''MyUserControlModel' (HashCode=1453241)' に見つかりません。BindingExpression:Path=MyControlVisible; DataItem='MyUserControlModel' (HashCode=1453241); ターゲット要素は 'MyUserControl' (Name='_myUserControl') です。ターゲット プロパティは「Visibility」(タイプ「Visibility」)

DataContext両方のコントロールで動作するようにバインドする必要がありますが、ユーザー コントロールがウィンドウに取って代わることは望ましくありません。

コードは次のとおりです。

<Window x:Class="Sandbox.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:Controls="clr-namespace:Sandbox.Controls" Title="Sandbox">
    <DockPanel LastChildFill="True">
        <DockPanel.Resources>
            <BooleanToVisibilityConverter x:Key="boolToVis" />
        </DockPanel.Resources>
        <Grid>
            <Controls:MyUserControl x:Name="_myUserControl" Visibility="{Binding MyControlVisible, Converter={StaticResource boolToVis}}"/>
        </Grid>
    </DockPanel>
</Window>

namespace Sandbox
{
    public partial class MainWindow
    {
        private MainWindowModel model;
        public MainWindow()
        {
            InitializeComponent();
            DataContext = model = new MainWindowModel();
            _myUserControl.Initialize(model.MyUControlModel);
        }
    }
}

using System.ComponentModel;
using Sandbox.Controls;

namespace Sandbox
{
    public class MainWindowModel : BaseModel
    {
        public MyUserControlModel MyUControlModel { get; set; }
        public bool MyControlVisible { get; set; }
        public MainWindowModel()
        {
            MyUControlModel = new MyUserControlModel();
            MyControlVisible = false;
            OnChange("");
        }
    }

    public class BaseModel : INotifyPropertyChanged
    {
        public event PropertyChangedEventHandler PropertyChanged;
        protected void OnChange(string s)
        {
            var handler = PropertyChanged;
            if (handler != null)
            {
                handler(this, new PropertyChangedEventArgs(s));
            }
        }
    }
}

<UserControl x:Class="Sandbox.Controls.MyUserControl"
             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" 
             mc:Ignorable="d">
    <Grid>
        <TextBlock Text="{Binding MyBoundText}"/>
    </Grid>
</UserControl>

namespace Sandbox.Controls
{
    public partial class MyUserControl
    {
        public MyUserControl()
        {
            InitializeComponent();
        }

        public void Initialize(MyUserControlModel context)
        {
            DataContext = context;
        }
    }

}

namespace Sandbox.Controls
{
    public class MyUserControlModel : BaseModel
    {
        public string MyBoundText { get; set; }
        public MyUserControlModel()
        {
            MyBoundText = "Hello World!";
            OnChange("");
        }
    }
}
4

1 に答える 1

5

DataContextこれは、 をそれ自体から直接設定してはならない多くの理由の 1 つですUserControl

DataContextこれを行うと、UserControl がハードコードされているため、他のものを使用できなくなりますDataContext

バインディングの場合、通常はDataContextが継承されるため、バインディングは現在Visibilityの でプロパティを見つけることができますが、UserControl のコンストラクターでをハードコーディングしたため、そのプロパティは見つかりません。MyControlVisibleDataContextDataContext

次のように、バインディングで別のバインディング ソースを指定できます。

<Controls:MyUserControl Visibility="{Binding 
    RelativeSource={RelativeSource AncestorType={x:Type Window}}, 
    Path=DataContext.MyControlVisible, 
    Converter={StaticResource boolToVis}}" ... />

ただし、これはこの特定のケースの問題の単なる回避策であり、私の見解では永続的な解決策ではありません。より良い解決策は、単にハードコードしないことDataContextですUserControl

UserControl の目的とアプリケーションの設計方法に応じて、いくつかの方法があります。

  • UserControl にDependencyPropertyを作成して値を渡し、それにバインドすることができます。

    <Controls:MyUserControl UcModel="{Binding MyUControlModelProperty}" ... />
    

    <UserControl x:Class="Sandbox.Controls.MyUserControl"
                 ElementName=MyUserControl...>
        <Grid DataContext="{Binding UCModel, ElementName=MyUserControl}">
            <TextBlock Text="{Binding MyBoundText}"/>
        </Grid>
    </UserControl>
    
  • またはUserControl、特定のプロパティがDataContext. これは通常、 と組み合わせて行うことですDataTemplates

    <Controls:MyUserControl DataContext="{Binding MyUControlModelProperty}" ... />
    

    <UserControl x:Class="Sandbox.Controls.MyUserControl"...>
        <Grid>
            <TextBlock Text="{Binding MyBoundText}"/>
        </Grid>
    </UserControl>
    
  • 上で述べたように、特定のタイプの を期待する を表示するために使用したいDataTemplatesので、通常、メイン ウィンドウの XAML は次のようになります。UserControlsModelDataContext

    <DataTemplate DataType="{x:Type local:MyUControlModel}">
        <Controls:MyUserControl />
    </DataTemplate>
    
    <ContentPresenter Content="{Binding MyUControlModelProperty}" ... />
    
于 2013-05-10T18:26:39.137 に答える