Spring Form Ajax Request + Проверка гибернации - PullRequest
2 голосов
/ 12 февраля 2012

Я всегда использовал @Valid и BindingResult для проверки полей формы. Теперь, используя Ajax, основанный на этой статье , невозможно использовать BindingResult вместо или дополнительно к HttpServletResponse, поскольку это приведет к неправильному запросу (код ошибки HTTP 400).

Как проверить мои поля формы сейчас?

  @RequestMapping( value = "/save.html", params = "json", method = RequestMethod.POST )
  public @ResponseBody Map<String, ? extends Object> saveJSON( @RequestBody Location location, /* BindingResult result, */ HttpServletResponse response )
  {
    return Collections.singletonMap( "foo", "foobar" );
  }

Это был старый способ без ajax:

  @RequestMapping( value = "/save.html", method = RequestMethod.POST )
  public String save( @ModelAttribute( "location" ) @Valid Location location, BindingResult result, Map<String, Object> map )
  {
    Location l;
    if ( ( l = service.findByTitle( location.getTitle() ) ) != null )
    {
      if ( location.getId() != l.getId() )
      {
        result.addError( new FieldError( "location", "title", messageSource.getMessage( "Unique.location.title", null, null ) ) );
      }
    }

    if ( result.hasErrors() )
    {
      return "locationform";
    }

    service.save( location );
    return "redirect:/locations/index.html";
  }

EDIT

Попробовал это, но в результате нет элемента errors, который содержит true при оставлении формы незаполненной (это должно вызвать сообщение @NotEmpty Constraint)

  @RequestMapping( value = "/save.html", params = "json", method = RequestMethod.POST )
  public @ResponseBody Map<String, ? extends Object> saveJSON( @RequestBody Location location, HttpServletResponse response, Map<String, Object> map )
  {
    BindingResult result = new BeanPropertyBindingResult( location, "" );
    if ( result.hasErrors() ) map.put( "errors", true );
    map.put( "foo", "foobar" );
    return map;
  }

1 Ответ

1 голос
/ 13 февраля 2012

Кажется, вы используете спящий валидатор. Если это так, попробуйте это

в вашем контроллере :

//other imports
import javax.validation.Validator;
@Controller()
class MyController{

    @autowire()
    @qualifier("myValidator")
    private Validator validator;
    public Validator getValidator() {
        return mValidator;
    }

    public void setValidator(Validator validator) {
        this.mValidator = validator;
    }

   @RequestMapping( value = "/save.html", params = "json", method = RequestMethod.POST )
   public @ResponseBody Map<String, ? extends Object> saveJSON( @RequestBody Location location, HttpServletResponse response )
   {
       Set<ConstraintViolation<Location>> errors = getValidator().validate(location);
       Map<String, String> validationMessages = new HashMap<String, String>();
       if(!errors.isEmpty())
       {
          //this map will contain the validation messages
          validationMessages = validationMessages(errors); 

          //probably you would like to send the errors back to client
          return validationMessages ;
       }
       else
       {
           //do whatever you like to do with the valid bean

       }
   }

   public Map<String, String> validationMessages(Set<ConstraintViolation<Location>> failures) {
        Map<String, String> failureMessages = new HashMap<String, String>();
        for (ConstraintViolation<Location> failure : failures) {

                failureMessages.put(failure.getPropertyPath().toString(), failure.getMessage());
        }
        return failureMessages;
    }



}

в файле spring context добавьте следующий bean

<beans:bean id="myValidator" class="org.springframework.validation.beanvalidation.LocalValidatorFactoryBean" />

надеюсь, это поможет:)

...