0

我正在开发一个使用 Silicon Labs SDK 的项目。在为他们的 SDK 制作包装器后,我制作了一个 WPF 项目来可视化 SDK 的一些结果。我的第一个测试是使用 GetNumDevices 函数。我的 WPF 知识不是很强,所以对任何让你眼睛流血的事情感到抱歉,但为了演示,让我向你展示我所拥有的。首先是我正在谈论的 API 方法。

[DllImport("CP210xManufacturing.dll", CallingConvention = CallingConvention.StdCall)]
public static extern CP210x_STATUS CP210x_GetNumDevices(ref int lpdwNumDevices);

我决定制作一个用户控件,而不是制作 2 个标签和 1 个按钮来可视化此方法。

CP210UserControl.xaml

<UserControl x:Class="SiliconLabsRosettaStone.CP210UserControl"
             x:Name="panel"
             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" 
             d:DesignHeight="50" d:DesignWidth="300"
             DataContext="{Binding ElementName=panel}">
    <Border BorderBrush="DarkGreen" BorderThickness="3" Margin="2">
        <Grid>
            <Grid.ColumnDefinitions>
                <ColumnDefinition Width="Auto"/>
                <ColumnDefinition />
                <ColumnDefinition Width="75"/>
            </Grid.ColumnDefinitions>

            <TextBlock Text="{Binding Caption}" Margin="3,0" />
            <Border Grid.Column="1" BorderBrush="Black" BorderThickness="2" Margin="3,0">
                <TextBlock x:Name="responseTextBlock" Text="{Binding Response}" />
            </Border>
            <Button Grid.Column="2" Content="Execute" Click="Button_Click" Margin="3,0" />
        </Grid>
    </Border>
</UserControl>

CP210UserControl.xaml.cs

public partial class CP210UserControl : UserControl
{
    public CP210UserControl()
    {
        InitializeComponent();
    }
    public void SetupUserControl(Func<CP210UserControl, CP210x_STATUS> function)
    {
        this.cp210Function = function;
    }
    private void Button_Click(object sender, RoutedEventArgs e)
    {
        CP210x_STATUS response = cp210Function(this);
        this.responseTextBlock.Background = response == CP210x_STATUS.CP210x_SUCCESS ? null : new SolidColorBrush(Colors.Pink);
    }

    public string Caption
    {
        get { return (string)GetValue(CaptionProperty); }
        set { SetValue(CaptionProperty, value); }
    }

    public string Response
    {
        get { return (string)GetValue(ResponseProperty); }
        set { SetValue(ResponseProperty, value); }
    }

    public static readonly DependencyProperty CaptionProperty = DependencyProperty.Register("Caption", typeof(string), typeof(CP210UserControl), new PropertyMetadata("CP210x"));
    public static readonly DependencyProperty ResponseProperty = DependencyProperty.Register("Response", typeof(string), typeof(CP210UserControl), new PropertyMetadata(""));

    private Func<CP210UserControl, CP210x_STATUS> cp210Function;
}

那是我对它的使用有疑问的文件。

这就是我在主窗口中使用它的方式

主窗口.xaml

<StackPanel>
    <local:CP210UserControl x:Name="cpGetNumDevices" Grid.Row="1" Grid.ColumnSpan="3" Caption="CP210x_GetNumDevices" />
</StackPanel>

现在来说我讨厌的部分。我如何连接它:/

主窗口.xaml.cs

    private void Window_Loaded(object sender, RoutedEventArgs e)
    {
        cpGetNumDevices.SetupUserControl(getNumberDevices);
    }
    private CP210x_STATUS getNumberDevices(CP210UserControl control)
    {
        int devices = 0;
        var result = CP210xWrapper.CP210x_GetNumDevices(ref devices);
        control.Response = string.Format("Number Of Devices:{0}", devices);
        return result;
    }

我认为我可以使 xaml 看起来像这样

    <local:CP210UserControl x:Name="cpGetNumDevices" Grid.Row="1" Grid.ColumnSpan="3" Caption="CP210x_GetNumDevices" cp210Function="getNumberDevices" />

如果我将我的用户控件更改为此。

    public Func<CP210UserControl, CP210x_STATUS> cp210Function
    {
        get { return (Func<CP210UserControl, CP210x_STATUS>)GetValue(cp210FunctionProperty); }
        set { SetValue(cp210FunctionProperty, value); }
    }

    public static readonly DependencyProperty cp210FunctionProperty = DependencyProperty.Register("cp210Function", typeof(Func<CP210UserControl, CP210x_STATUS>), typeof(CP210UserControl), new PropertyMetadata(null));

但是当我运行程序时出现此错误

A first chance exception of type 'System.Windows.Markup.XamlParseException' occurred in PresentationFramework.dll
Additional information: 'Func`2' type does not have a public TypeConverter class.  Error at Line 7 Position 122.

那么我该如何制作所说的TypeConverter?有可能吗?

编辑

这基本上是我的最终游戏。

我在拍摄什么

4

1 回答 1

1

你的班级结构对我来说似乎很尴尬。

UI 不是与任何外部库进行任何交互的正确位置。

如果您正在使用 WPF,那么您确实需要接受WPF Mentality

这就是我在 WPF 中的做法:

<UserControl x:Class="SiliconLabsRosettaStone.CP210UserControl"
             ...>
    <!-- ... Omitted for brevity -->

            <TextBlock Text="{Binding Caption}" ... />

            <TextBlock Text="{Binding NumberOfDevices}" />

            <Button Content="Execute" Click="Button_Click" ... />

    <!-- ... -->
</UserControl>

背后的用户控制代码:

public class CP210UserControl: UserControl
{
   public CP210UserControl()
   {
       InitializeComponent();
   }

   private void Button_Click(object sender, RoutedEventArgs e)
   {
      var VM = DataContext as MyViewModel;
      if (VM != null)
      {
          //Correct uses of Code-Behind:

          //1 - Call ViewModel Methods:
          var response = VM.GetNumberDevices();

          //2 - UI-specific code which does not contain business logic:
          responseTextBlock.Background = response ? null : Brushes.Pink;
      }
   }
}

窗户:

<Window ....>
   <local:CP210UserControl/>
</Window>

代码背后:

public class MainWindow: Window
{
    public MainWindow()
    {
       InitializeComponent();
       DataContext = new MyViewModel();
    }
}

视图模型:

public class MyViewModel: INotifyPropertyChanged
{
   public string Caption {get;set;} //NotifyPropertyChanged() required.

   public string NumberOfDevices {get;set;} //NotifyPropertyChanged() required.

   public bool GetNumberDevices()
   {
       int devices = 0;
       var result = CP210xWrapper.CP210x_GetNumDevices(ref devices);
       NumberOfDevices = string.Format("Number Of Devices:{0}", devices);

       return result == CP210x_STATUS.CP210x_SUCCESS;
   }

   //INotifyPropertyChanged implementation omitted for brevity.
}
  • 不需要奇怪Func
  • 不需要DependencyProperties
  • 只是简单,简单的属性和INotifyPropertyChanged. 这就是您在 WPF 中编码的方式。
  • 以这种方式使用后面的代码是完全有效的,只是调用 ViewModel 中的方法,而不是把实际的逻辑和行为放在后面的代码中。
  • WPF 摇滚
  • 如果您需要进一步的帮助,请告诉我。
于 2013-10-17T18:55:16.677 回答