Método de ação JSF não sendo chamado
Eu tenho uma visão JSF com uma tabela de dados Primefaces e um botão de comando, como fallows:
<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>
E aqui está o controlador:
@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;
}
}
}
Essa exibição é renderizada corretamente. A tabela de dados e botões parece bem. O problema é que quando eu clico no botão de comando "addWellsite" pela primeira vez, nada acontece. A página parece apenas atualizar. Se eu clicar novamente, como acontece a exceção:
java.lang.NumberFormatException: For input string: "null"
Usando um depurador, descobri que a ação "addWellsite" NÃO é chamada pela primeira vez e, portanto, o resultado não é gerado (assim, a atualização da página).
A exceção provavelmente está vindo da falta de inicialização nas visualizações atuais ou de destino (já que ambas as visualizações são exibidas a partir de métodos de ação que não foram chamados na atualização da página)
A questão é:Por que o método de ação não é chamado pela primeira vez?
A partir deesta resposta:
Sempre que umUICommand
componente falha ao chamar a ação associada, verifique o seguinte:
UICommand
componentes devem ser colocados dentro de umUIForm
componente (por ex.h:form
).Eu tenho um h: form
Você não podeninho múltiploUIForm
componentes uns nos outros (cuidado com arquivos include!).Há apenas um.
Nenhum erro de validação / conversão deveria ter ocorrido (useh:messages
para obtê-los todos).Eu tenho um h: mensagens que não exibem nenhum erro.
E seUICommand
componentes são colocados dentro de umUIData
componente, garantir que exatamente o mesmoDataModel
(o objeto por trás doUIData
'svalue
atributo) é preservada.O commandButton está dentro da dataTable, mas a visualização de destino não precisa do dataModel. Como mostra o código do meu controlador, o objeto é construído à medida que a visualização tenta recuperá-lo. A próxima solicitação não está usando essa dataTable, pois eu não ligo mais.
orendered
edisabled
atributos do componente e todos os componentes pai não devem avaliar parafalse
durante a fase de aplicação dos valores de solicitação.Não hárendered
oudisbled
atributos.
PhaseListener
ou qualquerEventListener
na cadeia de solicitação-resposta alterou o ciclo de vida do JSF para pular a fase de ação de chamada.Nenhum phaseListener é definido.
Certifique-se de que nãoFilter
ouServlet
na mesma cadeia de solicitação-resposta bloqueou o pedido para oFacesServlet
de alguma forma.Nenhum outro Servlet é definido. Eu nem sei o que é um filtro.
Por que o método de ação não é chamado pela primeira vez?