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

how to validate that a field is the same as another? #97

Closed
csantos1113 opened this issue Jun 5, 2017 · 19 comments
Closed

how to validate that a field is the same as another? #97

csantos1113 opened this issue Jun 5, 2017 · 19 comments

Comments

@csantos1113
Copy link

How can I have rules to validate that the user wrote the same account and confirmedAccount?

const inst = yup.object({
  account: yup.string().min(7).required(),
  confirmedAccount: yup.string().oneOf([yup.ref('account')])
 });
inst.validate({ account: '123456', confirmedAccount: '234' }, {
	abortEarly: false
}).then(function() { 
	console.log('ok', arguments);
})
.catch(function(error) {
	console.log('failed', error);
});

console.log(inst.cast({ account: '123456', confirmedAccount: '234' }));

Looks like yup.ref only works for casting data, not for validations?
So, how can I validate it?

@csantos1113
Copy link
Author

FYI: I created this extended validation that looks working:

function equalTo(ref, msg) {
	return this.test({
		name: 'equalTo',
		exclusive: false,
    message: msg || '${path} must be the same as ${reference}',
		params: {
			reference: ref.path
		},
		test: function(value) {
      return value === this.resolve(ref) 
		}
	})
};

yup.addMethod(yup.string, 'equalTo', equalTo);


 const inst = yup.object({
   account: yup.string().min(7).required(),
   confirmedAccount: yup.string().equalTo(yup.ref('account'))
 });

inst.validate({ account: '1234567', confirmedAccount: '12345678' }, {
	abortEarly: false
}).then(function() { 
	console.log('ok', arguments);
})
.catch(function(error) {
	console.log('failed', error);
})

@jquense
Copy link
Owner

jquense commented Jun 6, 2017

ah yeah I should note that in the documentation ,but yes that's the correct way to do it.

@vuhrmeister
Copy link

vuhrmeister commented Jan 2, 2018

I already posted (accidentally) here: jaredpalmer/formik#90 (comment)

To repeat my question: Did anything change in the way how ref() or resolve() works? It doesn't work for me (see linked issue comment).

@jquense
Copy link
Owner

jquense commented Jan 2, 2018

@vuhrmeister make sure you aren't using an arrow function for test:, nothing has changed, but it sounds like you getting schema.resolve which is a different method from test.resolve

@vuhrmeister
Copy link

Oh man … it can be so easy :)
Thanks, that was the point.

Anyhow: the docs doesn't say anything about test.resolve. Also I think the part about ref could be a bit more extended, maybe some examples. I don't think ref is such a rare use case.

@jquense
Copy link
Owner

jquense commented Jan 2, 2018

happy to take some doc PR's :)

@ghost
Copy link

ghost commented Mar 28, 2018

what is going on here i'm not getting ref value any ideas? https://codesandbox.io/s/zqlj1qp03p?expanddevtools=1

@jquense
Copy link
Owner

jquense commented Mar 28, 2018

@trisox refs work for sibilings and child paths, they can't walk up to parents

@ghost
Copy link

ghost commented Mar 28, 2018

@jquense / @jaredpalmer ah oke any tips for that?


i gave the whole object that needed to be validated as context...

https://github.com/mariocasciaro/object-path
then doing objectPath.get(this.options.context, ref.path) returns parent value.

@Yaojian
Copy link

Yaojian commented Dec 27, 2018

#49 (comment) has a solution:

password: yup.string().required('Password is required'),
confirmPassword: yup.string().oneOf([yup.ref('password'), null], "Passwords don't match").required('Confirm Password is required'),

@avinashreddy24
Copy link

where exactly can I add Yup.addmethod

I am not able to figure out how to add this Yup.addmethod and validate can some one help me Thanks

Below Code is written in single component file component.js Just looking how to compare fields by using addmethod

class Component extends React.Componenet {
render (){
return (

) } }

const formikEnhancer =withFormik({
validSchem :Yup.object().shape({
field1:Yup.string.required(),
field2:Yup.string.required()
}
})

const MyForm =props => {
const {
values,
touched,
dirty,
errors,
handleChange,
handleBlur,
handleSubmit,
handleReset,
setFieldValue,
setFieldTouched,
isSubmitting
} = props;
return (


/code for UI/

);
};
const MyEnhancedForm =formikEnhancer(MyForm);

export default connect (component);

@bradenmitchell
Copy link

Solution provided in #97 comment no longer works.

In the message parameter, ${path} and ${reference} need .this in front or node throws a Reference error that path (or reference) is not defined.

When this. is added, the error message returns undefined must be the same as undefined, not the desired confirmedAccount must be equal to account.

see related post on stack overflow.

@Axolodev
Copy link

Was having issues with the solution provided in this comment

Fixed this by adding a test:

  passwordValidation: yup
    .string()
    .test(
      'equal',
      'Passwords do not match!',
      function(v) { // Don't use arrow functions
        const ref = yup.ref('password');
        return v !== this.resolve(ref);
      }
    )

Works on v0.28.4 with formik 2.1.4

@shrmaky
Copy link

shrmaky commented Oct 14, 2020

@fponce-byu
Copy link

This was really helpful!! Thanks

@vendramini
Copy link

export const passwordSchema = object({
  currentPassword: string().optional(),
  newPassword: string().when('currentPassword', {
    is: (val: string) => val !== '',
    then: schema => schema.required(),
  }),
  newPasswordCheck: string().when('newPassword', {
    is: (val: string) => val !== '',
    then: schema => schema.required().oneOf([ref('newPassword')]),
  }),
});

@MatteoGauthier
Copy link

Finally, what's the recommended way ? :D

@crazyoptimist
Copy link

crazyoptimist commented Sep 5, 2022

Was having issues with the solution provided in this comment

Fixed this by adding a test:

  passwordValidation: yup
    .string()
    .test(
      'equal',
      'Passwords do not match!',
      function(v) { // Don't use arrow functions
        const ref = yup.ref('password');
        return v !== this.resolve(ref);
      }
    )

Works on v0.28.4 with formik 2.1.4

Thanks for sharing this!

And in the above snippet, it should be return v === this.resolve(ref);

@alissoonluan
Copy link

export const passwordSchema = object({
  currentPassword: string().optional(),
  newPassword: string().when('currentPassword', {
    is: (val: string) => val !== '',
    then: schema => schema.required(),
  }),
  newPasswordCheck: string().when('newPassword', {
    is: (val: string) => val !== '',
    then: schema => schema.required().oneOf([ref('newPassword')]),
  }),
});

Thanks you !!

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests