28 de fev. de 2016

Criando controles customizados com JavaFX

Essa é uma tradução do artigo criado por Hendrik. Veja mais no blog GuiGarage

Por Hendrik Ebbers


Uma coisa que eu frequentemente faço com Java Swing é a customização de componentes e a criação de novos tipos de componentes. Um exemplo é o JGrid. Desde que JavaFX foi lançado, eu quis portar o JGrid. Depois de muitos experimentos e protótipos ruins, eu acho que encontrei a forma correta de fazer isso. As apresentações de Gerrit Grunwald e Jonathan Giles no JavaOne me ajudaram muito a fazer isso. As gravações dessas apresentações estão disponívels online (link e link - NOTA: links quebrados), então eu recomendo quem estiver interessado nesse tópico a investir algum tempo e assistir elas.


Primeiros passos



Como componente de interface no JavaFX é composto por um control(controle), uma skin("casca") e um behaviour(comportamento). Em um caso ideal, há também a parte do CSS.


A melhor forma de começar é criar uma classe que herda de  javafx.scene.control.Control. Essa classe é basicamente compara ao JComponent. O Control deve ter as propriedades do componente e atuar como a classe principal dele porque instâncias dessa classe serão depois criadas automaticamente no códido da aplicação e serão adicionadas para a árvore da interface com o usuário(UI):

MyCustomControl myControl = new MyCustomControl();
panel.getChildren().add(myControl);

Quando programando componentes Swing na forma correta você coloca tudo que depende da visualização ou a interação com o usuário em uma classe de UI (veja LabelUI por exemplo). JavaFX vai além e disponibiliza uma classe de skin para toda a visualização e o layout relacionado com o código e a classe behaviour para todas as interações com o usuário.



Para fazer isso em JavaFX você deve entender como as classes se relacionam. Aqui está um pequeno diagrama que mostra as relações entre essas classes:



Criando o comportamento (classe Behaviour)

Se o seu componente é somente para visualização de dados e não há interação, é baste simples criar um comportamento. Assim, você simplesmente precisa criar um classe que herde de com.sun.javafx.scene.control.behavior.BehaviorBase.

public class MyCustomControlBehavior extends BehaviorBase {
public MyCustomControlSkin(MyCustomControl control) {
super(control, new MyCustomControlBehavior(control));
    }
}

Alguns de vocês podem ficar confusos quando ver o pacote da classe BehavioyBase. No momento é uma API privada e normalmente você não deveria usar essas classes no seu código, mas os caras da Oracle sabem desse problema e irão providencias a classe BehaviorBase como uma API com o JavaFX 8. Então uma boa prática é usar essa classe privada agora e refatorar assim que o JavaFX for lançado(*).

NOTA: O javaFX 8 foi lançado já!

Criando a casca (classe Skin)

Depois que a classe de comportamento é criada, podemos dar uma olhada na skin. Sua classe de skin vai muito provavelmente herdar de com.sun.javafx.scene.control.skin.BaseSkin e criar um novo comportamento para o seu Control. O código normalmente se parece com o seguinte:

public class MyCustomControlSkin extends SkinBase{
public MyCustomControlSkin(MyCustomControl control) {
super(control, new MyCustomControlBehavior(control)); }
}

Criando o controle

A última classe que precisamos é o controle. Primeiro criamos uma classe vazia:

public class MyCustomControl extends Control {
public MyCustomControl() { }
}

Nesse ponto nós temos uma falha nas dependências das nossas classes. A skin conhece o behavior e o control. Aqui tudo parece correto, no entanto, no código da aplicação você vaim simplesmente criar um novo controle e usar como eu mostrei antes. O problema é que a classe de controle não sabe nada sobre a skin ou o behavior. Esse foi um dos maiores desafios quando eu estava aprendendo JavaFX.

Juntando as coisas


O que inicialmente parece um grande problema, é na verdade parte do poder disponibilizado pelo JavaFX. Com JavaFX é muito fácil criar diferente visualizações (skins) para os controles. Nessa parte você pode customizar a aparência dos componentes usando CSS. Como a skin é a parte principal da aparência, ela deve ser definida no CSS também. Assim, invés criar um objeto skin para o controle manualmente, você simplesmente define a classe skin que deve ser usada no seu controle. A criação e tudo mais é automaticamente feito pelas APIs do JavaFX. Para fazer isso, você deve ligar o seu controle a uma classe CSS.
Primeiramente, crie um novo arquivo no seu projeto. Eu acredito que a melhor prática é usar o mesmo pacote que o controle está e não um arquivo CSS criado sob src/main/resource:


Dentro do seu CSS você deve especificar um novo seletor para o seu componente e adicionar a skin como uma propriedade. Isso deveria parecer como o seguinte exemplo:

.custom-control {
-fx-skin: "com.guigarage.customcontrol.MyCustomControlSkin";
}

Assim que você criar o CSS, você tem que definir o mesmo no seu controle. Portanto, você deve configurar o caminho para o arquivo css e o seletor dos seus componentes:

public class MyCustomControl extends Control {
public MyCustomControl() {
getStyleClass().add("custom-control");
}
protected String getUserAgentStylesheet() {
return MyCustomControl.class.getResource("customcontrol.css").toExternalForm();
}
}

Depois que todas essas coisas foram feitas corretamente, JavaFX vai criar uma instância de skin para o seu controle. Você não precisa se preocupar com instanciação ou o mecanismo de depdendëncias. Nesse ponto eu gostaria de agradecer Jonathan Giles, que dedicou um tempo para codificar a integração do css para o gridfx  comigo e explicou todo o mecanismo e benefícios.

Acesso à Skin e Behavior

Normalmente não há a necessidade de acessar a Skin e o Behavior através do controle, mas se você precisar disso, você pode acessá-los da seguinte forma:


Como controler.getSkin() rece um javafx.scene.control.Skin e não a classe SkinBase você deve fazer um cast se precisar do behaviour:


((SkinBase)getSkin()).getBehavior();

Uma solução para os que odeiam CSS

Para alguns de vocës esse mecanimos parece um pouco exagerado. Talvez vocë simplesmente precise de um controle na sua aplicação e você não planeja usar uma skin com CSS e fazer tudo isso. Para esse caso de uso, há um ótimo workaround na API do JavaFX. Você pode ignorar toda a parte do CSS e configurar a classe skin no seu controle usando código:

public class MyCustomControl extends Control {
public MyCustomControl() {
setSkinClassName(MyCustomControlSkin.class.getName());
}
}


O benefício dessa forma de trabalho é que refatorar pacotes e nome de classes não vai quebrar o seu código e você não precisa de um arquivo CSS extra. Por outro lado, há uma grande desvantagem. Você não poderia usar skins definidas por CSS em qualquer extensão do seu controle. Acho que toda API pública, como o GridFX, deveria usar a forma com CSS. Em alguns casos de uso internos, a forma hardcoded(essa que falamos agora) é muito mais rápida.

Conclusão

Agora nós criamos um controle, uma skin e um behavior que está funcionando bem e pode ser adicionado na árvore  UI da sua aplicação. Mas, como acontece com o Swing, simplesmente herdar de JComponent não levará você a ver nada na tela. Então o próximo passo é adicionar estilo e organizar o layout do seu componente. Irei falar disso no meu próximo artigo.
Se você quiser dar uma olhada em algum componente existente veja  jgridfx ou JFXtras. No jgridfx os seguintes arquivos batem com esse artigo:

  • com.guigarage.fx.grid.GridView (control)
  • com.guigarage.fx.grid.skin.GridViewSkin (skin)
  • com.guigarage.fx.grid.behavior.GridViewBehavior (behavior)
  • /src/main/resources/com/guigarage/fx/grid/gridview.css (css)





17 de fev. de 2016

Um CRUD com JavaFX usando banco de dados

Devido ao sucesso da postagem anterior Um CRUD com JavaFX, resolvi estender a mesma para falar sobre o acesso a um banco de dados MySQL utilizando a mesma estrutura do artigo anterior.

Acessando banco de dados com Java


Esse é talvez um dos assuntos mais buscados por quem está aprendendo Java. Há muitas referências a isso na internet, inclusive uma série de artigos que o criado desse blog  escreveu há mais de 5 anos atrás para o JavaFree, veja: Acessando dados com Java: 1º parte - Simples Dao - Acessando Dados com Java: Parte 2 - Prevendo problemas - Acessando Dados com Java: Parte 3 - Hibernate Annotations

Preparando o banco

O primeiro passo nosso é garantir que você tenha o MySQL instalado(que depende do sistema operacional sendo utilizado, Linux ou outros) e uma vez feito isso, entre no MySQL e então seguir alguns passos. Para esse artigo foi criado o seguinte:


  • Usuário contas_app com senha aprendajavafx;

CREATE USER 'contas_app'@'localhost' IDENTIFIED BY 'aprendajavafx';


  •  Banco de dados contas crud-contas; 

CREATE DATABASE `crud-contas`  DEFAULT CHARACTER SET utf8  DEFAULT COLLATE utf8_general_ci


  • Garanta os privilégios do usuário contas_app para o banco crud-contas

mysql> use crud-contas
Database changed
mysql> GRANT ALL PRIVILEGES ON * . * TO 'contas_app'@'localhost';


  • Tabela Contas com os campos que precisamos

create table contas(  
   id int auto_increment, 
   concessionaria varchar(30) not null,  
   descricao varchar(1000) not null,  
   data_vencimento date not null,
   primary key(id)         
); 

Notem que o ID é auto_increment, ou seja, o MySQL vai dar o ID para nós. Se inserirmos alguns valores de teste, podemos notar que o ID já foi preenchido:

Criando uma classe para acesso ao banco de dados

Conforme falamos no artigo anterior, para trocar onde os dados serão armazenados nós iremos criar uma implementação de ContasService e trocar o tipo retornado no método getNewInstance

Para falarmos com o nosso banco de dados MySQL, precisamos ter o Driver JDBC no classpath da aplicação, pois é ele quem possibilita a comunicação de uma aplicação Java com o banco. Por isso, baixe o driver JDBC do MySQL e coloque no classpath da aplicação.

A implementação do banco de dados simplesmente envia comandos SQL para o MySQL invés de abrir e manipular arquivos, assim temos que:

* Salvar significa realizar um INSERT com os dados da conta passada;
* Buscar é trazer dados usando SELECT e transformar em uma lista de contas;
* Apagar é realizar um DELETE;
* Por fim atualizar é a realização de um UPDATE.

Claro que tudo pode ser facilitado usando um framework que faz o mapeamento de objetos para o modelo relacional de banco de dados, como o Hibernate, mas por hoje vamos ficar no básico! Finalmente, veja como ficou nossa classe ContasDBService (lembre-se que todo o código pode ser encontrado no github):


package org.aprendendojavafx.crud.service.impl;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Date;
import java.util.List;
import org.aprendendojavafx.crud.model.Conta;
import org.aprendendojavafx.crud.service.ContasService;
/**
* Faz as operações de CRUD usando banco de dados MySQL. Para rodar essa classe
* vocë precisa <br />
* - Certificar-se que o driver do MySQL está no classpath da aplicação - Criar
* o banco de dados e atualizar o código abaixo de acordo com o seu banco -
* Criar a tabela contas.
*
* Se houver qualquer erro, o programa irá sair, assim, cheque os logs para ver o erro.
*
* @author wsiqueir
*
*/
public class ContasDBService implements ContasService {
// dados para acesso ao banco, atualize de acordo com o seu banco de dados
final String USUARIO = "contas_app";
final String SENHA = "aprendajavafx";
final String URL_BANCO = "jdbc:mysql://localhost:3306/crud-contas";
// constantes de acesso
final String CLASSE_DRIVER = "com.mysql.jdbc.Driver";
// comandos
final String INSERIR = "INSERT INTO contas(concessionaria, descricao, data_vencimento) VALUES(?, ?, STR_TO_DATE(?, '%d/%m/%Y'))";
final String ATUALIZAR = "UPDATE contas SET concessionaria=?, descricao=?, data_vencimento = STR_TO_DATE(?, '%d/%m/%Y') WHERE id = ?";
final String BUSCAR = "SELECT id, concessionaria, descricao, DATE_FORMAT(data_vencimento, %d/%m/%Y') FROM contas WHERE ID = ?";
final String BUSCAR_TODOS = "SELECT id, concessionaria, descricao, DATE_FORMAT(data_vencimento, '%d/%m/%Y') FROM contas";
final String APAGAR = "DELETE FROM contas WHERE id = ?";
// tratamento de data
final String FORMATO_DATA = "dd/MM/yyyy";
final SimpleDateFormat FORMATADOR = new SimpleDateFormat(FORMATO_DATA);
@Override
public void salvar(Conta conta) {
try {
Connection con = conexao();
PreparedStatement salvar = con.prepareStatement(INSERIR);
String dateStr = FORMATADOR.format(conta.getDataVencimento());
salvar.setString(1, conta.getConcessionaria());
salvar.setString(2, conta.getDescricao());
salvar.setString(3, dateStr);
salvar.executeUpdate();
salvar.close();
con.close();
} catch (Exception e) {
e.printStackTrace();
System.err.println("ERROR SALVANDO CONTA");
System.exit(0);
}
}
@Override
public List<Conta> buscarTodas() {
List<Conta> contas = new ArrayList<>();
try {
Connection con = conexao();
PreparedStatement buscarTodos = con.prepareStatement(BUSCAR_TODOS);
ResultSet resultadoBusca = buscarTodos.executeQuery();
while (resultadoBusca.next()) {
Conta conta = extraiConta(resultadoBusca);
contas.add(conta);
}
buscarTodos.close();
con.close();
} catch (Exception e) {
e.printStackTrace();
System.err.println("ERROR BUSCANDO TODAS AS CONTAS.");
System.exit(0);
}
return contas;
}
@Override
public Conta buscaPorId(int id) {
Conta conta = null;
try {
Connection con = conexao();
PreparedStatement buscar = con.prepareStatement(BUSCAR);
buscar.setInt(1, id);
ResultSet resultadoBusca = buscar.executeQuery();
resultadoBusca.next();
conta = extraiConta(resultadoBusca);
buscar.close();
con.close();
} catch (Exception e) {
e.printStackTrace();
System.err.println("ERROR BUSCANDO CONTA COM ID " + id);
System.exit(0);
}
return conta;
}
@Override
public void apagar(int id) {
try {
Connection con = conexao();
PreparedStatement apagar = con.prepareStatement(APAGAR);
apagar.setInt(1, id);
apagar.executeUpdate();
apagar.close();
con.close();
} catch (Exception e) {
e.printStackTrace();
System.err.println("ERROR APAGANDO CONTA COM ID " + id);
System.exit(0);
}
}
@Override
public void atualizar(Conta conta) {
try {
Connection con = conexao();
PreparedStatement atualizar = con.prepareStatement(ATUALIZAR);
String dateStr = FORMATADOR.format(conta.getDataVencimento());
atualizar.setString(1, conta.getConcessionaria());
atualizar.setString(2, conta.getDescricao());
atualizar.setString(3, dateStr);
atualizar.setInt(4, conta.getId());
atualizar.executeUpdate();
atualizar.close();
con.close();
} catch (Exception e) {
e.printStackTrace();
System.err.println("ERROR ATUALIZANDO CONTA COM ID " + conta.getId());
System.exit(0);
}
}
// abre uma nova conexão com o banco de dados. Se algum erro for lançado
// aqui, verifique o erro com atenção e se o banco está rodando
private Connection conexao() {
try {
Class.forName(CLASSE_DRIVER);
return DriverManager.getConnection(URL_BANCO, USUARIO, SENHA);
} catch (Exception e) {
e.printStackTrace();
if(e instanceof ClassNotFoundException) {
System.err.println("VERIFIQUE SE O DRIVER DO BANCO DE DADOS ESTÁ NO CLASSPATH");
} else {
System.err.println("VERIFIQUE SE O BANCO ESTÁ RODANDO E SE OS DADOS DE CONEXÃO ESTÃO CORRETOS");
}
System.exit(0);
// o sistema deverá sair antes de chegar aqui...
return null;
}
}
// extrain o objeto Conta do result set
private Conta extraiConta(ResultSet resultadoBusca) throws SQLException, ParseException {
Conta conta = new Conta();
conta.setId(resultadoBusca.getInt(1));
conta.setConcessionaria(resultadoBusca.getString(2));
conta.setDescricao(resultadoBusca.getString(3));
Date dataVencimento = FORMATADOR.parse(resultadoBusca.getString(4));
conta.setDataVencimento(dataVencimento);
return conta;
}
}

15 de fev. de 2016

Um CRUD com JavaFX

Finalmente!!! Uma das postagens mais esperadas desse blog é o CRUD! 

VEJA A PARTE 2 COM BANCO DE DADOS

Todo programador já fez(ou vai fazer) um CRUD na vida e é, no geral, a forma que aprendemos conceitos básicos de uma tecnologia. Nesse artigo vamos mostrar um simples CRUD em JavaFX para explorar algumas características da tecnologia.

Definição de CRUD


CRUD é uma sigla que vem das operações básicas que podemos fazer um algo armazenado em uma fonte de dados:

Create: Criar uma nova instância de algo na fonte dos dados;
Retrieve: Trazer os dados já armazenados;
Update: Atualizar dados já armazenados;
Delete: Apagar.

Onde armazenar os dados?


A fonte de dados pode ser um banco de dados(MySQL, MariaDB, Postgres, etc), um arquivo(arquivos em diversos formatos, como CSV), um Web Service (que após chamado, salva os dados em algum lugar) ou até mesmo a própria memória dinâmica do computador(sendo que os dados se perdem quando a aplicação é fechada). Aqui vamos usar um arquivo CSV, uma forma simples de armazenar objetos em um arquivo de texto.
O motivo de não usarmos um banco de dados tradicional é que o foco do artigo é mostrar JavaFX. Um banco de dados comum implica em termos que falar de SQL, conexão com o banco, select, etc, isso vai fugir do foco.

Nosso CRUD


Nossa aplicação simples vai ser um CRUD de Contas. Sim, bills, contas! Pagamos todo mês contas de luz, água, gás, faturas, etc, etc, argh. Para representar a conta, criamos um objeto Java chamado Conta com três atributos: id (gerado automaticamente para controle interno), concessionária, descrição e data de vencimento. É isso, simples, não? Em breve mostramos o código dessa classe, mas agoras que conhecemos os campos, veja a telinha que modelamos usando o SceneBuilder que gerou um FXML (se não sabe o que é isso, veja esse artigo sobre FXML)




Começamos com uma tabela com três colunas representando os campos, após a tabela temos os campos de texto para entrada do nome, descrição e um campo para entrada de data do tipo DatePicker, que infelizmente ainda não abordamos aqui por ter sido adicionado apenas no JavaFX 8, e por fim os botões de ações.

A lógica da aplicação é a seguinte:
  • A tabela tem ID tblContas e três colunas: clConc, clDesc e clVenc. Elas são populadas com os dados de um objeto do tipo Conta;
  • Os campos de texto e o campo de data tem um (txtConc, txtDesc, dpVenc) serão injetados no controller para que possamos saber o valor que o usuário entrou;
  • Cada um dos botões  ação:
    • salvar: salva o objeto de acordo com a informação entrada pelo usuário. Não está habilitado quando um campo está selecionado na tabela;
    • atualizar: Só está habilitado quando selecionamos uma linha da coluna e permite atualizar os dados dessa linha (os campos de entrada de dados vão ser atualizados com o valor selecionado para serem modificados pelo usuário);
    • apagar: apaga a linha selecionada;
    • limpar: limpa o campo selecionado atualmente.
As operações e os elementos da tela ficam na classe ContasController. Código que veremos já já.

Fazendo as operações com o banco de dados

As operações em sí com o banco ficam na interface ContasService. Ela contém os métodos salvar, que recebe uma instância de conta a ser salva, atualizar, que recebe a conta já salva para ser atualizada, apagar, que apaga uma conta e buscarTodos, que retorna todas as contas selecionadas. É nessa classe que fazemos as operações. 
Todo o código poderia ficar dentro do controller, MAS ISSO É COISA FEIA, temos que definir os métodos em uma interface e, vejam que interessante, usando a capacidade do Java 8 de definir métodos padrões, criamos um método getInstance para retornar a implementação que queremos dessa interface. Assim, criamos a clase ContasCSVService, que é uma implementação da interface, e retornamos uma nova instância nesse método! Podemos, obviamente, criar uma interface, por exemplo ContasBDService que faria a mesma coisa, mas que invés de usar um arquivo CSV, se comunica com um banco de dados, a mesma ideia poderia ser aplicada para um arquivo XLS, como ContasXLSService, e por aí vai. O código do controller, que os métodos da interface, não iria sofrer nenhuma modificação, pois só precisaríamos trocar a intância de ContasService retornada no método getNewInstance! (claro que com CDI e outras tecnologias, sequer criar manualmente precisaríamos). Veja a nossa interface



VAMOS AO CÓDIGO!


Agora, depois desse mooonnnntteee de papo, vamos ao código. Claro que se você leu acima com atenção, vai ser muito simples de entender tudo, mas mesmo assim o código está comentado na medida do possível.

Não vou colocar o todo código na postagem, pois já está muito extensa, mas veja o código da classe Conta, da interface ContasService e da classe ContasController. O restante está no github!



package org.aprendendojavafx.crud;
import java.util.Date;
/**
*
* Nossa classe de modelo do objeto que "sofrerá" as operações de CRUD
* @author wsiqueir
*
*/
public class Conta {
private int id;
private String concessionaria;
private String descricao;
private Date dataVencimento;
// gets e sets omitidos..
}
view raw Conta.java hosted with ❤ by GitHub
package org.aprendendojavafx.crud;
import java.net.URL;
import java.time.LocalDate;
import java.time.LocalDateTime;
import java.time.ZoneId;
import java.util.Date;
import java.util.ResourceBundle;
import javafx.beans.binding.BooleanBinding;
import javafx.fxml.FXML;
import javafx.fxml.Initializable;
import javafx.scene.control.Button;
import javafx.scene.control.DatePicker;
import javafx.scene.control.TableColumn;
import javafx.scene.control.TableView;
import javafx.scene.control.TextField;
import javafx.scene.control.cell.PropertyValueFactory;
/**
*
* O controller da aplicação, onde a mágica acontece
* @author wsiqueir
*
*/
public class ContasController implements Initializable {
@FXML
private TableView<Conta> tblContas;
@FXML
private TableColumn<Conta, String> clConsc;
@FXML
private TableColumn<Conta, String> clDesc;
@FXML
private TableColumn<Conta, Date> clVenc;
@FXML
private TextField txtConsc;
@FXML
private TextField txtDesc;
@FXML
private DatePicker dpVencimento;
@FXML
private Button btnSalvar;
@FXML
private Button btnAtualizar;
@FXML
private Button btnApagar;
@FXML
private Button btnLimpart;
private ContasService service;
// Esse método é chamado ao inicializar a aplicação, igual um construtor. Ele vem da interface Initializable
@Override
public void initialize(URL location, ResourceBundle resources) {
service = ContasService.getNewInstance();
configuraColunas();
configuraBindings();
atualizaDadosTabela();
}
// métodos públicos chamados quando o botão é clicado
public void salvar() {
Conta c = new Conta();
pegaValores(c);
service.salvar(c);
atualizaDadosTabela();
}
public void atualizar() {
Conta c = tblContas.getSelectionModel().getSelectedItem();
pegaValores(c);
service.atualizar(c);
atualizaDadosTabela();
}
public void apagar() {
Conta c = tblContas.getSelectionModel().getSelectedItem();
service.apagar(c.getId());
atualizaDadosTabela();
}
public void limpar() {
tblContas.getSelectionModel().select(null);
txtConsc.setText("");
txtDesc.setText("");
dpVencimento.setValue(null);
}
// métodos privados do controller
// pega os valores entrados pelo usuário e adiciona no objeto conta
private void pegaValores(Conta c) {
c.setConcessionaria(txtConsc.getText());
c.setDescricao(txtDesc.getText());
c.setDataVencimento(dataSelecionada());
}
// método utilitário para pega a data que foi selecionada (que usa o tipo novo do java 8 LocalDateTime)
private Date dataSelecionada() {
LocalDateTime time = dpVencimento.getValue().atStartOfDay();
return Date.from(time.atZone(ZoneId.systemDefault()).toInstant());
}
// chamado quando acontece alguma operação de atualização dos dados
private void atualizaDadosTabela() {
tblContas.getItems().setAll(service.buscarTodas());
limpar();
}
// configura as colunas para mostrar as propriedades da classe Conta
private void configuraColunas() {
clConsc.setCellValueFactory(new PropertyValueFactory<>("concessionaria"));
clDesc.setCellValueFactory(new PropertyValueFactory<>("descricao"));
clVenc.setCellValueFactory(new PropertyValueFactory<>("dataVencimento"));
}
// configura a lógica da tela
private void configuraBindings() {
// esse binding só e false quando os campos da tela estão preenchidos
BooleanBinding camposPreenchidos = txtConsc.textProperty().isEmpty().or(txtDesc.textProperty().isEmpty())
.or(dpVencimento.valueProperty().isNull());
// indica se há algo selecionado na tabela
BooleanBinding algoSelecionado = tblContas.getSelectionModel().selectedItemProperty().isNull();
// alguns botões só são habilitados se algo foi selecionado na tabela
btnApagar.disableProperty().bind(algoSelecionado);
btnAtualizar.disableProperty().bind(algoSelecionado);
btnLimpart.disableProperty().bind(algoSelecionado);
// o botão salvar só é habilitado se as informações foram preenchidas e não tem nada na tela
btnSalvar.disableProperty().bind(algoSelecionado.not().or(camposPreenchidos));
// quando algo é selecionado na tabela, preenchemos os campos de entrada com os valores para o
// usuário editar
tblContas.getSelectionModel().selectedItemProperty().addListener((b, o, n) -> {
if (n != null) {
LocalDate data = null;
data = n.getDataVencimento().toInstant().atZone(ZoneId.systemDefault()).toLocalDate();
txtConsc.setText(n.getConcessionaria());
txtDesc.setText(n.getDescricao());
dpVencimento.setValue(data);
}
});
}
}
package org.aprendendojavafx.crud;
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.List;
import java.util.stream.Collectors;
/**
*
* Uma implementação do ContasService para lidar com arquivo CSV
* @author wsiqueir
*
*/
public class ContasCSVService implements ContasService {
// divisor de colunas no arquivo
private static final String SEPARADOR = ";";
// o caminho para o arquivo deve ser selecionado aqui
private static final Path ARQUIVO_SAIDA = Paths.get("./dados.csv");
// os dados do arquivo
private List<Conta> contas;
// formato de data usado no arquivo
final SimpleDateFormat formatoData = new SimpleDateFormat("dd/MM/yyyy");
public ContasCSVService() {
carregaDados();
}
@Override
public void salvar(Conta conta) {
conta.setId(ultimoId() + 1);
contas.add(conta);
salvaDados();
}
@Override
public void atualizar(Conta conta) {
Conta contaAntiga = buscaPorId(conta.getId());
contaAntiga.setConcessionaria(conta.getConcessionaria());
contaAntiga.setDataVencimento(conta.getDataVencimento());
contaAntiga.setDescricao(conta.getDescricao());
salvaDados();
}
@Override
public List<Conta> buscarTodas() {
return contas;
}
@Override
public void apagar(int id) {
Conta conta = buscaPorId(id);
contas.remove(conta);
salvaDados();
}
public Conta buscaPorId(int id) {
return contas.stream().filter(c -> c.getId() == id).findFirst()
.orElseThrow(() -> new Error("Conta não encontrada"));
}
// salva a lista de dados no arquivo, gerando um novo CSV e escrevendo o arquivo completamente
private void salvaDados() {
StringBuffer sb = new StringBuffer();
for (Conta c : contas) {
String linha = criaLinha(c);
sb.append(linha);
sb.append(System.getProperty("line.separator"));
}
try {
Files.delete(ARQUIVO_SAIDA);
Files.write(ARQUIVO_SAIDA, sb.toString().getBytes());
} catch (IOException e) {
e.printStackTrace();
System.exit(0);
}
}
// o ID mais alto é retornado aqui para continuarmos contando os IDs
private int ultimoId() {
return contas.stream().mapToInt(Conta::getId).max().orElse(0);
}
// carrega os dados do arquivo para a lista contas
private void carregaDados() {
try {
if(!Files.exists(ARQUIVO_SAIDA)) {
Files.createFile(ARQUIVO_SAIDA);
}
contas = Files.lines(ARQUIVO_SAIDA).map(this::leLinha).collect(Collectors.toList());
} catch (IOException e) {
e.printStackTrace();
System.exit(0);
}
}
// transforma uma linha do CSV para o tipo Conta
private Conta leLinha(String linha) {
String colunas[] = linha.split(SEPARADOR);
int id = Integer.parseInt(colunas[0]);
Date dataVencimento = null;
try {
dataVencimento = formatoData.parse(colunas[3]);
} catch (ParseException e) {
e.printStackTrace();
System.exit(0);
}
Conta conta = new Conta();
conta.setId(id);
conta.setConcessionaria(colunas[1]);
conta.setDescricao(colunas[2]);
conta.setDataVencimento(dataVencimento);
return conta;
}
// transforma um objeto conta em um arquivo CSV
private String criaLinha(Conta c) {
String dataStr = formatoData.format(c.getDataVencimento());
String idStr = String.valueOf(c.getId());
String linha = String.join(SEPARADOR, idStr, c.getConcessionaria(), c.getDescricao(),
dataStr);
return linha;
}
}