繁体   English   中英

MVVM绑定无法正常工作

[英]MVVM Bindings not working properly

更新 托管以修复selectedIndex问题。 我也忘记设置SelectedItem,这自然会引起一些问题。

因此,今天上午9点,我们完成了24小时的作业,而我遇到了麻烦。 我们应该创建一个程序,允许主管添加和删除员工,并添加工作时间,总时数和总收入。 但是在MVVM模式之后成功实现此功能时遇到了一些问题。 由于某种原因,我的绑定根本无法工作,我只能看到有人在查看我的项目并帮助我进行故障排除的解决方案。

这是我的代码-对于必须发布整个内容,我感到非常抱歉,但是鉴于我不知道问题出在哪里,所以我没有看到任何其他选择。

员工模型

[Serializable]
public class WorkSessions : ObservableCollection<WorkSessionModel>
{
    public WorkSessions()
    {

    }
}
[Serializable]
public class WorkSessionModel : INotifyPropertyChanged
{
    private DateTime _dateTime;
    private string _id;
    private double _hours;

    public WorkSessionModel()
    {

    }

    public DateTime DateTime
    {
        get { return _dateTime; }
        set
        {
            _dateTime = value;
            NotifyPropertyChanged("DateTime");
        }
    }

    public string ID
    {
        get { return _id; }
        set
        {
            _id = value;
            NotifyPropertyChanged("ID");
        }
    }
    public double Hours
    {
        get { return _hours; }
        set
        {
            _hours = value;
            NotifyPropertyChanged("Hours");
            NotifyPropertyChanged("TotalHours");
        }
    }


    [field: NonSerialized]
    public event PropertyChangedEventHandler PropertyChanged;

    private void NotifyPropertyChanged(String info)
    {
        if (PropertyChanged != null)
        {
            PropertyChanged(this, new PropertyChangedEventArgs(info));
        }
    }
}

工作会话模型

    [Serializable]
public class WorkSessions : ObservableCollection<WorkSessionModel>
{
    public WorkSessions()
    {

    }
}
[Serializable]
public class WorkSessionModel : INotifyPropertyChanged
{
    private DateTime _dateTime;
    private string _id;
    private double _hours;

    public WorkSessionModel()
    {

    }

    public DateTime DateTime
    {
        get { return _dateTime; }
        set
        {
            _dateTime = value;
            NotifyPropertyChanged("DateTime");
        }
    }

    public string ID
    {
        get { return _id; }
        set
        {
            _id = value;
            NotifyPropertyChanged("ID");
        }
    }
    public double Hours
    {
        get { return _hours; }
        set
        {
            _hours = value;
            NotifyPropertyChanged("Hours");
            NotifyPropertyChanged("TotalHours");
        }
    }


    [field: NonSerialized]
    public event PropertyChangedEventHandler PropertyChanged;

    private void NotifyPropertyChanged(String info)
    {
        if (PropertyChanged != null)
        {
            PropertyChanged(this, new PropertyChangedEventArgs(info));
        }
    }
}

EmployeeViewModel

public class EmployeeViewModel : ViewModelBase
{
    private Employees _employeesModel = new Employees();
    public Employees EmployeesView = new Employees();

    public ObservableCollection<WorkSessionModel> WorkSessions { get; set; }

    private string _id = "0";
    private string _name = "noname";
    private double _wage = 0;
    private int _totalhours = 0;
    public string ID
    {
        get { return _id; }
        set { _id = value; RaisePropertyChanged("ID"); }
    }

    public string Name
    {
        get { return _name; }
        set
        {
            _name = value;
            RaisePropertyChanged("Name");
        }
    }
    public double Wage
    {
        get { return _wage; }
        set
        {
            _wage = value;
            RaisePropertyChanged("Wage");
        }
    }
    public int TotalHours
    {
        get { return _totalhours; }
        set
        {
            _totalhours = value;
            RaisePropertyChanged("TotalHours");
        }
    }

    private EmployeeModel _selectedEmployee = new EmployeeModel();
    public EmployeeModel SelectedEmployee
    {
        get { return _selectedEmployee; }
        set
        {
            _selectedEmployee = value;
            RaisePropertyChanged("SelectedEmployee");
        }
    }

    private int _selectedEmployeeIndex;
    public int SelectedEmployeeIndex
    {
        get { return _selectedEmployeeIndex; }
        set
        {
            _selectedEmployeeIndex = value;
            RaisePropertyChanged("SelectedEmployeeIndex");
        }
    }

    #region RelayCommands

    // Employee Relay Commands
    public RelayCommand EmployeeAddNewCommand { set; get; }
    public RelayCommand EmployeeDeleteCommand { set; get; }
    public RelayCommand EmployeeNextCommand { set; get; }
    public RelayCommand EmployeePrevCommand { set; get; }
    public RelayCommand EmployeeTotalHoursCommand { get; set; }

    #endregion

    public EmployeeViewModel()
    {
        InitCommands();
    }

    private void InitCommands()
    {           
        EmployeeAddNewCommand = new RelayCommand(EmployeeAddNewExecute, EmployeeAddNewCanExecute);
        EmployeeDeleteCommand = new RelayCommand(EmployeeDeleteNewExecute, EmployeeDeleteCanExecute);
        EmployeeNextCommand = new RelayCommand(EmployeeNextExecute, EmployeeNextCanExecute);
        EmployeePrevCommand = new RelayCommand(EmployeePrevExecute, EmployeePrevCanExecute);
        //EmployeeTotalHoursCommand = new RelayCommand(EmployeeTotalHoursExecute, EmployeeTotalHoursCanExecute);
    }

    //private void EmployeeTotalHoursExecute()
    //{
    //    _selectedEmployee.TotalHours();
    //}

    //private bool EmployeeTotalHoursCanExecute()
    //{
    //    return true;
    //}

    private void EmployeeAddNewExecute()
    {
        EmployeeModel newEmployee = new EmployeeModel();
        EmployeesView.Add(newEmployee);
        _employeesModel.Add(newEmployee);
        SelectedEmployee = newEmployee;
    }

    private bool EmployeeAddNewCanExecute()
    {
        return true;
    }

    private void EmployeeDeleteNewExecute()
    {
        if (MessageBox.Show("You are about delete all submissions for     Employee," + SelectedEmployee.Name + "(" + SelectedEmployee.ID +")\r\nAre you sure?", "This is a Warning!", MessageBoxButton.YesNo) == MessageBoxResult.Yes)
        {
            _employeesModel.Remove(SelectedEmployee);
            EmployeesView.Remove(SelectedEmployee);
        }
    }

    private bool EmployeeDeleteCanExecute()
    {
        if (SelectedEmployee != null)
            return true;
        else return false;
    }

    private void EmployeeNextExecute()
    {
        SelectedEmployeeIndex++;
    }
    private bool EmployeeNextCanExecute()
    {
        if (SelectedEmployeeIndex < EmployeesView.Count - 1)
            return true;
        return false;
    }

    private void EmployeePrevExecute()
    {
        SelectedEmployeeIndex--;
    }
    private bool EmployeePrevCanExecute()
    {
        if (SelectedEmployeeIndex > 0)
            return true;
        return false;
    }
}

视图

    public partial class MainWindow : Window
{
    public EmployeeViewModel EmployeeViewModel = new EmployeeViewModel();

    public MainWindow()
    {
        InitializeComponent();

        menu_employee.DataContext = EmployeeViewModel;
        sp_employees.DataContext = EmployeeViewModel;
        datagrid_employees.ItemsSource = EmployeeViewModel.EmployeesView;
        grid_selectedEmployee.DataContext =  EmployeeViewModel.SelectedEmployee;
    }
}

我可以看到您的代码有一些问题:

  • 当SelectedIndex更新时,SelectedItem保持不变,反之亦然。
  • 您的数据绑定似乎混乱了:

DataContext属性级联到特定依赖项对象的每个子级。

MainWindow构造函数中的代码可能应替换为:

this.DataContext = EmployeeViewModel;

然后在XAML中使用数据绑定设置其余属性。 您遇到的问题是所选雇员的DataContext仅设置一次。 这意味着,如果您选择其他员工,则不会更新。

您的SelectedEmployee网格的示例:

<Grid Name="grid_selectedEmployee" DataContext="{Binding SelectedEmployee, 
UpdateSourceTrigger=PropertyChanged}">...</Grid>

我看到的最大的事情之一是您正在设置属性,而不是绑定它们。

例如,

datagrid_employees.ItemsSource = EmployeeViewModel.EmployeesView;

您正在告诉DataGrid它的ItemsSource应该该特定对象。 您需要将其绑定到该值,以便告诉它指向该属性。 这将使您的UI正确反映ViewModel中的内容

我看到的另一个巨大的危险信号是您的ViewModel引用了一个称为和EmployeeView东西,这使我相信您的View和ViewModel紧密地结合在一起。 ViewModel应该包含所有业务逻辑和代码,而View通常是XAML,并以用户友好的方式简单地反映ViewModel。

View和ViewModel绝不能直接相互引用(在极少数情况下,我的View可以引用ViewModel,但绝对不能相反)

例如, EmployeesViewModel可能包含

  • ObservableCollection<Employee> Employees
  • Employee SelectedEmployee
  • ICommand AddEmployeeCommand
  • ICommand DeleteEmployeeCommand

而您的View(XAML)可能看起来像这样:

<StackPanel>
    <StackPanel Orientation="Horizontal">
        <Button Content="Add" Command="{Binding AddEmployeeCommand}" />
        <Button Content="Delete" Command="{Binding DeleteEmployeeCommand}" />
    </StackPanel>

    <DataGrid ItemsSource="{Binding Employees}"
              SelectedItem="{Binding SelectedEmployee}">
        ... etc
    </DataGrid>

    <UniformGrid DataContext="{Binding SelectedEmployee}" Columns="2" Rows="4">
        <TextBlock Text="ID" />
        <TextBox Text="{Binding Id}" />
        ... etc
    </UniformGrid >
</StackPanel>

而您唯一需要设置的就是整个Window的DataContext。 通常我会覆盖App.OnStartup()来启动我的应用程序:

public partial class App : Application
{
    protected override void OnStartup(StartupEventArgs e)
    {
        base.OnStartup(e);

        var view = new MainWindow();
        var vm = new EmployeesViewModel;
        view.DataContext = vm;
        view.Show();
    }
}

尽管我认为您的情况也可以:

public MainWindow()
{
    InitializeComponent();

    this.DataContext =  new EmployeesViewModel();
}

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM