将两个命令绑定到一个按钮?

Posted

技术标签:

【中文标题】将两个命令绑定到一个按钮?【英文标题】:Binding two commands to one button? 【发布时间】:2014-05-13 06:59:24 【问题描述】:

我有一个选项卡控件,它包含两个选项卡,一个显示正在运行的进程的消息,另一个显示一个网页!

我有三个按钮(开始、停止和清除) 当我按下开始时,进程运行并显示消息。

我需要的是当我按下开始按钮时,两个选项卡都会向我显示所需的内容。这意味着我需要两个将两个命令绑定到同一个按钮

你知道如何为 1 个按钮绑定两个命令吗?

我正在使用 MVVM,而且我是 MVVM 的新手。

<Button Content="Start" Command="Binding StartProcess_Command " Height="25" Width="60" Grid.Row="0" Grid.Column="0" Margin="10"/>
<Button Content="Stop"  Command="Binding StopProcess_Command" Height="25" Width="60" Grid.Row="0" Grid.Column="1" Margin="10"/>
<Button Content="Clear" Command="Binding ClearBtn_Command" Height="25" Width="60" Grid.Row="0" Grid.Column="2" Margin="10"/>
<telerik:RadTabControl HorizontalAlignment="Stretch" VerticalAlignment="Stretch"  Grid.Row="1" Grid.ColumnSpan="4" Margin="10">
    <telerik:RadTabItem Header="Maestro" IsSelected="Binding Path=MaestroSelected">
        <ScrollViewer VerticalScrollBarVisibility="Visible" HorizontalScrollBarVisibility="Visible">
            <local:WPFBrowser/>
        </ScrollViewer>
    </telerik:RadTabItem>
    <telerik:RadTabItem Header="Messages">
        <Grid Background="LightGray">
            <ScrollViewer VerticalScrollBarVisibility="Visible" HorizontalScrollBarVisibility="Visible">
                <TextBlock TextWrapping="Wrap" Text="Binding Path=output_process,UpdateSourceTrigger=Default"/>
            </ScrollViewer>
        </Grid>
    </telerik:RadTabItem>
</telerik:RadTabControl >

这是 WPFBrowser.cs 代码:

         public WPFBrowser()
            
                ServerString = "localhost"; //defualt value
                PortString = "9999"; //default value
                this.refreshCommand = new Lunch_Maestro.ViewModel.Command(DoRefreshCommand);
                m_WPFBrowser = new WebBrowser()
                
                    HorizontalAlignment = System.Windows.HorizontalAlignment.Stretch,
                    VerticalAlignment = System.Windows.VerticalAlignment.Stretch
                ;

                m_WPFBrowser.Navigated += WPFBrowser_Navigated;

                this.Content = m_WPFBrowser;
            

            private void WPFBrowser_Navigated(object sender, System.Windows.Navigation.NavigationEventArgs e)
            
                this.HideScriptErrors(m_WPFBrowser, true);
            


            /// <summary>
            /// Found this link with interesting scenerios for script erros...
            /// http://social.msdn.microsoft.com/Forums/vstudio/en-US/4f686de1-8884-4a8d-8ec5-ae4eff8ce6db/new-wpf-webbrowser-how-do-i-suppress-script-errors?forum=wpf
            /// </summary>
            public void HideScriptErrors(WebBrowser wb, bool Hide)
            
                FieldInfo fiComWebBrowser = typeof(WebBrowser).GetField("_axIWebBrowser2", BindingFlags.Instance | BindingFlags.NonPublic);
                if (fiComWebBrowser == null) return;

                object objComWebBrowser = fiComWebBrowser.GetValue(wb);

                if (objComWebBrowser == null) return;

                objComWebBrowser.GetType().InvokeMember("Silent", BindingFlags.SetProperty, null, objComWebBrowser, new object[]  Hide );

            


            private void DoRefreshCommand()
            
                m_WPFBrowser.Navigate(new Uri("http://" + ServerString + ":" + PortString + "/"));
            

            private bool _maestroSelected;
            public bool MaestroSelected
            
                get  return _maestroSelected; 
                set
                
                    _maestroSelected = value;
                    if (_maestroSelected == true)
                        m_WPFBrowser.Navigate(new Uri("http://" + ServerString + ":" + PortString + "/"));
                    OnPropertyChanged("MaestroSelected");
                
            

            private readonly WebBrowser m_WPFBrowser;

            private string _serverString;
            public string ServerString
            
                get  return _serverString; 
                set
                
                    _serverString = value;
                    OnPropertyChanged("ServerString");
                
            

            private string _portString;
            public string PortString
            
                get  return _portString; 
                set
                
                    _portString = value;
                    OnPropertyChanged("PortString");
                
            

            private Lunch_Maestro.ViewModel.Command refreshCommand;
            public Lunch_Maestro.ViewModel.Command RefreshCommand
            
                get  return refreshCommand; 
            
// and Inotify property changes here

【问题讨论】:

看看这个:codeproject.com/Articles/990113/… 【参考方案1】:

你可以像这样调用多个命令:

// xmlns:i="http://schemas.microsoft.com/xaml/behaviors"

<Button>
    <i:Interaction.Triggers>
        <i:EventTrigger EventName="Click">
            <i:InvokeCommandAction Command="Binding Cmd1" />
            <i:InvokeCommandAction Command="Binding Cmd2" />
            <i:InvokeCommandAction Command="Binding Cmd3" />
        </i:EventTrigger>
    </i:Interaction.Triggers>
</Button>

如果您需要,这也将允许您控制命令执行顺序

PS:你只需要安装Microsoft.Xaml.Behaviors.Wpf NuGet 包。

【讨论】:

【参考方案2】:

您可以使用来自Prism 框架的CompositeCommand。 在您的视图模型中创建一个额外的 CompositeCommand,使用此 CompositeCommand 注册两个普通命令并将按钮绑定到 CompositeCommand。

请参阅第 130 页上的 Prism documentation,了解如何使用 CompositeCommands。

您的应用需要引用 Prism 包中包含的 Microsoft.Practices.Prism.dll。

【讨论】:

【参考方案3】:

您想单击一个Button 并运行两段代码……听起来确实没有那么复杂。使用可用的RelayCommands 之一,可以像这样简单地解决问题:

public ICommand SomeCommand

    get  return new ActionCommand(action) =>  RunFirstFunction(action); 
        RunSecondFunction(action) , canExecute => someCondition); 

在 XAML 中:

<Button Command="Binding SomeCommand" CommandParameter="Binding SomeObject">
    Click Me
</Button>

ActionCommand 是我自己的RelayCommand 形式,但所有delegate ICommand 实现大致相同并且可以以这种方式工作。 CommandParameter 值作为action 变量进入代码并传递给这两个函数。

【讨论】:

以上是关于将两个命令绑定到一个按钮?的主要内容,如果未能解决你的问题,请参考以下文章

如何将命令绑定到 DataContext 之外的按钮中?

使用 Caliburn.Micro 将命令绑定到 ListView 内的按钮

如何将命令绑定添加到动态按钮?

wpf usercontrol,将按钮的命令参数绑定到父用户控件

MVVM:将带有 Observable 集合的命令绑定到 Listbox 并从文本框中获取值

DataGrid 绑定命令到行选择