Solucionado (ver solução)
Solucionado
(ver solução)
1
resposta

f:ajax levantando exception

Após incluir a linha:

<f:ajax execute="autor" render="tabelaAutores" />

no arquivo livro.xhtml , o eclipse começou a levantar uma exceção como essa:

fev 18, 2015 5:07:26 PM com.sun.faces.lifecycle.InvokeApplicationPhase execute
ADVERTÊNCIA: #{livroBean.gravarAutor}: java.lang.IllegalArgumentException: id to load is required for loading
javax.faces.FacesException: #{livroBean.gravarAutor}: java.lang.IllegalArgumentException: id to load is required for loading

Não acho que deva ser problema no método gravarAutor, pois o mesmo funciona sem problemas ao comentar a linha do ajax.

Seguem meus fontes: 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://java.sun.com/jsf/html"
    xmlns:f="http://java.sun.com/jsf/core">
<h:head />
<h:body>
    <h1>Novo Livro</h1>

    <h:messages />
    <h:form>
        <fieldset>
            <legend>Dados do Livro</legend>
            <h:panelGrid columns="2">

                <h:outputLabel value="Titulo:" for="titulo" />
                <h:inputText id="titulo" value="#{livroBean.livro.titulo}"
                    required="true" requiredMessage="Título Obrigatório!"
                    validatorMessage="Tamanho máximo: 40 !!!">
                    <f:validateLength maximum="40" />
                </h:inputText>
                <h:outputLabel value="ISBN:" for="isbn" />
                <h:inputText id="isbn" value="#{livroBean.livro.isbn}"
                    validator="#{livroBean.comecaComDigitoUm}" />

                <h:outputLabel value="Preço:" for="preco" />
                <h:inputText id="preco" value="#{livroBean.livro.preco}"
                    validatorMessage="O preço deve estar entre R$1,00 e R$1.000,00">
                    <f:validateDoubleRange minimum="1.00" maximum="1000.00" />
                    <f:convertNumber pattern="R$ 0.00" currencySymbol="R$"
                        locale="pt_BR" type="currency" />
                </h:inputText>

                <h:outputLabel value="Data de Lançamento:" for="dataLancamento" />
                <h:inputText id="dataLancamento"
                    value="#{livroBean.livro.dataLancamento.time}">
                    <f:convertDateTime pattern="dd/MM/yyyy"
                        timeZone="America/Sao_Paulo" />
                </h:inputText>
            </h:panelGrid>
        </fieldset>

        <fieldset>
            <legend>Dados do Autor</legend>
            <h:outputLabel value="Selecione Autor " for="autor" />
            <h:selectOneMenu value="#{livroBean.autorId}">
                <f:selectItems value="#{livroBean.autores}" var="autor"
                    itemLabel="#{autor.nome}" itemValue="#{autor.id}" id="autor" />
            </h:selectOneMenu>
            <h:commandButton value="Gravar Autor"
                action="#{livroBean.gravarAutor}">
                <f:ajax execute="autor" render="tabelaAutores" />
            </h:commandButton>
            <h:dataTable value="#{livroBean.autoresDoLivro}" var="autor"
                id="tabelaAutores">
                <h:column>
                    <h:outputText value="#{autor.nome}" />
                </h:column>
            </h:dataTable>
        </fieldset>

        <h:commandButton value="Gravar" action="#{livroBean.gravar}" />
    </h:form>

    <h:dataTable id="tabelaLivros" value="#{livroBean.livros}" var="livro">
        <h:column>
            <f:facet name="header">Título</f:facet>
            <h:outputText value="#{livro.titulo}" />
        </h:column>

        <h:column>
            <f:facet name="header">ISBN</f:facet>
            <h:outputText value="#{livro.isbn}" />
        </h:column>

        <h:column>
            <f:facet name="header">Preço</f:facet>
            <h:outputText value="#{livro.preco}">
                <f:convertNumber pattern="R$ 0.00" currencySymbol="R$"
                    locale="pt_BR" type="currency" />
            </h:outputText>
        </h:column>

        <h:column>
            <f:facet name="header">Data</f:facet>
            <h:outputText value="#{livro.dataLancamento.time}">
                <f:convertDateTime pattern="dd/MM/yyyy" timeZone="America/Sao_Paulo" />
            </h:outputText>
        </h:column>
    </h:dataTable>

</h:body>

</html>

LivroBean.java:

package br.com.caelum.livraria.bean;

import java.util.List;

import javax.faces.application.FacesMessage;
import javax.faces.bean.ManagedBean;
import javax.faces.bean.ViewScoped;
import javax.faces.component.UIComponent;
import javax.faces.context.FacesContext;
import javax.faces.validator.ValidatorException;

import br.com.caelum.livraria.dao.DAO;
import br.com.caelum.livraria.modelo.Autor;
import br.com.caelum.livraria.modelo.Livro;

@ManagedBean
@ViewScoped
public class LivroBean {

    private Livro livro = new Livro();
    private Integer autorId;

    public Integer getAutorId() {
        return autorId;
    }

    public void setAutorId(Integer autorId) {
        this.autorId = autorId;
    }

    public Livro getLivro() {
        return livro;
    }

    public List<Livro> getLivros() {
        return new DAO<Livro>(Livro.class).listaTodos();
    }

    public List<Autor> getAutoresDoLivro() {

        return livro.getAutores();
    }

    public void gravar() {
        System.out.println("Gravando livro " + this.livro.getTitulo());

        if (livro.getAutores().isEmpty()) {
            FacesContext.getCurrentInstance().addMessage("autor",
                    new FacesMessage("Livro deve ter pelo menos um Autor"));
            return;
        }

        new DAO<Livro>(Livro.class).adiciona(this.livro);
        this.livro = new Livro();
    }

    public List<Autor> getAutores() {

        return new DAO<Autor>(Autor.class).listaTodos();
    }

    public void gravarAutor() {

        Autor autor = new DAO<Autor>(Autor.class).buscaPorId(this.autorId);
        this.livro.adicionaAutor(autor);
        System.out.println("Livro escrito por: " + autor.getNome());
    }

    public void comecaComDigitoUm(FacesContext fc, UIComponent component,
            Object value) throws ValidatorException {

        String valor = value.toString();
        if (!valor.startsWith("1")) {
            throw new ValidatorException(new FacesMessage(
                    "Deveria começar com 1!"));
        }
    }

}
1 resposta
solução!

Descobri o erro !!!!

O id="autor" estava no local errado, estava no <f:selectItems ao invés de estar em <h:selectOneMenu.

Código correto:

            <h:selectOneMenu value="#{livroBean.autorId}" id="autor">
                <f:selectItems value="#{livroBean.autores}" var="autor"
                    itemLabel="#{autor.id} - #{autor.nome}" itemValue="#{autor.id}" />
            </h:selectOneMenu>
            <h:commandButton value="Gravar Autor"
                action="#{livroBean.gravarAutor}">
                <f:ajax execute="autor" render="tabelaAutores" />
            </h:commandButton>

O estranho é que sem utilizar AJAX, funcionava sem problemas ....