first commit

This commit is contained in:
2026-03-10 15:34:35 +00:00
commit a5ee481b39
68 changed files with 3187 additions and 0 deletions

View File

@@ -0,0 +1,26 @@
package com.example.pap;
import android.content.Context;
import androidx.test.platform.app.InstrumentationRegistry;
import androidx.test.ext.junit.runners.AndroidJUnit4;
import org.junit.Test;
import org.junit.runner.RunWith;
import static org.junit.Assert.*;
/**
* Instrumented test, which will execute on an Android device.
*
* @see <a href="http://d.android.com/tools/testing">Testing documentation</a>
*/
@RunWith(AndroidJUnit4.class)
public class ExampleInstrumentedTest {
@Test
public void useAppContext() {
// Context of the app under test.
Context appContext = InstrumentationRegistry.getInstrumentation().getTargetContext();
assertEquals("com.example.pap", appContext.getPackageName());
}
}

View File

@@ -0,0 +1,54 @@
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools">
<uses-permission android:name="android.permission.INTERNET" />
<application
android:allowBackup="true"
android:dataExtractionRules="@xml/data_extraction_rules"
android:fullBackupContent="@xml/backup_rules"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:roundIcon="@mipmap/ic_launcher_round"
android:supportsRtl="true"
android:theme="@style/Theme.PAP">
<activity
android:name=".Ingrediente"
android:exported="false" />
<activity
android:name=".PerfilActivity"
android:exported="false" />
<activity
android:name=".DesafiosActivity"
android:exported="false" />
<activity
android:name=".FotoActivity"
android:exported="false" />
<activity
android:name=".EstatisticasActivity"
android:exported="false" />
<activity
android:name=".ChatActivity"
android:exported="false" />
<activity
android:name=".HomeActivity"
android:exported="false" />
<activity
android:name=".RegisterActivity"
android:exported="false" />
<activity
android:name=".MainActivity"
android:exported="false" />
<activity
android:name=".LoginActivity"
android:exported="true">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
</manifest>

View File

@@ -0,0 +1,138 @@
package com.example.pap;
import android.graphics.Color;
import android.graphics.drawable.GradientDrawable;
import android.os.Bundle;
import android.os.Handler;
import android.os.Looper;
import android.view.Gravity;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.LinearLayout;
import android.widget.ScrollView;
import android.widget.TextView;
import androidx.appcompat.app.AppCompatActivity;
public class ChatActivity extends AppCompatActivity {
private LinearLayout chatLayout;
private ScrollView chatScrollView;
private EditText etNovaMensagem;
private Button btnEnviarMensagem;
private Button btnVoltarChat;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_chat);
// 1. Associar os componentes do layout
chatLayout = findViewById(R.id.chatLayout);
chatScrollView = findViewById(R.id.chatScrollView);
etNovaMensagem = findViewById(R.id.etNovaMensagem);
btnEnviarMensagem = findViewById(R.id.btnEnviarMensagem);
btnVoltarChat = findViewById(R.id.btnVoltarChat);
// Mensagem inicial da IA ao abrir o ecrã
adicionarMensagemBalao("Olá! Sou o teu IA Coach de Saúde. Como te posso ajudar hoje na tua jornada de emagrecimento?", false);
// 2. Clique no botão de Enviar
btnEnviarMensagem.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
String mensagemUser = etNovaMensagem.getText().toString().trim();
if (!mensagemUser.isEmpty()) {
// Adiciona a mensagem do utilizador no ecrã
adicionarMensagemBalao(mensagemUser, true);
// Limpa a caixa de texto
etNovaMensagem.setText("");
// Simula a IA a processar e a responder
simularRespostaDaIA(mensagemUser);
}
}
});
// 3. Clique no botão de Voltar
btnVoltarChat.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
finish();
}
});
}
// Passo 3: Função para desenhar os balões de conversa no ecrã
private void adicionarMensagemBalao(String texto, boolean isUser) {
TextView textView = new TextView(this);
textView.setText(texto);
textView.setTextSize(16f);
textView.setPadding(32, 24, 32, 24);
// Configurar as margens e o alinhamento (Direita = Utilizador, Esquerda = IA)
LinearLayout.LayoutParams params = new LinearLayout.LayoutParams(
LinearLayout.LayoutParams.WRAP_CONTENT,
LinearLayout.LayoutParams.WRAP_CONTENT
);
params.setMargins(0, 16, 0, 16);
// Configurar o design arredondado do balão
GradientDrawable background = new GradientDrawable();
background.setCornerRadius(32f);
if (isUser) {
// Estilo do Utilizador (Fundo Verde, Texto Branco, Alinhado à direita)
params.gravity = Gravity.END;
background.setColor(Color.parseColor("#4CAF50"));
textView.setTextColor(Color.WHITE);
} else {
// Estilo da IA (Fundo Cinza/Branco, Texto Escuro, Alinhado à esquerda)
params.gravity = Gravity.START;
background.setColor(Color.parseColor("#FFFFFF"));
textView.setTextColor(Color.parseColor("#2E3D32"));
textView.setElevation(4f); // Dá uma pequena sombra ao balão da IA
}
textView.setLayoutParams(params);
textView.setBackground(background);
// Adicionar o balão ao ecrã
chatLayout.addView(textView);
// Fazer scroll automático para baixo para ver a nova mensagem
chatScrollView.post(new Runnable() {
@Override
public void run() {
chatScrollView.fullScroll(ScrollView.FOCUS_DOWN);
}
});
}
// Passo 4: Função para simular a inteligência da IA
private void simularRespostaDaIA(String perguntaUtilizador) {
perguntaUtilizador = perguntaUtilizador.toLowerCase();
final String resposta;
// Regras simples para fingir que a IA está a entender a pergunta
if (perguntaUtilizador.contains("água") || perguntaUtilizador.contains("agua")) {
resposta = "A hidratação é fundamental! O ideal é beberes pelo menos 2 a 3 litros de água por dia. Já bebeste algum copo hoje?";
} else if (perguntaUtilizador.contains("comer") || perguntaUtilizador.contains("fome") || perguntaUtilizador.contains("dieta")) {
resposta = "Se sentires fome entre as refeições, opta por snacks saudáveis como uma peça de fruta ou um punhado de frutos secos. Evita os doces!";
} else if (perguntaUtilizador.contains("treino") || perguntaUtilizador.contains("exercício") || perguntaUtilizador.contains("correr")) {
resposta = "Excelente! O exercício acelera o metabolismo. Lembra-te que a consistência é mais importante do que a intensidade. 30 minutos de caminhada já fazem a diferença.";
} else {
resposta = "Essa é uma ótima questão! O segredo do emagrecimento é o défice calórico aliado a bons hábitos. Continua focado, estou aqui para te apoiar em cada passo!";
}
// Simular um tempo de "pensamento" da IA (espera 1.5 segundos antes de mostrar a resposta)
new Handler(Looper.getMainLooper()).postDelayed(new Runnable() {
@Override
public void run() {
adicionarMensagemBalao(resposta, false);
}
}, 1500);
}
}

View File

@@ -0,0 +1,147 @@
package com.example.pap;
import android.content.ActivityNotFoundException;
import android.content.Intent;
import android.content.SharedPreferences;
import android.graphics.Color;
import android.os.Bundle;
import android.os.Handler;
import android.os.Looper;
import android.provider.MediaStore;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;
import android.widget.Toast;
import androidx.activity.result.ActivityResultLauncher;
import androidx.activity.result.contract.ActivityResultContracts;
import androidx.appcompat.app.AppCompatActivity;
public class DesafiosActivity extends AppCompatActivity {
private TextView tvPontos, tvStatusDesafio1, tvStatusDesafio2;
private Button btnVideoDesafio1, btnVideoDesafio2, btnVoltarDesafios;
private int pontosAtuais = 0;
private SharedPreferences sharedPreferences;
// Variável para saber qual desafio está a ser gravado no momento
private int desafioAtualEmGravacao = -1;
// Ferramenta para abrir a câmara de vídeo e receber o resultado
private final ActivityResultLauncher<Intent> videoLauncher = registerForActivityResult(
new ActivityResultContracts.StartActivityForResult(),
result -> {
if (result.getResultCode() == RESULT_OK) {
// O utilizador gravou o vídeo com sucesso!
Toast.makeText(this, "Vídeo capturado! A enviar para a IA...", Toast.LENGTH_SHORT).show();
simularAnaliseDaIA(desafioAtualEmGravacao);
} else {
Toast.makeText(this, "Gravação cancelada.", Toast.LENGTH_SHORT).show();
}
}
);
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_desafios);
// 1. Inicializar os componentes
tvPontos = findViewById(R.id.tvPontos);
tvStatusDesafio1 = findViewById(R.id.tvStatusDesafio1);
tvStatusDesafio2 = findViewById(R.id.tvStatusDesafio2);
btnVideoDesafio1 = findViewById(R.id.btnVideoDesafio1);
btnVideoDesafio2 = findViewById(R.id.btnVideoDesafio2);
btnVoltarDesafios = findViewById(R.id.btnVoltarDesafios);
// 2. Carregar os dados guardados no telemóvel
sharedPreferences = getSharedPreferences("AppEmagrecimento", MODE_PRIVATE);
carregarDadosGuardados();
// 3. Configurar os cliques para gravar vídeo
btnVideoDesafio1.setOnClickListener(v -> abrirCameraVideo(1));
btnVideoDesafio2.setOnClickListener(v -> abrirCameraVideo(2));
// 4. Botão de voltar
btnVoltarDesafios.setOnClickListener(v -> finish());
}
// Função para abrir a câmara do telemóvel em modo VÍDEO
private void abrirCameraVideo(int numeroDoDesafio) {
desafioAtualEmGravacao = numeroDoDesafio;
Intent intent = new Intent(MediaStore.ACTION_VIDEO_CAPTURE);
try {
videoLauncher.launch(intent);
} catch (ActivityNotFoundException e) {
// Caso o emulador não tenha uma câmara configurada, abre a galeria para selecionar um vídeo
Toast.makeText(this, "Câmara não encontrada. Selecione um vídeo da galeria.", Toast.LENGTH_LONG).show();
Intent galleryIntent = new Intent(Intent.ACTION_GET_CONTENT);
galleryIntent.setType("video/*");
videoLauncher.launch(galleryIntent);
}
}
// Função que simula o tempo de pensamento da Inteligência Artificial
private void simularAnaliseDaIA(int numeroDoDesafio) {
TextView tvStatusAtual = (numeroDoDesafio == 1) ? tvStatusDesafio1 : tvStatusDesafio2;
Button btnAtual = (numeroDoDesafio == 1) ? btnVideoDesafio1 : btnVideoDesafio2;
// Muda a interface para mostrar que a IA está a trabalhar
tvStatusAtual.setText("A IA está a analisar os teus movimentos...");
tvStatusAtual.setTextColor(Color.parseColor("#2196F3")); // Azul
btnAtual.setEnabled(false);
btnAtual.setText("A processar...");
// Simula uma espera de 3.5 segundos (tempo que a IA demoraria a analisar o vídeo)
new Handler(Looper.getMainLooper()).postDelayed(() -> {
// Depois de 3.5 segundos, a IA aprova o desafio!
int pontosGanhos = (numeroDoDesafio == 1) ? 20 : 30;
// Atualizar UI
tvStatusAtual.setText("Aprovado pela IA! +" + pontosGanhos + " pts");
tvStatusAtual.setTextColor(Color.parseColor("#4CAF50")); // Verde
btnAtual.setText("Desafio Concluído");
btnAtual.setBackgroundTintList(android.content.res.ColorStateList.valueOf(Color.parseColor("#81C784")));
// Adicionar pontos e guardar
adicionarPontosEGuardar(numeroDoDesafio, pontosGanhos);
}, 3500); // 3500 milissegundos = 3.5 segundos
}
private void adicionarPontosEGuardar(int numeroDoDesafio, int pontos) {
pontosAtuais += pontos;
tvPontos.setText(String.valueOf(pontosAtuais));
SharedPreferences.Editor editor = sharedPreferences.edit();
editor.putInt("pontosTotais", pontosAtuais);
editor.putBoolean("desafio_" + numeroDoDesafio + "_concluido", true);
editor.apply();
}
private void carregarDadosGuardados() {
pontosAtuais = sharedPreferences.getInt("pontosTotais", 0);
tvPontos.setText(String.valueOf(pontosAtuais));
// Verificar se o Desafio 1 já foi feito anteriormente
if (sharedPreferences.getBoolean("desafio_1_concluido", false)) {
tvStatusDesafio1.setText("Aprovado pela IA!");
tvStatusDesafio1.setTextColor(Color.parseColor("#4CAF50"));
btnVideoDesafio1.setEnabled(false);
btnVideoDesafio1.setText("Desafio Concluído");
btnVideoDesafio1.setBackgroundTintList(android.content.res.ColorStateList.valueOf(Color.parseColor("#81C784")));
}
// Verificar se o Desafio 2 já foi feito anteriormente
if (sharedPreferences.getBoolean("desafio_2_concluido", false)) {
tvStatusDesafio2.setText("Aprovado pela IA!");
tvStatusDesafio2.setTextColor(Color.parseColor("#4CAF50"));
btnVideoDesafio2.setEnabled(false);
btnVideoDesafio2.setText("Desafio Concluído");
btnVideoDesafio2.setBackgroundTintList(android.content.res.ColorStateList.valueOf(Color.parseColor("#81C784")));
}
}
}

View File

@@ -0,0 +1,26 @@
package com.example.pap;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import androidx.appcompat.app.AppCompatActivity;
public class EstatisticasActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_estatisticas);
Button btnVoltarHome = findViewById(R.id.btnVoltarHome);
// Ao clicar no botão voltar, a janela de estatísticas fecha
// e o utilizador regressa naturalmente à HomeActivity que estava por baixo
btnVoltarHome.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
finish();
}
});
}
}

View File

@@ -0,0 +1,146 @@
package com.example.pap;
import android.content.ActivityNotFoundException;
import android.content.Intent;
import android.graphics.Bitmap;
import android.os.Bundle;
import android.os.Handler;
import android.os.Looper;
import android.provider.MediaStore;
import android.view.View;
import android.widget.Button;
import android.widget.ImageView;
import android.widget.LinearLayout;
import android.widget.TextView;
import android.widget.Toast;
import androidx.activity.result.ActivityResultLauncher;
import androidx.activity.result.contract.ActivityResultContracts;
import androidx.appcompat.app.AppCompatActivity;
import androidx.recyclerview.widget.LinearLayoutManager;
import androidx.recyclerview.widget.RecyclerView;
import java.util.ArrayList;
public class FotoActivity extends AppCompatActivity {
private ImageView ivFotoComida;
private Button btnTirarFoto, btnVoltar, btnAddManual, btnConfirmar;
private LinearLayout layoutResultadosIA;
private TextView tvTotalCalorias, tvNomePrato;
private RecyclerView recyclerView;
private IngredientesAdapter adapter;
private ArrayList<Ingrediente> listaIngredientes;
// Lançador da Câmara
private final ActivityResultLauncher<Intent> cameraLauncher = registerForActivityResult(
new ActivityResultContracts.StartActivityForResult(),
result -> {
if (result.getResultCode() == RESULT_OK && result.getData() != null) {
// Recebe a miniatura da foto
Bundle extras = result.getData().getExtras();
Bitmap imageBitmap = (Bitmap) extras.get("data");
ivFotoComida.setImageBitmap(imageBitmap);
// Inicia a simulação da IA
Toast.makeText(this, "A analisar a imagem com IA...", Toast.LENGTH_SHORT).show();
simularAnaliseIA();
}
}
);
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_foto);
// Inicializar vistas
ivFotoComida = findViewById(R.id.ivFotoComida);
btnTirarFoto = findViewById(R.id.btnTirarFotoComida);
btnVoltar = findViewById(R.id.btnVoltarFoto);
layoutResultadosIA = findViewById(R.id.layoutResultadosIA);
tvTotalCalorias = findViewById(R.id.tvTotalCalorias);
tvNomePrato = findViewById(R.id.tvNomePratoDetectado);
recyclerView = findViewById(R.id.recyclerViewIngredientes);
btnAddManual = findViewById(R.id.btnAddIngredienteManual);
btnConfirmar = findViewById(R.id.btnConfirmarRefeicao);
// Configurar a RecyclerView
listaIngredientes = new ArrayList<>();
recyclerView.setLayoutManager(new LinearLayoutManager(this));
adapter = new IngredientesAdapter(listaIngredientes);
recyclerView.setAdapter(adapter);
// Configurar o clique no botão de apagar (lixo) na lista
adapter.setOnItemClickListener(new IngredientesAdapter.OnItemClickListener() {
@Override
public void onDeleteClick(int position) {
removerIngrediente(position);
}
});
btnTirarFoto.setOnClickListener(v -> abrirCamera());
btnVoltar.setOnClickListener(v -> finish());
// Botão para simular adicionar um ingrediente extra manualmente
btnAddManual.setOnClickListener(v -> {
listaIngredientes.add(new Ingrediente("Azeite Extra (1 c.sopa)", 120));
adapter.notifyItemInserted(listaIngredientes.size() - 1);
calcularTotal();
Toast.makeText(this, "Ingrediente adicionado!", Toast.LENGTH_SHORT).show();
});
btnConfirmar.setOnClickListener(v -> {
Toast.makeText(this, "Refeição registada no diário!", Toast.LENGTH_LONG).show();
finish(); // Fecha o ecrã após confirmar
});
}
private void abrirCamera() {
Intent takePictureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
try {
cameraLauncher.launch(takePictureIntent);
} catch (ActivityNotFoundException e) {
Toast.makeText(this, "Erro ao abrir a câmara.", Toast.LENGTH_SHORT).show();
}
}
// Simula a resposta da IA após 2 segundos
private void simularAnaliseIA() {
new Handler(Looper.getMainLooper()).postDelayed(() -> {
// 1. Mostra a área de resultados
layoutResultadosIA.setVisibility(View.VISIBLE);
btnTirarFoto.setText("Tirar Outra Foto");
// 2. Simula os dados detetados
tvNomePrato.setText("Prato detetado: Salada César com Frango");
listaIngredientes.clear();
listaIngredientes.add(new Ingrediente("Peito de Frango Grelhado (150g)", 240));
listaIngredientes.add(new Ingrediente("Alface Romana (200g)", 30));
listaIngredientes.add(new Ingrediente("Molho César (2 c.sopa)", 160));
listaIngredientes.add(new Ingrediente("Croutons (30g)", 120));
listaIngredientes.add(new Ingrediente("Queijo Parmesão (20g)", 80));
// 3. Atualiza a lista e o total
adapter.notifyDataSetChanged();
calcularTotal();
}, 2000); // Espera 2 segundos
}
// Remove um item da lista e atualiza o total
private void removerIngrediente(int position) {
Ingrediente removido = listaIngredientes.get(position);
listaIngredientes.remove(position);
adapter.notifyItemRemoved(position);
calcularTotal();
Toast.makeText(this, "Removido: " + removido.getNome(), Toast.LENGTH_SHORT).show();
}
// Percorre a lista e soma as calorias
private void calcularTotal() {
int total = 0;
for (Ingrediente ing : listaIngredientes) {
total += ing.getCalorias();
}
tvTotalCalorias.setText(total + " kcal");
}
}

View File

@@ -0,0 +1,135 @@
package com.example.pap;
import android.content.Intent;
import android.content.SharedPreferences;
import android.os.Bundle;
import android.text.InputType;
import android.widget.EditText;
import android.widget.LinearLayout;
import android.widget.Toast;
import androidx.appcompat.app.AlertDialog;
import androidx.appcompat.app.AppCompatActivity;
import androidx.cardview.widget.CardView;
public class HomeActivity extends AppCompatActivity {
private SharedPreferences sharedPreferences;
// Agora são 6 cartões!
private CardView cardPerfil, cardEstatisticas, cardTirarFoto, cardChat, cardDesafios, cardDefinicoes;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_home);
// 1. Iniciar a memória do telemóvel
sharedPreferences = getSharedPreferences("AppEmagrecimento", MODE_PRIVATE);
// 2. Ligar os cartões do XML ao código
cardPerfil = findViewById(R.id.cardPerfil);
cardEstatisticas = findViewById(R.id.cardEstatisticas);
cardTirarFoto = findViewById(R.id.cardTirarFoto);
cardChat = findViewById(R.id.cardChat);
cardDesafios = findViewById(R.id.cardDesafios);
cardDefinicoes = findViewById(R.id.cardDefinicoes);
// 3. Configurar os cliques para abrir os outros ecrãs
cardPerfil.setOnClickListener(v -> {
Toast.makeText(this, "A abrir O Meu Perfil...", Toast.LENGTH_SHORT).show();
// startActivity(new Intent(HomeActivity.this, PerfilActivity.class));
});
cardEstatisticas.setOnClickListener(v -> {
Toast.makeText(this, "A abrir Estatísticas...", Toast.LENGTH_SHORT).show();
// startActivity(new Intent(HomeActivity.this, EstatisticasActivity.class));
});
cardTirarFoto.setOnClickListener(v -> {
startActivity(new Intent(HomeActivity.this, FotoActivity.class));
});
cardChat.setOnClickListener(v -> {
startActivity(new Intent(HomeActivity.this, ChatActivity.class));
});
cardDesafios.setOnClickListener(v -> {
startActivity(new Intent(HomeActivity.this, DesafiosActivity.class));
});
cardDefinicoes.setOnClickListener(v -> {
Toast.makeText(this, "A abrir Definições...", Toast.LENGTH_SHORT).show();
// startActivity(new Intent(HomeActivity.this, DefinicoesActivity.class));
});
// 4. Verifica se já passou o tempo para pedir o novo peso
verificarAtualizacaoSemanal();
}
// Função que calcula se já passou o tempo (1 semana ou 10 segundos)
private void verificarAtualizacaoSemanal() {
long dataUltimaAtualizacao = sharedPreferences.getLong("data_ultima_atualizacao", 0);
long dataAtual = System.currentTimeMillis();
if (dataUltimaAtualizacao == 0) {
sharedPreferences.edit().putLong("data_ultima_atualizacao", dataAtual).apply();
} else {
// TEMPO CONFIGURADO PARA TESTES: 10 segundos!
// Para a PAP final usa: long tempoNecessario = 7L * 24 * 60 * 60 * 1000;
long tempoNecessario = 10 * 1000;
if (dataAtual - dataUltimaAtualizacao >= tempoNecessario) {
mostrarPopupAtualizacao();
}
}
}
// Função que cria o pop-up para atualizar o peso
private void mostrarPopupAtualizacao() {
AlertDialog.Builder builder = new AlertDialog.Builder(this);
builder.setTitle("Hora da pesagem! ⚖️");
builder.setMessage("Já passou 1 semana! Atualiza o teu peso e altura para acompanharmos a tua evolução.");
builder.setCancelable(false);
LinearLayout layout = new LinearLayout(this);
layout.setOrientation(LinearLayout.VERTICAL);
layout.setPadding(50, 40, 50, 10);
final EditText etNovoPeso = new EditText(this);
etNovoPeso.setHint("Novo Peso (ex: 75.5)");
etNovoPeso.setInputType(InputType.TYPE_CLASS_NUMBER | InputType.TYPE_NUMBER_FLAG_DECIMAL);
layout.addView(etNovoPeso);
final EditText etNovaAltura = new EditText(this);
etNovaAltura.setHint("Nova Altura (ex: 1.75)");
etNovaAltura.setInputType(InputType.TYPE_CLASS_NUMBER | InputType.TYPE_NUMBER_FLAG_DECIMAL);
layout.addView(etNovaAltura);
builder.setView(layout);
builder.setPositiveButton("Atualizar Dados", null);
AlertDialog dialog = builder.create();
dialog.show();
dialog.getButton(AlertDialog.BUTTON_POSITIVE).setOnClickListener(v -> {
String pesoStr = etNovoPeso.getText().toString().trim();
String alturaStr = etNovaAltura.getText().toString().trim();
if (!pesoStr.isEmpty() && !alturaStr.isEmpty()) {
float novoPeso = Float.parseFloat(pesoStr);
float novaAltura = Float.parseFloat(alturaStr);
SharedPreferences.Editor editor = sharedPreferences.edit();
editor.putFloat("pesoAtual", novoPeso);
editor.putFloat("alturaAtual", novaAltura);
editor.putLong("data_ultima_atualizacao", System.currentTimeMillis());
editor.apply();
Toast.makeText(HomeActivity.this, "Dados atualizados com sucesso! 💪", Toast.LENGTH_SHORT).show();
dialog.dismiss();
} else {
Toast.makeText(HomeActivity.this, "Tens de preencher os dois campos!", Toast.LENGTH_SHORT).show();
}
});
}
}

View File

@@ -0,0 +1,25 @@
package com.example.pap;
// Esta classe é um modelo simples para guardar os dados de um ingrediente
public class Ingrediente {
private String nome;
private int calorias;
public Ingrediente(String nome, int calorias) {
this.nome = nome;
this.calorias = calorias;
}
public String getNome() {
return nome;
}
public int getCalorias() {
return calorias;
}
// Vamos precisar disto para alterar a quantidade se quisermos no futuro
public void setCalorias(int calorias) {
this.calorias = calorias;
}
}

View File

@@ -0,0 +1,79 @@
package com.example.pap;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ImageButton;
import android.widget.TextView;
import androidx.annotation.NonNull;
import androidx.recyclerview.widget.RecyclerView;
import java.util.ArrayList;
// O Adapter é o motor que liga os dados à lista visual
public class IngredientesAdapter extends RecyclerView.Adapter<IngredientesAdapter.ViewHolder> {
private ArrayList<Ingrediente> listaIngredientes;
private OnItemClickListener listener;
// Interface para avisar a Activity que um item foi apagado
public interface OnItemClickListener {
void onDeleteClick(int position);
}
public void setOnItemClickListener(OnItemClickListener listener) {
this.listener = listener;
}
// Construtor
public IngredientesAdapter(ArrayList<Ingrediente> listaIngredientes) {
this.listaIngredientes = listaIngredientes;
}
// Cria a "caixinha" visual para cada linha
@NonNull
@Override
public ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.activity_ingrediente, parent, false);
return new ViewHolder(view, listener);
}
// Preenche a "caixinha" com os dados do ingrediente atual
@Override
public void onBindViewHolder(@NonNull ViewHolder holder, int position) {
Ingrediente ingredienteAtual = listaIngredientes.get(position);
holder.tvNome.setText(ingredienteAtual.getNome());
holder.tvCalorias.setText(ingredienteAtual.getCalorias() + " kcal");
}
@Override
public int getItemCount() {
return listaIngredientes.size();
}
// Classe interna que guarda as referências aos elementos do layout de cada linha
public static class ViewHolder extends RecyclerView.ViewHolder {
public TextView tvNome;
public TextView tvCalorias;
public ImageButton btnDelete;
public ViewHolder(@NonNull View itemView, final OnItemClickListener listener) {
super(itemView);
tvNome = itemView.findViewById(R.id.tvNomeIngrediente);
tvCalorias = itemView.findViewById(R.id.tvCaloriasIngrediente);
btnDelete = itemView.findViewById(R.id.btnRemoverIngrediente);
// Configura o clique no botão de lixo
btnDelete.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
if (listener != null) {
int position = getAdapterPosition();
if (position != RecyclerView.NO_POSITION) {
listener.onDeleteClick(position);
}
}
}
});
}
}
}

View File

@@ -0,0 +1,93 @@
package com.example.pap;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.TextView;
import android.widget.Toast;
import androidx.appcompat.app.AppCompatActivity;
import retrofit2.Call;
import retrofit2.Callback;
import retrofit2.Response;
import retrofit2.Retrofit;
import retrofit2.converter.gson.GsonConverterFactory;
public class LoginActivity extends AppCompatActivity {
private EditText etLoginEmail, etLoginPassword;
private Button btnLogin;
private TextView tvGoToRegister;
private SupabaseApi api;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_login);
// 1. Inicializar os componentes
etLoginEmail = findViewById(R.id.etLoginEmail);
etLoginPassword = findViewById(R.id.etLoginPassword);
btnLogin = findViewById(R.id.btnLogin);
tvGoToRegister = findViewById(R.id.tvGoToRegister);
// 2. Iniciar a comunicação com a API do Supabase
Retrofit retrofit = new Retrofit.Builder()
.baseUrl(SupabaseConfig.URL)
.addConverterFactory(GsonConverterFactory.create())
.build();
api = retrofit.create(SupabaseApi.class);
// 3. Configurar os cliques
btnLogin.setOnClickListener(v -> efetuarLogin());
tvGoToRegister.setOnClickListener(v -> {
Intent intent = new Intent(LoginActivity.this, RegisterActivity.class);
startActivity(intent);
});
}
private void efetuarLogin() {
String email = etLoginEmail.getText().toString().trim();
String password = etLoginPassword.getText().toString().trim();
// Validação básica
if (email.isEmpty() || password.isEmpty()) {
Toast.makeText(this, "Preencha o email e a password!", Toast.LENGTH_SHORT).show();
return;
}
UserCredentials credentials = new UserCredentials(email, password);
// Faz o pedido de Login ao Supabase
api.login(SupabaseConfig.API_KEY, credentials).enqueue(new Callback<SupabaseResponse>() {
@Override
public void onResponse(Call<SupabaseResponse> call, Response<SupabaseResponse> response) {
if (response.isSuccessful() && response.body() != null) {
// Login com sucesso! Vai para a Home
Toast.makeText(LoginActivity.this, "Login com sucesso!", Toast.LENGTH_SHORT).show();
Intent intent = new Intent(LoginActivity.this, HomeActivity.class);
startActivity(intent);
finish(); // Fecha a tela de login
} else {
// SE DER ERRO (ex: 404, 400), MOSTRA O LINK EXATO E O CÓDIGO
String urlQueFalhou = response.raw().request().url().toString();
if (response.code() == 400) {
Toast.makeText(LoginActivity.this, "Email ou password incorretos!", Toast.LENGTH_LONG).show();
} else {
Toast.makeText(LoginActivity.this, "Erro " + response.code() + "! Link: " + urlQueFalhou, Toast.LENGTH_LONG).show();
}
}
}
@Override
public void onFailure(Call<SupabaseResponse> call, Throwable t) {
Toast.makeText(LoginActivity.this, "Falha na ligação: " + t.getMessage(), Toast.LENGTH_SHORT).show();
}
});
}
}

View File

@@ -0,0 +1,24 @@
package com.example.pap;
import android.os.Bundle;
import androidx.activity.EdgeToEdge;
import androidx.appcompat.app.AppCompatActivity;
import androidx.core.graphics.Insets;
import androidx.core.view.ViewCompat;
import androidx.core.view.WindowInsetsCompat;
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
EdgeToEdge.enable(this);
setContentView(R.layout.activity_main);
ViewCompat.setOnApplyWindowInsetsListener(findViewById(R.id.main), (v, insets) -> {
Insets systemBars = insets.getInsets(WindowInsetsCompat.Type.systemBars());
v.setPadding(systemBars.left, systemBars.top, systemBars.right, systemBars.bottom);
return insets;
});
}
}

View File

@@ -0,0 +1,39 @@
package com.example.pap;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import androidx.appcompat.app.AppCompatActivity;
public class PerfilActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_perfil);
Button btnVoltar = findViewById(R.id.btnVoltarPerfil);
Button btnSair = findViewById(R.id.btnSair);
// Voltar à Home
btnVoltar.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
finish();
}
});
// Terminar Sessão (Voltar ao Login e limpar a Home)
btnSair.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent(PerfilActivity.this, LoginActivity.class);
// Estas flags garantem que o utilizador não consegue clicar em "Voltar" no telemóvel e ir para a Home de novo
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TASK);
startActivity(intent);
finish();
}
});
}
}

View File

@@ -0,0 +1,134 @@
package com.example.pap;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.TextView;
import android.widget.Toast;
import androidx.appcompat.app.AppCompatActivity;
import retrofit2.Call;
import retrofit2.Callback;
import retrofit2.Response;
import retrofit2.Retrofit;
import retrofit2.converter.gson.GsonConverterFactory;
public class RegisterActivity extends AppCompatActivity {
private EditText etRegNome, etRegEmail, etRegPassword, etRegAltura, etRegPeso;
private Button btnRegister;
private TextView tvGoToLogin;
private SupabaseApi api;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_register);
// 1. Ligar os elementos do design ao código
etRegNome = findViewById(R.id.etRegNome);
etRegEmail = findViewById(R.id.etRegEmail);
etRegPassword = findViewById(R.id.etRegPassword);
etRegAltura = findViewById(R.id.etRegAltura);
etRegPeso = findViewById(R.id.etRegPeso);
btnRegister = findViewById(R.id.btnRegister);
tvGoToLogin = findViewById(R.id.tvGoToLogin);
// 2. Configurar o Retrofit (A ponte de comunicação com o Supabase)
Retrofit retrofit = new Retrofit.Builder()
.baseUrl(SupabaseConfig.URL)
.addConverterFactory(GsonConverterFactory.create())
.build();
api = retrofit.create(SupabaseApi.class);
// 3. Configurar os cliques dos botões
btnRegister.setOnClickListener(v -> efetuarRegisto());
tvGoToLogin.setOnClickListener(v -> finish());
}
private void efetuarRegisto() {
String nome = etRegNome.getText().toString().trim();
String email = etRegEmail.getText().toString().trim();
String password = etRegPassword.getText().toString().trim();
String pesoStr = etRegPeso.getText().toString().trim();
String alturaStr = etRegAltura.getText().toString().trim();
// Verificações para garantir que o utilizador preencheu tudo
if (nome.isEmpty() || email.isEmpty() || password.isEmpty() || pesoStr.isEmpty() || alturaStr.isEmpty()) {
Toast.makeText(this, "Preencha todos os campos!", Toast.LENGTH_SHORT).show();
return;
}
if (password.length() < 6) {
Toast.makeText(this, "A password precisa de pelo menos 6 caracteres.", Toast.LENGTH_SHORT).show();
return;
}
float peso = Float.parseFloat(pesoStr);
float altura = Float.parseFloat(alturaStr);
// FASE 1: Criar a conta (Email e Password) no Supabase Auth
UserCredentials credentials = new UserCredentials(email, password);
api.signUp(SupabaseConfig.API_KEY, credentials).enqueue(new Callback<SupabaseResponse>() {
@Override
public void onResponse(Call<SupabaseResponse> call, Response<SupabaseResponse> response) {
if (response.isSuccessful() && response.body() != null) {
// Sucesso! A conta foi criada. Vamos extrair o ID e o Token
String userId = response.body().id != null ? response.body().id : response.body().user.id;
String token = "Bearer " + response.body().access_token;
// FASE 2: Guardar o Nome, Peso e Altura na tabela "profiles"
salvarPerfil(userId, nome, email, peso, altura, token);
} else {
String urlQueFalhou = response.raw().request().url().toString();
if (response.code() == 400) {
Toast.makeText(RegisterActivity.this, "Erro 400: Este email já existe ou a password é fraca.", Toast.LENGTH_LONG).show();
} else {
Toast.makeText(RegisterActivity.this, "Erro " + response.code() + " no Auth! Link: " + urlQueFalhou, Toast.LENGTH_LONG).show();
}
}
}
@Override
public void onFailure(Call<SupabaseResponse> call, Throwable t) {
Toast.makeText(RegisterActivity.this, "Falha de rede (Sem internet?): " + t.getMessage(), Toast.LENGTH_SHORT).show();
}
});
}
// Método responsável por guardar os dados extras na tabela profiles
private void salvarPerfil(String id, String nome, String email, float peso, float altura, String token) {
ProfileData profile = new ProfileData(id, nome, email, peso, altura);
api.insertProfile(SupabaseConfig.API_KEY, token, "application/json", "return=minimal", profile)
.enqueue(new Callback<Void>() {
@Override
public void onResponse(Call<Void> call, Response<Void> response) {
if (response.isSuccessful()) {
// TUDO CORREU BEM!
Toast.makeText(RegisterActivity.this, "Conta e Perfil criados com sucesso!", Toast.LENGTH_LONG).show();
finish(); // Volta para o ecrã de login
} else {
// DEU ERRO A GUARDAR O PESO E ALTURA. VAMOS LER O ERRO EXATO!
try {
String erroExato = response.errorBody().string();
// Esta mensagem vai mostrar-te o que está mal configurado no painel do Supabase
Toast.makeText(RegisterActivity.this, "ERRO DA BASE DE DADOS: " + erroExato, Toast.LENGTH_LONG).show();
} catch (Exception e) {
Toast.makeText(RegisterActivity.this, "Erro a ler a resposta da base de dados.", Toast.LENGTH_SHORT).show();
}
}
}
@Override
public void onFailure(Call<Void> call, Throwable t) {
Toast.makeText(RegisterActivity.this, "Erro ao conectar à base de dados.", Toast.LENGTH_SHORT).show();
}
});
}
}

View File

@@ -0,0 +1,65 @@
package com.example.pap;
import retrofit2.Call;
import retrofit2.http.Body;
import retrofit2.http.Header;
import retrofit2.http.POST;
public interface SupabaseApi {
// Rota para criar a conta
@POST("auth/v1/signup")
Call<SupabaseResponse> signUp(@Header("apikey") String apiKey, @Body UserCredentials credentials);
// Rota para fazer login
@POST("auth/v1/token?grant_type=password")
Call<SupabaseResponse> login(@Header("apikey") String apiKey, @Body UserCredentials credentials);
// Rota para guardar o Nome, Peso e Altura na base de dados
@POST("rest/v1/profiles")
Call<Void> insertProfile(
@Header("apikey") String apiKey,
@Header("Authorization") String token,
@Header("Content-Type") String contentType,
@Header("Prefer") String prefer,
@Body ProfileData profile
);
}
// ---- Classes Auxiliares para Formatarem os Dados ----
class UserCredentials {
String email;
String password;
public UserCredentials(String email, String password) {
this.email = email;
this.password = password;
}
}
class ProfileData {
String id;
String nome;
String email;
float peso;
float altura;
public ProfileData(String id, String nome, String email, float peso, float altura) {
this.id = id;
this.nome = nome;
this.email = email;
this.peso = peso;
this.altura = altura;
}
}
class SupabaseResponse {
String id;
String access_token;
UserResponse user;
class UserResponse {
String id;
}
}

View File

@@ -0,0 +1,10 @@
package com.example.pap;
public class SupabaseConfig {
// O URL limpo, apenas com a barra no final!
public static final String URL = "https://lkjbbbgavoyknuxaskho.supabase.co";
// A tua chave está perfeita
public static final String API_KEY = "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpc3MiOiJzdXBhYmFzZSIsInJlZiI6ImxramJiYmdhdm95a251eGFza2hvIiwicm9sZSI6ImFub24iLCJpYXQiOjE3NzI2MjM4OTMsImV4cCI6MjA4ODE5OTg5M30.HV9ZXYCaF1V8dZwPxv_p5_gDi9cN_ioumDm9mgmEQPU";
}

View File

@@ -0,0 +1,170 @@
<?xml version="1.0" encoding="utf-8"?>
<vector xmlns:android="http://schemas.android.com/apk/res/android"
android:width="108dp"
android:height="108dp"
android:viewportWidth="108"
android:viewportHeight="108">
<path
android:fillColor="#3DDC84"
android:pathData="M0,0h108v108h-108z" />
<path
android:fillColor="#00000000"
android:pathData="M9,0L9,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M19,0L19,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M29,0L29,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M39,0L39,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M49,0L49,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M59,0L59,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M69,0L69,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M79,0L79,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M89,0L89,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M99,0L99,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,9L108,9"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,19L108,19"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,29L108,29"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,39L108,39"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,49L108,49"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,59L108,59"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,69L108,69"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,79L108,79"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,89L108,89"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,99L108,99"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M19,29L89,29"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M19,39L89,39"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M19,49L89,49"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M19,59L89,59"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M19,69L89,69"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M19,79L89,79"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M29,19L29,89"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M39,19L39,89"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M49,19L49,89"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M59,19L59,89"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M69,19L69,89"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M79,19L79,89"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
</vector>

View File

@@ -0,0 +1,30 @@
<vector xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:aapt="http://schemas.android.com/aapt"
android:width="108dp"
android:height="108dp"
android:viewportWidth="108"
android:viewportHeight="108">
<path android:pathData="M31,63.928c0,0 6.4,-11 12.1,-13.1c7.2,-2.6 26,-1.4 26,-1.4l38.1,38.1L107,108.928l-32,-1L31,63.928z">
<aapt:attr name="android:fillColor">
<gradient
android:endX="85.84757"
android:endY="92.4963"
android:startX="42.9492"
android:startY="49.59793"
android:type="linear">
<item
android:color="#44000000"
android:offset="0.0" />
<item
android:color="#00000000"
android:offset="1.0" />
</gradient>
</aapt:attr>
</path>
<path
android:fillColor="#FFFFFF"
android:fillType="nonZero"
android:pathData="M65.3,45.828l3.8,-6.6c0.2,-0.4 0.1,-0.9 -0.3,-1.1c-0.4,-0.2 -0.9,-0.1 -1.1,0.3l-3.9,6.7c-6.3,-2.8 -13.4,-2.8 -19.7,0l-3.9,-6.7c-0.2,-0.4 -0.7,-0.5 -1.1,-0.3C38.8,38.328 38.7,38.828 38.9,39.228l3.8,6.6C36.2,49.428 31.7,56.028 31,63.928h46C76.3,56.028 71.8,49.428 65.3,45.828zM43.4,57.328c-0.8,0 -1.5,-0.5 -1.8,-1.2c-0.3,-0.7 -0.1,-1.5 0.4,-2.1c0.5,-0.5 1.4,-0.7 2.1,-0.4c0.7,0.3 1.2,1 1.2,1.8C45.3,56.528 44.5,57.328 43.4,57.328L43.4,57.328zM64.6,57.328c-0.8,0 -1.5,-0.5 -1.8,-1.2s-0.1,-1.5 0.4,-2.1c0.5,-0.5 1.4,-0.7 2.1,-0.4c0.7,0.3 1.2,1 1.2,1.8C66.5,56.528 65.6,57.328 64.6,57.328L64.6,57.328z"
android:strokeWidth="1"
android:strokeColor="#00000000" />
</vector>

View File

@@ -0,0 +1,83 @@
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="#F4F9F5">
<LinearLayout
android:id="@+id/headerChat"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="#FFFFFF"
android:elevation="4dp"
android:orientation="horizontal"
android:padding="16dp"
android:gravity="center_vertical">
<Button
android:id="@+id/btnVoltarChat"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Voltar"
android:backgroundTint="#E0E0E0"
android:textColor="#000000"
android:layout_marginEnd="16dp"/>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="IA Coach de Saúde"
android:textSize="20sp"
android:textStyle="bold"
android:textColor="#2E3D32"/>
</LinearLayout>
<ScrollView
android:id="@+id/chatScrollView"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_below="@id/headerChat"
android:layout_above="@+id/layoutInput"
android:padding="16dp"
android:fillViewport="true">
<LinearLayout
android:id="@+id/chatLayout"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical"
android:paddingBottom="16dp"/>
</ScrollView>
<LinearLayout
android:id="@+id/layoutInput"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_alignParentBottom="true"
android:background="#FFFFFF"
android:elevation="8dp"
android:orientation="horizontal"
android:padding="8dp"
android:gravity="center_vertical">
<EditText
android:id="@+id/etNovaMensagem"
android:layout_width="0dp"
android:layout_height="50dp"
android:layout_weight="1"
android:background="#F0F0F0"
android:hint="Pede uma dica de saúde..."
android:paddingStart="16dp"
android:paddingEnd="16dp"
android:layout_marginEnd="8dp"/>
<Button
android:id="@+id/btnEnviarMensagem"
android:layout_width="wrap_content"
android:layout_height="50dp"
android:backgroundTint="#4CAF50"
android:text="Enviar"
android:textStyle="bold"/>
</LinearLayout>
</RelativeLayout>

View File

@@ -0,0 +1,149 @@
<?xml version="1.0" encoding="utf-8"?>
<ScrollView xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="#F4F9F5"
android:fillViewport="true">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical"
android:padding="24dp"
android:gravity="top">
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Desafios"
android:textSize="24sp"
android:textStyle="bold"
android:textColor="#2E3D32"
android:gravity="center"
android:layout_marginTop="8dp"
android:layout_marginBottom="16dp"/>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:background="#FFFFFF"
android:elevation="4dp"
android:padding="16dp"
android:gravity="center"
android:layout_marginBottom="24dp">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="🏋️"
android:textSize="32sp"
android:layout_marginEnd="16dp"/>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Pontos:"
android:textSize="20sp"
android:textStyle="bold"
android:textColor="#757575"
android:layout_marginEnd="8dp"/>
<TextView
android:id="@+id/tvPontos"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="0"
android:textSize="24sp"
android:textStyle="bold"
android:textColor="#4CAF50"/>
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical"
android:background="#FFFFFF"
android:elevation="2dp"
android:padding="16dp"
android:layout_marginBottom="16dp">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="1. Fazer 10 Flexões (+20 pts)"
android:textSize="18sp"
android:textStyle="bold"
android:textColor="#2E3D32"/>
<TextView
android:id="@+id/tvStatusDesafio1"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Estado: Pendente"
android:textColor="#FF9800"
android:layout_marginTop="4dp"
android:layout_marginBottom="8dp"/>
<Button
android:id="@+id/btnVideoDesafio1"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Gravar Vídeo"
android:backgroundTint="#2196F3"
android:drawableStart="@android:drawable/ic_menu_camera"
android:paddingStart="16dp"/>
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical"
android:background="#FFFFFF"
android:elevation="2dp"
android:padding="16dp"
android:layout_marginBottom="16dp">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="2. Prancha de 30 Segundos (+30 pts)"
android:textSize="18sp"
android:textStyle="bold"
android:textColor="#2E3D32"/>
<TextView
android:id="@+id/tvStatusDesafio2"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Estado: Pendente"
android:textColor="#FF9800"
android:layout_marginTop="4dp"
android:layout_marginBottom="8dp"/>
<Button
android:id="@+id/btnVideoDesafio2"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Gravar Vídeo"
android:backgroundTint="#2196F3"
android:drawableStart="@android:drawable/ic_menu_camera"
android:paddingStart="16dp"/>
</LinearLayout>
<View
android:layout_width="match_parent"
android:layout_height="0dp"
android:layout_weight="1"/>
<Button
android:id="@+id/btnVoltarDesafios"
android:layout_width="match_parent"
android:layout_height="60dp"
android:backgroundTint="#E0E0E0"
android:textColor="#000000"
android:text="Voltar ao Início"
android:layout_marginTop="16dp"/>
</LinearLayout>
</ScrollView>

View File

@@ -0,0 +1,82 @@
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
android:padding="32dp"
android:gravity="top|center_horizontal">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="O Teu Progresso"
android:textSize="28sp"
android:textStyle="bold"
android:layout_marginBottom="32dp"/>
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Objetivo de Peso"
android:textSize="18sp"
android:textStyle="bold"/>
<TextView
android:id="@+id/tvPesoStatus"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Faltam 5 kg para a tua meta!"
android:layout_marginTop="8dp"/>
<ProgressBar
android:id="@+id/progressPeso"
style="?android:attr/progressBarStyleHorizontal"
android:layout_width="match_parent"
android:layout_height="24dp"
android:max="100"
android:progress="60"
android:layout_marginTop="8dp"
android:layout_marginBottom="24dp"/>
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Calorias Consumidas (Hoje)"
android:textSize="18sp"
android:textStyle="bold"/>
<TextView
android:id="@+id/tvCalorias"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="1200 / 1800 kcal"
android:textSize="22sp"
android:textColor="#FF9800"
android:layout_marginTop="8dp"
android:layout_marginBottom="24dp"/>
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Água Bebida"
android:textSize="18sp"
android:textStyle="bold"/>
<TextView
android:id="@+id/tvAgua"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="1.5 L / 2.5 L"
android:textSize="22sp"
android:textColor="#03A9F4"
android:layout_marginTop="8dp"
android:layout_marginBottom="32dp"/>
<Button
android:id="@+id/btnVoltarHome"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Voltar ao Início"
android:layout_marginTop="16dp"/>
</LinearLayout>

View File

@@ -0,0 +1,129 @@
<?xml version="1.0" encoding="utf-8"?>
<ScrollView xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="#F4F9F5"
android:fillViewport="true">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical"
android:padding="24dp"
android:gravity="center_horizontal">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Scanner
"
android:textSize="24sp"
android:textStyle="bold"
android:textColor="#2E3D32"
android:layout_marginBottom="24dp"/>
<androidx.cardview.widget.CardView
android:layout_width="match_parent"
android:layout_height="250dp"
android:layout_marginBottom="24dp"
app:cardCornerRadius="16dp"
app:cardElevation="4dp"
xmlns:app="http://schemas.android.com/apk/res-auto">
<ImageView
android:id="@+id/ivFotoComida"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:src="@android:drawable/ic_menu_camera"
android:scaleType="centerCrop"
android:background="#E0E0E0"/>
</androidx.cardview.widget.CardView>
<Button
android:id="@+id/btnTirarFotoComida"
android:layout_width="match_parent"
android:layout_height="60dp"
android:text="Tirar Foto e Analisar"
android:backgroundTint="#4CAF50"
android:textStyle="bold"
android:layout_marginBottom="24dp"/>
<LinearLayout
android:id="@+id/layoutResultadosIA"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical"
android:visibility="gone"
android:background="#FFFFFF"
android:padding="16dp"
android:elevation="4dp">
<TextView
android:id="@+id/tvNomePratoDetectado"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Prato detetado: Salada César"
android:textSize="18sp"
android:textStyle="bold"
android:textColor="#2E3D32"
android:layout_marginBottom="16dp"/>
<androidx.recyclerview.widget.RecyclerView
android:id="@+id/recyclerViewIngredientes"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginBottom="16dp"/>
<View
android:layout_width="match_parent"
android:layout_height="1dp"
android:background="#E0E0E0"
android:layout_marginBottom="16dp"/>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:gravity="center_vertical">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Total Estimado:"
android:textSize="20sp"
android:layout_weight="1"/>
<TextView
android:id="@+id/tvTotalCalorias"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="0 kcal"
android:textSize="24sp"
android:textStyle="bold"
android:textColor="#4CAF50"/>
</LinearLayout>
<Button
android:id="@+id/btnAddIngredienteManual"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="+ Adicionar Ingrediente Extra"
android:backgroundTint="#2196F3"
android:layout_marginTop="16dp"/>
<Button
android:id="@+id/btnConfirmarRefeicao"
android:layout_width="match_parent"
android:layout_height="60dp"
android:text="Confirmar Refeição"
android:backgroundTint="#FF9800"
android:textStyle="bold"
android:layout_marginTop="16dp"/>
</LinearLayout>
<Button
android:id="@+id/btnVoltarFoto"
android:layout_width="match_parent"
android:layout_height="60dp"
android:text="Voltar"
android:backgroundTint="#E0E0E0"
android:textColor="#000000"
android:layout_marginTop="24dp"/>
</LinearLayout>
</ScrollView>

View File

@@ -0,0 +1,418 @@
<?xml version="1.0" encoding="utf-8"?>
<ScrollView xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="#F8FAFA"
android:fillViewport="true">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical"
android:padding="24dp">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Meu Progresso"
android:textSize="28sp"
android:textStyle="bold"
android:textColor="#111827"
android:layout_marginTop="16dp"/>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Acompanhe sua jornada de emagrecimento"
android:textSize="14sp"
android:textColor="#6B7280"
android:layout_marginTop="4dp"
android:layout_marginBottom="32dp"/>
<androidx.cardview.widget.CardView
android:id="@+id/cardPerfil"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginBottom="16dp"
android:clickable="true"
android:focusable="true"
android:foreground="?android:attr/selectableItemBackground"
app:cardCornerRadius="16dp"
app:cardElevation="2dp"
app:contentPadding="16dp">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:gravity="center_vertical">
<androidx.cardview.widget.CardView
android:layout_width="48dp"
android:layout_height="48dp"
app:cardCornerRadius="12dp"
app:cardBackgroundColor="#673AB7"
app:cardElevation="0dp">
<ImageView
android:layout_width="24dp"
android:layout_height="24dp"
android:layout_gravity="center"
android:src="@android:drawable/ic_menu_myplaces"
app:tint="#FFFFFF"/>
</androidx.cardview.widget.CardView>
<LinearLayout
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:orientation="vertical"
android:layout_marginStart="16dp">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="O Meu Perfil"
android:textSize="16sp"
android:textStyle="bold"
android:textColor="#111827"/>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Vê e edita os teus dados pessoais"
android:textSize="12sp"
android:textColor="#6B7280"
android:layout_marginTop="2dp"/>
</LinearLayout>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text=""
android:textSize="28sp"
android:textColor="#D1D5DB"
android:layout_marginStart="8dp"/>
</LinearLayout>
</androidx.cardview.widget.CardView>
<androidx.cardview.widget.CardView
android:id="@+id/cardEstatisticas"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginBottom="16dp"
android:clickable="true"
android:focusable="true"
android:foreground="?android:attr/selectableItemBackground"
app:cardCornerRadius="16dp"
app:cardElevation="2dp"
app:contentPadding="16dp">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:gravity="center_vertical">
<androidx.cardview.widget.CardView
android:layout_width="48dp"
android:layout_height="48dp"
app:cardCornerRadius="12dp"
app:cardBackgroundColor="#03A9F4"
app:cardElevation="0dp">
<ImageView
android:layout_width="24dp"
android:layout_height="24dp"
android:layout_gravity="center"
android:src="@android:drawable/ic_menu_sort_by_size"
app:tint="#FFFFFF"/>
</androidx.cardview.widget.CardView>
<LinearLayout
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:orientation="vertical"
android:layout_marginStart="16dp">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Estatísticas"
android:textSize="16sp"
android:textStyle="bold"
android:textColor="#111827"/>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Veja seus números e progresso"
android:textSize="12sp"
android:textColor="#6B7280"
android:layout_marginTop="2dp"/>
</LinearLayout>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text=""
android:textSize="28sp"
android:textColor="#D1D5DB"
android:layout_marginStart="8dp"/>
</LinearLayout>
</androidx.cardview.widget.CardView>
<androidx.cardview.widget.CardView
android:id="@+id/cardTirarFoto"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginBottom="16dp"
android:clickable="true"
android:focusable="true"
android:foreground="?android:attr/selectableItemBackground"
app:cardCornerRadius="16dp"
app:cardElevation="2dp"
app:contentPadding="16dp">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:gravity="center_vertical">
<androidx.cardview.widget.CardView
android:layout_width="48dp"
android:layout_height="48dp"
app:cardCornerRadius="12dp"
app:cardBackgroundColor="#E91E63"
app:cardElevation="0dp">
<ImageView
android:layout_width="24dp"
android:layout_height="24dp"
android:layout_gravity="center"
android:src="@android:drawable/ic_menu_camera"
app:tint="#FFFFFF"/>
</androidx.cardview.widget.CardView>
<LinearLayout
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:orientation="vertical"
android:layout_marginStart="16dp">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Tirar foto"
android:textSize="16sp"
android:textStyle="bold"
android:textColor="#111827"/>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Acompanhe a sua dieta"
android:textSize="12sp"
android:textColor="#6B7280"
android:layout_marginTop="2dp"/>
</LinearLayout>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text=""
android:textSize="28sp"
android:textColor="#D1D5DB"
android:layout_marginStart="8dp"/>
</LinearLayout>
</androidx.cardview.widget.CardView>
<androidx.cardview.widget.CardView
android:id="@+id/cardChat"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginBottom="16dp"
android:clickable="true"
android:focusable="true"
android:foreground="?android:attr/selectableItemBackground"
app:cardCornerRadius="16dp"
app:cardElevation="2dp"
app:contentPadding="16dp">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:gravity="center_vertical">
<androidx.cardview.widget.CardView
android:layout_width="48dp"
android:layout_height="48dp"
app:cardCornerRadius="12dp"
app:cardBackgroundColor="#00BFA5"
app:cardElevation="0dp">
<ImageView
android:layout_width="24dp"
android:layout_height="24dp"
android:layout_gravity="center"
android:src="@android:drawable/stat_notify_chat"
app:tint="#FFFFFF"/>
</androidx.cardview.widget.CardView>
<LinearLayout
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:orientation="vertical"
android:layout_marginStart="16dp">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Chat IA"
android:textSize="16sp"
android:textStyle="bold"
android:textColor="#111827"/>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Converse com seu assistente"
android:textSize="12sp"
android:textColor="#6B7280"
android:layout_marginTop="2dp"/>
</LinearLayout>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text=""
android:textSize="28sp"
android:textColor="#D1D5DB"
android:layout_marginStart="8dp"/>
</LinearLayout>
</androidx.cardview.widget.CardView>
<androidx.cardview.widget.CardView
android:id="@+id/cardDesafios"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginBottom="16dp"
android:clickable="true"
android:focusable="true"
android:foreground="?android:attr/selectableItemBackground"
app:cardCornerRadius="16dp"
app:cardElevation="2dp"
app:contentPadding="16dp">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:gravity="center_vertical">
<androidx.cardview.widget.CardView
android:layout_width="48dp"
android:layout_height="48dp"
app:cardCornerRadius="12dp"
app:cardBackgroundColor="#FF5722"
app:cardElevation="0dp">
<ImageView
android:layout_width="24dp"
android:layout_height="24dp"
android:layout_gravity="center"
android:src="@android:drawable/star_on"
app:tint="#FFFFFF"/>
</androidx.cardview.widget.CardView>
<LinearLayout
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:orientation="vertical"
android:layout_marginStart="16dp">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Desafios"
android:textSize="16sp"
android:textStyle="bold"
android:textColor="#111827"/>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Complete missões e ganhe prêmios"
android:textSize="12sp"
android:textColor="#6B7280"
android:layout_marginTop="2dp"/>
</LinearLayout>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text=""
android:textSize="28sp"
android:textColor="#D1D5DB"
android:layout_marginStart="8dp"/>
</LinearLayout>
</androidx.cardview.widget.CardView>
<androidx.cardview.widget.CardView
android:id="@+id/cardDefinicoes"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginBottom="32dp"
android:clickable="true"
android:focusable="true"
android:foreground="?android:attr/selectableItemBackground"
app:cardCornerRadius="16dp"
app:cardElevation="2dp"
app:contentPadding="16dp">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:gravity="center_vertical">
<androidx.cardview.widget.CardView
android:layout_width="48dp"
android:layout_height="48dp"
app:cardCornerRadius="12dp"
app:cardBackgroundColor="#37474F"
app:cardElevation="0dp">
<ImageView
android:layout_width="24dp"
android:layout_height="24dp"
android:layout_gravity="center"
android:src="@android:drawable/ic_menu_preferences"
app:tint="#FFFFFF"/>
</androidx.cardview.widget.CardView>
<LinearLayout
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:orientation="vertical"
android:layout_marginStart="16dp">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Definições"
android:textSize="16sp"
android:textStyle="bold"
android:textColor="#111827"/>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Personalize seu app"
android:textSize="12sp"
android:textColor="#6B7280"
android:layout_marginTop="2dp"/>
</LinearLayout>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text=""
android:textSize="28sp"
android:textColor="#D1D5DB"
android:layout_marginStart="8dp"/>
</LinearLayout>
</androidx.cardview.widget.CardView>
</LinearLayout>
</ScrollView>

View File

@@ -0,0 +1,39 @@
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:padding="12dp"
android:background="#FFFFFF"
android:layout_marginBottom="4dp"
android:elevation="2dp"
android:gravity="center_vertical">
<TextView
android:id="@+id/tvNomeIngrediente"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:text="Peito de Frango"
android:textSize="16sp"
android:textColor="#2E3D32"/>
<TextView
android:id="@+id/tvCaloriasIngrediente"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="150 kcal"
android:textStyle="bold"
android:textColor="#4CAF50"
android:layout_marginEnd="16dp"/>
<ImageButton
android:id="@+id/btnRemoverIngrediente"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:src="@android:drawable/ic_menu_delete"
android:background="?attr/selectableItemBackgroundBorderless"
app:tint="#F44336"
xmlns:app="http://schemas.android.com/apk/res-auto"/>
</LinearLayout>

View File

@@ -0,0 +1,88 @@
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
android:background="#F8FAFA"
android:padding="24dp"
android:gravity="center">
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Bem-vindo!"
android:textSize="32sp"
android:textStyle="bold"
android:textColor="#111827"
android:layout_marginBottom="8dp"
android:textAlignment="center"/>
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Faz login para continuar a tua jornada"
android:textSize="16sp"
android:textColor="#6B7280"
android:layout_marginBottom="40dp"
android:textAlignment="center"/>
<androidx.cardview.widget.CardView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginBottom="16dp"
app:cardCornerRadius="12dp"
app:cardElevation="2dp">
<EditText
android:id="@+id/etLoginEmail"
android:layout_width="match_parent"
android:layout_height="55dp"
android:background="@android:color/transparent"
android:hint="O teu Email"
android:inputType="textEmailAddress"
android:paddingStart="16dp"
android:paddingEnd="16dp"
android:textColor="#111827"/>
</androidx.cardview.widget.CardView>
<androidx.cardview.widget.CardView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginBottom="32dp"
app:cardCornerRadius="12dp"
app:cardElevation="2dp">
<EditText
android:id="@+id/etLoginPassword"
android:layout_width="match_parent"
android:layout_height="55dp"
android:background="@android:color/transparent"
android:hint="A tua Password"
android:inputType="textPassword"
android:paddingStart="16dp"
android:paddingEnd="16dp"
android:textColor="#111827"/>
</androidx.cardview.widget.CardView>
<Button
android:id="@+id/btnLogin"
android:layout_width="match_parent"
android:layout_height="60dp"
android:text="Entrar"
android:textSize="18sp"
android:textStyle="bold"
android:backgroundTint="#03A9F4"
android:textColor="#FFFFFF"
app:cornerRadius="12dp"
android:layout_marginBottom="24dp"/>
<TextView
android:id="@+id/tvGoToRegister"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Não tens conta? Regista-te aqui."
android:textSize="16sp"
android:textColor="#03A9F4"
android:textStyle="bold"
android:padding="8dp"/>
</LinearLayout>

View File

@@ -0,0 +1,19 @@
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/main"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".MainActivity">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Hello World!"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
</androidx.constraintlayout.widget.ConstraintLayout>

View File

@@ -0,0 +1,80 @@
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="#F4F9F5"
android:orientation="vertical"
android:padding="32dp"
android:gravity="center_horizontal">
<ImageView
android:layout_width="120dp"
android:layout_height="120dp"
android:src="@android:drawable/ic_menu_myplaces"
android:background="#E8F5E9"
android:padding="24dp"
android:layout_marginTop="32dp"
android:layout_marginBottom="16dp"/>
<TextView
android:id="@+id/tvNomePerfil"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Nome do Utilizador"
android:textSize="24sp"
android:textStyle="bold"
android:textColor="#2E3D32"
android:layout_marginBottom="32dp"/>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical"
android:background="#FFFFFF"
android:elevation="2dp"
android:padding="24dp"
android:layout_marginBottom="32dp">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Email: utilizador@email.com"
android:textSize="16sp"
android:textColor="#757575"
android:layout_marginBottom="12dp"/>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Altura: 175 cm"
android:textSize="16sp"
android:textColor="#757575"
android:layout_marginBottom="12dp"/>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Peso Inicial: 80 kg"
android:textSize="16sp"
android:textColor="#757575"/>
</LinearLayout>
<Button
android:id="@+id/btnSair"
android:layout_width="match_parent"
android:layout_height="60dp"
android:backgroundTint="#F44336"
android:text="Terminar Sessão"
android:textColor="#FFFFFF"
android:textStyle="bold"
android:layout_marginBottom="16dp"/>
<Button
android:id="@+id/btnVoltarPerfil"
android:layout_width="match_parent"
android:layout_height="60dp"
android:backgroundTint="#E0E0E0"
android:text="Voltar"
android:textColor="#000000" />
</LinearLayout>

View File

@@ -0,0 +1,152 @@
<?xml version="1.0" encoding="utf-8"?>
<ScrollView xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="#F8FAFA"
android:fillViewport="true">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical"
android:padding="24dp"
android:gravity="center">
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Criar Conta"
android:textSize="32sp"
android:textStyle="bold"
android:textColor="#111827"
android:layout_marginBottom="8dp"
android:textAlignment="center"/>
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Junta-te a nós e transforma a tua vida"
android:textSize="16sp"
android:textColor="#6B7280"
android:layout_marginBottom="32dp"
android:textAlignment="center"/>
<androidx.cardview.widget.CardView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginBottom="16dp"
app:cardCornerRadius="12dp"
app:cardElevation="2dp">
<EditText
android:id="@+id/etRegNome"
android:layout_width="match_parent"
android:layout_height="55dp"
android:background="@android:color/transparent"
android:hint="Nome completo"
android:inputType="textPersonName"
android:paddingStart="16dp"
android:paddingEnd="16dp"/>
</androidx.cardview.widget.CardView>
<androidx.cardview.widget.CardView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginBottom="16dp"
app:cardCornerRadius="12dp"
app:cardElevation="2dp">
<EditText
android:id="@+id/etRegEmail"
android:layout_width="match_parent"
android:layout_height="55dp"
android:background="@android:color/transparent"
android:hint="Email"
android:inputType="textEmailAddress"
android:paddingStart="16dp"
android:paddingEnd="16dp"/>
</androidx.cardview.widget.CardView>
<androidx.cardview.widget.CardView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginBottom="16dp"
app:cardCornerRadius="12dp"
app:cardElevation="2dp">
<EditText
android:id="@+id/etRegPassword"
android:layout_width="match_parent"
android:layout_height="55dp"
android:background="@android:color/transparent"
android:hint="Password (mín. 6 letras)"
android:inputType="textPassword"
android:paddingStart="16dp"
android:paddingEnd="16dp"/>
</androidx.cardview.widget.CardView>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:layout_marginBottom="32dp">
<androidx.cardview.widget.CardView
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:layout_marginEnd="8dp"
app:cardCornerRadius="12dp"
app:cardElevation="2dp">
<EditText
android:id="@+id/etRegAltura"
android:layout_width="match_parent"
android:layout_height="55dp"
android:background="@android:color/transparent"
android:hint="Altura (ex: 1.75)"
android:inputType="numberDecimal"
android:paddingStart="16dp"
android:paddingEnd="16dp"/>
</androidx.cardview.widget.CardView>
<androidx.cardview.widget.CardView
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:layout_marginStart="8dp"
app:cardCornerRadius="12dp"
app:cardElevation="2dp">
<EditText
android:id="@+id/etRegPeso"
android:layout_width="match_parent"
android:layout_height="55dp"
android:background="@android:color/transparent"
android:hint="Peso (ex: 75.5)"
android:inputType="numberDecimal"
android:paddingStart="16dp"
android:paddingEnd="16dp"/>
</androidx.cardview.widget.CardView>
</LinearLayout>
<Button
android:id="@+id/btnRegister"
android:layout_width="match_parent"
android:layout_height="60dp"
android:text="Registar Conta"
android:textSize="18sp"
android:textStyle="bold"
android:backgroundTint="#03A9F4"
android:textColor="#FFFFFF"
app:cornerRadius="12dp"
android:layout_marginBottom="24dp"/>
<TextView
android:id="@+id/tvGoToLogin"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Já tens conta? Entra aqui."
android:textSize="16sp"
android:textColor="#03A9F4"
android:textStyle="bold"
android:padding="8dp"/>
</LinearLayout>
</ScrollView>

View File

@@ -0,0 +1,6 @@
<?xml version="1.0" encoding="utf-8"?>
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
<background android:drawable="@drawable/ic_launcher_background" />
<foreground android:drawable="@drawable/ic_launcher_foreground" />
<monochrome android:drawable="@drawable/ic_launcher_foreground" />
</adaptive-icon>

View File

@@ -0,0 +1,6 @@
<?xml version="1.0" encoding="utf-8"?>
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
<background android:drawable="@drawable/ic_launcher_background" />
<foreground android:drawable="@drawable/ic_launcher_foreground" />
<monochrome android:drawable="@drawable/ic_launcher_foreground" />
</adaptive-icon>

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.4 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.8 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 982 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.7 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.9 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.8 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.8 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.8 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.8 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 7.6 KiB

View File

@@ -0,0 +1,7 @@
<resources xmlns:tools="http://schemas.android.com/tools">
<!-- Base application theme. -->
<style name="Base.Theme.PAP" parent="Theme.Material3.DayNight.NoActionBar">
<!-- Customize your dark theme here. -->
<!-- <item name="colorPrimary">@color/my_dark_primary</item> -->
</style>
</resources>

View File

@@ -0,0 +1,5 @@
<?xml version="1.0" encoding="utf-8"?>
<resources>
<color name="black">#FF000000</color>
<color name="white">#FFFFFFFF</color>
</resources>

View File

@@ -0,0 +1,3 @@
<resources>
<string name="app_name">PAP</string>
</resources>

View File

@@ -0,0 +1,9 @@
<resources xmlns:tools="http://schemas.android.com/tools">
<!-- Base application theme. -->
<style name="Base.Theme.PAP" parent="Theme.Material3.DayNight.NoActionBar">
<!-- Customize your light theme here. -->
<!-- <item name="colorPrimary">@color/my_light_primary</item> -->
</style>
<style name="Theme.PAP" parent="Base.Theme.PAP" />
</resources>

View File

@@ -0,0 +1,13 @@
<?xml version="1.0" encoding="utf-8"?><!--
Sample backup rules file; uncomment and customize as necessary.
See https://developer.android.com/guide/topics/data/autobackup
for details.
Note: This file is ignored for devices older than API 31
See https://developer.android.com/about/versions/12/backup-restore
-->
<full-backup-content>
<!--
<include domain="sharedpref" path="."/>
<exclude domain="sharedpref" path="device.xml"/>
-->
</full-backup-content>

View File

@@ -0,0 +1,19 @@
<?xml version="1.0" encoding="utf-8"?><!--
Sample data extraction rules file; uncomment and customize as necessary.
See https://developer.android.com/about/versions/12/backup-restore#xml-changes
for details.
-->
<data-extraction-rules>
<cloud-backup>
<!-- TODO: Use <include> and <exclude> to control what is backed up.
<include .../>
<exclude .../>
-->
</cloud-backup>
<!--
<device-transfer>
<include .../>
<exclude .../>
</device-transfer>
-->
</data-extraction-rules>

View File

@@ -0,0 +1,17 @@
package com.example.pap;
import org.junit.Test;
import static org.junit.Assert.*;
/**
* Example local unit test, which will execute on the development machine (host).
*
* @see <a href="http://d.android.com/tools/testing">Testing documentation</a>
*/
public class ExampleUnitTest {
@Test
public void addition_isCorrect() {
assertEquals(4, 2 + 2);
}
}