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: Add ValidationRule extensions method overloads to accept any object implementing IValidationState #147

Closed
thargy opened this issue Oct 20, 2020 · 1 comment · Fixed by #148

Comments

@thargy
Copy link
Contributor

thargy commented Oct 20, 2020

Is your feature request related to a problem? Please describe.
The ValidationRule extension methods added in #130 only accept and IValidationState, more usefully, they should accept any object/struct that implements that interface (including ValidationState).

Describe the solution you'd like
Adding the following 2 overloads will significantly increase the usability of the API:

        /// <summary>
        /// Setup a validation rule with a general observable based on <see cref="IValidationState"/>.
        /// </summary>
        /// <typeparam name="TViewModel">ViewModel type.</typeparam>
        /// <typeparam name="TValue">Validation observable type.</typeparam>
        /// <param name="viewModel">ViewModel instance.</param>
        /// <param name="validationObservable">Observable to define if the viewModel is valid or not.</param>
        /// <returns>Returns a <see cref="ValidationHelper"/> object.</returns>
        /// <remarks>
        /// It should be noted that the observable should provide an initial value, otherwise that can result
        /// in an inconsistent performance.
        /// </remarks>
        public static ValidationHelper ValidationRule<TViewModel, TValue>(
            this TViewModel viewModel,
            IObservable<TValue> validationObservable)
            where TViewModel : IReactiveObject, IValidatableViewModel
            where TValue : IValidationState
        {
            if (viewModel is null)
            {
                throw new ArgumentNullException(nameof(viewModel));
            }

            if (validationObservable is null)
            {
                throw new ArgumentNullException(nameof(validationObservable));
            }

            return viewModel.RegisterValidation(
                new ObservableValidation<TViewModel, bool>(
                    validationObservable.Select(s => s as IValidationState)));
        }

        /// <summary>
        /// Setup a validation rule with a general observable based on <see cref="IValidationState"/>.
        /// </summary>
        /// <typeparam name="TViewModel">ViewModel type.</typeparam>
        /// <typeparam name="TViewModelProp">ViewModel property type.</typeparam>
        /// <typeparam name="TValue">Validation observable type.</typeparam>
        /// <param name="viewModel">ViewModel instance.</param>
        /// <param name="viewModelProperty">ViewModel property referenced in viewModelObservableProperty.</param>
        /// <param name="validationObservable">Observable to define if the viewModel is valid or not.</param>
        /// <returns>Returns a <see cref="ValidationHelper"/> object.</returns>
        /// <remarks>
        /// It should be noted that the observable should provide an initial value, otherwise that can result
        /// in an inconsistent performance.
        /// </remarks>
        public static ValidationHelper ValidationRule<TViewModel, TViewModelProp, TValue>(
            this TViewModel viewModel,
            Expression<Func<TViewModel, TViewModelProp>> viewModelProperty,
            IObservable<TValue> validationObservable)
            where TViewModel : IReactiveObject, IValidatableViewModel
            where TValue : IValidationState
        {
            if (viewModel is null)
            {
                throw new ArgumentNullException(nameof(viewModel));
            }

            if (viewModelProperty is null)
            {
                throw new ArgumentNullException(nameof(viewModelProperty));
            }

            if (validationObservable is null)
            {
                throw new ArgumentNullException(nameof(validationObservable));
            }

            return viewModel.RegisterValidation(
                new ObservableValidation<TViewModel, bool, TViewModelProp>(
                    viewModelProperty, validationObservable.Select(v => v as IValidationState)));
        }

These will now accept observables such as IObservable<ValidationState.

Describe alternatives you've considered
Without these methods consumers need to write code like this:

IObservable<ValidationState> observable = ...;
model.ValidationRule(observable.Select(v => v as IValidationState));

Describe suggestions on how to achieve the feature
Implementation above.

thargy added a commit to thargy/ReactiveUI.Validation that referenced this issue Oct 20, 2020
worldbeater pushed a commit that referenced this issue Oct 20, 2020
… object implementing IValidationState (#148)

* resolves #147
* Added tests for new overloads.
@github-actions
Copy link

This issue has been automatically locked since there has not been any recent activity after it was closed. Please open a new issue for related bugs.

@github-actions github-actions bot locked as resolved and limited conversation to collaborators Nov 25, 2021
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Labels
None yet
Projects
None yet
1 participant