commit 8219b97196382c255a4dc767b5d7fe4dd4379238 Author: 230402 <230402@epvc.pt> Date: Thu Mar 19 10:27:14 2026 +0000 DigimonAPI diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..aa724b7 --- /dev/null +++ b/.gitignore @@ -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 diff --git a/.idea/.gitignore b/.idea/.gitignore new file mode 100644 index 0000000..eaf91e2 --- /dev/null +++ b/.idea/.gitignore @@ -0,0 +1,3 @@ +# Default ignored files +/shelf/ +/workspace.xml diff --git a/.idea/.name b/.idea/.name new file mode 100644 index 0000000..698d4d1 --- /dev/null +++ b/.idea/.name @@ -0,0 +1 @@ +apidigimon \ No newline at end of file diff --git a/.idea/AndroidProjectSystem.xml b/.idea/AndroidProjectSystem.xml new file mode 100644 index 0000000..d58d49b --- /dev/null +++ b/.idea/AndroidProjectSystem.xml @@ -0,0 +1,6 @@ + + + + + \ No newline at end of file diff --git a/.idea/compiler.xml b/.idea/compiler.xml new file mode 100644 index 0000000..8b3f102 --- /dev/null +++ b/.idea/compiler.xml @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/.idea/deploymentTargetSelector.xml b/.idea/deploymentTargetSelector.xml new file mode 100644 index 0000000..0d46093 --- /dev/null +++ b/.idea/deploymentTargetSelector.xml @@ -0,0 +1,10 @@ + + + + + + + + + \ No newline at end of file diff --git a/.idea/gradle.xml b/.idea/gradle.xml new file mode 100644 index 0000000..d33973f --- /dev/null +++ b/.idea/gradle.xml @@ -0,0 +1,18 @@ + + + + + + + \ No newline at end of file diff --git a/.idea/misc.xml b/.idea/misc.xml new file mode 100644 index 0000000..b2c751a --- /dev/null +++ b/.idea/misc.xml @@ -0,0 +1,9 @@ + + + + + + + + \ No newline at end of file diff --git a/.idea/runConfigurations.xml b/.idea/runConfigurations.xml new file mode 100644 index 0000000..5bd6771 --- /dev/null +++ b/.idea/runConfigurations.xml @@ -0,0 +1,17 @@ + + + + + + \ No newline at end of file diff --git a/.idea/vcs.xml b/.idea/vcs.xml new file mode 100644 index 0000000..94a25f7 --- /dev/null +++ b/.idea/vcs.xml @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/app/.gitignore b/app/.gitignore new file mode 100644 index 0000000..42afabf --- /dev/null +++ b/app/.gitignore @@ -0,0 +1 @@ +/build \ No newline at end of file diff --git a/app/build.gradle.kts b/app/build.gradle.kts new file mode 100644 index 0000000..235fe3e --- /dev/null +++ b/app/build.gradle.kts @@ -0,0 +1,53 @@ +plugins { + alias(libs.plugins.android.application) +} + +android { + namespace = "com.example.apidigimon" + compileSdk { + version = release(36) { + minorApiLevel = 1 + } + } + + defaultConfig { + applicationId = "com.example.apidigimon" + 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("com.squareup.retrofit2:retrofit:2.9.0") + implementation("com.squareup.retrofit2:converter-gson:2.9.0") + implementation("com.github.bumptech.glide:glide:4.16.0") + annotationProcessor("com.github.bumptech.glide:compiler:4.16.0") + implementation("androidx.appcompat:appcompat:1.6.1") + implementation("com.google.android.material:material:1.9.0") + implementation("androidx.constraintlayout:constraintlayout:2.1.4") + implementation(libs.appcompat) + implementation(libs.material) + implementation(libs.activity) + implementation(libs.constraintlayout) + testImplementation(libs.junit) + androidTestImplementation(libs.ext.junit) + androidTestImplementation(libs.espresso.core) +} \ No newline at end of file diff --git a/app/proguard-rules.pro b/app/proguard-rules.pro new file mode 100644 index 0000000..481bb43 --- /dev/null +++ b/app/proguard-rules.pro @@ -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 \ No newline at end of file diff --git a/app/src/androidTest/java/com/example/apidigimon/ExampleInstrumentedTest.java b/app/src/androidTest/java/com/example/apidigimon/ExampleInstrumentedTest.java new file mode 100644 index 0000000..a70b230 --- /dev/null +++ b/app/src/androidTest/java/com/example/apidigimon/ExampleInstrumentedTest.java @@ -0,0 +1,26 @@ +package com.example.apidigimon; + +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 Testing documentation + */ +@RunWith(AndroidJUnit4.class) +public class ExampleInstrumentedTest { + @Test + public void useAppContext() { + // Context of the app under test. + Context appContext = InstrumentationRegistry.getInstrumentation().getTargetContext(); + assertEquals("com.example.apidigimon", appContext.getPackageName()); + } +} \ No newline at end of file diff --git a/app/src/main/AndroidManifest.xml b/app/src/main/AndroidManifest.xml new file mode 100644 index 0000000..9f5897f --- /dev/null +++ b/app/src/main/AndroidManifest.xml @@ -0,0 +1,31 @@ + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/ic_launcher-playstore.png b/app/src/main/ic_launcher-playstore.png new file mode 100644 index 0000000..b70bb4f Binary files /dev/null and b/app/src/main/ic_launcher-playstore.png differ diff --git a/app/src/main/java/com/example/apidigimon/DetailsActivity.java b/app/src/main/java/com/example/apidigimon/DetailsActivity.java new file mode 100644 index 0000000..e861d6f --- /dev/null +++ b/app/src/main/java/com/example/apidigimon/DetailsActivity.java @@ -0,0 +1,34 @@ +package com.example.apidigimon; + +import android.os.Bundle; +import android.view.View; +import android.widget.ImageView; +import android.widget.TextView; +import androidx.appcompat.app.AppCompatActivity; +import com.bumptech.glide.Glide; + +public class DetailsActivity extends AppCompatActivity { + + @Override + protected void onCreate(Bundle savedInstanceState) { + super.onCreate(savedInstanceState); + setContentView(R.layout.activity_details); + + ImageView img = findViewById(R.id.imgDetalhe); + TextView nome = findViewById(R.id.txtNomeDetalhe); + TextView level = findViewById(R.id.txtLevelDetalhe); + + // Receber o objeto Digimon que foi enviado + Digimon d = (Digimon) getIntent().getSerializableExtra("digimon_clicado"); + + if (d != null) { + nome.setText(d.getName()); + level.setText("Level: " + d.getLevel()); + Glide.with(this).load(d.getImg()).into(img); + } + } + + public void voltar(View v) { + finish(); // Fecha esta tela e volta para a lista + } +} diff --git a/app/src/main/java/com/example/apidigimon/Digimon.java b/app/src/main/java/com/example/apidigimon/Digimon.java new file mode 100644 index 0000000..5b2f754 --- /dev/null +++ b/app/src/main/java/com/example/apidigimon/Digimon.java @@ -0,0 +1,13 @@ +package com.example.apidigimon; + +import java.io.Serializable; + +public class Digimon implements Serializable { + private String name; + private String img; + private String level; + + public String getName() { return name; } + public String getImg() { return img; } + public String getLevel() { return level; } +} diff --git a/app/src/main/java/com/example/apidigimon/DigimonAdapter.java b/app/src/main/java/com/example/apidigimon/DigimonAdapter.java new file mode 100644 index 0000000..9ef45bc --- /dev/null +++ b/app/src/main/java/com/example/apidigimon/DigimonAdapter.java @@ -0,0 +1,89 @@ +package com.example.apidigimon; + +import android.content.Context; +import android.content.Intent; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.ImageView; +import android.widget.TextView; +import androidx.annotation.NonNull; +import androidx.recyclerview.widget.RecyclerView; +import com.bumptech.glide.Glide; +import java.util.ArrayList; +import java.util.List; + +public class DigimonAdapter extends RecyclerView.Adapter { + + private Context context; + private List listaMostrada; + private List listaOriginal; + + public DigimonAdapter(Context context, List lista) { + this.context = context; + this.listaMostrada = lista; + this.listaOriginal = new ArrayList<>(lista); + } + + public void filtrar(String texto) { + listaMostrada.clear(); + if (texto.isEmpty()) { + listaMostrada.addAll(listaOriginal); + } else { + String busca = texto.toLowerCase().trim(); + for (Digimon item : listaOriginal) { + if (item.getName().toLowerCase().contains(busca) || + item.getLevel().toLowerCase().contains(busca)) { + listaMostrada.add(item); + } + } + } + notifyDataSetChanged(); + } + + @NonNull + @Override + public DigimonViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) { + View v = LayoutInflater.from(context).inflate(R.layout.item_digimon, parent, false); + return new DigimonViewHolder(v); + } + + @Override + public void onBindViewHolder(@NonNull DigimonViewHolder holder, int position) { + Digimon d = listaMostrada.get(position); + holder.txtName.setText(d.getName()); + holder.txtLevel.setText("Level: " + d.getLevel()); + + Glide.with(context) + .load(d.getImg()) + .placeholder(android.R.drawable.ic_menu_gallery) + .into(holder.imgDigimon); + + // AÇÃO DE CLIQUE: Abre o ecrã de detalhes + holder.itemView.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View v) { + Intent intent = new Intent(context, DetailsActivity.class); + intent.putExtra("digimon_clicado", d); // Envia o objeto Digimon + context.startActivity(intent); + } + }); + } + + @Override + public int getItemCount() { + return listaMostrada.size(); + } + + public static class DigimonViewHolder extends RecyclerView.ViewHolder { + ImageView imgDigimon; + TextView txtName, txtLevel; + + public DigimonViewHolder(@NonNull View itemView) { + super(itemView); + imgDigimon = itemView.findViewById(R.id.imgDigimon); + txtName = itemView.findViewById(R.id.txtName); + txtLevel = itemView.findViewById(R.id.txtLevel); + } + } +} \ No newline at end of file diff --git a/app/src/main/java/com/example/apidigimon/DigimonService.java b/app/src/main/java/com/example/apidigimon/DigimonService.java new file mode 100644 index 0000000..0d996b2 --- /dev/null +++ b/app/src/main/java/com/example/apidigimon/DigimonService.java @@ -0,0 +1,10 @@ +package com.example.apidigimon; + +import java.util.List; +import retrofit2.Call; +import retrofit2.http.GET; + +public interface DigimonService { + @GET("digimon") + Call> getDigimons(); +} diff --git a/app/src/main/java/com/example/apidigimon/MainActivity.java b/app/src/main/java/com/example/apidigimon/MainActivity.java new file mode 100644 index 0000000..60016e2 --- /dev/null +++ b/app/src/main/java/com/example/apidigimon/MainActivity.java @@ -0,0 +1,78 @@ +package com.example.apidigimon; + +import android.os.Bundle; +import android.widget.Toast; +import androidx.appcompat.app.AppCompatActivity; +import androidx.appcompat.widget.SearchView; +import androidx.recyclerview.widget.LinearLayoutManager; +import androidx.recyclerview.widget.RecyclerView; +import java.util.List; +import retrofit2.Call; +import retrofit2.Callback; +import retrofit2.Response; +import retrofit2.Retrofit; +import retrofit2.converter.gson.GsonConverterFactory; + +public class MainActivity extends AppCompatActivity { + + private RecyclerView rv; + private DigimonAdapter adapter; + private SearchView searchView; + + @Override + protected void onCreate(Bundle savedInstanceState) { + super.onCreate(savedInstanceState); + setContentView(R.layout.activity_main); + + // Iniciar componentes + rv = findViewById(R.id.recyclerViewDigimon); + searchView = findViewById(R.id.searchView); + rv.setLayoutManager(new LinearLayoutManager(this)); + + // Configurar Retrofit + Retrofit retrofit = new Retrofit.Builder() + .baseUrl("https://digimon-api.vercel.app/api/") + .addConverterFactory(GsonConverterFactory.create()) + .build(); + + DigimonService service = retrofit.create(DigimonService.class); + + // Chamar a API + service.getDigimons().enqueue(new Callback>() { + @Override + public void onResponse(Call> call, Response> response) { + if (response.isSuccessful() && response.body() != null) { + // Criar o adapter com a lista que veio da internet + adapter = new DigimonAdapter(MainActivity.this, response.body()); + rv.setAdapter(adapter); + + // Configurar o filtro da barra de pesquisa só depois de termos os dados + configurarPesquisa(); + } + } + + @Override + public void onFailure(Call> call, Throwable t) { + Toast.makeText(MainActivity.this, "Erro de rede!", Toast.LENGTH_SHORT).show(); + } + }); + } + + private void configurarPesquisa() { + searchView.setOnQueryTextListener(new SearchView.OnQueryTextListener() { + @Override + public boolean onQueryTextSubmit(String query) { + return false; + } + + @Override + public boolean onQueryTextChange(String newText) { + // Filtra a lista enquanto o utilizador escreve + if (adapter != null) { + adapter.filtrar(newText); + } + return true; + } + }); + } +} \ No newline at end of file diff --git a/app/src/main/res/drawable/ic_launcher_background.xml b/app/src/main/res/drawable/ic_launcher_background.xml new file mode 100644 index 0000000..953ca56 --- /dev/null +++ b/app/src/main/res/drawable/ic_launcher_background.xml @@ -0,0 +1,74 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/app/src/main/res/drawable/ic_launcher_foreground.xml b/app/src/main/res/drawable/ic_launcher_foreground.xml new file mode 100644 index 0000000..2b068d1 --- /dev/null +++ b/app/src/main/res/drawable/ic_launcher_foreground.xml @@ -0,0 +1,30 @@ + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/activity_details.xml b/app/src/main/res/layout/activity_details.xml new file mode 100644 index 0000000..5ce5c50 --- /dev/null +++ b/app/src/main/res/layout/activity_details.xml @@ -0,0 +1,38 @@ + + + + + + + + + +