1

好吧,我是一名 C++ 开发人员,目前我正在开发 WPF 应用程序,看起来这是一个棘手的情况。我已经动态生成了一组按钮、标签等,其中文本框和按钮都相互绑定。我之前在 C++ 代码中完成了此操作,现在我需要在 WPF 应用程序中执行此操作。

XAML:

<ListBox x:Name="myViewChannelList" HorizontalAlignment="Stretch" Height="Auto" ItemsSource="{Binding VoltageCollection}" Margin="0" VerticalAlignment="Stretch" Width="Auto" >
            <ListBox.Resources>
                <convert:BooleanToVisibilityConverter x:Key="booltovisibility"/>
            </ListBox.Resources>

            <ListBox.ItemTemplate>
                <DataTemplate >
                    <Grid Visibility="{Binding IsAvailable, Converter={StaticResource booltovisibility}}">
                        <Grid.ColumnDefinitions>
                            <ColumnDefinition Width="170"  />
                            <ColumnDefinition />
                            <ColumnDefinition  />
                            <ColumnDefinition />
                        </Grid.ColumnDefinitions>

                        <Label Grid.Column="0" Content="{Binding ChannelName}" Margin="50,20,0,0"></Label>

                        <Grid Grid.Column="1">
                            <Grid.ColumnDefinitions>
                                <ColumnDefinition />
                                <ColumnDefinition />
                            </Grid.ColumnDefinitions>
                            <TextBox Grid.Column="0" Text="{Binding VoltageText, Mode=TwoWay, UpdateSourceTrigger=PropertyChanged}" Height="25" Width="50" HorizontalAlignment="Center" VerticalAlignment="Center" Margin="170,20,0,0" />
                            <Button Grid.Column="1" Content="Set" Height="25" CommandParameter="{Binding VoltageText}" Command="{Binding VoltageCommand}" Width="50" HorizontalAlignment="Center" VerticalAlignment="Center" Margin="20,20,0,0" ></Button>
                        </Grid>
                    </Grid>
                </DataTemplate>
            </ListBox.ItemTemplate>
        </ListBox>

视图模型:

private ICommand m_voltageCommand;

    public ChannelList()
    {
         m_voltageCommand = new DelegateVoltageCommand(x => SetCommandExecute(x));
    }

public void Initialize()
{
    VoltageCollection = new ObservableCollection<VoltageModel> { new VoltageModel() { ChannelName = "", IsAvailable = false, VoltageText = String.Empty, VoltageCommand = m_voltageCommand },
                                                                 new VoltageModel() { ChannelName = "VDD__Main", IsAvailable = true, VoltageText = String.Empty, VoltageCommand = m_voltageCommand }, 
                                                                 new VoltageModel() { ChannelName = "VDD__IO__AUD", IsAvailable = true, VoltageText = String.Empty, VoltageCommand = m_voltageCommand }, 
                                                                 new VoltageModel() { ChannelName = "VDD__CODEC__AUD", IsAvailable = true, VoltageText = String.Empty, VoltageCommand = m_voltageCommand } 
                                                               }; 
}

ObservableCollection<VoltageModel> _voltages;
public ObservableCollection<VoltageModel> VoltageCollection
{
    get
    {
        return _voltages;
    }
    set
    {
        _voltages = value;
        OnPropertyChanged("VoltageCollection");
    }
} 

// Event when SET Button is clicked
public void SetCommandExecute(object voltageText)
{       
    string value = voltageText.ToString();
    int val = Convert.ToInt32(value);
}

因此,它会生成 Button + Textbox + Label 3 次,如Initialize()方法所示。现在VoltageCommand = m_voltageCommand给我在文本框中输入的文本,它调用电压文本SetCommandExecute(object voltageText)给我输入值的方法。

模型:

string voltageText = string.Empty;
    public string VoltageText
    {
        get
        {
            return voltageText;
        }

        set
        {
            voltageText = value;
            OnPropertyChanged("VoltageText");
        }
    }

**C++ Code:**

// Since we have 3 channels, channel maintains count
if(button == m_setButton[channel])
{
    unsigned cmd = 0x0300;
    int numBytes = 0;

    cmd |= (channel & 0xFF);
            // Some code

在这里,它告诉用户哪个按钮已被单击,并采用channel 的值,即如果单击第二个按钮则channel = 2

在这里,我需要实现用 C++ 编写的代码。如何获取频道,即点击了哪个按钮。看看cmd |= (channel & 0xFF);,它使用了channel值。如何在我的应用程序中实现它?

4

5 回答 5

2

您可以简单地将 ID 属性添加到您的 VoltageBoardChannel 类。

int index ; 
public int ID 
{ 
    get 
    { 
        return index; 
    } 

    set 
    { 
        index = value; 
        OnPropertyChanged("ID"); 
    }
}

然后将您的 CommandParameter Binding 更改为CommandParameter="{Binding}"CommandParameter="{Binding VoltageText}"您现在不仅会收到 Text,还会收到现在拥有 ID 的 VoltageBoardChannel 类的实例。

在您的命令执行方法中

public void DoSomethingExecute(object param) 
{ 
    VoltageBoardChannel result = param as VoltageBoardChannel; 
    string value = result.VoltageText;
    int index = result.ID;
}
于 2012-10-16T09:45:42.747 回答
1

这是您实现 MVVM 的基本问题:

ViewModel 永远不应该知道 View;它应该可以独立于任何 View 元素进行测试

在您的 SetCommandExecute 方法中,期望执行一些从视图发送的基于文本的工作。如果你要为 SetCommandExecute 方法编写一个单元测试,只使用来自 ViewModel 其他部分的信息,你会传入什么?

相反,您的 SecCommandExecute 应该是:

SetCommandExecute(object voltageModel)
{
    VoltageModel myModel = voltageModel as VoltageModel; // Cast the model to the correct object type
    if (myModel != null)
    { // myModel will be null of voltageModel is not a VoltageModel instance
        // TODO: Whatever work you need to do based on the values of the myModel
    }
}

您的 XML 应为:

<Button Grid.Column="1" Content="Set" Height="25" CommandParameter="{Binding }" Command="{Binding VoltageCommand}" Width="50" HorizontalAlignment="Center" VerticalAlignment="Center" Margin="20,20,0,0" ></Button>

这是如何运作的?那么它归结为DataContext。由于网格中每条线的 DataContext 都是一个电压模型,因此您可以直接绑定到它的几个属性。例子Text="{Binding VoltageText ...}"

数据网格中的每一项都有每行的对象实例的隐含 DataContext。由于每一行都绑定到 VoltageModel 的一个实例,因此您可以直接在代码中使用该事实。View 知道它正在使用哪些 ViewModel 属性和实例,并且可以将用户操作的特定电压模型“向下”传递给 ViewModel。

推理:

当您的命令事件运行时,它应该传入 VoltageModel 对象,这使您可以直接访问所有实例属性。请记住,这是最佳实践,因为您希望能够对SetCommandExecute方法进行单元测试,而不需要视图传递一些文本、解析它、在视图上找到一些控件以及所有这些东西。

简而言之:ViewModel 应该是完全独立的,并且能够基于仅 ViewModel 可用的数据运行所有单元测试。

于 2012-10-15T19:19:34.827 回答
0

我自己只做了一点WPF。当我将数据绑定到控件时,我可以通过某种方式将该数据项与其他数据项唯一区分开来。

伪代码:

private void Button_Clicked(object sender, EventArgs e) {
  MyType obj = (MyType)listView1.SelectedItem[0];
  if (obj.UniqueItem == whatINeed) {
    DoStuffFunction();
  }
}

我不知道这是否适用于您的情况,但这就是我解决问题的方式。

于 2012-10-15T17:38:19.667 回答
0

嗨,而不是将电压文本绑定到命令参数,而是将按钮绑定到它,您可以从 ButtonDataContext 或通过将 ListBox 的 SelectedItem 绑定到 ViewModel 属性来获取电压文本。我希望这会有所帮助。

<ListBox x:Name="lb">
        <ListBoxItem>
            <Button x:Name="btn" CommandParameter="btn" Command="{Binding MyCommand}" VerticalAlignment="Bottom" Height="30"/>
        </ListBoxItem>
        <ListBoxItem>
            <Button  CommandParameter="{Binding SelectedIndex, ElementName=lb}" Command="{Binding MyCommand}" VerticalAlignment="Bottom" Height="30"/> 
        </ListBoxItem>
    </ListBox>

 public MainWindow()
    {
        InitializeComponent();
        DataContext = this;
    }
    private ICommand _myCommand;
    public ICommand MyCommand { get { return _myCommand ?? (new CommandHandler((o) => FireCommand(o),()=> true)); } }
    public void FireCommand(object obj)
    {
        var a = lb.SelectedIndex; 
    }

public class CommandHandler:ICommand
{
        private Action<object> del;
        public CommandHandler(Action<object> action, Func<bool> b=null)
        {
            del = action;
        }

        #region ICommand Members

        public bool CanExecute(object parameter)
        {
            return true;
        }

        public event EventHandler CanExecuteChanged;

        public void Execute(object parameter)
        {
            del(parameter);
        }

        #endregion
}

您可以尝试以上两种方法之一。

于 2012-10-15T17:46:34.583 回答
0

您已经将电压文本框绑定到一个属性,因此不需要将该值作为命令参数传递。相反,您可以将源指定为命令:

<Button CommandParameter="TheButton" />

在你的命令处理程序的实现中:

public void SetCommandExecute(object source)
{       
    string source = source as string;

    if (source == "TheButton")
    {
       int val = Convert.ToInt32(this.VoltageText);
    }
}
于 2012-10-15T17:43:06.877 回答