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

最常见的是ElementNameor RelativeSource,它将在 VisualTree 中找到另一个 UI 元素,以便您可以绑定到它的属性。

例如,以下用于ElementName告诉绑定它应该MyGridView用作绑定源,并绑定到MyGridView.DataContext.OpenLinkCommand

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

您还可以RelativeSource在绑定中使用以在指定对象类型的 VisualTree 中进一步查找对象,并将其用作绑定源。这个例子和上面的例子做同样的事情,除了它使用RelativeSource而不是ElementName,所以你GridView不需要Name指定。

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

第三种选择是将绑定的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 回答