WPF - 一个 ViewModel 与多个 Model 实例进行交互

3
我有一个WorkspaceViewModel,通过ObservableCollection动态处理选项卡项目的添加和删除。每次将选项卡连接到PayslipModel时,所有绑定都可以正常工作,但我遇到的一个问题是;
在UserControl中我有一个保存按钮,它的DataContext设置为WorkspaceViewModel,我想保存所选选项卡中显示的任何信息。现在,每次添加选项卡时,都会创建一个新的PayslipModel实例,这正是我想要的,因为我不希望所有选项卡共享绑定。但是,由于PayslipModel有多个实例,因此无法保存显示的内容(临时使用MessageBox测试是否检索到信息),因此不返回任何内容当我点击保存时。
我创建了一个图表来更好地解释我的情况: enter image description here 是否可能在选择选项卡时访问当前实例或循环遍历所有实例并执行类似批量保存的操作?

1
ItemsControls有一个ItemsSource和SelectedItem。在TabControl中,SelectedItem是当前显示的选项卡。这不足以告诉您正在编辑哪个PayslipModel吗? - user1228
@Will,我正在尝试通过ViewModel实现保存功能,因为我使用命令来绑定按钮。这是否意味着对于这种特殊情况,无法通过ViewModel来完成? - bruh1234
@Patrick,同样的问题^ - bruh1234
如果我理解正确的话,按钮后面的代码无法访问TabControl,因此需要将ViewModel中的属性绑定到TabControl的SelectedItem上,然后在按钮的代码后台查看该属性。 - patrick
@Patrick,那似乎也不起作用,所以我考虑将保存按钮移动到具有选项卡内容的视图内部,这样它就可以直接访问属性。它确实会破坏布局的流程,但现在我可以处理它。 - bruh1234
显示剩余2条评论
1个回答

1
这是一个工作示例,展示了其中的一种可能性:
视图
    <TabControl DataContext="{Binding}" ItemsSource="{Binding Models}"  >
        <TabControl.ItemTemplate>
            <DataTemplate>
                <TextBlock Text="{Binding Name}" > 
                </TextBlock>
            </DataTemplate>
        </TabControl.ItemTemplate>
        <TabControl.ContentTemplate>
            <DataTemplate>
                <DockPanel>
                    <Button DockPanel.Dock="Top" Content="Click Me" Command="{Binding DataContext.PCommand, 
                                                            RelativeSource={RelativeSource Mode=FindAncestor, AncestorType=TabControl}}"
                            CommandParameter="{Binding Desc}"/>
                    <TextBlock Text="{Binding Desc}" >
                    </TextBlock>
                </DockPanel> 
            </DataTemplate>
        </TabControl.ContentTemplate>
    </TabControl>

模型视图
public class ModelView
{
    public ModelView()
    {
        _models = new ObservableCollection<Model>(); 
        _pCommand = new Command(DoParameterisedCommand);
    }
    ObservableCollection<Model> _models;
    public ObservableCollection<Model> Models { get { return _models; } }

    private void DoParameterisedCommand(object parameter)
    {
        MessageBox.Show("Parameterised Command; Parameter is '" +
                     parameter.ToString() + "'.");
    }
    Command _pCommand;
    public Command PCommand
    {
        get { return _pCommand; }
    }
}

模型

public class Model : INotifyPropertyChanged
{
    string _desc;
    public string Desc { get { return _desc; } set { _desc = value; RaisePropertyChanged("Desc"); } }

    string _name;
    public string Name { get { return _name; } set { _name = value; RaisePropertyChanged("Name"); } }

    public event PropertyChangedEventHandler PropertyChanged;
    void RaisePropertyChanged(string propname)
    {
        PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propname));
    }
}

命令

public class Command : ICommand
{ 
    public Command(Action<object> parameterizedAction, bool canExecute = true)
    {
        _parameterizedAction = parameterizedAction;
        _canExecute = canExecute;
    }


    Action<object> _parameterizedAction = null;
    bool _canExecute = false;
    public bool CanExecute
    {
        get { return _canExecute; }
        set
        {
            if (_canExecute != value)
            {
                _canExecute = value;
                CanExecuteChanged?.Invoke(this, EventArgs.Empty);
            }
        }
    }
    public event EventHandler CanExecuteChanged;
    bool ICommand.CanExecute(object parameter)
    {
        return _canExecute;
    }

    void ICommand.Execute(object parameter)
    {
        this.DoExecute(parameter);
    }
    public virtual void DoExecute(object param)
    { if (_parameterizedAction != null)
            _parameterizedAction(param);
        else
            throw new Exception();
    }


}

使用这个来初始化:
    public MainWindow()
    {
        InitializeComponent();
        ModelView mv = new ModelView();
        mv.Models.Add(new Model() { Name = "a", Desc = "aaa" });
        mv.Models.Add(new Model() { Name = "b" , Desc = "bbb"});
        mv.Models.Add(new Model() { Name = "c", Desc = "cccc" });
        this.DataContext = mv; 

    }

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接