org.zkoss.bind.validator.BeanValidator Maven / Gradle / Ivy
The newest version!
/* BeanValidator.java
Purpose:
Description:
History:
2011/12/22 Created by Dennis Chen
Copyright (C) 2011 Potix Corporation. All Rights Reserved.
*/
package org.zkoss.bind.validator;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;
import java.util.List;
import java.util.Set;
import jakarta.validation.ConstraintViolation;
import jakarta.validation.Validator;
import org.zkoss.bind.Property;
import org.zkoss.bind.ValidationContext;
/**
* A JSR 303 compatible validator for a property-binding.
* Notice : Before use this validator, you have to configure your environment (depends on the implementation you chosen).
* Here is a article Integrate ZK with JSR 303: Bean Validation
* talks about how to set up JSR 303 in ZK with Hibernate implementation.
*
* It validates a single property of a bean and sets the invalid message by
* {@link AbstractValidator#addInvalidMessage(ValidationContext, String)}.
* Since 8.5, you can assign a self-defined message key by appending a key parameter.
*
*
* To use this class, you have to add @validator('beanValidator')
or @validator('org.zkoss.bind.validator.BeanValidator')
to the property-binding
*
* Example
* {@code
*
*
*
*
* }
* {@code
*
*
*
*
* }
*
* Note
* It doesn't supports to validate a property of a form which properties are load from a bean,
* if you want to validate the form property of the bean, you could use org.zkoss.zkmax.bind.BeanValodator
* @author dennis
* @since 6.0.0
*/
public class BeanValidator extends AbstractValidator {
protected Validator getValidator() {
return BeanValidations.getValidator();
}
/**
* Validate the value
* @param clz the class of bean
* @param property the property of bean
* @param value the value to be validated.
* @param groups the validation groups (since 8.0.0)
* @return the ConstraintViolation set.
* @since 6.0.1
*/
@SuppressWarnings({ "unchecked", "rawtypes" })
protected Set> validate(Class clz, String property, Object value, Class>[] groups) {
if (groups != null)
return getValidator().validateValue(clz, property, value, groups);
return getValidator().validateValue(clz, property, value);
}
public void validate(ValidationContext ctx) {
final Property p = ctx.getProperty();
final Object base = p.getBase();
String property = p.getProperty();
final Object value = p.getValue();
Object[] info = getValidationInfo(ctx, base, property);
Class> clz = (Class>) info[0];
property = (String) info[1];
final Class>[] groups = (Class>[]) ctx.getValidatorArg("groups");
Set> violations = validate(clz, property, value, groups);
handleConstraintViolation(ctx, violations);
}
/**
* Sort the violations, make multiple violation order more predictable.
* By default, sort it by the constraint name.
* @param viloations
*/
protected void sort(List> viloations) {
Collections.sort(viloations, new Comparator>() {
public int compare(ConstraintViolation> o1, ConstraintViolation> o2) {
String s1 = o1.getConstraintDescriptor().getAnnotation().toString();
String s2 = o2.getConstraintDescriptor().getAnnotation().toString();
return s1.compareTo(s2);
}
});
}
/**
* Get the bean class of the base object and property to validate.
* By default, it returns the class of base object directly.
* @param ctx the validation context
* @param base the base object
* @param property the property to validate
* @return a object array, the first item is the bean class of base object, 2nd item is the tuned property regards to the bean class
* @since 6.0.1
*/
protected Object[] getValidationInfo(ValidationContext ctx, Object base, String property) {
Class> clz = base.getClass();
return new Object[] { clz, property };
}
/**
* Handle hibernate ConstraintViolation. by default, it add to invalid messages.
* @param ctx
* @param violations
* @since 6.0.1
*/
protected void handleConstraintViolation(ValidationContext ctx, Set> violations) {
final String key = (String) ctx.getValidatorArg("key");
final int s = violations.size();
if (s == 1) {
addInvalidMessage(ctx, key, violations.iterator().next().getMessage());
} else if (s > 0) {
String[] msgs = new String[violations.size()];
// it is a Set, I tested in hibernate 4 , it doesn't guarantee the
// order, so I sort it by annottion.toString
List> l = new ArrayList>(violations);
sort(l);
for (int i = 0; i < msgs.length; i++) {
msgs[i] = l.get(i).getMessage();
}
addInvalidMessages(ctx, key, msgs);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy