Olá pessoal! Eu sou Alex Felipe, instrutor da Alura, e venho apresentar o curso de Flutter com Web API. Para esse treinamento, assumirei que você tem todo o conhecimento do curso de Persistência de Dados Internos utilizando o SQFlight. Como reaproveitaremos o projeto desenvolvido durante esse curso, é recomendado ter feito a implementação de todo o projeto, entendendo suas peculiaridades e dificuldades, e até mesmo tê-lo pronto para então seguirmos sem nenhum problema.
Agora que conhecemos esses pré-requisitos, vamos entender o que será desenvolvido durante o treinamento para, posteriormente, conversarmos sobre conteúdos mais técnicos que serão utilizados por baixo dos panos. Perceba que teremos o mesmo visual inicial do projeto anterior, mas com algumas funcionalidades diferentes. Enquanto antes tínhamos a funcionalidade de contatos, agora teremos uma de transferência e outra que lista as transferências realizadas no uso do aplicativo.
Clicando em "Transfer", acessaremos uma tela na qual ainda mantemos a lista de contatos - ou seja, nossa lista também foi reutilizada para esse novo curso, mas agora com uma nova funcionalidade.
Ao clicarmos em um dos contatos, será aberto um novo formulário para criação de uma transferência.
Isso nos traz uma nova exigência, que é fazer a comunicação com uma Web API, um serviço online onde as transferências enviadas serão registradas e poderão ser buscadas.
Para visualizarmos as transferências realizadas, voltaremos ao dashboard e acessaremos o Transaction Feed, onde buscaremos todas as transferências que foram cadastradas. Quando não há nenhuma inforação, a mensagem "No transactions found" será exibida para o usuário.
Para testarmos essas novas funcionalidades, simularemos uma transferência para o Alex no valor de 2000
reais. Voltando ao Transaction Feed, essa transferência será exibida corretamente.
Nesse curso, aprenderemos a fazer tanto a comunicação de mandar as informações para a Web API quanto a de buscá-las. Para termos certeza de que essa comunicação está funcionando, podemos colocar o emulador do Android Studio no movo avião, cortando o acesso à rede. Feito isso, se acessarmos o Transactions Feed, a mensagem "No transactions found" voltará a ser exibida, afinal nenhuma informação foi encontrada.
Isso significa que realmente estamos fazendo uma comunicação externa com um serviço online ao invés de trabalharmos com um banco de dados local. Agora que conhecemos o fluxo do produto, vamos entender as técnicas que serão utilizadas por baixo dos panos. Primeiramente, utilizaremos um pacote do próprio Dart, que é justamente o HTTP, para fazermos as comunicações no protocolo HTTP.
Veremos como é feita sua configuração inicial e como configuramos o interceptador para identificarmos o que está acontecendo durante a comunicação, aprenderemos a fazer a conversão dos dados que são enviados e recebidos via HTTP, além de quais cuidados devem ser tomados quando fazemos uma comunicação (seja ela bem sucedida ou não).
Também veremos como aplicar as novas funcionalidades que temos, por exemplo, na lista de contatos, onde o clique agora envia uma informação diferente para o formulário - quando clicamos no Alex, abrimos as informações desse contato, e assim por diante.
Ao longo do nosso treinamento aprenderemos todas essas abordagens, passando por problemas e erros comuns do desenvolvimento, considerando as boas práticas de programação e e conhecendo maneiras mas sucintas de trabalhar com o Dart no Flutter. Bons estudos!
Agora que o cliente nos apresentou uma nova necessidade, vamos entender o que precisa ser modificado no código para começarmos a implementação. Para isso, consultaremos o documento da proposta de implementação, focado na integração de uma Web API. Logo de cara, perceberemos que o *Dashboard *está de volta, justamente porque agora ele exibirá novas funcionalidades.
Inclusive, podemos comparar a Dashboard desejada com a que temos atualmente, que possui uma única funcionalidade (Contacts) que será substituída por duas outras (Transfer, para transferências, e Transaction Feed, uma lista das transferências realizadas). Na próxima página, que mostra os pormenores da tela de transferência, perceberemos esta é, na verdade, a tela Contacts com um novo nome e um comportamento adicional: ao clicarmos em um dos contatos cadastrados, será aberto um formulário permitindo a inserção de um valor para a transferência. Essa é uma adaptação que também teremos que fazer.
Como já vimos que será necessário modificar bastante o fluxo inicial, vamos adaptar o nosso código para que ele dê suporte a essas múltiplas funcionalidades no dashboard e para que a lista de contatos tenha o novo nome. O primeiro ajuste no código será modificarmos o visual do botão, ou seja, o nome e o ícone. Na classe Dashboard
, substituiremos "Contacts" por "Transfer" e o Icons.people
por Icons.monetization_on
.
children: <Widget>[
Icon(
Icons.monetization_on,
color: Colors.white,
size: 24.0,
),
Text('Transfer',
style: TextStyle(
color: Colors.white,
fontSize: 16.0,
)),
],
Feito isso, utilizaremos o Hot Restart para visualizarmos a tela atualizada, que ficou exatamente como desejado.
Também modificaremos a classe ContactsList
, alterando o Text('Contacts')
por Text('Transfer')
. Após um novo Hot Restart, faremos a navegação da Dashboard para a tela Transfer, que terá sido atualizada com o novo título.
class ContactsList extends StatelessWidget {
final ContactDao _dao = ContactDao();
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text('Transfer'),
//...
Terminada essa primeira modificação, agora precisamos adaptar o código para que consigamos adicionar mais de uma funcionalidade. Para isso, necessitamos de uma nova estrutura que mantenha tal funcionalidade, que é justamente todo o código dentro do segundo Padding()
, incluindo Material()
, InkWell()
, Container()
, Column()
, etc. Minimizaremos então esse código, adicionaremos uma vírgula ao final da linha e pressionaremos "Ctrl + D" para duplicá-lo. Perceba que ficaremos então com dois paddings:
body: Column(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
crossAxisAlignment: CrossAxisAlignment.start,
children: <Widget>[
Padding(
padding: const EdgeInsets.all(8.0),
child: Image.asset('images/bytebank_logo.png'),
),
Padding(...),
Padding(...),
],
),
//...
Ao fazermos o Hot Restart, teremos na tela dois botões - ou seja, duas funcionalidades - em uma mesma coluna na vertical.
Porém, na proposta de implementação, a ideia é que as funcionalidades sejam apresentadas horizontalmente. Para isso, precisaremos de outra estrutura, chamada Row()
("linha"), cujo comportamento é muito similar ao de uma coluna, mas no modo horizontal.
Para adicionarmos essa linha, usaremos os recursos do IntelliJ. Após clicarmos em um widget, no caso o Padding()
, usaremos o atalho "Alt + Enter" para acessarmos algumas opções, dentre elas "Wrap with Row". Isso basicamente agrupará toda a estrutura selecionada em uma Row()
. Em seguida, recortaremos o segundo Padding()
com "Ctrl + X" e o colocaremos também dentro da nova Row()
.
Row(
children: <Widget>[
Padding(...),
Padding(...),
],
),
//...
Com essa estrutura pronta, as duas funcionalidades passarão a ser exibidas lado a lado na nossa Dashboard.
Claro, como replicamos o código, o comportamento ainda não é o esperado (Transfer e Transaction Feed). O ideal seria adaptarmos o código para que essa modificação de conteúdo seja feita facilmente. Pensando nisso, faz sentido extrairmos a estrutura do Padding()
, Material()
, InkWell()
e assim por diante para um widget próprio, que pode ser nomeado, por exemplo, como FeatureItem()
, ou seja, um item que representa uma funcionalidade.
Ainda no arquivo dashboard.dart
, ao final do código, começaremos a criação desse widget com a estrutura stless
seguida de um "Enter", o que criará um StatelessWidget
. Como somente a nossa Dashboard utilizará esse recurso, ele será privado, e se chamará _FeatureItem
.
class _FeatureItem extends StatelessWidget {
@override
Widget build(BuildContext context) {
return Container()
}
}
Nele, ao invés de Container()
, retornaremos toda a estrutura do nosso Padding()
:
class _FeatureItem extends StatelessWidget {
@override
Widget build(BuildContext context) {
return Padding(...);
}
}
Agora teremos esse código com um nome mais significativo e, ao invés de utilizarmos os paddings na classe Dashboard
, passaremos a utilizar o _FeatureItem()
, o widget que foi extraído.
body: Column(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
crossAxisAlignment: CrossAxisAlignment.start,
children: <Widget>[
Padding(
padding: const EdgeInsets.all(8.0),
child: Image.asset('images/bytebank_logo.png'),
),
Row(
children: <Widget>[
_FeatureItem(),
_FeatureItem(),
],
),
],
),
/...
Atualizando a aplicação, ela manterá o mesmo comportamento. Agora podemos indicar quais são os valores esperados nesse Widget. Por exemplo, a string "Transfer" pode ser representada por um name
("nome"), e Icons.monetization_on
pode ser representado como icon
("ícone"). Criaremos esses atributos no widget prestando atenção aos seus devidos tipos (String
e IconData
, respectivamente), e os passaremos em um construtor _FeatureItem()
.
class _FeatureItem extends StatelessWidget {
final String name;
final IconData icon;
_FeatureItem(this.name, this.icon);
//...
Agora, cada vez que o _FeatureItem()
for utilizado, exigiremos que os seus valores sejam enviados. Na primeira chamada,, enviaremos Transfer
e Icons.monetization_on
, e na segunda Transaction Feed
e Icons.description
(que é o ícone de "descrição" exibido na proposta de implementação).
Row(
children: <Widget>[
_FeatureItem('Transfer', Icons.monetization_on),
_FeatureItem('Transaction Feed', Icons.description),
],
),
Após essa adaptação, teremos um visual bastante semelhar ao que foi apresentado na proposta.
Se clicarmos em "Transfer", a lista de contatos é aberta corretamente. Porém, o mesmo acontece se clicarmos em "Transaction Feed", afinal estamos reutilizando o mesmo código. A seguir trabalharemos de forma que cada funcionalidade tenha sua própria navegação.
Precisamos modificar o nosso código para que cada uma das suas funcionalidades consiga acessar uma tela específica, já que, como vimos anteriormente, tanto a "Transfer" quanto a "Transaction Feed" atualmente acessam a mesma tela, que é a lista de contatos.
Em _FeatureItem()
, que é o código compartilhado por essas funcionalidades, temos em comum a propriedade onTap
, que é justamente a identificação de que o widget foi clicado. Dentro dela é feita a navegação para a lista de contatos - ou seja, é aqui que precisaremos atuar. Mas o que deve ser feito para que um código específico seja executado a depender da funcionalidade?
Existem diversas formas de resolver esse problema. Uma delas seria incluirmos mais um argumento no construtor, informando o que a funcionalidade clicada representa, como "Transaction Feed", "Transfer" e assim por diante. Porém, esse tipo de implementação possui um problema: como o código tende a crescer, pode ser que acabemos trabalhando com diversas funcionalidades, exigindo mais responsabilidades do _FeatureItem()
, um widget cujo propósito é ser reutilizado e que deveria ter menos responsabilidades.
Pensando nisso, ao invés de mantermos todo o código que toma essas decisões dentro do _FeatureItem()
, faz todo sentido trabalharmos com uma outra solução, na qual o Dashboard
, após ser notificado de que uma funcionalidade foi clicada, será responsável por tomar a decisão sobre o que deverá ser feito.
Essa implementação pode ser realizada por meio de callbacks, utilizando funções para nos indicar que algo foi clicado e para criar o comportamento esperado. No _FeatureItem()
, criaremos um novo atributo cujo tipo será Function
, tipo este que determina um callback do Dart. Podemos utilizar um nome como onTap
ou onClick
para referenciar a ação responsável por sua execução. No caso, utilizaremos onClick
para diferenciarmos do onTap
que já existe no nosso widget.
Para recebermos o onClick
via construtor, usaremos uma técnica um pouco diferente daquilo que já vimos no Flutter, de modo a conhecermos um comportamento comum nesse tipo de abordagem que é o recebimento de callbacks. No construtor, indicaremos que o callback this.onClick
será recebido como um parâmetro opcional, o que é feito com chaves ({}
).
class _FeatureItem extends StatelessWidget {
final String name;
final IconData icon;
final Function onClick;
_FeatureItem(this.name, this.icon, {this.onClick});
//...
Agora, na utilização do _FeatureItem()
, poderemos mandar também o onClick
. Dado que ele representa uma função sem nenhum tipo de argumento, podemos implementá-lo vazio e então definir o seu comportamento.
Row(
children: <Widget>[
_FeatureItem(
'Transfer',
Icons.monetization_on,
onClick: () {},
),
_FeatureItem(
'Transaction Feed',
Icons.description,
),
],
),
Agora vamos à implementação "diferente" que foi citada. Ao invés de simplesmente colocarmos um parâmetro opcional, faremos com que o Dart indique para qualquer pessoa que utilize o nosso componente que a implementação do onClick
é uma exigência, da mesma maneira que aconteceu quando utilizamos o RaisedButton()
ou o FloatingActionButton()
, nos quais precisamos implementar o onPress
, onTap
e assim por diante. Para isso, usaremos a anotação @required
.
class _FeatureItem extends StatelessWidget {
final String name;
final IconData icon;
final Function onClick;
_FeatureItem(this.name, this.icon, {@required this.onClick});
//...
O @required
indica que o atributo onClick
precisa ser implementado para que o widget funcione da maneira esperada, e pode ser utilizada em vários parâmetros. Feito isso, a IDE passará a informar, na chamada de _FeatureItem()
, que o parâmetro onClick
é requerido, ainda que o código funcione sem a sua implementação.
Feito o nosso callback, quando identificarmos que o onTap
foi executado, bastará executarmos também o onClick()
, Dessa forma, ele irá notificar o escopo que implementamos no _FeatureItem()
da Dashboard.
child: InkWell(
onTap: () {
onClick();
Navigator.of(context).push(
(MaterialPageRoute(
builder: (context) => ContactsList(),
)),
);
},
//...
Para testarmos isso, executaremos um print()
simples com a mensagem "transfer was clicked".
_FeatureItem(
'Transfer',
Icons.monetization_on,
onClick: () {
print('transfer was clicked');
},
),
Após fazermos o Hot reload e clicarmos no "Transfer" da nossa aplicação, a mensagem "transfer was clicked" será exibida no console, exatamente como planejamos.
I/flutter ( 7236): transfer was clicked
Já se clicarmos em "Transaction Feed", como não implementamos nenhum tipo de ação, receberemos um "null". Agora precisamos adaptar as nossas funcionalidades para que cada uma tenha seu próprio comportamento.
O primeiro passo será removermos a navegação específica do nosso Padding()
, deixando apenas o onClick()
. Podemos ainda simplificá-lo com uma expressão (arrow function):
return Padding(
padding: const EdgeInsets.all(8.0),
child: Material(
color: Theme.of(context).primaryColor,
child: InkWell(
onTap: () => onClick(),
//...
Criaremos então, dentro da classe Dashboard
, a função _showContactsList()
, que navegará para a lista de contatos. No corpo da função colaremos o código de navegação que criamos anteriormente, e como argumento ela receberá a dependência BuildContext
.
void _showContactsList(BuildContext context) {
Navigator.of(context).push(
MaterialPageRoute(
builder: (context) => ContactsList(),
),
);
}
Com isso, quando fizermos o onClick
do nosso "Transfer", poderemos chamar o _showContactsList()
mandando como argumento o context
.
_FeatureItem(
'Transfer',
Icons.monetization_on,
onClick: () {
_showContactsList(context);
},
),
//...
Já no "Transaction Feed", faremos de forma um pouco diferente. Usaremos o atalho "Alt + Enter" do IntelliJ para implementar o onClick
, que se iniciará como null
, e então indicaremos o comportamento esperado. Dado que ainda não temos a tela específica para essa funcionalidade, vamos trabalhar com o print()
, passando o texto "transaction feed was clicked" para confirmarmos que tudo está funcionando corretamente.
Após o Hot Restart, voltaremos à aplicação para realizarmos nossos testes. Se clicarmos em "Transfer", seremos corretamente redirecionados para a lista de contatos. Já clicando em "Transaction Feed", a mensagem "transaction feed was clicked" será exibida no console.
I/flutter ( 8617): transaction feed was clicked
Esse tipo de abordagem, delegando responsabilidades de determinados eventos nos nossos widgets, é muito comum e será utilizada toda vez que precisarmos subir algum nível para implementar alguma funcionalidade específica, seja com base em um click ou um gesto.
Para fecharmos essa parte de adição de funcionalidades, vamos verificar o que acontece quando replicamos mais uma feature. Para isso, selecionaremos o _FeatureItem()
do nosso Transaction Feed e utilizaremos o atalho "Ctrl + D" para duplicá-lo. Executando novamente a aplicação, teremos um problema, pois estamos ultrapassando o limite da lateral da tela, impedindo o acesso a essa e às demais funcionalidades.
Isso acontece pois o Row()
tem um comportamento bem parecido com o do Column()
, ou seja, possui um tamanho fixo e não inclui comportamentos de scroll (rolagem). Nesse caso, teremos que fazer algumas adaptações. Uma delas será a adição do widget SingleChildScrollView()
.
SingleChildScrollView(
child: Row(
children: <Widget>[
_FeatureItem(...),
_FeatureItem(...),
_FeatureItem(...),
],
),
),
//...
Entretanto, com isso, nossa aplicação continuará apresentando o mesmo problema. Isso porque, por padrão, o SingleChildScrollView
opera no modo vertical. Sendo assim, precisaremos adaptar a sua direção utilizando a propriedade scrollDirection
, que recebe como valor um Axis
que permite modificar a sua direção, e que nesse caso definiremos como Axis.horizontal
.
SingleChildScrollView(
scrollDirection: Axis.horizontal,
child: Row(
children: <Widget>[
_FeatureItem(...),
_FeatureItem(...),
_FeatureItem(...),
],
),
),
//...
Dessa vez conseguiremos incluir corretamente o comportamento de scroll, permitindo o acesso às outras funcionalidades na horizontal, tornando nossa aplicação mais flexível.
Claro, também é possível trabalhar com um conceito que já vimos anteriormente, que é o ListView()
. Para isso, removeremos o SingleChildScrollView
usando "Alt + Enter" e clicando na opção "Replace widget with its children". Em seguida, modificaremos o Row()
para ListView()
, o que fará com que nossa aplicação deixe de exibir as funcionalidades (já que ainda precisamos determinar um tamanho para o ListView()
).
Antes de adaptarmos o tamanho, adicionaremos a propriedade scrollDirection
, novamente com Axis.horizontal
, para que as funcionalidades sejam exibidas na horizontal.
ListView(
scrollDirection: Axis.horizontal,
children: <Widget>[
_FeatureItem(...),
_FeatureItem(...),
_FeatureItem(...),
],
),
//...
Entretanto, elas ainda não serão exibidas corretamente na tela. Como não definimos um tamanho, o ListView()
tende a ocupar o máximo que ele pode da tela, impedindo a visualização. Para resolvermos isso, o colocaremos dentro de um Container
("Alt + Enter" seguido de "Wrap with Container") e estabeleceremos um height
de 100
, o mesmo que estávamos utilizando para nossas features.
Container(
height: 100,
child: ListView(
scrollDirection: Axis.horizontal,
children: <Widget>[
_FeatureItem(...),
_FeatureItem(...),
_FeatureItem(...),
],
),
),
Feito isso, as funcionalidades passarão a ser exibidas na tela, ainda que não atendam exatamente às nossas necessidades por não estarem no tamanho ideal.
Corrigiremos isso removendo a propriedade height
do Container()
que está localizado no nosso _FeatureItem()
e alterando o valor dessa mesma propriedade, agora no Container()
que contém o ListView()
, para 120
.
Com isso, teremos conseguindo implementar as nossas duas funcionalidades utilizando soluções que também nos permitem adicionar diversas outras, cada uma mantendo o seu próprio clique. Antes de partirmos para o próximo passo, removeremos a funcionalidade duplicada.
O curso Flutter com Web API: integrando sua app mobile possui 152 minutos de vídeos, em um total de 45 atividades. Gostou? Conheça nossos outros cursos de Flutter em Mobile, ou leia nossos artigos de Mobile.
Matricule-se e comece a estudar com a gente hoje! Conheça outros tópicos abordados durante o curso:
Impulsione a sua carreira com os melhores cursos e faça parte da maior comunidade tech.
1 ano de Alura
Assine o PLUS e garanta:
Formações com mais de 1500 cursos atualizados e novos lançamentos semanais, em Programação, Inteligência Artificial, Front-end, UX & Design, Data Science, Mobile, DevOps e Inovação & Gestão.
A cada curso ou formação concluído, um novo certificado para turbinar seu currículo e LinkedIn.
No Discord, você tem acesso a eventos exclusivos, grupos de estudos e mentorias com especialistas de diferentes áreas.
Faça parte da maior comunidade Dev do país e crie conexões com mais de 120 mil pessoas no Discord.
Acesso ilimitado ao catálogo de Imersões da Alura para praticar conhecimentos em diferentes áreas.
Explore um universo de possibilidades na palma da sua mão. Baixe as aulas para assistir offline, onde e quando quiser.
Acelere o seu aprendizado com a IA da Alura e prepare-se para o mercado internacional.
1 ano de Alura
Todos os benefícios do PLUS e mais vantagens exclusivas:
Luri é nossa inteligência artificial que tira dúvidas, dá exemplos práticos, corrige exercícios e ajuda a mergulhar ainda mais durante as aulas. Você pode conversar com a Luri até 100 mensagens por semana.
Aprenda um novo idioma e expanda seus horizontes profissionais. Cursos de Inglês, Espanhol e Inglês para Devs, 100% focado em tecnologia.
Transforme a sua jornada com benefícios exclusivos e evolua ainda mais na sua carreira.
1 ano de Alura
Todos os benefícios do PRO e mais vantagens exclusivas:
Mensagens ilimitadas para estudar com a Luri, a IA da Alura, disponível 24hs para tirar suas dúvidas, dar exemplos práticos, corrigir exercícios e impulsionar seus estudos.
Envie imagens para a Luri e ela te ajuda a solucionar problemas, identificar erros, esclarecer gráficos, analisar design e muito mais.
Escolha os ebooks da Casa do Código, a editora da Alura, que apoiarão a sua jornada de aprendizado para sempre.