2

名フィールドを持つフォームを持つ春の mvc アプリケーションがあります。このフィールドは、少なくとも 3 文字以上である必要があります。バッキング Bean クラスで次のように休止状態の @Size バリデーターを使用しています。

@Size(min=3, message="message.key")
String firstName;

この検証が失敗すると、適切なエラー メッセージが表示されます。ただし、失敗後にページがリロードされると、名に入力された値が入力フィールドからクリアされます。その値を編集のためにフィールドに残すにはどうすればよいですか? もし可能なら....

コードは次のとおりです。

JSP スニペット - これはポートレット アプリケーションではありません

<portlet:renderURL var="createUserAction">
    <portlet:param name="action" value="createUser"/>
</portlet:renderURL>
<form:form method="post" commandName="userInformation" action="${createUserAction}" htmlEscape="false">
<h2>
    <fmt:message key="msg.label.form.title" bundle="${msg}" />
</h2>
<form:errors path="*" cssClass="errorblock" element="div"></form:errors>
<p>
    <form:label path="firstName">
        <fmt:message key="msg.label.form.fname" bundle="${msg}"/>
    </form:label>
   <form:input path="firstName" />
</p>
<p>
    <form:label path="lastName">
        <fmt:message key="msg.label.form.lname" bundle="${msg}"/>
    </form:label>
    <form:input path="lastName" />
</p>
<div style="margin-left: 150px; margin-top: 20px">
    <input type="submit" value="Save" /> 
    <input type="reset" value="Reset" />
</div>
</form:form>

@Component
public class UserInformation {


  @NotEmpty(message="msg.error.required.lname")
  private String lastName;

  @NotEmpty(message="msg.error.required.fname")
  @Size(min=3, message="msg.error.length.fname")
  private String firstName;

  public UserInformation() {
    super();
  }

  public String getLastName() {
    return lastName;
  }

  public void setLastName(String lastName) {
    this.lastName = lastName;
  }

  public String getFirstName() {
    return firstName;
  }

  public void setFirstName(String firstName) {
    this.firstName = firstName;
  }

}

コントローラ

@Controller
@RequestMapping("VIEW")
public class UserManagement {

  @Autowired
  private UserInformation userInformation;

  @Autowired
  private Validator validator;

  @Autowired
  private MessageSource messageSource;

  @RequestMapping(params="page=addUser")
  public String addUser(Model model){
    userInformation = new UserInformation();
    model.addAttribute("userInformation", userInformation); 
    return Page.ADD_USER;
  }

  @RequestMapping(params="action=createUser")
  public String createUser(
   @ModelAttribute(value="userInformation") UserInformation userInformation,
   BindingResult result, Model model) throws ApplicationException{

    // get values
    String firstName = userInformation.getFirstName();

    System.out.println("fname="+firstName);

    Set<ConstraintViolation<UserInformation>> constraintViolations = 
     validator.validate(userInformation);

   for(ConstraintViolation<UserInformation> constraintViolation : constraintViolations) {
     String propertyPath = constraintViolation.getPropertyPath().toString();
     String message = constraintViolation.getMessage();
     result.addError(
       new FieldError(
         "member", 
         propertyPath, 
         messageSource.getMessage(
           message, 
           null, 
           Locale.ENGLISH
         )
       )
     );
  }
  // Errors found
  if(result.hasErrors()){
    return UMConstants.Page.ADD_USER;
  }

  String successMsg = messageSource.getMessage(
    "msg.user.added", 
    new Object[]{userInformation.getFirstName()}, 
    Locale.ENGLISH
   );

   model.addAttribute("successMsg", successMsg);

   return UMConstants.Page.INDEX;
  }
}

ユーザーはaddUserメソッドを実行するリンクをクリックして、上記の JSP スニペットで示されているフォームを含むページをロードします。ユーザーが送信ボタンをクリックすると、createUserメソッドが呼び出されます。ここで検証が行われます。

4

2 に答える 2

1

私は同じ問題を抱えていて、Spring MVC form:inputjsp タグのソース コードを (痛烈に) 繰り返し処理することで解決策を見つけました...ここで共有すると思いますが、誰かの助けになるかもしれません。

一言で言えば、問題はform:inputタグにあります。このタグは、その値を評価するときに、このフィールド (属性) に関連付けられたエラーをチェックし、エラーpathが見つかった場合は、コマンドまたはフォーム オブジェクトのフィールドの値ではなく、インスタンスのrejectedValue属性を使用します。FieldError

@Validコントローラー メソッドの引数でアノテーションを使用すると、Spring はオブジェクトのrejectedValueプロパティを正しく埋めFieldErrorます。

ただしFieldError、短いコンストラクター (3 つの引数) を使用して手動でオブジェクトを作成すると、のrejectedValue属性はFieldError設定されず、タグはこの null 値を表示に使用します...

FieldError解決策は、オブジェクトを作成するときにコンストラクターの長いバージョンを使用することです。次に例を示します。

result.addError(
       new FieldError(
         "member", 
         "firstName", 
         userInformation.getFirstName(),
         false,
         new String[0],
         new Object[0],
         messageSource.getMessage(
           message, 
           null, 
           Locale.ENGLISH
         )
       )
     );

form:inputタグは、この長いコンストラクターの 3 番目の引数であり、より単純な 3 つの引数コンストラクターを使用するときに設定されなかった のプロパティをrejectedValue使用します...FieldError

コマンドまたはフォーム オブジェクト (ここでは userInformation) の実際の値は常に正しいため、このバグ/癖を追跡するのはかなり困難であることに注意してください。

最終的に誰かを助けることを願っています...

于 2013-03-19T12:01:28.323 に答える
0

リターン前にエラーが発生した場合は、モデルにユーザー情報を追加してみてください

if(result.hasErrors()){
model.addAttribute("userInformation", userInformation); 
return UMConstants.Page.ADD_USER;

}

更新しました

私はあなたの問題を理解したと思います。エラーが発生した場合、addUser メソッドを呼び出し、内部でユーザー メソッドを追加すると、新しい UserInformation オブジェクトが作成され、ページに表示する値がありません。addUser メソッドに対して以下のようなことを試すことができます。

@RequestMapping(params="page=addUser")
public String addUser(Model model,HttpServletRequest request){
    userInformation =  (UserInformation)request.getAttribute("userInformation");

   if(userInformation == null){
    userInformation = new UserInformation();
   }
    model.addAttribute("userInformation", userInformation); 
    return Page.ADD_USER;
  }
于 2012-11-10T01:54:19.877 に答える