Ainda não tem acesso? Estude com a gente! Matricule-se
Ainda não tem acesso? Estude com a gente! Matricule-se

Buscando dados sob demanda com LazyDataModel do Primefaces

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;
    }
3 respostas

Opa, acho que vc deixou fixo argumento titulo no método load. Usa o parâmetro campoOrdenacao.

Bom dia, Alberto! Alterei o código do método load conforme a sugestão, mas não funcionou e passei a ter erro de java.lang.IllegalArgumentException na linha alterada do método.

Observação: da forma como estava antes a paginação funciona e o filtro por título e o que não funciona são os sortBy(ordenação) por título, gênero, isbn, preco, data e o filtro por genero.

Método load do DAO alterado conforme a sugestão

@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, campoOrdenacao, titulo);
    }

Erro de IllegalArgumentException após a mudança do método load

GRAVE: java.lang.IllegalArgumentException: Unable to resolve attribute [null] against path
    at org.hibernate.ejb.criteria.path.AbstractPathImpl.unknownAttribute(AbstractPathImpl.java:120)
    at org.hibernate.ejb.criteria.path.AbstractPathImpl.locateAttribute(AbstractPathImpl.java:229)
    at org.hibernate.ejb.criteria.path.AbstractPathImpl.get(AbstractPathImpl.java:200)
    at br.com.caelum.livraria.dao.DAO.listaTodosPaginada(DAO.java:87)
    at br.com.caelum.livraria.modelo.LivroDataModel.load(LivroDataModel.java:25)

Então, vc precisa verificar o motivo do campoOrdenacao tá indo nulo.. Pq esse eh o motivo que a exception ta informando.