4

我有一个 JSF 视图,其中包含 Primefaces 数据表和其中的命令按钮,如下所示:

<p:messages id="statusMessages" showDetail="true" />
    <h:form id="listForm">
        <p:panel header="Wellsite List">
            <br />
            <h:outputLabel value="Welcome, #{wellsiteController.loggedUser.login}" />
            <br />
            <br />
            
            <p:dataTable id="dataTable" var="wellsite" value="#{wellsiteController.wellsiteDataTableModel}"
                         paginator="true" rows="10" selection="#{wellsiteController.wellsite}">

                <p:column selectionMode="single" style="width:18px" id="radioSelect" />

                <p:column sortBy="#{wellsite.reference}" headerText="Wellsite ID">
                    <h:outputText value="#{wellsite.reference}" />
                </p:column>

                <p:column headerText="Allowed Groups">
                    <h:outputText value="#{wellsite.allowedGroups.toString()}" />
                </p:column>

                <f:facet name="footer">
                    <h:panelGrid columns="3">
                        <p:commandButton id="addWellsite" value="Add New Wellsite" icon="ui-icon-flag" ajax="false" action="#{wellsiteController.showAddWellsite}"/>
                        <p:commandButton id="editWellsite" value="Edit Selected Wellsite" icon="ui-icon-wrench" ajax="false" action="#{wellsiteController.showEditWellsite}"/>

                        <p:commandButton id="deleteWellsiteButton" value="Remove Selected Wellsite" icon="ui-icon-trash" onclick="confirmation.show()" type="button"/>
                         
                    </h:panelGrid>
                </f:facet>
            </p:dataTable>
            <p:spacer height="20" />
        </p:panel>
        <p:confirmDialog id="confirmDialog" message="Are you sure you want to remove the selected Wellsite along with all it's data?" header="Confirmation" severity="alert" widgetVar="confirmation">  
            <p:commandButton id="confirm" value="Yes" ajax="false" oncomplete="confirmation.hide()" action="#{wellsiteController.deleteWellsite}" />
            <p:commandButton id="decline" value="Cancel" onclick="confirmation.hide()" type="button" />   

        </p:confirmDialog>
    </h:form>

这是控制器:

@ManagedBean(name = "wellsiteController")
@RequestScoped
public class WellsiteController implements Serializable {
private static final long serialVersionUID = 1L;

@ManagedProperty("#{wellsiteDao}")
private WellsiteDao wellsiteDao;

@ManagedProperty("#{userDao}")
private UserDao userDao;

@ManagedProperty("#{groupDao}")
private GroupDao groupDao;

@ManagedProperty("#{userController.loggedUser}")
private UserEnt loggedUser;

private WellsiteEnt wellsite;
private List<WellsiteEnt> wellsiteList;
DualListModel<GroupEnt> pickGroupsModel;

public WellsiteController(){
}

@PostConstruct
public void build(){
    wellsite = new WellsiteEnt();
    wellsite.setAllowedGroups(new ArrayList<GroupEnt>());
}

/*some getters & setters*/

public WellsiteDataTableModel getWellsiteDataTableModel(){
    return new WellsiteDataTableModel(getWellsiteList());
}

public void setPickGroupsModel(DualListModel<GroupEnt> model){
    pickGroupsModel = model;
}

public DualListModel<GroupEnt> getPickGroupsModel() {
    if(pickGroupsModel == null){
        List<GroupEnt> allGroups = groupDao.getAll();
        List<GroupEnt> currentGroups = wellsite.getAllowedGroups();
        for(GroupEnt g : currentGroups){
            allGroups.remove(g);
        }
        pickGroupsModel = new DualListModel<GroupEnt>(allGroups, currentGroups);
    }
    return pickGroupsModel;
}

public String listWellsites(){
    getWellsiteList();
    return "listWellsites";
}

public String showAddWellsite(){
    FacesContext context = FacesContext.getCurrentInstance();
    setWellsite(new WellsiteEnt());
    wellsite.setAllowedGroups(new ArrayList<GroupEnt>());
    pickGroupsModel = null;
    context.addMessage(null, new FacesMessage(FacesMessage.SEVERITY_INFO,
                                            "Fields annotated with a ' * ' are mandatory",""));
    return "addWellsite";
}

public String addWellsite(){
    FacesContext context = FacesContext.getCurrentInstance();
    
    wellsite.setDate(new Date());
    wellsite.setLastUpdate(wellsite.getDate());
    try {
        wellsiteDao.addWell(wellsite);
        
        for(GroupEnt g : pickGroupsModel.getTarget()){
            GroupEnt group = groupDao.getOne(g.getGroupId());
            group.getGroupWellsites().add(wellsite);
            groupDao.update(group);
        }
        return listWellsites();
        
    } catch (Exception ex) {
        Logger.getLogger(WellsiteController.class.getName()).log(Level.SEVERE, null, ex);
        context.addMessage(null, new FacesMessage(FacesMessage.SEVERITY_ERROR,
                                            ex.getMessage(),""));
        return null;
    }
}
}

该视图被正确渲染。数据表和按钮看起来不错。问题是当我第一次单击“addWellsite”命令按钮时,什么也没有发生。该页面似乎正在刷新。如果我再次单击它,则会发生异常:

java.lang.NumberFormatException: For input string: "null"

使用调试器,我发现第一次没有调用“addWellsite”的操作,因此没有生成结果(因此,页面刷新)。

异常可能是由于当前视图或目标视图中缺少初始化(因为两个视图都是从页面刷新中未调用的操作方法显示的)

问题是:为什么第一次没有调用action方法?

这个答案

每当UICommand组件未能调用关联的操作时,请验证以下内容:

  1. UICommand组件必须放置在UIForm组件内(例如h:form)。

我有啊:表格

  1. 您不能将多个组件相互嵌套(注意包含文件!)。UIForm

只有一个。

  1. 不应发生验证/转换错误(用于h:messages获取所有错误)。

我有 ah:messages 没有显示任何错误。

  1. 如果UICommand组件被放置在组件内,请确保保留UIData完全相同(位于's属性DataModel后面的对象)。UIDatavalue

commandButton 在 dataTable 中,但目标视图不需要 dataModel。正如我的控制器代码所示,该对象是在视图尝试检索它时构建的。下一个请求不使用此数据表,我不再处理它。

  1. 在应用请求值阶段不应评估组件和所有父组件的rendered和属性。disabledfalse

没有rendereddisbled属性。

  1. 确保请求-响应链中没有PhaseListener或有任何EventListener更改了 JSF 生命周期以跳过调用操作阶段。

没有定义 phaseListener。

  1. 确保没有FilterServlet在同一个请求-响应链中以某种方式阻止了请求FacesServlet

没有定义其他 Servlet。我什至不知道过滤器是什么。

为什么第一次没有调用 action 方法?

4

1 回答 1

5

当它的父级<h:form>另一个 <h:form>事先发起的 ajax 请求呈现时,可能会发生这种情况。渲染/更新<h:form>后将失去其视图状态。这是由 JavaScript API 中的错误引起的,如JSF 问题 790中所述,该错误已在即将推出的 JSF 2.2 中修复。

同时,在 JSF 2.0/2.1 中,您需要在另一种形式的动作<h:form>render(或 PrimeFaces 的update)属性中明确指定客户端 ID。

例如

<h:commandButton ...>
   <f:ajax ... render=":listForm" />
</h:commandButton>

或者

<p:commandButton ... update=":listForm" />

或者只是让它成为一个普通的(非ajax)请求。

也可以看看:

于 2012-05-14T16:29:43.887 回答