9

私の質問ヘッダーが私の問題を正確に表しているかどうかはわかりません。説明するために最善を尽くします。

私はグリッドセルのDataTemplateを持っています:(グリッドはサードパーティの会社に属していますが、私の質問には重要ではありません)

<DataTemplate>
    <TextBlock>
        <Hyperlink Command="{Binding OpenLinkCommand}"> 
            <Hyperlink.ToolTip>
                <TextBlock Text="{Binding Data.MapLink}"/>
            </Hyperlink.ToolTip>
            <TextBlock Text="{Binding Data.MapLink}" TextDecorations="underline">
        </Hyperlink>
    </TextBlock>
</DataTemplate>

この DataTemplate にハイパーリンクを表示させたい (「Data.MapLink」はリンク値を含むオブジェクトです)。このリンクをクリックするたびに、コマンド「OpenLinkCommand」が起動されます。

問題は、「Data.MapLink」と「OpenLinkCommand」が異なる dataContext にあり、次の選択肢のいずれかを選択する必要があることです。

  1. ハイパーリンクの dataContext をそのままにしておきます。コマンドは機能せず、ハイパーリンクは Data.MapLink 値を取得します。

  2. ハイパーリンク dataContext をコマンド datacontext に変更します - コマンドは機能しますが、ハイパーリンク名は空になります。

残念ながら、これらの項目を同じ dataContext に配置するオプションがないため、dataContext が「X」であることをコマンドに伝え、dataContext が「Y」であることをハイパーリンクに伝える方法を見つける必要があります。

私の質問が明確であることを願っています。どうすればこの問題を解決できますか?

4

3 に答える 3

21

Sourceデフォルトとは異なるバインディングを指定するために使用できるバインディング プロパティがいくつかあります。DataContext

最も一般的なものはElementNameまたはRelativeSourceで、VisualTree 内の別の UI 要素を見つけて、そのプロパティにバインドできるようにします。

たとえば、次の使用法は、バインディング ソースとしてElementName使用する必要があることをバインディングに通知し、バインドするために使用します。MyGridViewMyGridView.DataContext.OpenLinkCommand

<Hyperlink Command="{Binding ElementName=MyGridView, 
                             Path=DataContext.OpenLinkCommand}"> 

バインディングで使用RelativeSourceして、指定されたオブジェクト タイプの VisualTree のさらに上のオブジェクトを検索し、それをバインディング ソースとして使用することもできます。この例は上記の例と同じことを行いますが、代わりに を使用しているためRelativeSource、を指定する必要はありません。ElementNameGridViewName

<Hyperlink Command="{Binding 
               RelativeSource={RelativeSource AncestorType={x:Type GridView}}, 
               Path=DataContext.OpenLinkCommand}"> 

3 番目のオプションは、次のSourceようにバインディングのプロパティを静的オブジェクトに設定することです。

<Hyperlink Command="{Binding 
               Source={x:Static local:MyStaticClass.OpenLinkCommand}}"> 

ここでのシングルトンへのバインドに関するコメントに基づいて、これがおそらく最適なオプションです。

于 2013-03-19T13:09:27.927 に答える
0

目的のデータコンテキストのインスタンスが必要になります(通常はコントロールまたはウィンドウのリソースにあります)。これができたら、親データコンテキストを自動的に継承するのではなく、テキストブロックのデータコンテキストを明示的に設定できるようになります。

例えば:

<TextBlock DataContext="{StaticResource MyDataMapLinkDataContext}" Text="{Binding Data.MapLink}" TextDecorations="underline"/>
于 2013-03-19T12:06:24.807 に答える
0

追加のデータ コンテキストに別のプロパティを使用する必要がある場合は、添付プロパティを使用できます。

XAML

    <Window.Resources>
        <Style TargetType="Button">
            <Setter Property="Template">
                <Setter.Value>
                    <ControlTemplate TargetType="Button">
                        <ContentPresenter Content="{Binding (local:ExtraDataContextProvider.ExtraDataContext), RelativeSource={RelativeSource TemplatedParent}}"/>
                    </ControlTemplate>
                </Setter.Value>
            </Setter>
        </Style>
    </Window.Resources>

    <Grid>
        <Button Margin="172,122,131,79" Foreground="Green" local:ExtraDataContextProvider.ExtraDataContext="A test">
            test
        </Button>
    </Grid>
</Window>

コード

using System;
using System.Collections.Generic;
using System.Linq;
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.Navigation;
using System.Windows.Shapes;

namespace WpfApplication1
{

    public class ExtraDataContextProvider : DependencyObject
    {
        public static object GetExtraDataContext(DependencyObject obj)
        {
            return (object)obj.GetValue(ExtraDataContextProperty);
        }

        public static void SetExtraDataContext(DependencyObject obj, object value)
        {
            obj.SetValue(ExtraDataContextProperty, value);
        }

        public static readonly DependencyProperty ExtraDataContextProperty = DependencyProperty.RegisterAttached("ExtraDataContext", typeof(object), typeof(ExtraDataContextProvider), new PropertyMetadata(null));
    }

    /// <summary>
    /// Interaction logic for MainWindow.xaml
    /// </summary>
    public partial class MainWindow : Window
    {
        public MainWindow()
        {
            InitializeComponent();
        }
    }
}
于 2013-03-19T12:21:17.303 に答える