如何解决当用户返回表单时,如何告诉Facelets页面应加载什么对象?
| 我遇到了一个问题,已经解决了,但是我觉得我的解决方案是一个糟糕的破解。有没有更好的办法? 我有一个页面,在该页面上放置了表单,该表单显示了某些对象的属性,例如示例(省略了明显的细节)。 Ticket.java:@Entity
public class Ticket {
@Id
private Long id;
private String title;
private byte priority;
// Getters,setters...
}
TicketController.java
@RequestScoped
public class TicketController {
private Ticket ticket = new Ticket();
// Getters,setters...
public String doUpdateTicket() {
Ticket t = ticketEJB.getTicketById(ticket.getId());
t.setTitle(ticket.getTitle());
t.setPriority(ticket.getPriority());
ticketEJB.updateTicket(t);
ticket = t;
return \"view.faces\";
}
}
edit.xhtml(只是形式,其他所有都是样板)
<h:form>
<h:inputHidden value=\"#{ticketController.ticket.id}\" />
<h:panelGrid columns=\"2\">
<h:outputLabel value=\"ID\"/>
<h:outputLabel value=\"#{ticketController.ticket.id}\"/>
<h:outputLabel value=\"Title: \"/>
<h:inputText value=\"#{ticketController.ticket.title}\"/>
<h:outputLabel value=\"Priority: \"/>
<h:inputText value=\"#{ticketController.ticket.priority}\" />
<h:commandButton value=\"Submit\"
action=\"#{ticketController.doUpdateTicket}\" />
</h:panelGrid>
</h:form>
还有TicketEJB,它负责获取那些票证,持久化等等。
因此,我在表单中创建了一个隐藏的输入,然后(在托管bean中)使用提供的id查找票证,然后将托管bean的3个对象中的所有字段手动复制到获取的票证中,然后将其持久保存...违反了DRY原则(当我向Ticket添加字段时,我已经偶然发现了一个错误,但是忘记将其复制到doUpdateTicket()
中。
因此,也许有更好的方法可以做到这一点?
解决方法
只需在视图作用域bean的“ 5”期间从EJB获得原始票证,而不是自己创建一个新票证。假设票证ID作为名称为“ 6”的请求参数传递:
edit.xhtml
<f:metadata>
<f:viewParam name=\"id\" value=\"#{ticketController.id}\" />
<f:event type=\"preRenderView\" listener=\"#{ticketController.preLoad}\" />
</f:metadata>
...
TicketController
@ManagedBean
@ViewScoped
public class TicketController {
private Long id;
private Ticket ticket;
@EJB
private TicketEJB ticketEJB;
public void preLoad() {
ticket = ticketEJB.getTicketById(id);
}
public String doUpdateTicket() {
ticketEJB.updateTicket(ticket);
return \"view.faces\";
}
// ...
}
唯一的区别是输入字段不会空白。但是,这不只是“编辑”表格背后的全部想法吗?然后,该问题也将立即以这种方式解决。
哦,你的
<h:outputLabel value=\"#{ticketController.ticket.id}\"/>
真的需要成为一个
<h:outputText value=\"#{ticketController.ticket.id}\"/>
, 您可以自己将票证添加为ManagedBean,但可以使用@SessionScoped。这样,票证域对象在请求之间保留其ID,JSF可以直接更新它。当然,您失去了使用这种方法保持数据短暂生存的优势,而这种方法目前是通过Request范围获得的。然后,您将展开有关绑定到域对象本身的辩论。
使用JSF 2,您还具有View Scope,可以在其中存储针对UIViewRoot的属性,在这种情况下,最好避免使用隐藏字段,例如,将票证或Controller的HAS-A票证存储在viewScope中-因此,当用户将故障单回发到编辑页面,该故障单将保留在范围内。某些人可能会说您应该在此处使用传输对象来将Service实体与表示层分离开-因此更新TO,将其传递给EJB,并让EJB处理实体的更新和持久性。
或者,您可以仅将Long ID服务器端存储在@SessionScoped或@ViewScoped中,因为将其存储为隐藏字段可能是不安全的,因为客户端可以更改它以更新另一个票证。如果确实使用票证的另一个实例来捕获UI输入,则可以在票证对象上提供一个Copy构造函数,因此doUpdateTicket方法本身不包括从一个票证到另一个代码的繁琐的复制字段。
为了避免重复,我希望直接绑定到JPA实体AKA域对象。我会使用@ViewScoped。
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。