繁体   English   中英

ReactiveUI:从自定义对话框返回值

[英]ReactiveUI: Return value from a custom dialog

我有一个窗口,使用ReactiveUI Interaction打开第二个窗口作为模式对话框,然后从第二个窗口中的ListBox返回数据。

问题是当.ShowDialog()完成时,ViewModel的SelectedItem总是求值为null。 我已经确认绑定工作正常,所选项目正在对话框窗口的ViewModel中正确更新。 只有当我返回到Interaction时,属性才会重置为默认值(null)。

我在这里汇总了一个最小的问题示例:

https://github.com/replicaJunction/ReactiveDialogTest

测试中的主要逻辑是在MainWindowViewModel.cs中。

编辑:这是代码中的一个基本概念片段:

GetNumberFromDialog = new Interaction<Unit, int>();
GetNumberFromDialog.RegisterHandler(interaction =>
{
    var vm = new DialogWindowViewModel();

    // Get a reference to the view for this VM
    var view = Locator.Current.GetService<IViewFor<DialogWindowViewModel>>();

    // Set its VM to our current reference
    view.ViewModel = vm;

    var window = view as Window;
    var dialogResult = window.ShowDialog();

    // At this point, vm.SelectedNumber is expected be the number the user selected -
    // but instead, it always evaluates to 0.

    if (true == dialogResult)
        interaction.SetOutput(vm.SelectedNumber);
    else
        interaction.SetOutput(-1);
});

OpenDialog = ReactiveCommand.Create(() =>
{
    GetNumberFromDialog.Handle(Unit.Default)
        .Where(retVal => -1 != retVal) // If the dialog did not return true, don't update
        .Subscribe(retVal =>
        {
            this.MyNumber = retVal;
        });
});

重现问题的步骤:

  1. 运行该项目。 请注意带有-5000的标签 - 这是要更新的数字。
  2. 单击“打开对话框”按钮。 将打开一个对话窗口。
  3. 在ListBox中选择一个数字。 请注意,“当前选择”下的标签更新 - 这与ListBox.SelectedItem绑定的值相同。
  4. 单击确定。 对话框关闭。

预期行为:“我的号码是”下方主窗口中的标签应更新为您在ListBox中选择的值。

实际行为:标签更新为0(int的默认值)。

当对话框关闭时,为什么我的ViewModel会自行重置?

在GitHub上查看您的示例有助于揭示问题。 您的DialogWindow如下所示:

public partial class DialogWindow : Window, IViewFor<DialogWindowViewModel>
{
    public DialogWindow()
    {
        InitializeComponent();
        this.ViewModel = new DialogWindowViewModel();
        this.DataContext = this.ViewModel;

        this.ViewModel
            .WhenAnyValue(x => x.DialogResult)
            .Where(x => null != x)
            .Subscribe(val =>
            {
                this.DialogResult = val;
                this.Close();
            });
    }

    public DialogWindowViewModel ViewModel { get; set; }
    object IViewFor.ViewModel
    {
        get => ViewModel;
        set => ViewModel = (DialogWindowViewModel)value;
    }
}

在你MainWindowViewModel ,您将DialogWindow.ViewModel属性为您的实例DialogWindowViewModel 这个问题在这一点上很突出。 你的问题是,设置DialogWindow.ViewModel属性设置视图的DataContext或重新创建WhenAnyValue观察到。 这意味着视图仍然绑定到DialogWindowViewModel实例(在DialogWindow构造函数中创建的实例)上的SelectedNumber属性。 要修复上面的示例代码,您可以简单地避免设置ViewModel属性,并使用已在对话框中设置的ViewModel:

GetNumberFromDialog.RegisterHandler(interaction =>
{
    // Get a reference to the view for this VM
    var view = Locator.Current.GetService<IViewFor<DialogWindowViewModel>>();

    var window = view as Window;
    var dialogResult = window.ShowDialog();

    // use the ViewModel here that's already set on the DialogWindow
    if (true == dialogResult)
        interaction.SetOutput(view.ViewModel.SelectedNumber);
    else
        interaction.SetOutput(-1);
});

暂无
暂无

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

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