org.apache.wicket.bean.validation.BeanValidationConfiguration Maven / Gradle / Ivy
package org.apache.wicket.bean.validation;
import java.lang.annotation.Annotation;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.function.Supplier;
import javax.validation.Validator;
import javax.validation.constraints.Size;
import org.apache.wicket.Application;
import org.apache.wicket.MetaDataKey;
import org.apache.wicket.markup.html.form.FormComponent;
import org.apache.wicket.util.lang.Args;
/**
* Configures bean validation and integrates it with Wicket
*
* @author igor
*
*/
public class BeanValidationConfiguration implements BeanValidationContext
{
private static final MetaDataKey KEY = new MetaDataKey()
{
};
private Supplier validatorProvider = new DefaultValidatorProvider();
private IViolationTranslator violationTranslator = new DefaultViolationTranslator();
private List propertyResolvers = new CopyOnWriteArrayList<>();
private Map, ITagModifier extends Annotation>> tagModifiers = new ConcurrentHashMap<>();
public BeanValidationConfiguration()
{
add(new DefaultPropertyResolver());
register(Size.class, new SizeTagModifier());
}
/**
* Registeres a tag modifier for a specific constraint annotation
*
* @param annotationType
* constraint annotation such as {@link Size}
* @param modifier
* tag modifier to use
* @return {@code this}
*/
public BeanValidationConfiguration register(Class annotationType,
ITagModifier modifier)
{
Args.notNull(annotationType, "annotationType");
Args.notNull(modifier, "modifier");
tagModifiers.put(annotationType, modifier);
return this;
}
@SuppressWarnings("unchecked")
public ITagModifier getTagModifier(Class annotationType)
{
Args.notNull(annotationType, "annotationType");
return (ITagModifier)tagModifiers.get(annotationType);
}
/**
* Adds a property resolver to the configuration. Property resolvers registered last are the
* first to be allowed to resolve the property.
*
* @param resolver
* @return {@code this}
*/
public BeanValidationConfiguration add(IPropertyResolver resolver)
{
Args.notNull(resolver, "resolver");
// put newly added resolvers in the beginning so its possible to override previously added
// resolvers with newer ones
propertyResolvers.add(0, resolver);
return this;
}
/**
* @return validator
*/
@Override
public Validator getValidator()
{
return validatorProvider.get();
}
/**
* Sets the provider used to retrieve {@link Validator} instances
*
* @param validatorProvider
*/
public void setValidatorProvider(Supplier validatorProvider)
{
Args.notNull(validatorProvider, "validatorProvider");
this.validatorProvider = validatorProvider;
}
/**
* Binds this configuration to the application instance
*
* @param application
*/
public void configure(Application application)
{
application.setMetaData(KEY, this);
}
/** @return registered violation translator */
@Override
public IViolationTranslator getViolationTranslator()
{
return violationTranslator;
}
/**
* Registers a violation translator
*
* @param violationTranslator
* A violation translator that will convert {@link javax.validation.ConstraintViolation}s
* into Wicket's {@link org.apache.wicket.validation.ValidationError}s
*/
public void setViolationTranslator(IViolationTranslator violationTranslator)
{
Args.notNull(violationTranslator, "violationTranslator");
this.violationTranslator = violationTranslator;
}
/**
* Retrieves the validation context (read only version of the configuration). This is how
* components retrieve the configuration.
*
* @return validation context
*/
public static BeanValidationContext get()
{
BeanValidationConfiguration config = Application.get().getMetaData(KEY);
if (config == null)
{
throw new IllegalStateException(
"Application instance has not yet been configured for bean validation. See BeanValidationConfiguration#configure(Application)");
}
return config;
}
@Override
public Property resolveProperty(FormComponent> component)
{
for (IPropertyResolver resolver : propertyResolvers)
{
Property property = resolver.resolveProperty(component);
if (property != null)
{
return property;
}
}
return null;
}
}