Проверка ReactiveUI не вызывает уведомление об ошибке и не привела к BindValidation - PullRequest
0 голосов
/ 27 марта 2020

Здравствуйте, Я пытаюсь использовать Пакет ReactiveUI.Validation Для проверки Wpf Simple App и проверки работоспособности i can see it in ViewModel valid state Но я не вижу никаких уведомлений для просмотра элемента, я имею в виду не красный появляются границы или ошибки любого типа, даже при использовании MaterialDesignPackage, Custome Templates nothing works!

ViewModel.cs

public class AddNewUserViewModel : ReactiveValidationObject<AddNewUserViewModel>
{
    private string _email;
    public string Email
    {
        get => _email;
        set => this.RaiseAndSetIfChanged(ref _email, value);
    }

    public ReactiveCommand<Unit,Unit> Start { get; }
    public AddNewUserViewModel()
    {
        var vEmailHelper = this
            .ValidationRule(viewModel => viewModel.Email, e => e?.Length > 2, "Error Email Message");
        var canStart = this.IsValid();
        Start = ReactiveCommand.Create(() => { }, canStart);
    }
}

View.cs

public partial class AddNewUserDialogView : ReactiveWindow<AddNewUserViewModel>
{
    public AddNewUserDialogView()
    {
        ViewModel = new AddNewUserViewModel();
        InitializeComponent();
        this.WhenActivated(disposable =>
        {
            this.Bind(ViewModel, viewModel => viewModel.Email, v => v.EmailTextBox.Text)
                .DisposeWith(disposable);
            this.BindCommand(ViewModel, viewModel => viewModel.Start, v => v.ButStart).DisposeWith(disposable);
        });
    }
}

View. cs.xaml

<reactiveUi:ReactiveWindow x:TypeArguments="dialogs:AddNewUserViewModel"
                           x:Class="UsersManager.UI.Views.Body.Dialogs.AddNewUserDialogView"
                           xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
                           xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
                           xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
                           xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
                           xmlns:local="clr-namespace:UsersManager.UI.Views.Body.Dialogs"
                           xmlns:reactiveUi="http://reactiveui.net"
                           xmlns:dialogs="clr-namespace:UsersManager.Backend.ViewModels.Body.Dialogs;assembly=UsersManager.Backend"
                           mc:Ignorable="d"

                           Title="test" Height="450" Width="800">
    <Grid>
        <Grid.RowDefinitions>
            <RowDefinition Height="2*"/>
            <RowDefinition Height="2*"/>
            <RowDefinition Height="10*"/>
        </Grid.RowDefinitions>

        <TextBox
            Grid.Row="0"
            x:Name="EmailTextBox"
            materialDesign:HintAssist.Hint="Name"
            materialDesign:HintAssist.HelperText="Your user name"
        >
        </TextBox>

        <Button
            Grid.Row="1"
            x:Name="ButStart"
            >Start</Button>
    </Grid>
</reactiveUi:ReactiveWindow>

Если мой EmailText не соответствует правилу, кнопка отключена, и это нормально. Но я не вижу никакого поведения проверки в самом элементе управления TextBox? I try it with ErrorTemplate but not work also

Так как я могу показать ошибку или даже получить какие-либо уведомления об ошибках в представлении

Также обратите внимание: May open it as new question

Когда я пытаюсь связать сообщение проверки указанного свойства c, это Ошибка


public partial class AddNewUserDialogView : ReactiveWindow<AddNewUserViewModel>
{
    public AddNewUserDialogView()
    {
        ViewModel = new AddNewUserViewModel();
        InitializeComponent();
        this.WhenActivated(disposable =>
        {
            this.Bind(ViewModel, viewModel => viewModel.Email, v => v.EmailTextBox.Text)
                .DisposeWith(disposable);
            this.BindCommand(ViewModel, viewModel => viewModel.Start, v => v.ButStart).DisposeWith(disposable);

            // here
            this.BindValidation(ViewModel, viewModel => viewModel.Email, v => v.ButStart.Content).DisposeWith(disposable);
        });
    }
}

С ошибкой:

System.ArgumentOutOfRangeException: Index was out of range. Must be non-negative and less than the size of the collection.

Эта ошибка возникает в CurrentThreadScheduler.cs ReactiveUI в строке 107

Это местоположение ошибки фрагмента кода


// Line 85
 SchedulerQueue<TimeSpan> queue;

            // There is no timed task and no task is currently running
            if (!_running)
            {
                _running = true;

                if (dueTime > TimeSpan.Zero)
                {
                    ConcurrencyAbstractionLayer.Current.Sleep(dueTime);
                }

                // execute directly without queueing
                IDisposable d;
                try
                {
                    d = action(this, state);
                }
                catch
                {
                    SetQueue(null);
                    _running = false;
                    throw; // Line 107
                }


Но

Когда я пытаюсь связать сообщения об ошибках Total / Ultimate ViewModel, это работает!


public partial class AddNewUserDialogView : ReactiveWindow<AddNewUserViewModel>
{
    public AddNewUserDialogView()
    {
        ViewModel = new AddNewUserViewModel();
        InitializeComponent();
        this.WhenActivated(disposable =>
        {
            this.Bind(ViewModel, viewModel => viewModel.Email, v => v.EmailTextBox.Text)
                .DisposeWith(disposable);
            this.BindCommand(ViewModel, viewModel => viewModel.Start, v => v.ButStart).DisposeWith(disposable);

            // here
            this.BindValidation(ViewModel, v => v.ButStart.Content).DisposeWith(disposable);
        });
    }
}

ресурсы: я думаю, может быть полезно
 https://github.com/reactiveui/ReactiveUI.Validation/issues/16
 https://github.com/reactiveui/ReactiveUI.Validation/pull/17
 https://github.com/reactiveui/ReactiveUI.Validation/pull/44

Пакеты


  <ItemGroup>
    <Reference Include="ActiproSoftware.BarCode.Wpf, Version=17.2.663.0, Culture=neutral, PublicKeyToken=36ff2196ab5654b9, processorArchitecture=MSIL" />
    <Reference Include="ActiproSoftware.Shared.Wpf, Version=17.2.663.0, Culture=neutral, PublicKeyToken=36ff2196ab5654b9, processorArchitecture=MSIL" />
    <Reference Include="DynamicData, Version=6.14.0.0, Culture=neutral, PublicKeyToken=null">
      <HintPath>..\packages\DynamicData.6.14.8\lib\net461\DynamicData.dll</HintPath>
      <Private>True</Private>
    </Reference>
    <Reference Include="ICSharpCode.AvalonEdit, Version=5.0.3.0, Culture=neutral, PublicKeyToken=9cc39be672370310, processorArchitecture=MSIL">
      <HintPath>..\packages\AvalonEdit.5.0.4\lib\Net40\ICSharpCode.AvalonEdit.dll</HintPath>
    </Reference>
    <Reference Include="MaterialDesignColors, Version=1.2.2.920, Culture=neutral, processorArchitecture=MSIL">
      <HintPath>..\packages\MaterialDesignColors.1.2.2\lib\net45\MaterialDesignColors.dll</HintPath>
    </Reference>
    <Reference Include="MaterialDesignThemes.Wpf, Version=3.0.1.920, Culture=neutral, processorArchitecture=MSIL">
      <HintPath>..\packages\MaterialDesignThemes.3.0.1\lib\net45\MaterialDesignThemes.Wpf.dll</HintPath>
    </Reference>
    <Reference Include="mscorlib" />
    <Reference Include="Pharmacist.Common, Version=1.5.0.0, Culture=neutral, PublicKeyToken=null">
      <HintPath>..\packages\Pharmacist.Common.1.5.15\lib\netstandard2.0\Pharmacist.Common.dll</HintPath>
      <Private>True</Private>
    </Reference>
    <Reference Include="PresentationFramework.Aero" />
    <Reference Include="ReactiveUI, Version=11.2.0.0, Culture=neutral, PublicKeyToken=null">
      <HintPath>..\packages\ReactiveUI.11.2.3\lib\net461\ReactiveUI.dll</HintPath>
      <Private>True</Private>
    </Reference>
    <Reference Include="ReactiveUI.Events.WPF, Version=11.2.0.0, Culture=neutral, PublicKeyToken=null">
      <HintPath>..\packages\ReactiveUI.Events.WPF.11.2.3\lib\net461\ReactiveUI.Events.WPF.dll</HintPath>
      <Private>True</Private>
    </Reference>
    <Reference Include="ReactiveUI.Fody.Helpers, Version=11.2.0.0, Culture=neutral, PublicKeyToken=null">
      <HintPath>..\packages\ReactiveUI.Fody.11.2.3\lib\net461\ReactiveUI.Fody.Helpers.dll</HintPath>
      <Private>True</Private>
    </Reference>
    <Reference Include="ReactiveUI.Validation, Version=1.4.0.0, Culture=neutral, processorArchitecture=MSIL">
      <HintPath>..\packages\ReactiveUI.Validation.1.4.10\lib\net461\ReactiveUI.Validation.dll</HintPath>
    </Reference>
    <Reference Include="ReactiveUI.WPF, Version=11.2.0.0, Culture=neutral, PublicKeyToken=null">
      <HintPath>..\packages\ReactiveUI.WPF.11.2.3\lib\net461\ReactiveUI.WPF.dll</HintPath>
      <Private>True</Private>
    </Reference>
    <Reference Include="ShowMeTheXAML, Version=1.0.0.0, Culture=neutral, processorArchitecture=MSIL">
      <HintPath>..\packages\ShowMeTheXAML.1.0.12\lib\net45\ShowMeTheXAML.dll</HintPath>
    </Reference>
    <Reference Include="ShowMeTheXAML.AvalonEdit, Version=1.0.0.0, Culture=neutral, processorArchitecture=MSIL">
      <HintPath>..\packages\ShowMeTheXAML.AvalonEdit.1.0.12\lib\net45\ShowMeTheXAML.AvalonEdit.dll</HintPath>
    </Reference>
    <Reference Include="Splat, Version=9.3.0.0, Culture=neutral, PublicKeyToken=null">
      <HintPath>..\packages\Splat.9.3.11\lib\net461\Splat.dll</HintPath>
      <Private>True</Private>
    </Reference>
    <Reference Include="System" />
    <Reference Include="System.Core" />
    <Reference Include="System.Data" />
    <Reference Include="System.Reactive, Version=4.3.0.0, Culture=neutral, PublicKeyToken=94bc3704cddfc263">
      <HintPath>..\packages\System.Reactive.4.3.2\lib\net46\System.Reactive.dll</HintPath>
      <Private>True</Private>
    </Reference>
    <Reference Include="System.Runtime.CompilerServices.Unsafe, Version=4.0.4.1, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a">
      <HintPath>..\packages\System.Runtime.CompilerServices.Unsafe.4.5.2\lib\netstandard2.0\System.Runtime.CompilerServices.Unsafe.dll</HintPath>
      <Private>True</Private>
    </Reference>
    <Reference Include="System.Threading.Tasks.Extensions, Version=4.2.0.1, Culture=neutral, PublicKeyToken=cc7b13ffcd2ddd51">
      <HintPath>..\packages\System.Threading.Tasks.Extensions.4.5.3\lib\netstandard2.0\System.Threading.Tasks.Extensions.dll</HintPath>
      <Private>True</Private>
    </Reference>
    <Reference Include="System.ValueTuple, Version=4.0.3.0, Culture=neutral, PublicKeyToken=cc7b13ffcd2ddd51">
      <HintPath>..\packages\System.ValueTuple.4.5.0\lib\net47\System.ValueTuple.dll</HintPath>
      <Private>True</Private>
    </Reference>
    <Reference Include="System.Windows" />
    <Reference Include="System.Windows.Forms" />
    <Reference Include="System.Xml" />
    <Reference Include="System.Xaml">
      <RequiredTargetFramework>4.0</RequiredTargetFramework>
    </Reference>
    <Reference Include="WindowsBase" />
    <Reference Include="PresentationCore" />
    <Reference Include="PresentationFramework" />
  </ItemGroup>

Обновления

Решаю вторую проблему с очень странной модификацией

Это не должно рассматриваться как реальное решение

Я просто слушаю подтверждение изменения статуса

В представлении

ViewModel.ValidationContext.ValidationStatusChange.Subscribe(x =>
                {
                    // if i omite next line , the solution will not work
                    MessageBox.Show("Start");
                });

И теперь появляется сообщение проверки

 // so: now this line will not produce argument out of range exception
 this.BindValidation(ViewModel, viewModel => viewModel.Email, v => v.ButStart.Content).DisposeWith(disposable);

Также обратите внимание: если я слушаю напрямую из viewModel, решение также будет работать

I tested it in another project

Но здесь я отделяю это, потому что я использую различные проекты из UI / Backe nd contains ViewModels

Также обратите внимание, что это будет работать, если я слушаю любой другой валидатор

Так что если я выставлю электронную почту ValidationHelper как свойство вне ViewModel и подпишусь на него, это будет работать,

Также если Я просто подписываюсь / слушаю электронную почту ValidationHelper из ViewModel, это также будет работать

Но помните, мне все еще нужно вызвать MessageBox.Show ("")

Примечание: это будет работать, если я вызову ShowDialog () в любом окне, даже MainWindow

Так что следующее решение также будет работать

ViewModel.ValidationContext.ValidationStatusChange.Subscribe(x =>
                {
                    var v = new MainWindow();
                    v.ShowDialog();
                });

1 Ответ

0 голосов
/ 03 апреля 2020

Привязка проверки к шаблону элемента управления в настоящее время не поддерживается, но вы можете отобразить сообщение об ошибке проверки в представлении, если привязать его к отдельному текстовому блоку ( Документация ).

Если вы просто хотите использовать MaterialDesign HelperText для отображения сообщения об ошибке, вам не нужно использовать ReactiveUI.Validation, но вы можете сделать что-то вроде этого:

    // Viewmodel
    private ObservableAsPropertyHelper<bool> isEmailValid;
    public bool IsEmailValid => isEmailValid.Value;
    ...
    isEmailValid = this.WhenAnyValue(x => x.Email).Select(e => e?.Length > 2).ToProperty(this, nameof(IsEmailValid));

    // View
    this.WhenAnyValue(x => x.ViewModel.IsEmailValid)
        .Do(isValid => HintAssist.SetHint(EmailTextBox, isValid ? string.Empty : "Error Email Message"))
        .Subscribe();

Ваша вторая проблема должна быть исправлена ​​при использовании последняя версия ReactiveUI.Validation (1.4.13) .

...