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

15
.gitignore vendored Normal file
View File

@@ -0,0 +1,15 @@
*.iml
.gradle
/local.properties
/.idea/caches
/.idea/libraries
/.idea/modules.xml
/.idea/workspace.xml
/.idea/navEditor.xml
/.idea/assetWizardSettings.xml
.DS_Store
/build
/captures
.externalNativeBuild
.cxx
local.properties

3
.idea/.gitignore generated vendored Normal file
View File

@@ -0,0 +1,3 @@
# Default ignored files
/shelf/
/workspace.xml

1
.idea/.name generated Normal file
View File

@@ -0,0 +1 @@
PAP

6
.idea/AndroidProjectSystem.xml generated Normal file
View File

@@ -0,0 +1,6 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="AndroidProjectSystem">
<option name="providerId" value="com.android.tools.idea.GradleProjectSystem" />
</component>
</project>

6
.idea/compiler.xml generated Normal file
View File

@@ -0,0 +1,6 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="CompilerConfiguration">
<bytecodeTargetLevel target="21" />
</component>
</project>

10
.idea/deploymentTargetSelector.xml generated Normal file
View File

@@ -0,0 +1,10 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="deploymentTargetSelector">
<selectionStates>
<SelectionState runConfigName="app">
<option name="selectionMode" value="DROPDOWN" />
</SelectionState>
</selectionStates>
</component>
</project>

19
.idea/gradle.xml generated Normal file
View File

@@ -0,0 +1,19 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="GradleMigrationSettings" migrationVersion="1" />
<component name="GradleSettings">
<option name="linkedExternalProjectsSettings">
<GradleProjectSettings>
<option name="testRunner" value="CHOOSE_PER_TEST" />
<option name="externalProjectPath" value="$PROJECT_DIR$" />
<option name="gradleJvm" value="#GRADLE_LOCAL_JAVA_HOME" />
<option name="modules">
<set>
<option value="$PROJECT_DIR$" />
<option value="$PROJECT_DIR$/app" />
</set>
</option>
</GradleProjectSettings>
</option>
</component>
</project>

10
.idea/migrations.xml generated Normal file
View File

@@ -0,0 +1,10 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="ProjectMigrations">
<option name="MigrateToGradleLocalJavaHome">
<set>
<option value="$PROJECT_DIR$" />
</set>
</option>
</component>
</project>

10
.idea/misc.xml generated Normal file
View File

@@ -0,0 +1,10 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="ExternalStorageConfigurationManager" enabled="true" />
<component name="ProjectRootManager" version="2" languageLevel="JDK_21" default="true" project-jdk-name="jbr-21" project-jdk-type="JavaSDK">
<output url="file://$PROJECT_DIR$/build/classes" />
</component>
<component name="ProjectType">
<option name="id" value="Android" />
</component>
</project>

17
.idea/runConfigurations.xml generated Normal file
View File

@@ -0,0 +1,17 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="RunConfigurationProducerService">
<option name="ignoredProducers">
<set>
<option value="com.intellij.execution.junit.AbstractAllInDirectoryConfigurationProducer" />
<option value="com.intellij.execution.junit.AllInPackageConfigurationProducer" />
<option value="com.intellij.execution.junit.PatternConfigurationProducer" />
<option value="com.intellij.execution.junit.TestInClassConfigurationProducer" />
<option value="com.intellij.execution.junit.UniqueIdConfigurationProducer" />
<option value="com.intellij.execution.junit.testDiscovery.JUnitTestDiscoveryConfigurationProducer" />
<option value="org.jetbrains.kotlin.idea.junit.KotlinJUnitRunConfigurationProducer" />
<option value="org.jetbrains.kotlin.idea.junit.KotlinPatternConfigurationProducer" />
</set>
</option>
</component>
</project>

6
.idea/vcs.xml generated Normal file
View File

@@ -0,0 +1,6 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="VcsDirectoryMappings">
<mapping directory="$PROJECT_DIR$" vcs="Git" />
</component>
</project>

1
app/.gitignore vendored Normal file
View File

@@ -0,0 +1 @@
/build

46
app/build.gradle.kts Normal file
View File

@@ -0,0 +1,46 @@
plugins {
alias(libs.plugins.android.application)
}
android {
namespace = "com.example.pap"
compileSdk = 36
defaultConfig {
applicationId = "com.example.pap"
minSdk = 24
targetSdk = 36
versionCode = 1
versionName = "1.0"
testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner"
}
buildTypes {
release {
isMinifyEnabled = false
proguardFiles(
getDefaultProguardFile("proguard-android-optimize.txt"),
"proguard-rules.pro"
)
}
}
compileOptions {
sourceCompatibility = JavaVersion.VERSION_11
targetCompatibility = JavaVersion.VERSION_11
}
}
dependencies {
implementation(libs.appcompat)
implementation(libs.material)
implementation(libs.activity)
implementation(libs.constraintlayout)
testImplementation(libs.junit)
androidTestImplementation(libs.ext.junit)
androidTestImplementation(libs.espresso.core)
implementation("com.squareup.retrofit2:retrofit:2.9.0")
implementation("com.squareup.retrofit2:converter-gson:2.9.0")
implementation("com.squareup.okhttp3:logging-interceptor:4.9.0")
}

21
app/proguard-rules.pro vendored Normal file
View File

@@ -0,0 +1,21 @@
# Add project specific ProGuard rules here.
# You can control the set of applied configuration files using the
# proguardFiles setting in build.gradle.
#
# For more details, see
# http://developer.android.com/guide/developing/tools/proguard.html
# If your project uses WebView with JS, uncomment the following
# and specify the fully qualified class name to the JavaScript interface
# class:
#-keepclassmembers class fqcn.of.javascript.interface.for.webview {
# public *;
#}
# Uncomment this to preserve the line number information for
# debugging stack traces.
#-keepattributes SourceFile,LineNumberTable
# If you keep the line number information, uncomment this to
# hide the original source file name.
#-renamesourcefileattribute SourceFile

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);
}
}

4
build.gradle.kts Normal file
View File

@@ -0,0 +1,4 @@
// Top-level build file where you can add configuration options common to all sub-projects/modules.
plugins {
alias(libs.plugins.android.application) apply false
}

21
gradle.properties Normal file
View File

@@ -0,0 +1,21 @@
# Project-wide Gradle settings.
# IDE (e.g. Android Studio) users:
# Gradle settings configured through the IDE *will override*
# any settings specified in this file.
# For more details on how to configure your build environment visit
# http://www.gradle.org/docs/current/userguide/build_environment.html
# Specifies the JVM arguments used for the daemon process.
# The setting is particularly useful for tweaking memory settings.
org.gradle.jvmargs=-Xmx2048m -Dfile.encoding=UTF-8
# When configured, Gradle will run in incubating parallel mode.
# This option should only be used with decoupled projects. For more details, visit
# https://developer.android.com/r/tools/gradle-multi-project-decoupled-projects
# org.gradle.parallel=true
# AndroidX package structure to make it clearer which packages are bundled with the
# Android operating system, and which are packaged with your app's APK
# https://developer.android.com/topic/libraries/support-library/androidx-rn
android.useAndroidX=true
# Enables namespacing of each library's R class so that its R class includes only the
# resources declared in the library itself and none from the library's dependencies,
# thereby reducing the size of the R class for that library
android.nonTransitiveRClass=true

22
gradle/libs.versions.toml Normal file
View File

@@ -0,0 +1,22 @@
[versions]
agp = "8.13.2"
junit = "4.13.2"
junitVersion = "1.3.0"
espressoCore = "3.7.0"
appcompat = "1.7.1"
material = "1.13.0"
activity = "1.12.4"
constraintlayout = "2.2.1"
[libraries]
junit = { group = "junit", name = "junit", version.ref = "junit" }
ext-junit = { group = "androidx.test.ext", name = "junit", version.ref = "junitVersion" }
espresso-core = { group = "androidx.test.espresso", name = "espresso-core", version.ref = "espressoCore" }
appcompat = { group = "androidx.appcompat", name = "appcompat", version.ref = "appcompat" }
material = { group = "com.google.android.material", name = "material", version.ref = "material" }
activity = { group = "androidx.activity", name = "activity", version.ref = "activity" }
constraintlayout = { group = "androidx.constraintlayout", name = "constraintlayout", version.ref = "constraintlayout" }
[plugins]
android-application = { id = "com.android.application", version.ref = "agp" }

BIN
gradle/wrapper/gradle-wrapper.jar vendored Normal file

Binary file not shown.

View File

@@ -0,0 +1,6 @@
#Tue Mar 03 15:23:41 WET 2026
distributionBase=GRADLE_USER_HOME
distributionPath=wrapper/dists
distributionUrl=https\://services.gradle.org/distributions/gradle-8.13-bin.zip
zipStoreBase=GRADLE_USER_HOME
zipStorePath=wrapper/dists

185
gradlew vendored Executable file
View File

@@ -0,0 +1,185 @@
#!/usr/bin/env sh
#
# Copyright 2015 the original author or authors.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# https://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
##############################################################################
##
## Gradle start up script for UN*X
##
##############################################################################
# Attempt to set APP_HOME
# Resolve links: $0 may be a link
PRG="$0"
# Need this for relative symlinks.
while [ -h "$PRG" ] ; do
ls=`ls -ld "$PRG"`
link=`expr "$ls" : '.*-> \(.*\)$'`
if expr "$link" : '/.*' > /dev/null; then
PRG="$link"
else
PRG=`dirname "$PRG"`"/$link"
fi
done
SAVED="`pwd`"
cd "`dirname \"$PRG\"`/" >/dev/null
APP_HOME="`pwd -P`"
cd "$SAVED" >/dev/null
APP_NAME="Gradle"
APP_BASE_NAME=`basename "$0"`
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"'
# Use the maximum available, or set MAX_FD != -1 to use that value.
MAX_FD="maximum"
warn () {
echo "$*"
}
die () {
echo
echo "$*"
echo
exit 1
}
# OS specific support (must be 'true' or 'false').
cygwin=false
msys=false
darwin=false
nonstop=false
case "`uname`" in
CYGWIN* )
cygwin=true
;;
Darwin* )
darwin=true
;;
MINGW* )
msys=true
;;
NONSTOP* )
nonstop=true
;;
esac
CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
# Determine the Java command to use to start the JVM.
if [ -n "$JAVA_HOME" ] ; then
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
# IBM's JDK on AIX uses strange locations for the executables
JAVACMD="$JAVA_HOME/jre/sh/java"
else
JAVACMD="$JAVA_HOME/bin/java"
fi
if [ ! -x "$JAVACMD" ] ; then
die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
Please set the JAVA_HOME variable in your environment to match the
location of your Java installation."
fi
else
JAVACMD="java"
which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
Please set the JAVA_HOME variable in your environment to match the
location of your Java installation."
fi
# Increase the maximum file descriptors if we can.
if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then
MAX_FD_LIMIT=`ulimit -H -n`
if [ $? -eq 0 ] ; then
if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
MAX_FD="$MAX_FD_LIMIT"
fi
ulimit -n $MAX_FD
if [ $? -ne 0 ] ; then
warn "Could not set maximum file descriptor limit: $MAX_FD"
fi
else
warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
fi
fi
# For Darwin, add options to specify how the application appears in the dock
if $darwin; then
GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
fi
# For Cygwin or MSYS, switch paths to Windows format before running java
if [ "$cygwin" = "true" -o "$msys" = "true" ] ; then
APP_HOME=`cygpath --path --mixed "$APP_HOME"`
CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
JAVACMD=`cygpath --unix "$JAVACMD"`
# We build the pattern for arguments to be converted via cygpath
ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
SEP=""
for dir in $ROOTDIRSRAW ; do
ROOTDIRS="$ROOTDIRS$SEP$dir"
SEP="|"
done
OURCYGPATTERN="(^($ROOTDIRS))"
# Add a user-defined pattern to the cygpath arguments
if [ "$GRADLE_CYGPATTERN" != "" ] ; then
OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
fi
# Now convert the arguments - kludge to limit ourselves to /bin/sh
i=0
for arg in "$@" ; do
CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option
if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition
eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
else
eval `echo args$i`="\"$arg\""
fi
i=`expr $i + 1`
done
case $i in
0) set -- ;;
1) set -- "$args0" ;;
2) set -- "$args0" "$args1" ;;
3) set -- "$args0" "$args1" "$args2" ;;
4) set -- "$args0" "$args1" "$args2" "$args3" ;;
5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
esac
fi
# Escape application args
save () {
for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done
echo " "
}
APP_ARGS=`save "$@"`
# Collect all arguments for the java command, following the shell quoting and substitution rules
eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS"
exec "$JAVACMD" "$@"

89
gradlew.bat vendored Normal file
View File

@@ -0,0 +1,89 @@
@rem
@rem Copyright 2015 the original author or authors.
@rem
@rem Licensed under the Apache License, Version 2.0 (the "License");
@rem you may not use this file except in compliance with the License.
@rem You may obtain a copy of the License at
@rem
@rem https://www.apache.org/licenses/LICENSE-2.0
@rem
@rem Unless required by applicable law or agreed to in writing, software
@rem distributed under the License is distributed on an "AS IS" BASIS,
@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
@rem See the License for the specific language governing permissions and
@rem limitations under the License.
@rem
@if "%DEBUG%" == "" @echo off
@rem ##########################################################################
@rem
@rem Gradle startup script for Windows
@rem
@rem ##########################################################################
@rem Set local scope for the variables with windows NT shell
if "%OS%"=="Windows_NT" setlocal
set DIRNAME=%~dp0
if "%DIRNAME%" == "" set DIRNAME=.
set APP_BASE_NAME=%~n0
set APP_HOME=%DIRNAME%
@rem Resolve any "." and ".." in APP_HOME to make it shorter.
for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi
@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m"
@rem Find java.exe
if defined JAVA_HOME goto findJavaFromJavaHome
set JAVA_EXE=java.exe
%JAVA_EXE% -version >NUL 2>&1
if "%ERRORLEVEL%" == "0" goto execute
echo.
echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
echo.
echo Please set the JAVA_HOME variable in your environment to match the
echo location of your Java installation.
goto fail
:findJavaFromJavaHome
set JAVA_HOME=%JAVA_HOME:"=%
set JAVA_EXE=%JAVA_HOME%/bin/java.exe
if exist "%JAVA_EXE%" goto execute
echo.
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
echo.
echo Please set the JAVA_HOME variable in your environment to match the
echo location of your Java installation.
goto fail
:execute
@rem Setup the command line
set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
@rem Execute Gradle
"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %*
:end
@rem End local scope for the variables with windows NT shell
if "%ERRORLEVEL%"=="0" goto mainEnd
:fail
rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
rem the _cmd.exe /c_ return code!
if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
exit /b 1
:mainEnd
if "%OS%"=="Windows_NT" endlocal
:omega

24
settings.gradle.kts Normal file
View File

@@ -0,0 +1,24 @@
pluginManagement {
repositories {
google {
content {
includeGroupByRegex("com\\.android.*")
includeGroupByRegex("com\\.google.*")
includeGroupByRegex("androidx.*")
}
}
mavenCentral()
gradlePluginPortal()
}
}
dependencyResolutionManagement {
repositoriesMode.set(RepositoriesMode.FAIL_ON_PROJECT_REPOS)
repositories {
google()
mavenCentral()
}
}
rootProject.name = "PAP"
include(":app")