Conversation Scoped bean用于相同的视图

时间:2012-09-03 10:15:16

标签: jsf-2 richfaces cdi

我有一个有一个清单的DTO。我想在用户点击添加按钮时向datatable添加新行。但是,当我单击添加时,将调用dto,即构造函数并初始化值并且列表大小为0.该bean是会话作用域。我应该在使用会话范围的bean时开始和结束相同视图的对话吗?我正在使用相同的bean进行编辑,它运行良好。如何在使用richfaces 4和jsf 2以及ajax时解决初始化问题。

查看:

                    <rich:panel id ="dataPnl">
                        <rich:dataTable value="#{legendbean.legendDTO.list}" var="legend" style="width:100%">

                            <rich:column>
                                <f:facet name="header">
                                    <h:outputText value="SN"/>
                                </f:facet>
                                <h:inputText value="#{legend.sn}"/>
                            </rich:column>

                            <rich:column>
                                <f:facet name="header">
                                    <h:outputText value="Description"/>
                                </f:facet>
                                <h:inputText value="#{legend.desc}"/>
                            </rich:column>


                            <rich:column>
                                <a4j:commandLink value="Add" actionListener="#{legendbean.addLegendRange()}" render="nisForm:dataPnl"/>
                                <h:outputText value=" / "/>
                                <a4j:commandLink value="Remove" actionListener="#{legendbean.removeLegendRange(legend)}" render="nisForm:dataPnl"/>
                            </rich:column>

                        </rich:dataTable>
                    </rich:panel>                 

Bean:

@Named("legendbean")
@ConversationScoped
public class LegendController implements Serializable {

    LegendDTO legendDTO = new LegendDTO();
    String selectedLegend;
    boolean edit;
    @Inject
    private Conversation conversation;

    public boolean isEdit() {
        return edit;
    }

    public void setEdit(boolean edit) {
        this.edit = edit;
    }

    public LegendController() {
        Logger.getLogger(LegendController.class.getName()).warning("The value of Edit is : " + edit);
        if (!edit) {
            legendDTO.getList().add(new Legend());
            Logger.getLogger(LegendController.class.getName()).warning("The size of list" + legendDTO.getList().size());
        }
    }

    public LegendDTO getLegendDTO() {
        return legendDTO;
    }

    public void setLegendDTO(LegendDTO legendDTO) {
        this.legendDTO = legendDTO;
    }

    public void addLegendRange() {
        Logger.getLogger(LegendController.class.getName()).warning("List Size " + legendDTO.getList().size());
        legendDTO.getList().add(new Legend());
        Logger.getLogger(LegendController.class.getName()).warning("List Size " + legendDTO.getList().size());
    }

    public void removeLegendRange(Legend legend) {
        if (legendDTO.getList().size() != 1) {
            legendDTO.getList().remove(legend);
        }
    }

    public String saveLegend() {
        Logger.getLogger(LegendController.class.getName()).warning("Save Legend Edit" + edit);
        LegendDAO dao = new LegendDAO();
        if (dao.addLegend(legendDTO, edit)) {
            if (edit) {
                conversation.end();
                edit = false;
                Logger.getLogger(LegendController.class.getName()).warning("Save Legend Edit" + edit);
                return "VIEWLEGEND";
            } else {
                legendDTO = new LegendDTO();
                legendDTO.getList().add(new Legend());              
                FacesContext.getCurrentInstance().addMessage(null, new FacesMessage("Saved !"));
                return "";
            }
        } else {
            FacesContext.getCurrentInstance().addMessage(null, new FacesMessage("Could Not Save Confim if you have already defined Legend " + legendDTO.getLegendName() + "!"));
            return "";
        }
    }

    public List<LegendDTO> getLegends() {
        LegendDAO dao = new LegendDAO();
        return dao.getLegendDTO();
    }

    //All function from here are for legend  delete
    public void deleteLegendType(LegendDTO dto) {
        LegendDAO dao = new LegendDAO();
        if (dao.deleteLegendType(dto.getLegendName())) {
            FacesContext.getCurrentInstance().addMessage(null, new FacesMessage("Deleted !"));
        } else {
            FacesContext.getCurrentInstance().addMessage(null, new FacesMessage("Deleted Error !"));
        }
    }

    //All function from here is to legend edit
    public String editLegendType(LegendDTO dto) {
        conversation.begin();
        edit = true;
        legendDTO = dto;
        LegendDAO dao = new LegendDAO();
        dto.getList().clear();
        try {
            List<Legend> legends = dao.getDetailForEditLegend(dto.getLegendName());
            dto.setList(legends);
        } catch (SQLException ex) {
            Logger.getLogger(LegendController.class.getName()).warning("SQL EXception has occoured");
        }
        Logger.getLogger(LegendController.class.getName()).warning("The size of list" + dto.getList().size());
        return "addLegend";
    }

    public String cancel() {
        conversation.end();
        return "VIEWLEGEND";
    }
}

2 个答案:

答案 0 :(得分:2)

是的,您需要start长时间对话才能使您的对话(以及对话范围内的bean)跨越多个请求。否则会话在JSF请求结束时被终止(默认情况下会话是暂时的:请参阅ConversationScoped javadoc)。

在这种情况下,常见的解决方案是使用ViewScoped bean,但注释是特定于JSF2的,并且不在CDI中呈现(您可以将其移植到CDI或使用seam-faces模块,更多细节:http://www.verborgh.be/articles/2010/01/06/porting-the-viewscoped-jsf-annotation-to-cdi/)。

答案 1 :(得分:2)

如果您不限制使用CDI / Seam注释,则可以将bean更改为使用@ManagedBean(name="legendbean")包中的javax.faces.bean.ManagedBean,然后在您的类上使用@ViewScoped注释保证只要用户在同一页面上,您将使用托管bean的相同实例。根据您的设置,任何其他内容都无需更改,所有@Inject都将正常工作。要初始化支持legendDTO.list,请使用@PostConstruct JSF注释在ViewScoped JSF bean中注释一个方法,并将列表填充逻辑放在那里。您可以安全地添加/删除列表,而不会将其重新初始化为空。但是您必须记住将对此列表的更改提交回数据库。

只是想一想,您可能希望显示一个弹出窗口,允许您的用户确认他们想要从您的数据库中删除任何内容,这是安全的做法。干杯