我正在尝试编写一个简单的创建用户页面。我希望用户能够键入所需的用户名,如果已经使用了用户名,则会显示输出文本并显示“用户名已在使用中”。
这是我的xhtml页面
<tr>
<td>Username: </td>
<td>
<p:inputText id="username" value="#{createUserManagedBean.username}" required="true" requiredMessage="Username is required.">
<p:ajax event="keyup" update="uniqueUsernameMessage"/>
</p:inputText>
</td>
<td>
<h:outputText id="uniqueUsernameMessage" value="Username already in use" rendered="#{!createUserManagedBean.checkUniqueUsername()}" />
</td>
</tr>
这是我的托管bean
public boolean checkUniqueUsername()
{
if(!StringUtils.isBlank(getUsername()))
{
UserDTO userDTO = new UserDTO();
userDTO.setUsername(username);
boolean result = getUserService().validateUniqueUsername(userDTO);
return result;
}
else
return false;
}
我的问题是每个keyup事件的消息都没有更新。正在调用该服务,但该元素是否会根据方法结果显示或不显示。
答案 0 :(得分:1)
使用rendered
属性绝对不是验证输入组件的正确方法。您应该使用真正的Validator
实现。在其中,如果失效,只需抛出一个ValidatorException
FacesMessage
。然后,JSF会关注FacesMessage
最终在与输入组件关联的右侧<h:message>
。
总而言之,这应该做到:
<p:inputText id="username" value="#{createUserManagedBean.username}" ...>
<f:validator binding="#{uniqueUsernameValidator}" />
<p:ajax event="keyup" update="usernameMessage" />
</p:inputText>
<h:message id="usernameMessage" for="username" />
使用
@ManagedBean
@RequestScoped
public class UniqueUsernameValidator implements Validator {
@EJB
private UserService userService;
@Override
public void validate(FacesContext context, UIComponent component, Object value) throws ValidatorException {
if (value == null || value.isEmpty()) {
return; // Let required="true" handle.
}
String username = (String) value;
if (userService.findByUsername(username) != null) {
throw new ValidatorException(new FacesMessage(
FacesMessage.SEVERITY_ERROR, "Username already in use. Please choose another", null));
}
}
}
请注意,验证工具为@ManagedBean
而不是@FacesValidator
,因为否则无法注入@EJB
。但是如果你没有使用EJB并且手动创建服务类并自己摆弄事务,那么你可能只需要保持它真实@FacesValidator
:
@FacesValidator("uniqueUsernameValidator")
public class UniqueUsernameValidator implements Validator {
然后将其引用如下:
<f:validator validatorId="uniqueUsernameValidator" />