Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feature: Use IValidationState Instead of ValidationState #130

Merged
merged 12 commits into from
Oct 17, 2020

Conversation

worldbeater
Copy link
Collaborator

@worldbeater worldbeater commented Oct 16, 2020

What kind of change does this PR introduce?

This PR implements the API specification eloquently argued and formulated by @thargy in #123 (comment)

What is the current behavior?

Currently, there is no way to customize a ValidationState.
Also, there is no ValidationRule overload accepting a ValidationState.
See #123 for more information regarding this topic.

What is the new behavior?

Now, we are using IValidationState where possible, and this simplifies the use of complex and extended validation rules. Now we have two more overloads for the ValidationRule extension method, one of the rules accepts a view model and IObservable<IValidationState>, and another one accepts a view model, a property, and an IObservable<IValidationState>. This allows ReactiveUI.Validation users to fully control how an object implementing IValidtionState is created.

What might this PR break?

One potential breaking change is described in #123 (comment) e.g. there is no .Component property on the new IValidationState interface, so the code relying on that property might break. Hopefully, there aren't many people who are relying on that .Component property. And if they do, a potential workaround would be to just cast IObservable<IValidationState> to IObservable<ValidationState> by adding a call to .OfType<ValidationState>(), e.g:

IValidationComponent validation = new ValidationContext();
validation.ValidationStatusChange.OfType<ValidationState>() // IValidationState -> ValidationState

However, we discourage the use of ValidationState.Component obsolete property as we are going to remove it in the future.

Please check if the PR fulfills these requirements

  • Tests for the changes have been added (the work is in progress)
  • Docs have been added / updated

Other information:

We need to carefully review the API changes to avoid introducing additional breaking changes.

@codecov
Copy link

codecov bot commented Oct 16, 2020

Codecov Report

Merging #130 into main will increase coverage by 0.04%.
The diff coverage is n/a.

Impacted file tree graph

@@            Coverage Diff             @@
##             main     #130      +/-   ##
==========================================
+ Coverage   61.52%   61.57%   +0.04%     
==========================================
  Files          16       16              
  Lines         863      877      +14     
==========================================
+ Hits          531      540       +9     
- Misses        332      337       +5     
Impacted Files Coverage Δ
Comparators/ValidationStateComparer.cs 37.50% <0.00%> (-6.95%) ⬇️
Extensions/ValidatableViewModelExtensions.cs 72.41% <0.00%> (-0.59%) ⬇️
Components/ObservableValidation.cs 92.50% <0.00%> (-0.10%) ⬇️
States/ValidationState.cs 100.00% <0.00%> (ø)
Helpers/ReactiveValidationObject.cs 97.36% <0.00%> (+0.07%) ⬆️

Continue to review full report at Codecov.

Legend - Click here to learn more
Δ = absolute <relative> (impact), ø = not affected, ? = missing data
Powered by Codecov. Last update 86e9e8a...93e0f7e. Read the comment docs.

@thargy
Copy link
Contributor

thargy commented Oct 16, 2020

It's hard to keep up with you @worldbeater! I've had my head down on some other code today, so I only just noticed you'd pushed this. I've gone through and reviewed the code changes and I think you've struck the right balance between backwards compatibility and making, what I feel, is a fundamentally significant improvement. Not only will this positively impact extensibility it should impact performance. I look forward to the NuGet release so I can rip out a bunch of existing workaround code. I'll definitely use the IValidationState to efficiently drive async validation directly.

@worldbeater
Copy link
Collaborator Author

worldbeater commented Oct 16, 2020

Btw @thargy, how do you think, should we remove the generic version of IPropertyValidationComponent<TViewModel> completely? Having the generic version of this interface could help in compile-time generic type resolution https://github.com/reactiveui/ReactiveUI.Validation/blob/main/src/ReactiveUI.Validation/Components/Abstractions/IPropertyValidationComponent.cs#L18 in extension methods like .ContainsProperty<TViewModel, TProp>

public static bool ContainsProperty<TViewModel, TProp>(
Also probably this could encourage people to follow the pattern "one validation component can only contain validations for one and only one view model".

GitHub
Validation helpers for ReactiveUI based solutions, functioning in reactive way. - reactiveui/ReactiveUI.Validation

@thargy
Copy link
Contributor

thargy commented Oct 16, 2020

@worldbeater we're overlapping a bit. Just created a PR for Readme update (#131), but whilst doing it, I noticed that, now that ValidationState.Component is redundant (and can be null). There's a strong case for adding:

public static readonly ValidationState Valid = new ValidationState(true, string.Empty);

To ValidationState to allow for quickly, and efficiently, returning ValidationState.Valid.

I've made the necessary changes in #132.

Proposed update to indicate use of `IValidationState` or `ValidationState` with `ValidationRule`.
@thargy
Copy link
Contributor

thargy commented Oct 16, 2020

Btw @thargy, how do you think, should we remove the generic version of IPropertyValidationComponent completely? Having the generic version of this interface could help in compile-time generic type resolution https://github.com/reactiveui/ReactiveUI.Validation/blob/main/src/ReactiveUI.Validation/Components/Abstractions/IPropertyValidationComponent.cs#L18 in extension methods like .ContainsProperty<TViewModel, TProp>

Actually, having the interface would mean that the extension method will only work with IPropertyValidationComponents that have implemented the generic interface, when, in reality, any component that implements that non-generic interface is capable of being consumed by the extension method.

As the extension method is largely only used internally, you get the problem where it might not be obvious to those implementing components why their component isn't working (i.e. they didn't explicitly implement the empty generic interface. Also, the generic type parameter would be unused in their component code. As such it isn't a promising idea IMHO, and I'd be in favour of dropping the generic interface entirely, as it is redundant.

Also probably this could encourage people to follow the pattern "one validation component can only contain validations for one and only one view model".

This is the strongest argument, but I don't personally feel it justifies the potential confusion. Again, the extension method is largely used by the library, not consumers or implementors. If you wish to force a component to be bound to a view model type that should be part of the component design and logically it would require the generic interface to have a TViewModel ViewModel { get; } property, forcing implementors to provide it. Fundamentally, that property isn't a required, or used, at the moment, so adding it would require more memory, for no use. This is a strong indicator that the interface is not necessary.

Although there are scenarios where having an empty interface is of use, in general, it is a strong indicator that a careful review is required to see if the interface actually provides value. Interfaces (and particularly generic ones) are not free to use. In this case, my view is that it is redundant and should be obsolesced.

GitHub
Validation helpers for ReactiveUI based solutions, functioning in reactive way. - reactiveui/ReactiveUI.Validation

thargy added a commit to thargy/ReactiveUI.Validation that referenced this pull request Oct 16, 2020
Added static Valid property, for `ValidationState.Valid` as mentioned in reactiveui#130 (comment)
* Update ValidationState.cs

Added static Valid property, for `ValidationState.Valid` as mentioned in #130 (comment)

* Update README.md

Updated document to use `ValidationState.Valid` as introduced in #131

* Replaced construction of valid validation states with `ValidationState.Valid`.

* Added `ValidationState.Valid` to public API.
@worldbeater
Copy link
Collaborator Author

worldbeater commented Oct 17, 2020

Going to spend some time on improving the public API surface of ObservableValidation before merging this. Suddenly realized that I introduced some breaking changes in ObservableValidationBase yesterday that aren't required indeed. Thanks @thargy for helping out with the API design!

@worldbeater worldbeater marked this pull request as ready for review October 17, 2020 13:00
@worldbeater worldbeater merged commit e6aadc5 into main Oct 17, 2020
@worldbeater worldbeater deleted the ivalidationstate branch October 17, 2020 14:24
@reactiveui reactiveui locked as resolved and limited conversation to collaborators Nov 26, 2021
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants