如何在没有代码隐藏的情况下实现OnPreviewMouseDown?

时间:2019-07-20 08:41:44

标签: c# wpf mvvm data-binding code-behind

我需要将我的代码制作为MVVM(这意味着无代码后置植入),并且我想单击任意更改我的LabelBackground >(即使我单击按钮)。

这意味着我需要从代码隐藏中删除我的MainWindowView_OnPreviewMouseDownMainWindowView_OnPreviewMouseUp

这是我的工作项目:

代码隐藏

    public partial class MainWindowView : Window
    {

        private readonly MainWindowViewModel _viewModel;
        public MainWindowView()
        {
            InitializeComponent();
            _viewModel = new MainWindowViewModel();
            // The DataContext serves as the starting point of Binding Paths
            DataContext = _viewModel;
        }

        private void MainWindowView_OnPreviewMouseDown(object sender, MouseButtonEventArgs e)
        {
            _viewModel.LabelBackground = Brushes.Black;
        }

        private void MainWindowView_OnPreviewMouseUp(object sender, MouseButtonEventArgs e)
        {
            _viewModel.LabelBackground = Brushes.Blue;
        }
    }
}

XAML

<Window x:Class="WpfExample.MainWindowView" PreviewMouseDown="MainWindowView_OnPreviewMouseDown" PreviewMouseUp="MainWindowView_OnPreviewMouseUp"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        Title="MainWindow" Height="350" Width="525"
        xmlns:local="clr-namespace:WpfExample">
    <Grid>
        <Grid.RowDefinitions>
            <RowDefinition />
            <RowDefinition/>
        </Grid.RowDefinitions>
        <Button Grid.Row="0" Command="{Binding HiButtonCommand}" CommandParameter="Hai" Content="click to hi"
                Height="100" Margin="208,30,203,30"  />
        <Label Content="Disco Background" Background="{Binding LabelBackground}" Margin="208,66,203,69" Grid.Row="1"/>
    </Grid>
</Window>

VIEWMODEL

class MainWindowViewModel : INotifyPropertyChanged
{
    private ICommand hiButtonCommand;
    public ICommand HiButtonCommand
    {
        get
        { return hiButtonCommand; }
        set
        { hiButtonCommand = value; }
    }

    private SolidColorBrush labelBackground;
    public SolidColorBrush LabelBackground
    {
        get { return labelBackground; }
        set
        {
            if (labelBackground != value)
            {
                labelBackground = value;
                OnPropertyChanged("LabelBackground");
            }
        }
    }

    public MainWindowViewModel()
    {
        HiButtonCommand = new RelayCommand(ShowMessage);
    }

    public void ShowMessage(object obj)
    {
        MessageBox.Show(obj.ToString());
    }

    public event PropertyChangedEventHandler PropertyChanged;

    [NotifyPropertyChangedInvocator]
    protected virtual void OnPropertyChanged(string propertyName)
    {
        PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
    }
}

最好的方法是什么?

编辑:

  

我尝试使用EventTrigger方法,但对我来说效果不佳,因为   它不会更改标签背景,而是会更改背景   而不是窗口的有什么办法可以定位标签而不是   窗户?

XAML:

<Window x:Class="WpfExample.MainWindowView" PreviewMouseDown="MainWindowView_OnPreviewMouseDown" PreviewMouseUp="MainWindowView_OnPreviewMouseUp"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        Title="MainWindow" Height="350" Width="525"
        xmlns:local="clr-namespace:WpfExample">

    <Window.Triggers>
        <EventTrigger RoutedEvent="Mouse.PreviewMouseDown">
            <BeginStoryboard>
                <Storyboard>
                    <ColorAnimation To="Black" 
                                    Storyboard.TargetProperty="(Label.Background).(SolidColorBrush.Color)" 
                                    Duration="0"/>
                </Storyboard>
            </BeginStoryboard>
        </EventTrigger>

        <EventTrigger RoutedEvent="Mouse.PreviewMouseUp">
            <BeginStoryboard>
                <Storyboard>
                    <ColorAnimation To="Blue" 
                                    Storyboard.TargetProperty="(Label.Background).(SolidColorBrush.Color)" 
                                    Duration="0"/>
                </Storyboard>
            </BeginStoryboard>
        </EventTrigger>
    </Window.Triggers>
    <Grid>
        <Grid.RowDefinitions>
            <RowDefinition />
            <RowDefinition/>
        </Grid.RowDefinitions>
        <Button Grid.Row="0" Command="{Binding HiButtonCommand}" CommandParameter="Hai" Content="click to hi" Height="100" Margin="208,30,203,30"/>
        <Label Name="Label" Content="Disco Background" Background="White" Margin="208,66,203,59" Grid.Row="1"/>
    </Grid>
</Window>

2 个答案:

答案 0 :(得分:2)

我的新XAML解决了该问题 无需代码隐藏:

新的XAML:

<Window x:Class="WpfExample.MainWindowView"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        Title="MainWindow" Height="350" Width="525"
        xmlns:local="clr-namespace:WpfExample">

    <Window.Triggers>
        <EventTrigger RoutedEvent="Mouse.PreviewMouseDown">
            <BeginStoryboard>
                <Storyboard>
                    <ColorAnimation To="Black" 
                                    Storyboard.TargetProperty="(Label.Background).(SolidColorBrush.Color)" 
                                    Storyboard.TargetName="Label"
                                    Duration="0"/>
                </Storyboard>
            </BeginStoryboard>
        </EventTrigger>

        <EventTrigger RoutedEvent="Mouse.PreviewMouseUp">
            <BeginStoryboard>
                <Storyboard>
                    <ColorAnimation To="Blue" 
                                    Storyboard.TargetProperty="(Label.Background).(SolidColorBrush.Color)"
                                    Storyboard.TargetName="Label"
                                    Duration="0"/>
                </Storyboard>
            </BeginStoryboard>
        </EventTrigger>
    </Window.Triggers>
    <Grid>
        <Grid.RowDefinitions>
            <RowDefinition />
            <RowDefinition/>
        </Grid.RowDefinitions>
        <Button Grid.Row="0" Command="{Binding HiButtonCommand}" CommandParameter="Hai" Content="click to hi" Height="100" Margin="208,30,203,30"/>
        <Label Name="Label" Content="Disco Background" Background="White" Margin="208,66,203,59" Grid.Row="1"/>
    </Grid>
</Window>

答案 1 :(得分:1)

这是一个简单的任务,没有最佳方法。

  1. 使用https://github.com/Fody/Home/属性已更改的https://github.com/Fody/PropertyChanged型玩具
  2. 创建一个实现通知属性已更改的基类视图
  3. 使用支持MV *模式的框架,例如Caliburn.Micro https://caliburnmicro.com/documentation/cheat-sheet

并且代码应该是以下行中的内容:

XAML

<Button 
    cal:Message.Attach="[Event MouseDown] = [Action Action1]" 
    cal:Message.Attach="[Event MouseUp] = [Action Action2]" 
    Content="{Binding ButtonContent}" />
<Label 
    Content="Disco Background" 
    Background="{Binding Importance, Converter={StaticResource ImportanceToBgConverter}}"/>

ViewModel

    public String ButtonContent { get; set; } = "click to hi";
    public ImportanceType Importance { get; set; }
    public void Action1()
    {
        Importance = Importance.NotImportant;
    }

    public void Action2()
    {
        Importance = Importance.Important;
    }