2008-10-08 71 views

回答

2

我同意Joel的首选方法是通过触发器或通过将Enabled值绑定到另一个依赖项属性(可能在父元素上)来触发设置按钮的Enabled属性和xaml标记, ValueConverter的帮助。但是,如果必须在C#中明确执行此操作,则可以在按钮的模板属性上使用FindName()方法。有一个MSDN'如何'linked here

3

你真的不应该明确这么做,而应该在ControlTemplate本身中设置触发器。这些触发器会对其他更改作出反应,并将设置ButtonEnabled属性。

4

我建议按钮很容易通过RoutedUICommands进行控制 - 包括轻松启用和禁用它们。命令不需要对Button的任何类型的引用,所以这也解决了你的问题。

在这里看到:

<Window.CommandBindings> 
    <CommandBinding 
     Command="{x:Static local:MyCommands.MyCommand}" 
     Executed="CommandBinding_Executed" 
     CanExecute="CommandBinding_CanExecute" /> 
</Window.CommandBindings> 

<Window.Resources> 
    <ControlTemplate x:Key="MyTemplate" TargetType="Label"> 
     <Button Command="{x:Static local:MyCommands.MyCommand}"> 
      <TextBlock> 
       Click Me 
       <TextBlock Text="{TemplateBinding Content}" /> 
      </TextBlock> 
     </Button> 
    </ControlTemplate> 
</Window.Resources> 

<StackPanel> 
    <Label Template="{StaticResource MyTemplate}">1</Label> 
    <Label Template="{StaticResource MyTemplate}">2</Label> 
    <Label Template="{StaticResource MyTemplate}">3</Label> 
    <Label Template="{StaticResource MyTemplate}">4</Label> 
    <Label Template="{StaticResource MyTemplate}">5</Label> 
</StackPanel> 

使用此:

public partial class MainWindow : Window 
{ 
    public MainWindow() 
    { 
     InitializeComponent(); 
    } 

    private void CommandBinding_CanExecute(object sender, System.Windows.Input.CanExecuteRoutedEventArgs e) 
    { 
     // your logic here 
     int _Integer = -1; 
     int.TryParse(e.Parameter.ToString(), out _Integer); 
     e.CanExecute = _Integer % 2 == 0; 
    } 

    private void CommandBinding_Executed(object sender, System.Windows.Input.ExecutedRoutedEventArgs e) 
    { 
     // do something when clicked 
    } 
} 

public static class MyCommands 
{ 
    public static RoutedUICommand MyCommand = new RoutedUICommand(); 
} 

的样子:

screenshot

0

我有一个问题,在CONTROLTEMPLATES &的DataTemplates按钮。我发现触发器通常太麻烦,无法准确提供我的客户想要的东西,最后我得到了太多的转换器。我第一次尝试的方法是定义这些更难的模板内联,以便他们可以绑定到我的屏幕ViewModel上的命令(&代码隐藏事件)。

如果你是模板化自己的控制,我要提到的最简单的方法是简单地命名按钮,然后用这样的:

hourHand = this.Template.FindName("PART_HourHand", this) as Rectangle; 

然而,最近我已经改变了圆通(回indevidual文件..重新使用!),但开始添加一个ViewModel(各种),我后缀...TemplateCommandStore到所有我的模板有点复杂。我甚至模板化时,我customcontrols(一致性,大部分)使用这个

public class BookingDetailsTemplateCommandStore 
{ 
    public OpenWindowCommand_Command OpenWindowCommand_Command { get; set; } 

    public BookingDetailsTemplateCommandStore() 
    { 
     OpenWindowCommand_Command = new OpenWindowCommand_Command(this); 
    }  
} 

然后我就可以开心地在模板中使用此命令。您还可以通过绑定(依赖性?)属性来将控件引用传递到命令存储区,以捕获事件&可以更精确地控制模板。

<DataTemplate DataType="{x:Type LeisureServices:BookingSummary}"> 
    <Border Height="50" otherStyling="xyz"> 
     <Border.Resources> 
      <local:BookingDetailsTemplateCommandStore x:Key="CommandStore" /> 
     </Border.Resources> 
     <DockPanel> 
      <Button otherStyling="xyz" 
        Command="{Binding Source={StaticResource CommandStore}, 
             Path=OpenWindowCommand_Command}" /> 

MVVM:我发现,只要图形化的逻辑就是这样,从业务逻辑完全不同,这种方法不会干扰MVVM。实质上,我为模板添加了一致的C#代码能力,任何花费太多时间在winform上的人都可能会像我一样错过。

相关问题