Boa tarde a todos! Após a realização dos exercícios "Buscando dados sob demanda com LazyDataModel do Primefaces" e "Filtrando elementos com p:selectOneMenu" os sortBy (ordenação) não funcionam mais para as colunas: Título, Gênero, ISBN, Preço e Data.
Alguém poderia me orientar dizendo o que faltou eu implementar ou o que eu estou fazendo errado?
Agradeço antecipadamente qualquer ajuda
Segue abaixo o código:
livro.xhtml
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml"
xmlns:h="http://xmlns.jcp.org/jsf/html"
xmlns:f="http://xmlns.jcp.org/jsf/core"
xmlns:p="http://primefaces.org/ui"
xmlns:ui="http://xmlns.jcp.org/jsf/facelets">
<ui:composition template="_template.xhtml">
<ui:define name="titulo">
Novo Livro
</ui:define>
<ui:define name="conteudo">
<h:form>
<p:messages id="messages"/>
<p:fieldset legend="Dados do Livro">
<p:panelGrid columns="2">
<p:outputLabel value="Titulo:" for="titulo" />
<p:inputText id="titulo" value="#{livroBean.livro.titulo}" required="true" requiredMessage="Título obrigatório"
validatorMessage="Título não pode ser superior a 40">
<f:validateLength maximum="40"/>
<f:ajax event="blur" render="messages"/>
</p:inputText>
<p:outputLabel value="Gênero:" for="genero" />
<p:selectOneMenu value="#{livroBean.livro.genero}" id="genero">
<f:selectItem itemLabel="Selecione..." itemValue="#{null}" />
<f:selectItems value="#{livroBean.generos}" />
</p:selectOneMenu>
<p:outputLabel value="ISBN:" for="isbn" />
<p:inputMask id="isbn" value="#{livroBean.livro.isbn}" validator="#{livroBean.comecaComDigitoUm}" mask = "999-9-99-999999-9"/>
<p:outputLabel value="Preço:" for="preco" />
<p:inputText id="preco" value="#{livroBean.livro.preco}" />
<p:outputLabel value="Data de Lançamento:" for="dataLancamento" />
<p:calendar id="dataLancamento" value="#{livroBean.livro.dataLancamento.time}"
timeZone="America/Sao_Paulo" pattern="dd/MM/yyyy" mask="true" />
</p:panelGrid>
</p:fieldset>
<p:fieldset legend="Dados do Autor">
<p:panelGrid columns="4">
<p:outputLabel value="Selecione Autor:" for="autor"/>
<p:selectOneMenu value="#{livroBean.autorId}" id="autor">
<f:selectItems value="#{livroBean.autores}" var="autor"
itemLabel="#{autor.nome}" itemValue="#{autor.id}"/>
</p:selectOneMenu>
<p:commandButton value="Gravar Autor" action="#{livroBean.gravarAutor}"
process="@this autor" update="tabelaAutores">
</p:commandButton>
<p:commandLink value="ou cadastrar novo autor" action="#{livroBean.formAutor}" immediate="true" update="@all" />
</p:panelGrid>
<p:dataTable value="#{livroBean.getAutoresDoLivro()}" var="autor" id="tabelaAutores" emptyMessage="Nenhum autor">
<p:column>
<h:outputText value="#{autor.nome}"/>
</p:column>
<p:column>
<p:commandLink value="X" action="#{livroBean.removerAutorDoLivro(autor)}" update="tabelaAutores" process="@this" />
</p:column>
</p:dataTable>
</p:fieldset>
<p:commandButton value="Gravar" action="#{livroBean.gravar()}" process="@form" update="@form :formTabelaLivros:tabelaLivros" />
</h:form>
<h:form id="formTabelaLivros">
<p:dataTable value="#{livroBean.livroDataModel}" widgetVar="tabelaLivros" var="livro" id="tabelaLivros" paginator="true" rows="5" lazy="true">
<f:facet name="header">Livros</f:facet>
<p:column headerText="Título" sortBy="#{livro.titulo}" filterBy="#{livro.titulo}" filterMatchMode="startsWith">
<h:outputText value="#{livro.titulo}"/>
</p:column>
<p:column headerText="Gênero" sortBy="#{livro.genero}" filterBy="#{livro.genero}" filterMatchMode="startsWith">
<f:facet name="filter">
<p:selectOneMenu onchange="PF('tabelaLivros').filter()">
<f:selectItem itemLabel="Selecione..." itemValue="#{null}" noSelectionOption="true" />
<f:selectItems value="#{livroBean.generos}" />
</p:selectOneMenu>
</f:facet>
<h:outputText value="#{livro.genero}"/>
</p:column>
<p:column sortBy="#{livro.isbn}">
<f:facet name="header">ISBN</f:facet>
<h:outputText value="#{livro.isbn}">
<f:convertNumber type="currency" pattern="R$ #0.00" currencySymbol="R$" locale="pt_BR" />
</h:outputText>
</p:column>
<p:column sortBy="#{livro.preco}">
<f:facet name="header">Preço</f:facet>
<h:outputText value="#{livro.preco}"/>
</p:column>
<p:column sortBy="#{livro.dataLancamento.time}">
<f:facet name="header">Data</f:facet>
<h:outputText value="#{livro.dataLancamento.time}">
<f:convertDateTime pattern="dd/MM/yyyy HH:mm" />
</h:outputText>
</p:column>
<p:column>
<f:facet name="header">Alterar</f:facet>
<h:commandLink value="Altera" action="#{livroBean.alterar(livro)}"/>
</p:column>
<p:column>
<f:facet name="header">Remover</f:facet>
<h:commandLink value="Remove" action="#{livroBean.remover(livro)}"/>
</p:column>
</p:dataTable>
</h:form>
</ui:define>
</ui:composition>
</html>
LivroBean sem os imports
@ManagedBean
@ViewScoped
public class LivroBean implements Serializable{
private Livro livro = new Livro();
private Integer autorId;
private List<Livro> livros;
private DAO<Livro> dao = new DAO<Livro>(Livro.class);
private LivroDataModel livroDataModel = new LivroDataModel();
private List<String> generos = Arrays.asList("Romance", "Drama", "Ação");
public void gravar() {
System.out.println("Gravando livro " + this.livro.getTitulo());
if (this.livro.getAutores().isEmpty()) {
FacesContext.getCurrentInstance().addMessage("autor", new FacesMessage("Livro deve ter pelo menos um Autor."));
return;
}else{
if (this.livro.getId() == null) {
dao.adiciona(this.livro);
this.livros = dao.listaTodos();
}else{
this.dao.atualiza(this.livro);
}
this.livro = new Livro();
}
}
public void alterar(Livro livro){
System.out.println("Carregando livro");
this.livro = livro;
}
public void remover(Livro livro){
System.out.println("Removendo livro");
this.dao.remove(livro);
this.livros = dao.listaTodos();
}
public void removerAutorDoLivro(Autor autor){
this.livro.removeAutor(autor);
}
public String formAutor(){
System.out.println("Chamando o formulário do Autor");
return "autor?faces-redirect=true";
}
public void comecaComDigitoUm(FacesContext fc, UIComponent component, Object value) throws ValidatorException{
String valor = value.toString();
if (!valor.startsWith("1")) {
throw new ValidatorException(new FacesMessage("ISBN deveria começar com 1"));
}
}
public void gravarAutor() {
Autor autor = new DAO<Autor>(Autor.class).buscaPorId(this.autorId);
this.livro.adicionaAutor(autor);
}
public Livro getLivro() {
return livro;
}
@SuppressWarnings({ "unchecked", "rawtypes" })
public List<Livro> getLivros(){
if (this.livros == null) {
this.livros = dao.listaTodos();
}
return livros;
}
@SuppressWarnings({ "unchecked", "rawtypes" })
public List<Autor> getAutores() {
return new DAO(Autor.class).listaTodos();
}
public List<Autor> getAutoresDoLivro(){
return this.livro.getAutores();
}
public Integer getAutorId() {
return autorId;
}
public void setAutorId(Integer autorId) {
this.autorId = autorId;
}
public LivroDataModel getLivroDataModel() {
return livroDataModel;
}
public List<String> getGeneros() {
return generos;
}
}
LivroDataModel sem os imports
public class LivroDataModel extends LazyDataModel<Livro> {
private DAO<Livro> dao = new DAO<Livro>(Livro.class);
public LivroDataModel() {
super.setRowCount(this.dao.quantidadeDeElementos());
}
@Override
public List<Livro> load(int inicio, int quantidade, String campoOrdenacao, SortOrder sentidoOrdenacao, Map<String, Object> filtros) {
String titulo = (String) filtros.get("titulo");
return dao.listaTodosPaginada(inicio, quantidade, "titulo", titulo);
}
}
Método "listaTodosPaginada" do DAO
public List<T> listaTodosPaginada(int firstResult, int maxResults, String coluna, String valor) {
EntityManager em = new JPAUtil().getEntityManager();
CriteriaQuery<T> query = em.getCriteriaBuilder().createQuery(classe);
Root<T> root = query.from(classe);
if(valor != null)
query = query.where(em.getCriteriaBuilder().like(root.<String>get(coluna), valor + "%"));
List<T> lista = em.createQuery(query).setFirstResult(firstResult).setMaxResults(maxResults).getResultList();
em.close();
return lista;
}