0

私はWPFに比較的慣れていないので、少し問題が発生しました。HierarchicalDataTemplatesを使用して、XMLをTreeViewコントロールに正常にバインドしました。各ノードレンダラーは正しく(SubstepTemplateのラベルを含む)。

SubstepTemplateのボタンからViewModelのバインドされたコマンドを取得することもできますが、CommandParameterにハードコードされた値(999など)を入力した場合に限ります。XMLのcommandID属性にバインドしようとする試みはすべて失敗しました。

これが私が今持っているものです:

XML:

<root xmlns="">
  <step label="Step Label 1">
    <button label="Button Title 1A" commandID="701" />
    <button label="Button Title 1B" commandID="702" />
    <button label="Button Title 1C" commandID="703" />
  </step>
  <step label="Step Label 2">
    <button label="Button Title 2A" commandID="801" />
    <button label="Button Title 2B" commandID="802" />
  </step>
</root>

XAML:

<ResourceDictionary
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="clr-namespace:SidePanel">

<HierarchicalDataTemplate
            x:Key="SubstepTemplate"
            DataType="button"
            ItemsSource="{Binding XPath=*}">
    <StackPanel Orientation="Horizontal">
        <Button Margin="2" Width="32" Height="32" Command="{Binding ElementName=MyTreeView, Path=DataContext.PluginCommand}" CommandParameter="{Binding XPath=@commandID}" />
        <Label VerticalAlignment="Center" Margin="8,0,0,0" Content="{Binding XPath=@label}" />
    </StackPanel>
</HierarchicalDataTemplate>

<HierarchicalDataTemplate
            x:Key="StepTemplate"
            DataType="step"
            ItemsSource="{Binding XPath=*}"
            ItemTemplate="{StaticResource SubstepTemplate}">
    <Expander Header="{Binding Mode=OneWay, XPath=@label}" HorizontalAlignment="Stretch">
    </Expander>
</HierarchicalDataTemplate>

<Style TargetType="{x:Type local:SidePanelControl}">
    <Setter Property="Template">
        <Setter.Value>
            <ControlTemplate TargetType="{x:Type local:SidePanelControl}">
                <TreeView
                    Name="MyTreeView"
                    ItemsSource="{Binding XmlRoot}"
                    ItemTemplate="{StaticResource StepTemplate}"
                    Background="{TemplateBinding Background}"
                    BorderBrush="{TemplateBinding BorderBrush}"
                    BorderThickness="{TemplateBinding BorderThickness}"
                    HorizontalAlignment="Stretch">
                </TreeView>
            </ControlTemplate>
        </Setter.Value>
    </Setter>
</Style>
</ResourceDictionary>

ViewModelスニペット:

    public DelegateCommand<string> PluginCommand
    {
        get
        {
            if (pluginCommand == null)
            {
                pluginCommand = new DelegateCommand<string>(ExecPluginCommand, canExecPluginCommand);
            }
            return pluginCommand;
        }
    }

    public void ExecPluginCommand(string param)
    {
        LogMessage("In ExecPluginCommand, param = " + param);
    }

    public bool canExecPluginCommand(string param)
    {
        return true;
    }

これを機能させるためにCommandParameterに入れる必要がある正しいBinding式は何ですか?{Binding XPath = @ commandID}が機能していないようで、その理由がわかりません。

4

1 に答える 1

0

問題を見つけました。XPathがCommandParameterバインディングとして使用される場合、テンプレート化されたDelegateCommandは、文字列ではなく、タイプSystem.Xml.XmlNodeのパラメーターを受け取ります。

コマンドデータ型を取得しようとすると、コマンドで例外が発生していました

    void ICommand.Execute(object parameter)
    {
        TParam value = GetCommandDataType(parameter);
        Execute(value);
    }

TypeConverter :: CanConvertFromは、System.Xml.XmlNode型の引数をどう処理するかを知らなかったためです。

明示的にテストすることでcommandID値を取得できました

    data is System.Xml.XmlNode

GetCommandDataType(object data)で、.Valueメンバーを返します。

        else if (data is System.Xml.XmlNode)
        {
            System.Xml.XmlNode foo = (System.Xml.XmlNode)data;
            string fooVal = foo.Value;
            TypeConverter converter = TypeDescriptor.GetConverter(typeof(string));
            result = (TParam)converter.ConvertFrom(fooVal);
        }
于 2011-08-03T23:20:31.033 に答える