Update de muchos cambios, nuevas funcionalidades en la app, juegos finalizados y todos juntos en una app)
|  | @ -4,7 +4,7 @@ | |||
|     <selectionStates> | ||||
|       <SelectionState runConfigName="app"> | ||||
|         <option name="selectionMode" value="DROPDOWN" /> | ||||
|         <DropdownSelection timestamp="2025-04-18T20:03:38.478308800Z"> | ||||
|         <DropdownSelection timestamp="2025-05-23T14:38:41.985444400Z"> | ||||
|           <Target type="DEFAULT_BOOT"> | ||||
|             <handle> | ||||
|               <DeviceId pluginId="LocalEmulator" identifier="path=C:\Users\moran\.android\avd\Pixel_9_Pro.avd" /> | ||||
|  |  | |||
|  | @ -48,5 +48,9 @@ dependencies { | |||
|     androidTestImplementation(libs.espresso.core) | ||||
| 
 | ||||
|     implementation("com.github.PhilJay:MPAndroidChart:v3.1.0") | ||||
|     implementation("androidx.swiperefreshlayout:swiperefreshlayout:1.1.0") | ||||
|     implementation ("androidx.security:security-crypto:1.1.0-alpha06") | ||||
|     implementation("com.github.bumptech.glide:glide:4.15.1") | ||||
| 
 | ||||
| 
 | ||||
| } | ||||
|  | @ -18,6 +18,7 @@ | |||
|         <activity | ||||
|             android:name=".ui.MainActivity" | ||||
|             android:exported="true" | ||||
|             android:screenOrientation="portrait" | ||||
|             android:label="@string/app_name"> | ||||
|             <intent-filter> | ||||
|                 <action android:name="android.intent.action.MAIN" /> | ||||
|  |  | |||
|  | @ -0,0 +1,12 @@ | |||
| package com.andresgmoran.apptrabajadores.exceptions; | ||||
| 
 | ||||
| public class ParserException extends RuntimeException { | ||||
| 
 | ||||
|     public ParserException(String message) { | ||||
|         super(message); | ||||
|     } | ||||
| 
 | ||||
|     public ParserException(String message, Throwable cause) { | ||||
|         super(message, cause); | ||||
|     } | ||||
| } | ||||
|  | @ -6,5 +6,6 @@ import com.andresgmoran.apptrabajadores.models.ActivityResident; | |||
| import java.util.List; | ||||
| 
 | ||||
| public interface IOClickOnActivityListener { | ||||
|     void onClickOnActivity(Activity activity, List<ActivityResident> activityResidents); | ||||
|     void onClickOnActivity(Activity activity , List<ActivityResident> participants); | ||||
|     void onDeleteActivitie(Activity activity); | ||||
| } | ||||
|  |  | |||
|  | @ -0,0 +1,11 @@ | |||
| package com.andresgmoran.apptrabajadores.interfaces; | ||||
| 
 | ||||
| import com.andresgmoran.apptrabajadores.models.Activity; | ||||
| import com.andresgmoran.apptrabajadores.models.ActivityResident; | ||||
| import com.andresgmoran.apptrabajadores.models.Resident; | ||||
| 
 | ||||
| import java.util.List; | ||||
| 
 | ||||
| public interface IOClickOnAddParticipantListener { | ||||
|     void onClickOnAddParticipant(Activity activity, List<ActivityResident> participants , List<Resident> residents); | ||||
| } | ||||
|  | @ -6,4 +6,5 @@ import com.andresgmoran.apptrabajadores.models.gameStats.GameStat; | |||
| 
 | ||||
| public interface IOClickOnGameStatsListener { | ||||
|     void onClickOnLatestGame(GameStat gameStat, Resident gameStatResident, Game gameStatGame); | ||||
|     void onDeleteGameStat(GameStat gameStat, Game gameStatGame); | ||||
| } | ||||
|  |  | |||
|  | @ -0,0 +1,12 @@ | |||
| package com.andresgmoran.apptrabajadores.interfaces; | ||||
| 
 | ||||
| import com.andresgmoran.apptrabajadores.models.ActivityResident; | ||||
| 
 | ||||
| public interface IOClickOnParticipantListener { | ||||
|     void onClickOnParticipant(ActivityResident participant); | ||||
|     void onClickOnAssistance(ActivityResident participant, boolean isTrue); | ||||
|     void onClickOnOpinion(ActivityResident participant, boolean isPreOpinion); | ||||
|     void onClickOnMaterialHelp(ActivityResident participant, boolean isTrue); | ||||
|     void onClickOnHumanHelp(ActivityResident participant, boolean isTrue); | ||||
| 
 | ||||
| } | ||||
|  | @ -4,4 +4,5 @@ import com.andresgmoran.apptrabajadores.models.Resident; | |||
| 
 | ||||
| public interface IOClickOnResidentListener { | ||||
|     void onClickOnResident(Resident resident); | ||||
|     void onTakeOutResident(Resident resident); | ||||
| } | ||||
|  |  | |||
|  | @ -0,0 +1,8 @@ | |||
| package com.andresgmoran.apptrabajadores.interfaces; | ||||
| 
 | ||||
| import com.andresgmoran.apptrabajadores.models.Activity; | ||||
| import com.andresgmoran.apptrabajadores.models.ActivityState; | ||||
| 
 | ||||
| public interface IOnChageStateActivityListener { | ||||
|     void onChangeStateActivity(Activity activity, ActivityState state); | ||||
| } | ||||
|  | @ -2,49 +2,60 @@ package com.andresgmoran.apptrabajadores.models; | |||
| 
 | ||||
| import java.io.Serializable; | ||||
| import java.time.LocalDate; | ||||
| import java.time.LocalDateTime; | ||||
| import java.util.List; | ||||
| import java.util.Objects; | ||||
| 
 | ||||
| public class Activity implements Serializable { | ||||
|     private final Long id; | ||||
|     private final String name; | ||||
|     private final String description; | ||||
|     private final LocalDate date; | ||||
|     private final LocalDateTime date; | ||||
|     private final ActivityState state; | ||||
|     private final List<Resident> residents; | ||||
|     private final List<Long> residentIds; | ||||
|     private final Long residenceId; | ||||
| 
 | ||||
|     public Activity(String name, String description, LocalDate date, ActivityState state, List<Resident> residents) { | ||||
|     public Activity(Long id, String name, String description, LocalDateTime date, ActivityState state, List<Long> residentIds, Long residenceId) { | ||||
|         this.id = id; | ||||
|         this.name = name; | ||||
|         this.description = description; | ||||
|         this.date = date; | ||||
|         this.state = state; | ||||
|         this.residents = residents; | ||||
|         this.residentIds = residentIds; | ||||
|         this.residenceId = residenceId; | ||||
|     } | ||||
| 
 | ||||
|     public Long getId() { | ||||
|         return id; | ||||
|     } | ||||
|     public String getName() { | ||||
|         return name; | ||||
|     } | ||||
|     public String getDescription() { | ||||
|         return description; | ||||
|     } | ||||
|     public LocalDate getDate() { | ||||
|     public LocalDateTime getDate() { | ||||
|         return date; | ||||
|     } | ||||
|     public ActivityState getState() { | ||||
|         return state; | ||||
|     } | ||||
|     public List<Resident> getResidents() { | ||||
|         return residents; | ||||
|     public List<Long> getResidentIds() { | ||||
|         return residentIds; | ||||
|     } | ||||
|     public Long getResidenceId() { | ||||
|         return residenceId; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public boolean equals(Object o) { | ||||
|         if (o == null || getClass() != o.getClass()) return false; | ||||
|         Activity activity = (Activity) o; | ||||
|         return Objects.equals(name, activity.name) && Objects.equals(description, activity.description) && Objects.equals(date, activity.date) && state == activity.state && Objects.equals(residents, activity.residents); | ||||
|         return Objects.equals(id, activity.id) && Objects.equals(name, activity.name) && Objects.equals(description, activity.description) && Objects.equals(date, activity.date) && state == activity.state && Objects.equals(residentIds, activity.residentIds) && Objects.equals(residenceId, activity.residenceId); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public int hashCode() { | ||||
|         return Objects.hash(name, description, date, state, residents); | ||||
|         return Objects.hash(id, name, description, date, state, residentIds, residenceId); | ||||
|     } | ||||
| } | ||||
|  |  | |||
|  | @ -1,24 +1,48 @@ | |||
| package com.andresgmoran.apptrabajadores.models; | ||||
| 
 | ||||
| import java.io.Serializable; | ||||
| import java.time.LocalDate; | ||||
| import java.util.Objects; | ||||
| 
 | ||||
| public class ActivityResident extends Resident{ | ||||
|     private boolean asistanceState; | ||||
| public class ActivityResident implements Serializable { | ||||
|     private final Long id; | ||||
|     private final Long activityId; | ||||
|     private final Long idResident; | ||||
|     private final boolean assistance; | ||||
|     private final boolean humanHelp; | ||||
|     private final boolean materialHelp; | ||||
|     private String preOpinion; | ||||
|     private String postOpinion; | ||||
| 
 | ||||
|     public ActivityResident(Long id, String name, String surnames, LocalDate birthDate, Long residenceId, boolean asistanceState, String preOpinion, String postOpinion) { | ||||
|         super(id, name, surnames, birthDate, residenceId); | ||||
|         this.asistanceState = asistanceState; | ||||
|     public ActivityResident(Long id, Long activityId, Long idResident,boolean assistance, boolean humanHelp, boolean materialHelp, String preOpinion, String postOpinion) { | ||||
|         this.id = id; | ||||
|         this.activityId = activityId; | ||||
|         this.idResident = idResident; | ||||
|         this.assistance = assistance; | ||||
|         this.humanHelp = humanHelp; | ||||
|         this.materialHelp = materialHelp; | ||||
|         this.preOpinion = preOpinion; | ||||
|         this.postOpinion = postOpinion; | ||||
|     } | ||||
|     public boolean isAsistanceState() { | ||||
|         return asistanceState; | ||||
|     public Long getId() { | ||||
|         return id; | ||||
|     } | ||||
|     public void setAsistanceState(boolean asistanceState) { | ||||
|         this.asistanceState = asistanceState; | ||||
| 
 | ||||
|     public Long getActivityId() { | ||||
|         return activityId; | ||||
|     } | ||||
|     public Long getIdResident() { | ||||
|         return idResident; | ||||
|     } | ||||
|     public boolean isAssistance() { | ||||
|         return assistance; | ||||
|     } | ||||
| 
 | ||||
|     public boolean isHumanHelp() { | ||||
|         return humanHelp; | ||||
|     } | ||||
|     public boolean isMaterialHelp() { | ||||
|         return materialHelp; | ||||
|     } | ||||
|     public String getPreOpinion() { | ||||
|         return preOpinion; | ||||
|  | @ -36,13 +60,12 @@ public class ActivityResident extends Resident{ | |||
|     @Override | ||||
|     public boolean equals(Object o) { | ||||
|         if (o == null || getClass() != o.getClass()) return false; | ||||
|         if (!super.equals(o)) return false; | ||||
|         ActivityResident that = (ActivityResident) o; | ||||
|         return asistanceState == that.asistanceState && Objects.equals(preOpinion, that.preOpinion) && Objects.equals(postOpinion, that.postOpinion); | ||||
|         return humanHelp == that.humanHelp && materialHelp == that.materialHelp && Objects.equals(id, that.id) && Objects.equals(activityId, that.activityId) && Objects.equals(idResident, that.idResident) && Objects.equals(preOpinion, that.preOpinion) && Objects.equals(postOpinion, that.postOpinion); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public int hashCode() { | ||||
|         return Objects.hash(super.hashCode(), asistanceState, preOpinion, postOpinion); | ||||
|         return Objects.hash(id, activityId, idResident, humanHelp, materialHelp, preOpinion, postOpinion); | ||||
|     } | ||||
| } | ||||
|  |  | |||
|  | @ -4,7 +4,9 @@ import com.andresgmoran.apptrabajadores.models.gameStats.Difficulty; | |||
| 
 | ||||
| public enum ActivityState { | ||||
|     ABIERTO, | ||||
|     CERRADO; | ||||
|     CERRADO, | ||||
|     EN_CURSO, | ||||
|     FINALIZADA; | ||||
| 
 | ||||
|     public static ActivityState fromString(String value) { | ||||
|         switch (value.toLowerCase()) { | ||||
|  | @ -12,6 +14,10 @@ public enum ActivityState { | |||
|                 return ABIERTO; | ||||
|             case "cerrado": | ||||
|                 return CERRADO; | ||||
|             case "en_curso": | ||||
|                 return EN_CURSO; | ||||
|             case "finalizada": | ||||
|                 return FINALIZADA; | ||||
|             default: | ||||
|                 throw new IllegalArgumentException("Valor de estado no válido: " + value); | ||||
|         } | ||||
|  |  | |||
|  | @ -7,12 +7,10 @@ import java.util.Objects; | |||
| public class Game implements Serializable { | ||||
|     private final Long id; | ||||
|     private final String name; | ||||
|     private final Long residenceId; | ||||
| 
 | ||||
|     public Game(Long id, String name, Long residenceId) { | ||||
|     public Game(Long id, String name) { | ||||
|         this.id = id; | ||||
|         this.name = name; | ||||
|         this.residenceId = residenceId; | ||||
|     } | ||||
| 
 | ||||
|     public Long getId() { | ||||
|  | @ -23,19 +21,15 @@ public class Game implements Serializable { | |||
|         return name; | ||||
|     } | ||||
| 
 | ||||
|     public Long getResidenceId() { | ||||
|         return residenceId; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public boolean equals(Object o) { | ||||
|         if (o == null || getClass() != o.getClass()) return false; | ||||
|         Game game = (Game) o; | ||||
|         return Objects.equals(id, game.id) && Objects.equals(name, game.name) && Objects.equals(residenceId, game.residenceId); | ||||
|         return Objects.equals(id, game.id) && Objects.equals(name, game.name); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public int hashCode() { | ||||
|         return Objects.hash(id, name, residenceId); | ||||
|         return Objects.hash(id, name); | ||||
|     } | ||||
| } | ||||
|  | @ -10,13 +10,15 @@ public class Resident implements Serializable { | |||
|     private final String name; | ||||
|     private final String surnames; | ||||
|     private final LocalDate birthDate; | ||||
|     private final String identityCard; | ||||
|     private final Long residenceId; | ||||
| 
 | ||||
|     public Resident(Long id, String name, String surnames, LocalDate birthDate, Long residenceId) { | ||||
|     public Resident(Long id, String name, String surnames, LocalDate birthDate, String identityCard , Long residenceId) { | ||||
|         this.id = id; | ||||
|         this.name = name; | ||||
|         this.surnames = surnames; | ||||
|         this.birthDate = birthDate; | ||||
|         this.identityCard = identityCard; | ||||
|         this.residenceId = residenceId; | ||||
|     } | ||||
| 
 | ||||
|  |  | |||
|  | @ -10,16 +10,19 @@ public class User implements Serializable { | |||
|     private final String email; | ||||
|     private final boolean enabled; | ||||
|     private final Long residenceId; | ||||
|     private final String role; //Decirle a kevin que lo ponga | ||||
|     private final String accountImage; | ||||
|     private final boolean takenOut; | ||||
| 
 | ||||
|     public User(Long id, String name, String surnames, String email, boolean enabled, Long residenceId) { | ||||
|     public User(Long id, String name, String surnames, String email, boolean enabled, Long residenceId, String accountImage, boolean takenOut) { | ||||
|         this.id = id; | ||||
|         this.name = name; | ||||
|         this.surnames = surnames; | ||||
|         this.email = email; | ||||
|         this.enabled = enabled; | ||||
|         this.residenceId = residenceId; | ||||
|         this.role = "Trabajador"; | ||||
|         this.accountImage = accountImage; | ||||
|         this.takenOut = takenOut; | ||||
| 
 | ||||
| 
 | ||||
|     } | ||||
| 
 | ||||
|  | @ -47,19 +50,23 @@ public class User implements Serializable { | |||
|         return residenceId; | ||||
|     } | ||||
| 
 | ||||
|     public String getRole() { | ||||
|         return role; | ||||
|     public String getAccountImage() { | ||||
|         return accountImage; | ||||
|     } | ||||
| 
 | ||||
|     public boolean isTakenOut() { | ||||
|         return takenOut; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public boolean equals(Object o) { | ||||
|         if (o == null || getClass() != o.getClass()) return false; | ||||
|         User user = (User) o; | ||||
|         return enabled == user.enabled && Objects.equals(id, user.id) && Objects.equals(name, user.name) && Objects.equals(surnames, user.surnames) && Objects.equals(email, user.email) && Objects.equals(residenceId, user.residenceId) && Objects.equals(role, user.role); | ||||
|         return enabled == user.enabled && takenOut == user.takenOut && Objects.equals(id, user.id) && Objects.equals(name, user.name) && Objects.equals(surnames, user.surnames) && Objects.equals(email, user.email) && Objects.equals(residenceId, user.residenceId) && Objects.equals(accountImage, user.accountImage); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public int hashCode() { | ||||
|         return Objects.hash(id, name, surnames, email, enabled, residenceId, role); | ||||
|         return Objects.hash(id, name, surnames, email, enabled, residenceId, accountImage, takenOut); | ||||
|     } | ||||
| } | ||||
|  |  | |||
|  | @ -1,49 +1,135 @@ | |||
| package com.andresgmoran.apptrabajadores.models.adapters; | ||||
| 
 | ||||
| import static java.security.AccessController.getContext; | ||||
| 
 | ||||
| import android.app.AlertDialog; | ||||
| import android.content.Context; | ||||
| import android.view.ContentInfo; | ||||
| import android.view.LayoutInflater; | ||||
| import android.view.View; | ||||
| import android.view.ViewGroup; | ||||
| import android.widget.ImageButton; | ||||
| import android.widget.PopupMenu; | ||||
| import android.widget.TextView; | ||||
| 
 | ||||
| import androidx.annotation.NonNull; | ||||
| import androidx.recyclerview.widget.RecyclerView; | ||||
| 
 | ||||
| import com.andresgmoran.apptrabajadores.R; | ||||
| import com.andresgmoran.apptrabajadores.interfaces.IOClickOnActivityListener; | ||||
| import com.andresgmoran.apptrabajadores.interfaces.IOnChageStateActivityListener; | ||||
| import com.andresgmoran.apptrabajadores.models.Activity; | ||||
| import com.andresgmoran.apptrabajadores.models.ActivityResident; | ||||
| import com.andresgmoran.apptrabajadores.models.ActivityState; | ||||
| import com.andresgmoran.apptrabajadores.models.Game; | ||||
| 
 | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| 
 | ||||
| public class ActivitiesAdapter extends RecyclerView.Adapter<ActivitiesAdapter.ActivitiesViewHolder> { | ||||
| 
 | ||||
|     public ActivitiesAdapter(){ | ||||
|         // Constructor implementation | ||||
|     private final Context context; | ||||
|     private List<Activity> activities; | ||||
|     private List<ActivityResident> participants; | ||||
|     private IOClickOnActivityListener listener; | ||||
|     private IOnChageStateActivityListener changeStateListener; | ||||
| 
 | ||||
| 
 | ||||
|     public ActivitiesAdapter(Context context, List<Activity> activities, List<ActivityResident> participants, IOClickOnActivityListener listener, IOnChageStateActivityListener changeStateListener) { | ||||
|         this.context = context; | ||||
|         this.activities = activities; | ||||
|         this.participants = participants; | ||||
|         this.listener = listener; | ||||
|         this.changeStateListener = changeStateListener; | ||||
|     } | ||||
|     @NonNull | ||||
|     @Override | ||||
|     public ActivitiesAdapter.ActivitiesViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) { | ||||
|         final View view = LayoutInflater.from( parent.getContext()).inflate(R.layout.item_resident_activity, parent, false); | ||||
|         final View view = LayoutInflater.from( parent.getContext()).inflate(R.layout.item_activity, parent, false); | ||||
|         return new ActivitiesAdapter.ActivitiesViewHolder(view); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public void onBindViewHolder(@NonNull ActivitiesViewHolder holder, int position) { | ||||
|         Activity activity = activities.get(position); | ||||
|         holder.bindActivity(activity); | ||||
|         List<ActivityResident> activityResidents = new ArrayList<>(); | ||||
|         for (ActivityResident activityResident : participants) { | ||||
|             if (activityResident.getActivityId() == activity.getId()) { | ||||
|                 activityResidents.add(activityResident); | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         if (activity.getState() != ActivityState.CERRADO){ | ||||
|             holder.activityStateButton.setOnClickListener( v -> { | ||||
|                 new AlertDialog.Builder(context) | ||||
|                         .setTitle("Confirmación") | ||||
|                         .setMessage("No se podra cambiar el estado de la actividad una vez cerrada. ¿Desea continuar?") | ||||
|                         .setPositiveButton("Aceptar", (dialog, which) -> { | ||||
|                             changeStateListener.onChangeStateActivity(activity, ActivityState.CERRADO); | ||||
|                         }) | ||||
|                         .setNegativeButton("Cancelar", (dialog, which) -> { | ||||
|                             dialog.dismiss(); | ||||
|                         }) | ||||
|                         .show(); | ||||
|             }); | ||||
|         } | ||||
| 
 | ||||
|         holder.optionsButton.setOnClickListener(v -> { | ||||
|             PopupMenu popupMenu = new PopupMenu(v.getContext(), v); | ||||
|             popupMenu.getMenuInflater().inflate(R.menu.menu_options, popupMenu.getMenu()); | ||||
| 
 | ||||
|             popupMenu.setOnMenuItemClickListener(item -> { | ||||
|                 if (item.getItemId() == R.id.action_delete) { | ||||
|                     if (activities != null) | ||||
|                         listener.onDeleteActivitie(activity); | ||||
|                     return true; | ||||
|                 } | ||||
|                 return false; | ||||
|             }); | ||||
| 
 | ||||
|             popupMenu.show(); | ||||
|         }); | ||||
| 
 | ||||
|         holder.itemView.setOnClickListener(new View.OnClickListener() { | ||||
|             @Override | ||||
|             public void onClick(View view) { | ||||
| 
 | ||||
|                 listener.onClickOnActivity(activity, activityResidents); | ||||
|             } | ||||
|         }); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public int getItemCount() { | ||||
|         return 0; | ||||
|         return activities.size(); | ||||
|     } | ||||
| 
 | ||||
|     public class ActivitiesViewHolder extends RecyclerView.ViewHolder { | ||||
| 
 | ||||
|         private final TextView activityName; | ||||
|         private final TextView activityDate; | ||||
|         private final ImageButton optionsButton; | ||||
|         private final ImageButton activityStateButton; | ||||
| 
 | ||||
| 
 | ||||
|         public ActivitiesViewHolder(View view) { | ||||
|             super(view); | ||||
|             activityName = itemView.findViewById(R.id.name_salida); | ||||
|             activityDate = itemView.findViewById(R.id.date_text_activityItem); | ||||
|             optionsButton = itemView.findViewById(R.id.more_options_activity_item); | ||||
|             activityStateButton = itemView.findViewById(R.id.activity_status_button); | ||||
| 
 | ||||
|         } | ||||
| 
 | ||||
|         public void bindActivity(Activity activity) { | ||||
|             activityName.setText(activity.getName()); | ||||
|             activityDate.setText(activity.getDate().toString()); | ||||
|             if (activity.getState() == ActivityState.ABIERTO) { | ||||
|                 activityStateButton.setImageResource(R.drawable.open_activity_status); | ||||
|             } else { | ||||
|                 activityStateButton.setImageResource(R.drawable.closed_activity_status); | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|     } | ||||
| } | ||||
|  |  | |||
|  | @ -3,6 +3,7 @@ package com.andresgmoran.apptrabajadores.models.adapters; | |||
| import android.view.LayoutInflater; | ||||
| import android.view.View; | ||||
| import android.view.ViewGroup; | ||||
| import android.widget.ImageView; | ||||
| import android.widget.TextView; | ||||
| 
 | ||||
| import androidx.annotation.NonNull; | ||||
|  | @ -15,7 +16,7 @@ import com.andresgmoran.apptrabajadores.models.Game; | |||
| import java.util.List; | ||||
| 
 | ||||
| public class GamesAdapter extends RecyclerView.Adapter<GamesAdapter.GamesViewHolder>{ | ||||
|     private final List<Game> games; | ||||
|     private List<Game> games; | ||||
|     private final IOClickOnGameListener listener; | ||||
| 
 | ||||
|     public GamesAdapter(List<Game> games, IOClickOnGameListener listener) { | ||||
|  | @ -23,6 +24,11 @@ public class GamesAdapter extends RecyclerView.Adapter<GamesAdapter.GamesViewHol | |||
|         this.listener = listener; | ||||
|     } | ||||
| 
 | ||||
|     public void updateData(List<Game> newGames) { | ||||
|         this.games = newGames; | ||||
|         notifyDataSetChanged(); | ||||
|     } | ||||
| 
 | ||||
|     @NonNull | ||||
|     @Override | ||||
|     public GamesAdapter.GamesViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) { | ||||
|  | @ -50,15 +56,24 @@ public class GamesAdapter extends RecyclerView.Adapter<GamesAdapter.GamesViewHol | |||
|     public class GamesViewHolder extends RecyclerView.ViewHolder { | ||||
| 
 | ||||
|         private final TextView gameName; | ||||
|         private final ImageView gameImage; | ||||
| 
 | ||||
| 
 | ||||
|         public GamesViewHolder(View view) { | ||||
|             super(view); | ||||
|             gameName = view.findViewById(R.id.name_list_item); | ||||
|             gameImage = view.findViewById(R.id.item_list_image); | ||||
|         } | ||||
| 
 | ||||
|         public void bindGame(Game game) { | ||||
|             gameName.setText(game.getName()); | ||||
|             String name = game.getName(); | ||||
|             String formattedName = name.substring(0, 1).toUpperCase() + name.substring(1); | ||||
|             gameName.setText(formattedName); | ||||
| 
 | ||||
|             String resourceName = "logo_" + game.getName().toLowerCase().replace(" ", "_"); | ||||
|             int imageResId = itemView.getContext().getResources().getIdentifier(resourceName, "drawable", itemView.getContext().getPackageName()); | ||||
|             if (imageResId != 0) | ||||
|                 gameImage.setImageResource(imageResId); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  |  | |||
|  | @ -4,6 +4,8 @@ import android.util.Log; | |||
| import android.view.LayoutInflater; | ||||
| import android.view.View; | ||||
| import android.view.ViewGroup; | ||||
| import android.widget.ImageButton; | ||||
| import android.widget.PopupMenu; | ||||
| import android.widget.TextView; | ||||
| 
 | ||||
| import androidx.annotation.NonNull; | ||||
|  | @ -76,9 +78,26 @@ public class LastGamesAdapter extends RecyclerView.Adapter<LastGamesAdapter.Last | |||
| 
 | ||||
|         holder.bindObservation(gameStat, gameStatResident, gameStatGame); | ||||
| 
 | ||||
| 
 | ||||
|         Resident finalGameStatResident = gameStatResident; | ||||
|         Game finalGameStatGame = gameStatGame; | ||||
| 
 | ||||
|         holder.optionsButton.setOnClickListener(v -> { | ||||
|             PopupMenu popupMenu = new PopupMenu(v.getContext(), v); | ||||
|             popupMenu.getMenuInflater().inflate(R.menu.menu_options, popupMenu.getMenu()); | ||||
| 
 | ||||
|             popupMenu.setOnMenuItemClickListener(item -> { | ||||
|                 if (item.getItemId() == R.id.action_delete) { | ||||
|                     if (finalGameStatResident != null && finalGameStatGame != null) | ||||
|                         listener.onDeleteGameStat(gameStat, finalGameStatGame); | ||||
|                     return true; | ||||
|                 } | ||||
|                 return false; | ||||
|             }); | ||||
| 
 | ||||
|             popupMenu.show(); | ||||
|         }); | ||||
| 
 | ||||
|         holder.itemView.setOnClickListener(v -> { | ||||
|             if (finalGameStatResident != null && finalGameStatGame != null) | ||||
|                 listener.onClickOnLatestGame(gameStat, finalGameStatResident, finalGameStatGame); | ||||
|  | @ -96,6 +115,7 @@ public class LastGamesAdapter extends RecyclerView.Adapter<LastGamesAdapter.Last | |||
|         private final TextView gameName; | ||||
|         private final TextView date; | ||||
|         private final TextView time; | ||||
|         private final ImageButton optionsButton; | ||||
| 
 | ||||
|         public LastGamesViewHolder(View view) { | ||||
|             super(view); | ||||
|  | @ -103,10 +123,11 @@ public class LastGamesAdapter extends RecyclerView.Adapter<LastGamesAdapter.Last | |||
|             gameName = view.findViewById(R.id.gameName_text_latestGameItem); | ||||
|             date = view.findViewById(R.id.date_text_latestGameItem); | ||||
|             time = view.findViewById(R.id.time_text_latestGameItem); | ||||
|             optionsButton = view.findViewById(R.id.more_options_latestGameItem); | ||||
|         } | ||||
| 
 | ||||
|         public void bindObservation(GameStat gameStat, Resident gameStatResident, Game gameStatGame) { | ||||
|             if (gameStatResident == null || gameStatGame == null) { | ||||
|             if (gameStat == null) { | ||||
|                 residentName.setText("Residente no encontrado"); | ||||
|                 gameName.setText("Juego desconocido"); | ||||
|                 date.setText("-"); | ||||
|  | @ -114,8 +135,25 @@ public class LastGamesAdapter extends RecyclerView.Adapter<LastGamesAdapter.Last | |||
|                 return; | ||||
|             } | ||||
| 
 | ||||
|             residentName.setText(gameStatResident.getName() + " " + gameStatResident.getSurnames()); | ||||
|             gameName.setText(gameStatGame.getName()); | ||||
|             if (gameStatResident == null) { | ||||
|                 residentName.setText("Residente no encontrado"); | ||||
|             } else { | ||||
|                 residentName.setText(gameStatResident.getName() + " " + gameStatResident.getSurnames()); | ||||
|             } | ||||
| 
 | ||||
|             if (gameStatGame == null) { | ||||
|                 gameName.setText("Juego desconocido"); | ||||
|             } else { | ||||
|                 if (gameStatGame.getName().equalsIgnoreCase("flecha y reacciona") || gameStatGame.getName().equalsIgnoreCase("bingo auditivo")){ | ||||
|                     if (gameStat.getNum() == 0){ | ||||
|                         gameName.setText(gameStatGame.getName() + " - PERDIDA"); | ||||
|                     } else if ( gameStat.getNum() == 1){ | ||||
|                         gameName.setText(gameStatGame.getName() + " - GANADA"); | ||||
|                     } | ||||
|                 } else { | ||||
|                     gameName.setText(gameStatGame.getName()); | ||||
|                 } | ||||
|             } | ||||
| 
 | ||||
|             DateTimeFormatter formatter = DateTimeFormatter.ofPattern("EEEE, d 'de' MMMM 'de' yyyy", new Locale("es", "ES")); | ||||
|             String fechaFormateada = gameStat.getDateTime().format(formatter); | ||||
|  |  | |||
|  | @ -1,58 +0,0 @@ | |||
| package com.andresgmoran.apptrabajadores.models.adapters; | ||||
| 
 | ||||
| import android.view.LayoutInflater; | ||||
| import android.view.View; | ||||
| import android.view.ViewGroup; | ||||
| import android.widget.TextView; | ||||
| 
 | ||||
| import androidx.annotation.NonNull; | ||||
| import androidx.recyclerview.widget.RecyclerView; | ||||
| 
 | ||||
| import com.andresgmoran.apptrabajadores.R; | ||||
| import com.andresgmoran.apptrabajadores.models.Observation; | ||||
| 
 | ||||
| import java.util.List; | ||||
| 
 | ||||
| public class ObservationsAdapter extends RecyclerView.Adapter<ObservationsAdapter.ObservationsViewHolder> { | ||||
|     private final List<Observation> observations; | ||||
| 
 | ||||
|     public ObservationsAdapter(List<Observation> observations){ | ||||
|         this.observations = observations; | ||||
|     } | ||||
| 
 | ||||
|     @NonNull | ||||
|     @Override | ||||
|     public ObservationsViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) { | ||||
|         final View view = LayoutInflater.from( parent.getContext()).inflate(R.layout.item_observation, parent, false); | ||||
|         return new ObservationsViewHolder(view); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public void onBindViewHolder(@NonNull ObservationsViewHolder holder, int position) { | ||||
|         Observation observation = observations.get(position); | ||||
|         holder.bindObservation(observation); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public int getItemCount() { | ||||
|         return observations.size(); | ||||
|     } | ||||
| 
 | ||||
|     public class ObservationsViewHolder extends RecyclerView.ViewHolder { | ||||
| 
 | ||||
|         private final TextView observation; | ||||
|         private final TextView observationOwner; | ||||
|         private final TextView observationOwnerRole; | ||||
| 
 | ||||
|         public ObservationsViewHolder(View view) { | ||||
|             super(view); | ||||
|             observation = view.findViewById(R.id.observation_text); | ||||
|             observationOwner = view.findViewById(R.id.home_user_name); | ||||
|             observationOwnerRole = view.findViewById(R.id.observation_person_role); | ||||
|         } | ||||
|         public void bindObservation(Observation observation){ | ||||
|             observationOwner.setText(observation.getObservationOwner().getName()); | ||||
|             this.observation.setText(observation.getObservationText()); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | @ -0,0 +1,162 @@ | |||
| package com.andresgmoran.apptrabajadores.models.adapters; | ||||
| 
 | ||||
| import android.content.Context; | ||||
| import android.util.Log; | ||||
| 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 com.andresgmoran.apptrabajadores.R; | ||||
| import com.andresgmoran.apptrabajadores.interfaces.IOClickOnParticipantListener; | ||||
| import com.andresgmoran.apptrabajadores.models.Activity; | ||||
| import com.andresgmoran.apptrabajadores.models.ActivityResident; | ||||
| import com.andresgmoran.apptrabajadores.models.ActivityState; | ||||
| import com.andresgmoran.apptrabajadores.models.Resident; | ||||
| 
 | ||||
| import java.util.List; | ||||
| 
 | ||||
| public class ParticipantsAdapter extends RecyclerView.Adapter<ParticipantsAdapter.ParticipantsViewHolder> { | ||||
| 
 | ||||
|     private final Context context; | ||||
|     private final Activity activity; | ||||
|     private List<Resident> residents; | ||||
|     private List<ActivityResident> activityResidents; | ||||
|     private IOClickOnParticipantListener listener; | ||||
| 
 | ||||
| 
 | ||||
|     public ParticipantsAdapter(Context context, Activity activity, List<ActivityResident> activityResidents, List<Resident> residents  , IOClickOnParticipantListener listener) { | ||||
|         this.context = context; | ||||
|         this.activity = activity; | ||||
|         this.residents = residents; | ||||
|         this.activityResidents = activityResidents; | ||||
|         this.listener = listener; | ||||
|     } | ||||
|     @NonNull | ||||
|     @Override | ||||
|     public ParticipantsAdapter.ParticipantsViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) { | ||||
|         final View view = LayoutInflater.from( parent.getContext()).inflate(R.layout.item_resident_activity, parent, false); | ||||
|         return new ParticipantsAdapter.ParticipantsViewHolder(view); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public void onBindViewHolder(@NonNull ParticipantsViewHolder holder, int position) { | ||||
|         ActivityResident activityResident = activityResidents.get(position); | ||||
|         holder.bindActivity(activityResident); | ||||
|         holder.itemView.setOnClickListener(new View.OnClickListener() { | ||||
|             @Override | ||||
|             public void onClick(View view) { | ||||
|                 listener.onClickOnParticipant(activityResident); | ||||
|             } | ||||
|         }); | ||||
| 
 | ||||
|         holder.assitanceButton.setOnClickListener(new View.OnClickListener() { | ||||
|             @Override | ||||
|             public void onClick(View view) { | ||||
|                 if (activity.getState() == ActivityState.ABIERTO){ | ||||
|                     if (activityResident.isAssistance()){ | ||||
|                         listener.onClickOnAssistance(activityResident, false); | ||||
|                     } else { | ||||
|                         listener.onClickOnAssistance(activityResident, true); | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         }); | ||||
| 
 | ||||
|         holder.opinionButton.setOnClickListener(new View.OnClickListener() { | ||||
|             @Override | ||||
|             public void onClick(View view) { | ||||
|                 if (activity.getState() == ActivityState.ABIERTO && activityResident.getPreOpinion().isEmpty()) { | ||||
|                     // Dejar al usuario añadir un comentario y cambiar boolean de preOpinion a true | ||||
|                     listener.onClickOnOpinion(activityResident, true); | ||||
|                 } else if (activity.getState() == ActivityState.FINALIZADA && activityResident.getPostOpinion().isEmpty()){ | ||||
|                     // Dejar al usuario añadir un comentario y cambiar boolean de postOpinion a true | ||||
|                     listener.onClickOnOpinion(activityResident, false); | ||||
|                 } | ||||
|             } | ||||
|         }); | ||||
| 
 | ||||
|         holder.materialHelp.setOnClickListener(new View.OnClickListener() { | ||||
|             @Override | ||||
|             public void onClick(View view) { | ||||
|                 if (activity.getState() == ActivityState.ABIERTO){ | ||||
|                     if (activityResident.isMaterialHelp()){ | ||||
|                         listener.onClickOnMaterialHelp(activityResident, false); | ||||
|                     } else { | ||||
|                         listener.onClickOnMaterialHelp(activityResident, true); | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         }); | ||||
| 
 | ||||
|         holder.humanHelp.setOnClickListener(new View.OnClickListener() { | ||||
|             @Override | ||||
|             public void onClick(View view) { | ||||
|                 if (activity.getState() == ActivityState.ABIERTO){ | ||||
|                     if (activityResident.isHumanHelp()){ | ||||
|                         listener.onClickOnHumanHelp(activityResident, false); | ||||
|                     } else { | ||||
|                         listener.onClickOnHumanHelp(activityResident, true); | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         }); | ||||
| 
 | ||||
|         if (activityResident.isAssistance()){ | ||||
|             holder.assitanceButton.setBackgroundResource(R.drawable.activity_button_pressed_background); | ||||
|         } | ||||
|         if (activityResident.isMaterialHelp()){ | ||||
|             holder.materialHelp.setBackgroundResource(R.drawable.activity_button_pressed_background); | ||||
|         } | ||||
|         if (activityResident.isHumanHelp()){ | ||||
|             holder.humanHelp.setBackgroundResource(R.drawable.activity_button_pressed_background); | ||||
|         } | ||||
|         Log.e( "TAG", "State: " + activity.getState()); | ||||
|         Log.e( "TAG", "PreOpinion: " + activityResident.getPreOpinion()); | ||||
|         if (activity.getState() == ActivityState.ABIERTO && !activityResident.getPreOpinion().isEmpty()){ | ||||
|             holder.opinionButton.setBackgroundResource(R.drawable.activity_button_pressed_background); | ||||
|         } | ||||
|         if (activity.getState() == ActivityState.FINALIZADA && !activityResident.getPostOpinion().isEmpty()){ | ||||
|             Log.e( "TAG", "Opinion: " + activityResident.getPostOpinion()); | ||||
|             holder.opinionButton.setBackgroundResource(R.drawable.activity_button_pressed_background); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public int getItemCount() { | ||||
|         return activityResidents.size(); | ||||
|     } | ||||
| 
 | ||||
|     public class ParticipantsViewHolder extends RecyclerView.ViewHolder { | ||||
| 
 | ||||
|         private final TextView name; | ||||
|         private final ImageButton assitanceButton; | ||||
|         private final ImageButton opinionButton; | ||||
|         private final ImageButton materialHelp; | ||||
|         private final ImageButton humanHelp; | ||||
| 
 | ||||
| 
 | ||||
|         public ParticipantsViewHolder(View view) { | ||||
|             super(view); | ||||
|             name = itemView.findViewById(R.id.tv_name_participant_item); | ||||
|             assitanceButton = itemView.findViewById(R.id.participant_asistencia_button); | ||||
|             opinionButton = itemView.findViewById(R.id.participant_opinion_button); | ||||
|             materialHelp = itemView.findViewById(R.id.participant_material_help_button); | ||||
|             humanHelp = itemView.findViewById(R.id.participant_human_help_button); | ||||
|         } | ||||
| 
 | ||||
|         public void bindActivity(ActivityResident activityResident) { | ||||
|             for( Resident resident : residents){ | ||||
|                 if(resident.getId() == activityResident.getIdResident()){ | ||||
|                     name.setText(resident.getName() + " " + resident.getSurnames()); | ||||
|                     break; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|     } | ||||
| } | ||||
|  | @ -3,6 +3,8 @@ package com.andresgmoran.apptrabajadores.models.adapters; | |||
| import android.view.LayoutInflater; | ||||
| import android.view.View; | ||||
| import android.view.ViewGroup; | ||||
| import android.widget.ImageButton; | ||||
| import android.widget.PopupMenu; | ||||
| import android.widget.TextView; | ||||
| 
 | ||||
| import androidx.annotation.NonNull; | ||||
|  | @ -15,7 +17,7 @@ import com.andresgmoran.apptrabajadores.models.Resident; | |||
| import java.util.List; | ||||
| 
 | ||||
| public class ResidentsAdapter extends RecyclerView.Adapter<ResidentsAdapter.ResidentsViewHolder> { | ||||
|     private final List<Resident> residents; | ||||
|     private List<Resident> residents; | ||||
|     private final IOClickOnResidentListener listener; | ||||
| 
 | ||||
|     public ResidentsAdapter(List<Resident> residents, IOClickOnResidentListener listener) { | ||||
|  | @ -23,6 +25,11 @@ public class ResidentsAdapter extends RecyclerView.Adapter<ResidentsAdapter.Resi | |||
|         this.listener = listener; | ||||
|     } | ||||
| 
 | ||||
|     public void updateData(List<Resident> newResidents) { | ||||
|         this.residents = newResidents; // O la lista que uses internamente | ||||
|         notifyDataSetChanged(); | ||||
|     } | ||||
| 
 | ||||
|     @NonNull | ||||
|     @Override | ||||
|     public ResidentsViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) { | ||||
|  | @ -35,6 +42,21 @@ public class ResidentsAdapter extends RecyclerView.Adapter<ResidentsAdapter.Resi | |||
|         Resident resident = residents.get(position); | ||||
|         holder.bindResident(resident); | ||||
| 
 | ||||
|         holder.optionsButton.setOnClickListener(v -> { | ||||
|             PopupMenu popupMenu = new PopupMenu(v.getContext(), v); | ||||
|             popupMenu.getMenuInflater().inflate(R.menu.menu_options, popupMenu.getMenu()); | ||||
| 
 | ||||
|             popupMenu.setOnMenuItemClickListener(item -> { | ||||
|                 if (item.getItemId() == R.id.action_delete) { | ||||
|                     listener.onTakeOutResident(resident); | ||||
|                     return true; | ||||
|                 } | ||||
|                 return false; | ||||
|             }); | ||||
| 
 | ||||
|             popupMenu.show(); | ||||
|         }); | ||||
| 
 | ||||
|         holder.itemView.setOnClickListener(new View.OnClickListener() { | ||||
|             @Override | ||||
|             public void onClick(View view) { | ||||
|  | @ -50,11 +72,13 @@ public class ResidentsAdapter extends RecyclerView.Adapter<ResidentsAdapter.Resi | |||
|     public class ResidentsViewHolder extends RecyclerView.ViewHolder { | ||||
| 
 | ||||
|         private final TextView residentName; | ||||
|         private final ImageButton optionsButton; | ||||
| 
 | ||||
| 
 | ||||
|         public ResidentsViewHolder(View view) { | ||||
|             super(view); | ||||
|             residentName = view.findViewById(R.id.name_list_item); | ||||
|             optionsButton = view.findViewById(R.id.more_options_item_list); | ||||
|         } | ||||
| 
 | ||||
|         public void bindResident(Resident resident) { | ||||
|  |  | |||
|  | @ -1,18 +1,18 @@ | |||
| package com.andresgmoran.apptrabajadores.models.gameStats; | ||||
| 
 | ||||
| public enum Difficulty { | ||||
|     FACIL, | ||||
|     MEDIO, | ||||
|     DIFICIL; | ||||
|     DIFICULTAD1, | ||||
|     DIFICULTAD2, | ||||
|     DIFICULTAD3; | ||||
| 
 | ||||
|     public static Difficulty fromString(String value) { | ||||
|         switch (value.toLowerCase()) { | ||||
|             case "facil": | ||||
|                 return FACIL; | ||||
|             case "medio": | ||||
|                 return MEDIO; | ||||
|             case "dificil": | ||||
|                 return DIFICIL; | ||||
|             case "dificultad1": | ||||
|                 return DIFICULTAD1; | ||||
|             case "dificultad2": | ||||
|                 return DIFICULTAD2; | ||||
|             case "dificultad3": | ||||
|                 return DIFICULTAD3; | ||||
|             default: | ||||
|                 throw new IllegalArgumentException("Valor de dificultad no válido: " + value); | ||||
|         } | ||||
|  |  | |||
|  | @ -13,8 +13,9 @@ public class GameStat implements Serializable { | |||
|     private final Double duration; | ||||
|     private final Difficulty difficulty; | ||||
|     private final LocalDateTime dateTime; | ||||
|     private final String observation; | ||||
| 
 | ||||
|     public GameStat(Long id, Long residentId, Long gameId, Long userId, Integer num, Double duration, Difficulty difficulty, LocalDateTime dateTime) { | ||||
|     public GameStat(Long id, Long residentId, Long gameId, Long userId, Integer num, Double duration, Difficulty difficulty, LocalDateTime dateTime, String observation) { | ||||
|         this.id = id; | ||||
|         this.residentId = residentId; | ||||
|         this.gameId = gameId; | ||||
|  | @ -23,6 +24,7 @@ public class GameStat implements Serializable { | |||
|         this.duration = duration; | ||||
|         this.difficulty = difficulty; | ||||
|         this.dateTime = dateTime; | ||||
|         this.observation = observation; | ||||
|     } | ||||
| 
 | ||||
|     public Long getId() { | ||||
|  | @ -57,15 +59,19 @@ public class GameStat implements Serializable { | |||
|         return dateTime; | ||||
|     } | ||||
| 
 | ||||
|     public String getObservation() { | ||||
|         return observation; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public boolean equals(Object o) { | ||||
|         if (o == null || getClass() != o.getClass()) return false; | ||||
|         GameStat gameStat = (GameStat) o; | ||||
|         return Objects.equals(id, gameStat.id) && Objects.equals(residentId, gameStat.residentId) && Objects.equals(gameId, gameStat.gameId) && Objects.equals(userId, gameStat.userId) && Objects.equals(num, gameStat.num) && Objects.equals(duration, gameStat.duration) && difficulty == gameStat.difficulty && Objects.equals(dateTime, gameStat.dateTime); | ||||
|         return Objects.equals(id, gameStat.id) && Objects.equals(residentId, gameStat.residentId) && Objects.equals(gameId, gameStat.gameId) && Objects.equals(userId, gameStat.userId) && Objects.equals(num, gameStat.num) && Objects.equals(duration, gameStat.duration) && difficulty == gameStat.difficulty && Objects.equals(dateTime, gameStat.dateTime) && Objects.equals(observation, gameStat.observation); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public int hashCode() { | ||||
|         return Objects.hash(id, residentId, gameId, userId, num, duration, difficulty, dateTime); | ||||
|         return Objects.hash(id, residentId, gameId, userId, num, duration, difficulty, dateTime, observation); | ||||
|     } | ||||
| } | ||||
|  |  | |||
|  | @ -0,0 +1,63 @@ | |||
| package com.andresgmoran.apptrabajadores.models.parsers; | ||||
| 
 | ||||
| import com.andresgmoran.apptrabajadores.exceptions.ParserException; | ||||
| import com.andresgmoran.apptrabajadores.models.Activity; | ||||
| import com.andresgmoran.apptrabajadores.models.ActivityResident; | ||||
| import com.andresgmoran.apptrabajadores.models.ActivityState; | ||||
| 
 | ||||
| import org.json.JSONArray; | ||||
| import org.json.JSONException; | ||||
| import org.json.JSONObject; | ||||
| 
 | ||||
| import java.time.LocalDate; | ||||
| import java.time.LocalDateTime; | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| 
 | ||||
| public class ActivityParser { | ||||
| 
 | ||||
|     public static List<Activity> parseActivities(String jsonText) throws ParserException { | ||||
|         List<Activity> activities = new ArrayList<>(); | ||||
| 
 | ||||
|         try { | ||||
|             JSONArray array = new JSONArray(jsonText); | ||||
| 
 | ||||
|             for (int i = 0; i < array.length(); i++) { | ||||
|                 JSONObject obj = array.getJSONObject(i); | ||||
| 
 | ||||
|                 Long id = obj.getLong("id"); | ||||
|                 String name = obj.getString("nombre"); | ||||
|                 String description = obj.getString("descripcion"); | ||||
| 
 | ||||
|                 // Asegúrate de que la fecha tenga formato "yyyy-MM-dd" | ||||
|                 String fechaStr = obj.getString("fechaInicio").trim(); | ||||
|                 LocalDateTime date = LocalDateTime.parse(fechaStr); | ||||
| 
 | ||||
|                 // Validar estado | ||||
|                 ActivityState state = null; | ||||
|                 if (obj.has("estado") && !obj.isNull("estado")) { | ||||
|                     String estadoStr = obj.getString("estado").trim(); | ||||
|                     if (!estadoStr.isEmpty()) { | ||||
|                         state = ActivityState.fromString(estadoStr); | ||||
|                     } | ||||
|                 } | ||||
| 
 | ||||
|                 Long residenceId = obj.getLong("idResidencia"); | ||||
| 
 | ||||
|                 JSONArray jsonResidentsArray = obj.getJSONArray("participantes"); | ||||
|                 List<Long> residentIds = new ArrayList<>(); | ||||
| 
 | ||||
|                 for (int j = 0; j < jsonResidentsArray.length(); j++) { | ||||
|                     residentIds.add(jsonResidentsArray.getLong(j)); | ||||
|                 } | ||||
| 
 | ||||
|                 activities.add(new Activity(id, name, description, date, state, residentIds, residenceId)); | ||||
|             } | ||||
| 
 | ||||
|         } catch (JSONException | IllegalArgumentException e) { | ||||
|             throw new ParserException("Error al parsear actividades: " + e.getMessage(), e); | ||||
|         } | ||||
| 
 | ||||
|         return activities; | ||||
|     } | ||||
| } | ||||
|  | @ -0,0 +1,45 @@ | |||
| package com.andresgmoran.apptrabajadores.models.parsers; | ||||
| 
 | ||||
| import com.andresgmoran.apptrabajadores.exceptions.ParserException; | ||||
| import com.andresgmoran.apptrabajadores.models.ActivityResident; | ||||
| 
 | ||||
| import org.json.JSONArray; | ||||
| import org.json.JSONException; | ||||
| import org.json.JSONObject; | ||||
| 
 | ||||
| import java.time.LocalDate; | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| 
 | ||||
| public class ActivityResidentParser { | ||||
| 
 | ||||
|     public static List<ActivityResident> parseActivityResidents(String jsonText) throws ParserException{ | ||||
|         List<ActivityResident> activityResidents = new ArrayList<>(); | ||||
| 
 | ||||
|         try { | ||||
|             JSONArray jsonArray = new JSONArray(jsonText); | ||||
| 
 | ||||
|             for (int i = 0; i < jsonArray.length(); i++) { | ||||
|                 JSONObject jsonResident = jsonArray.getJSONObject(i); | ||||
| 
 | ||||
|                 Long id = jsonResident.getLong("id"); | ||||
|                 Long activityId = jsonResident.getLong("idEvento"); | ||||
|                 Long residentId = jsonResident.getLong("idResidente"); | ||||
|                 boolean humanHelp = jsonResident.getBoolean("recursosHumanos"); | ||||
|                 boolean materialHelp = jsonResident.getBoolean("recursosMateriales"); | ||||
|                 boolean assistance = jsonResident.getBoolean("asistenciaPermitida"); | ||||
|                 String preOpinion = jsonResident.optString("preOpinion", ""); | ||||
|                 String postOpinion = jsonResident.optString("postOpinion", ""); | ||||
| 
 | ||||
|                 ActivityResident activityResident = new ActivityResident(id, activityId, residentId, assistance, humanHelp, materialHelp, preOpinion, postOpinion); | ||||
| 
 | ||||
|                 activityResidents.add(activityResident); | ||||
|             } | ||||
| 
 | ||||
|         } catch (JSONException | IllegalArgumentException e) { | ||||
|             throw new ParserException("Error al parsear residentes de actividades", e); | ||||
|         } | ||||
| 
 | ||||
|         return activityResidents; | ||||
|     } | ||||
| } | ||||
|  | @ -1,5 +1,6 @@ | |||
| package com.andresgmoran.apptrabajadores.models.parsers; | ||||
| 
 | ||||
| import com.andresgmoran.apptrabajadores.exceptions.ParserException; | ||||
| import com.andresgmoran.apptrabajadores.models.Game; | ||||
| 
 | ||||
| import org.json.JSONArray; | ||||
|  | @ -11,7 +12,7 @@ import java.util.List; | |||
| 
 | ||||
| public class GameParser { | ||||
| 
 | ||||
|     public static List<Game> parseGames(String jsonText) { | ||||
|     public static List<Game> parseGames(String jsonText) throws ParserException{ | ||||
|         List<Game> games = new ArrayList<>(); | ||||
| 
 | ||||
|         try { | ||||
|  | @ -22,14 +23,13 @@ public class GameParser { | |||
| 
 | ||||
|                 Long id = jsonGame.getLong("id"); | ||||
|                 String name = jsonGame.getString("nombre"); | ||||
|                 Long residenceId = jsonGame.getLong("idResidencia"); | ||||
| 
 | ||||
|                 Game game = new Game(id, name, residenceId); | ||||
|                 Game game = new Game(id, name); | ||||
|                 games.add(game); | ||||
|             } | ||||
| 
 | ||||
|         } catch (JSONException e) { | ||||
|             e.printStackTrace(); //TODO: Throw personalized exception | ||||
|         } catch (JSONException | IllegalArgumentException e) { | ||||
|             throw new ParserException("Error al parsear juegos", e); | ||||
|         } | ||||
| 
 | ||||
|         return games; | ||||
|  |  | |||
|  | @ -1,6 +1,7 @@ | |||
| package com.andresgmoran.apptrabajadores.models.parsers; | ||||
| 
 | ||||
| 
 | ||||
| import com.andresgmoran.apptrabajadores.exceptions.ParserException; | ||||
| import com.andresgmoran.apptrabajadores.models.gameStats.Difficulty; | ||||
| import com.andresgmoran.apptrabajadores.models.gameStats.GameStat; | ||||
| 
 | ||||
|  | @ -15,7 +16,7 @@ import java.util.List; | |||
| 
 | ||||
| public class GameStatParser { | ||||
| 
 | ||||
|     public static List<GameStat> parseStats(String jsonText) { | ||||
|     public static List<GameStat> parseStats(String jsonText) throws ParserException { | ||||
|         List<GameStat> stats = new ArrayList<>(); | ||||
| 
 | ||||
|         try { | ||||
|  | @ -27,7 +28,7 @@ public class GameStatParser { | |||
|                 Long id = obj.getLong("id"); | ||||
|                 Long residentId = obj.getLong("idResidente"); | ||||
|                 Long gameId = obj.getLong("idJuego"); | ||||
|                 Long userId = obj.getLong("idUsario"); | ||||
|                 Long userId = obj.getLong("idUsuario"); | ||||
|                 Integer num = obj.getInt("num"); | ||||
|                 Double duration = obj.getDouble("duracion"); | ||||
|                 Difficulty difficulty = null; | ||||
|  | @ -41,12 +42,13 @@ public class GameStatParser { | |||
|                         obj.getString("fecha"), | ||||
|                         DateTimeFormatter.ISO_DATE_TIME // ejemplo: "2025-04-18T22:31:00" | ||||
|                 ); | ||||
|                 String observation = obj.getString("observacion"); | ||||
| 
 | ||||
|                 stats.add(new GameStat(id, residentId, gameId, userId, num, duration, difficulty, dateTime)); | ||||
|                 stats.add(new GameStat(id, residentId, gameId, userId, num, duration, difficulty, dateTime, observation)); | ||||
|             } | ||||
| 
 | ||||
|         } catch (JSONException e) { | ||||
|             e.printStackTrace(); | ||||
|         } catch (JSONException | IllegalArgumentException e) { | ||||
|             throw new ParserException("Error al parsear partidas", e); | ||||
|         } | ||||
| 
 | ||||
|         return stats; | ||||
|  |  | |||
|  | @ -1,8 +1,10 @@ | |||
| package com.andresgmoran.apptrabajadores.models.parsers; | ||||
| 
 | ||||
| import com.andresgmoran.apptrabajadores.exceptions.ParserException; | ||||
| import com.andresgmoran.apptrabajadores.models.Resident; | ||||
| 
 | ||||
| import org.json.JSONArray; | ||||
| import org.json.JSONException; | ||||
| import org.json.JSONObject; | ||||
| 
 | ||||
| import java.time.LocalDate; | ||||
|  | @ -11,7 +13,7 @@ import java.util.List; | |||
| 
 | ||||
| public class ResidentParser { | ||||
| 
 | ||||
|     public static List<Resident> parseResidents(String jsonText) { | ||||
|     public static List<Resident> parseResidents(String jsonText) throws ParserException { | ||||
|         List<Resident> residents = new ArrayList<>(); | ||||
| 
 | ||||
|         try { | ||||
|  | @ -28,12 +30,14 @@ public class ResidentParser { | |||
|                 String birthDateStr = obj.getString("fechaNacimiento"); | ||||
|                 LocalDate birthDate = LocalDate.parse(birthDateStr); // Usa formato ISO (yyyy-MM-dd) | ||||
| 
 | ||||
|                 String identityCard = obj.getString("documentoIdentidad"); | ||||
| 
 | ||||
|                 Long residenceId = obj.getLong("idResidencia"); | ||||
| 
 | ||||
|                 residents.add(new Resident(id, name, surnames, birthDate, residenceId)); | ||||
|                 residents.add(new Resident(id, name, surnames, birthDate, identityCard, residenceId)); | ||||
|             } | ||||
|         } catch (Exception e) { | ||||
|             e.printStackTrace(); //TODO: Throw personalized exception | ||||
|         } catch (JSONException | IllegalArgumentException e) { | ||||
|             throw new ParserException("Error al parsear residentes", e); | ||||
|         } | ||||
| 
 | ||||
|         return residents; | ||||
|  |  | |||
|  | @ -1,8 +1,10 @@ | |||
| package com.andresgmoran.apptrabajadores.models.parsers; | ||||
| 
 | ||||
| import com.andresgmoran.apptrabajadores.exceptions.ParserException; | ||||
| import com.andresgmoran.apptrabajadores.models.User; | ||||
| 
 | ||||
| import org.json.JSONArray; | ||||
| import org.json.JSONException; | ||||
| import org.json.JSONObject; | ||||
| 
 | ||||
| import java.util.ArrayList; | ||||
|  | @ -10,7 +12,7 @@ import java.util.List; | |||
| 
 | ||||
| public class UserParser { | ||||
| 
 | ||||
|     public static List<User> parseUsers(String jsonText) { | ||||
|     public static List<User> parseUsers(String jsonText) throws ParserException { | ||||
|         List<User> usuarios = new ArrayList<>(); | ||||
| 
 | ||||
|         try { | ||||
|  | @ -25,14 +27,36 @@ public class UserParser { | |||
|                 String email = obj.getString("email"); | ||||
|                 boolean enabled = obj.getBoolean("enabled"); | ||||
|                 Long residentId = obj.has("idResidencia") && !obj.isNull("residentId") ? obj.getLong("residentId") : null; | ||||
|                 String accountImage = obj.getString("fotoPerfil"); | ||||
|                 boolean takenOut = obj.getBoolean("baja"); | ||||
| 
 | ||||
|                 usuarios.add(new User(id, name, surnames, email, enabled, residentId)); | ||||
|                 usuarios.add(new User(id, name, surnames, email, enabled, residentId, accountImage, takenOut)); | ||||
|             } | ||||
|         } catch (Exception e) { | ||||
|             e.printStackTrace(); //TODO: Throw personalized exception | ||||
|         } catch (JSONException | IllegalArgumentException e) { | ||||
|             throw new ParserException("Error al parsear usuarios", e); | ||||
|         } | ||||
| 
 | ||||
|         return usuarios; | ||||
|     } | ||||
| 
 | ||||
|     public static User parseUser(String jsonText) throws ParserException { | ||||
|         try { | ||||
|             JSONObject obj = new JSONObject(jsonText); | ||||
| 
 | ||||
|             Long id = obj.getLong("id"); | ||||
|             String name = obj.getString("nombre"); | ||||
|             String surnames = obj.getString("apellido"); | ||||
|             String email = obj.getString("email"); | ||||
|             boolean enabled = obj.getBoolean("enabled"); | ||||
|             Long residentId = obj.has("residentId") && !obj.isNull("residentId") ? obj.getLong("residentId") : null; | ||||
|             String accountImage = obj.getString("fotoPerfil"); | ||||
|             boolean takenOut = obj.getBoolean("baja"); | ||||
| 
 | ||||
|             return new User(id, name, surnames, email, enabled, residentId, accountImage, takenOut); | ||||
|         } catch (JSONException | IllegalArgumentException e) { | ||||
|             throw new ParserException("Error al parsear el usuario", e); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
| 
 | ||||
| } | ||||
|  |  | |||
|  | @ -1,39 +1,109 @@ | |||
| package com.andresgmoran.apptrabajadores.network; | ||||
| 
 | ||||
| import android.content.Context; | ||||
| import android.content.SharedPreferences; | ||||
| import android.graphics.Bitmap; | ||||
| import android.graphics.BitmapFactory; | ||||
| import android.os.Handler; | ||||
| import android.os.Looper; | ||||
| import android.util.Log; | ||||
| 
 | ||||
| import androidx.security.crypto.EncryptedSharedPreferences; | ||||
| import androidx.security.crypto.MasterKey; | ||||
| 
 | ||||
| import com.andresgmoran.apptrabajadores.models.ActivityState; | ||||
| import com.andresgmoran.apptrabajadores.models.User; | ||||
| import com.andresgmoran.apptrabajadores.utils.SecurePreferencesUtil; | ||||
| 
 | ||||
| import java.io.BufferedReader; | ||||
| import java.io.InputStream; | ||||
| import java.io.InputStreamReader; | ||||
| import java.net.HttpURLConnection; | ||||
| import java.net.URL; | ||||
| import java.time.LocalDate; | ||||
| import java.time.LocalDateTime; | ||||
| 
 | ||||
| public class ApiClient { | ||||
| 
 | ||||
|     private static final String BASE_URL = "http://10.0.2.2:8080/resi/"; | ||||
|     private static final String BASE_URL = "http://10.0.2.2:8080"; | ||||
| 
 | ||||
|     public interface RawCallback { | ||||
|         void onSuccess(String jsonText); | ||||
|         void onError(String error); | ||||
|     } | ||||
| 
 | ||||
|     public static void getUsers(RawCallback callback) { | ||||
|         makeGetRequest("users", callback); | ||||
|     public interface ImageCallback { | ||||
|         void onSuccess(Bitmap bitmap); | ||||
|         void onError(String error); | ||||
|     } | ||||
| 
 | ||||
|     public static void getResidents(RawCallback callback) { | ||||
|         makeGetRequest("residents", callback); | ||||
| 
 | ||||
|     /** | ||||
|      * | ||||
|      * @param context | ||||
|      * @param callback | ||||
|      */ | ||||
|     public static void getUsers(Context context, RawCallback callback) { | ||||
|         long idResidence = SecurePreferencesUtil.getLong(context, "idResidence", -1); | ||||
|         makeGetRequest(context, "/resi/user/getAll", callback); | ||||
|     } | ||||
| 
 | ||||
|     public static void getGames(RawCallback callback) { | ||||
|         makeGetRequest("juegos", callback); | ||||
|     } | ||||
|     public static void getGamesStats(RawCallback callback) { | ||||
|         makeGetRequest("juegos/stats", callback); | ||||
|     public static void getActualUser(Context context, RawCallback callback) { | ||||
|         makeGetRequest(context, "/resi/user/me", callback); | ||||
|     } | ||||
| 
 | ||||
|     private static void makeGetRequest(String endpoint, RawCallback callback) { | ||||
|     /** | ||||
|      * | ||||
|      * @param context | ||||
|      * @param callback | ||||
|      */ | ||||
|     public static void getResidents(Context context, RawCallback callback) { | ||||
|         long idResidence = SecurePreferencesUtil.getLong(context, "idResidence", -1); | ||||
|         makeGetRequest(context, "/resi/resident/getAll", callback); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * | ||||
|      * @param context | ||||
|      * @param callback | ||||
|      */ | ||||
|     public static void getGames(Context context, RawCallback callback) { | ||||
|         long idResidence = SecurePreferencesUtil.getLong(context, "idResidence", -1); | ||||
|         makeGetRequest(context, "/resi/juego/getAll", callback); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * | ||||
|      * @param context | ||||
|      * @param callback | ||||
|      */ | ||||
|     public static void getGamesStats(Context context, RawCallback callback) { | ||||
|         long idResidence = SecurePreferencesUtil.getLong(context, "idResidence", -1); | ||||
|         makeGetRequest(context, "/resi/registro/getAll", callback); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * | ||||
|      * @param context | ||||
|      * @param callback | ||||
|      */ | ||||
|     public static void getAllActivities(Context context, RawCallback callback) { | ||||
|         long idResidence = SecurePreferencesUtil.getLong(context, "idResidence", -1); | ||||
|         makeGetRequest(context, "/resi/evento/getAll", callback); | ||||
|     } | ||||
| 
 | ||||
|     public static void getAllParticipants(Context context, Long idActivity, RawCallback callback) { | ||||
|         long idResidence = SecurePreferencesUtil.getLong(context, "idResidence", -1); | ||||
|         makeGetRequest(context, "/resi/evento/" + idActivity + "/participante/getAll", callback); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * | ||||
|      * @param context | ||||
|      * @param endpoint | ||||
|      * @param callback | ||||
|      */ | ||||
|     private static void makeGetRequest(Context context, String endpoint, RawCallback callback) { | ||||
|         new Thread(() -> { | ||||
|             HttpURLConnection connection = null; | ||||
|             BufferedReader reader = null; | ||||
|  | @ -43,6 +113,25 @@ public class ApiClient { | |||
|                 connection = (HttpURLConnection) url.openConnection(); | ||||
|                 connection.setRequestMethod("GET"); | ||||
|                 connection.setRequestProperty("User-Agent", "Android"); | ||||
| 
 | ||||
|                 // Obtener el token desde EncryptedSharedPreferences | ||||
|                 MasterKey masterKey = new MasterKey.Builder(context) | ||||
|                         .setKeyScheme(MasterKey.KeyScheme.AES256_GCM) | ||||
|                         .build(); | ||||
| 
 | ||||
|                 SharedPreferences encryptedPrefs = EncryptedSharedPreferences.create( | ||||
|                         context, | ||||
|                         "secure_auth", | ||||
|                         masterKey, | ||||
|                         EncryptedSharedPreferences.PrefKeyEncryptionScheme.AES256_SIV, | ||||
|                         EncryptedSharedPreferences.PrefValueEncryptionScheme.AES256_GCM | ||||
|                 ); | ||||
| 
 | ||||
|                 String token = encryptedPrefs.getString("token", null); | ||||
|                 if (token != null) { | ||||
|                     connection.setRequestProperty("Authorization", "Bearer " + token); | ||||
|                 } | ||||
| 
 | ||||
|                 connection.setConnectTimeout(5000); | ||||
|                 connection.setReadTimeout(5000); | ||||
| 
 | ||||
|  | @ -56,7 +145,6 @@ public class ApiClient { | |||
|                         json.append(line); | ||||
|                     } | ||||
| 
 | ||||
|                     // Devolver el JSON plano al hilo principal | ||||
|                     new Handler(Looper.getMainLooper()).post(() -> | ||||
|                             callback.onSuccess(json.toString())); | ||||
|                 } else { | ||||
|  | @ -76,5 +164,344 @@ public class ApiClient { | |||
|             } | ||||
|         }).start(); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * | ||||
|      * @param email | ||||
|      * @param password | ||||
|      * @param callback | ||||
|      */ | ||||
|     public static void postLogin(String email, String password, RawCallback callback) { | ||||
|         String jsonBody = String.format("{\"email\":\"%s\", \"password\":\"%s\"}", email, password); | ||||
|         makePostRequest("/auth/login", jsonBody, callback); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * | ||||
|      * @param context | ||||
|      * @param nombre | ||||
|      * @param descripcion | ||||
|      * @param fecha | ||||
|      * @param callback | ||||
|      */ | ||||
|     public static void postEvento(Context context, String nombre, String descripcion, LocalDateTime fecha, RawCallback callback) { | ||||
|         String jsonBody = String.format("{\"nombre\":\"%s\", \"descripcion\":\"%s\", \"fecha\":\"%s\", \"estado\":\"%s\"}", nombre, descripcion, fecha.toString(), ActivityState.ABIERTO); | ||||
|         makePostRequest(context,"/resi/evento/add", jsonBody, callback); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * | ||||
|      * @param context | ||||
|      * @param residentId | ||||
|      * @param asistencia | ||||
|      * @param opinionPre | ||||
|      * @param opinionPost | ||||
|      * @param idActivity | ||||
|      * @param callback | ||||
|      */ | ||||
|     public static void postParticipant( Context context, Long residentId, boolean humanHelp, boolean materialHelp, String opinionPre, String opinionPost, Long idActivity, RawCallback callback) { | ||||
|         String jsonBody = String.format("{\"idResidente\":%d, \"recursosHumanos\":%b, \"recursosMateriales\":%b, \"preOpinion\":\"%s\", \"postOpinion\":\"%s\"}", residentId, humanHelp, materialHelp, opinionPre, opinionPost);        long idResidence = SecurePreferencesUtil.getLong(context, "idResidence", -1); | ||||
|         makePostRequest(context,"/resi/evento/" + idActivity + "/participante/add", jsonBody, callback); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * | ||||
|      * @param endpoint | ||||
|      * @param jsonBody | ||||
|      * @param callback | ||||
|      */ | ||||
|     public static void makePostRequest(String endpoint, String jsonBody, RawCallback callback) { | ||||
|         new Thread(() -> { | ||||
|             HttpURLConnection connection = null; | ||||
| 
 | ||||
|             try { | ||||
|                 URL url = new URL(BASE_URL + endpoint); | ||||
|                 connection = (HttpURLConnection) url.openConnection(); | ||||
|                 connection.setRequestMethod("POST"); | ||||
|                 connection.setRequestProperty("Content-Type", "application/json; utf-8"); | ||||
|                 connection.setRequestProperty("Accept", "application/json"); | ||||
|                 connection.setRequestProperty("User-Agent", "Android"); | ||||
|                 connection.setDoOutput(true); | ||||
|                 connection.setConnectTimeout(5000); | ||||
|                 connection.setReadTimeout(5000); | ||||
| 
 | ||||
|                 try (java.io.OutputStream os = connection.getOutputStream()) { | ||||
|                     byte[] input = jsonBody.getBytes("utf-8"); | ||||
|                     os.write(input, 0, input.length); | ||||
|                 } | ||||
| 
 | ||||
|                 int responseCode = connection.getResponseCode(); | ||||
|                 java.io.InputStream inputStream = (responseCode >= 200 && responseCode < 300) | ||||
|                         ? connection.getInputStream() | ||||
|                         : connection.getErrorStream(); | ||||
| 
 | ||||
|                 BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream, "utf-8")); | ||||
|                 StringBuilder response = new StringBuilder(); | ||||
|                 String line; | ||||
| 
 | ||||
|                 while ((line = reader.readLine()) != null) { | ||||
|                     response.append(line.trim()); | ||||
|                 } | ||||
| 
 | ||||
|                 final String result = response.toString(); | ||||
|                 new Handler(Looper.getMainLooper()).post(() -> { | ||||
|                     if (responseCode >= 200 && responseCode < 300) { | ||||
|                         callback.onSuccess(result); | ||||
|                     } else { | ||||
|                         callback.onError("Error (" + responseCode + "): " + result); | ||||
|                     } | ||||
|                 }); | ||||
| 
 | ||||
|             } catch (Exception e) { | ||||
|                 new Handler(Looper.getMainLooper()).post(() -> | ||||
|                         callback.onError("Excepción: " + e.getMessage())); | ||||
|             } finally { | ||||
|                 if (connection != null) connection.disconnect(); | ||||
|             } | ||||
|         }).start(); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * | ||||
|      * @param context | ||||
|      * @param endpoint | ||||
|      * @param jsonBody | ||||
|      * @param callback | ||||
|      */ | ||||
|     public static void makePostRequest(Context context, String endpoint, String jsonBody, RawCallback callback) { | ||||
|         new Thread(() -> { | ||||
|             HttpURLConnection connection = null; | ||||
| 
 | ||||
|             try { | ||||
|                 URL url = new URL(BASE_URL + endpoint); | ||||
|                 connection = (HttpURLConnection) url.openConnection(); | ||||
|                 connection.setRequestMethod("POST"); | ||||
|                 connection.setRequestProperty("Content-Type", "application/json; utf-8"); | ||||
|                 connection.setRequestProperty("Accept", "application/json"); | ||||
|                 connection.setRequestProperty("User-Agent", "Android"); | ||||
|                 connection.setDoOutput(true); | ||||
|                 connection.setConnectTimeout(5000); | ||||
|                 connection.setReadTimeout(5000); | ||||
| 
 | ||||
|                 String token = SecurePreferencesUtil.getString(context, "token", null); | ||||
|                 if (token != null) { | ||||
|                     connection.setRequestProperty("Authorization", "Bearer " + token); | ||||
|                 } | ||||
| 
 | ||||
|                 try (java.io.OutputStream os = connection.getOutputStream()) { | ||||
|                     byte[] input = jsonBody.getBytes("utf-8"); | ||||
|                     os.write(input, 0, input.length); | ||||
|                 } | ||||
| 
 | ||||
|                 int responseCode = connection.getResponseCode(); | ||||
|                 java.io.InputStream inputStream = (responseCode >= 200 && responseCode < 300) | ||||
|                         ? connection.getInputStream() | ||||
|                         : connection.getErrorStream(); | ||||
| 
 | ||||
|                 BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream, "utf-8")); | ||||
|                 StringBuilder response = new StringBuilder(); | ||||
|                 String line; | ||||
| 
 | ||||
|                 while ((line = reader.readLine()) != null) { | ||||
|                     response.append(line.trim()); | ||||
|                 } | ||||
| 
 | ||||
|                 final String result = response.toString(); | ||||
|                 new Handler(Looper.getMainLooper()).post(() -> { | ||||
|                     if (responseCode >= 200 && responseCode < 300) { | ||||
|                         callback.onSuccess(result); | ||||
|                     } else { | ||||
|                         callback.onError("Error (" + responseCode + "): " + result); | ||||
|                     } | ||||
|                 }); | ||||
| 
 | ||||
|             } catch (Exception e) { | ||||
|                 new Handler(Looper.getMainLooper()).post(() -> | ||||
|                         callback.onError("Excepción: " + e.getMessage())); | ||||
|             } finally { | ||||
|                 if (connection != null) connection.disconnect(); | ||||
|             } | ||||
|         }).start(); | ||||
|     } | ||||
| 
 | ||||
|     public static void patchObservation(Context context, String observation, long idGameStat, RawCallback callback) { | ||||
|         String jsonBody = String.format("{\"observacion\":\"%s\"}", observation); | ||||
|         long idResidence = SecurePreferencesUtil.getLong(context, "idResidence", -1); | ||||
|         makePatchRequest(context,"/resi/registro/" + idGameStat + "/addComment", jsonBody, callback); | ||||
|     } | ||||
| 
 | ||||
|     public static void patchActivityState(Context context, long idActivity, ActivityState state, RawCallback callback) { | ||||
|         String jsonBody = String.format("{\"estado\":\"%s\"}", state); | ||||
|         makePatchRequest(context,"/resi/evento/" + idActivity + "/update", jsonBody, callback); | ||||
|     } | ||||
| 
 | ||||
|     public static void patchParticipant( Context context, long idActivity, long idParticipant, String jsonBody, RawCallback callback) { | ||||
|         makePatchRequest(context,"/resi/evento/" + idActivity + "/participante/" + idParticipant + "/update", jsonBody, callback); | ||||
|     } | ||||
| 
 | ||||
|     public static void patchTakeOutResident(Context context,long idResident, RawCallback callback) { | ||||
|         makePatchRequest(context,"/resi/resident/" + idResident + "/baja", "", callback); | ||||
|     } | ||||
| 
 | ||||
|     public static void makePatchRequest(Context context, String endpoint, String jsonBody, RawCallback callback) { | ||||
|         new Thread(() -> { | ||||
|             HttpURLConnection connection = null; | ||||
| 
 | ||||
|             try { | ||||
|                 URL url = new URL(BASE_URL + endpoint); | ||||
|                 connection = (HttpURLConnection) url.openConnection(); | ||||
|                 connection.setRequestMethod("PATCH"); | ||||
|                 connection.setRequestProperty("Content-Type", "application/json; utf-8"); | ||||
|                 connection.setRequestProperty("Accept", "application/json"); | ||||
|                 connection.setRequestProperty("User-Agent", "Android"); | ||||
|                 connection.setDoOutput(true); | ||||
|                 connection.setConnectTimeout(5000); | ||||
|                 connection.setReadTimeout(5000); | ||||
| 
 | ||||
|                 String token = SecurePreferencesUtil.getString(context, "token", null); | ||||
|                 if (token != null) { | ||||
|                     connection.setRequestProperty("Authorization", "Bearer " + token); | ||||
|                 } | ||||
| 
 | ||||
|                 try (java.io.OutputStream os = connection.getOutputStream()) { | ||||
|                     byte[] input = jsonBody.getBytes("utf-8"); | ||||
|                     os.write(input, 0, input.length); | ||||
|                 } | ||||
| 
 | ||||
|                 int responseCode = connection.getResponseCode(); | ||||
|                 java.io.InputStream inputStream = (responseCode >= 200 && responseCode < 300) | ||||
|                         ? connection.getInputStream() | ||||
|                         : connection.getErrorStream(); | ||||
| 
 | ||||
|                 BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream, "utf-8")); | ||||
|                 StringBuilder response = new StringBuilder(); | ||||
|                 String line; | ||||
| 
 | ||||
|                 while ((line = reader.readLine()) != null) { | ||||
|                     response.append(line.trim()); | ||||
|                 } | ||||
| 
 | ||||
|                 final String result = response.toString(); | ||||
|                 new Handler(Looper.getMainLooper()).post(() -> { | ||||
|                     if (responseCode >= 200 && responseCode < 300) { | ||||
|                         callback.onSuccess(result); | ||||
|                     } else { | ||||
|                         callback.onError("Error (" + responseCode + "): " + result); | ||||
|                     } | ||||
|                 }); | ||||
| 
 | ||||
|             } catch (Exception e) { | ||||
|                 new Handler(Looper.getMainLooper()).post(() -> | ||||
|                         callback.onError("Excepción: " + e.getMessage())); | ||||
|             } finally { | ||||
|                 if (connection != null) connection.disconnect(); | ||||
|             } | ||||
|         }).start(); | ||||
|     } | ||||
| 
 | ||||
|     public static void deleteGameStat(Context context, long idGameStat, RawCallback callback) { | ||||
|         makeDeleteRequest(context,"/resi/registro/" + idGameStat + "/delete", callback); | ||||
|     } | ||||
| 
 | ||||
|     public static void deleteActivity(Context context, long idActivity, RawCallback callback) { | ||||
|         makeDeleteRequest(context,"/resi/evento/" + idActivity + "/delete", callback); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * | ||||
|      * @param context | ||||
|      * @param endpoint | ||||
|      * @param callback | ||||
|      */ | ||||
|     public static void makeDeleteRequest(Context context, String endpoint, RawCallback callback) { | ||||
|         new Thread(() -> { | ||||
|             HttpURLConnection connection = null; | ||||
| 
 | ||||
|             try { | ||||
|                 URL url = new URL(BASE_URL + endpoint); | ||||
|                 connection = (HttpURLConnection) url.openConnection(); | ||||
|                 connection.setRequestMethod("DELETE"); | ||||
|                 connection.setRequestProperty("Accept", "application/json"); | ||||
|                 connection.setRequestProperty("User-Agent", "Android"); | ||||
|                 connection.setConnectTimeout(5000); | ||||
|                 connection.setReadTimeout(5000); | ||||
| 
 | ||||
|                 // Añadir el token si está disponible | ||||
|                 String token = SecurePreferencesUtil.getString(context, "token", null); | ||||
|                 if (token != null) { | ||||
|                     connection.setRequestProperty("Authorization", "Bearer " + token); | ||||
|                 } | ||||
| 
 | ||||
|                 int responseCode = connection.getResponseCode(); | ||||
|                 java.io.InputStream inputStream = (responseCode >= 200 && responseCode < 300) | ||||
|                         ? connection.getInputStream() | ||||
|                         : connection.getErrorStream(); | ||||
| 
 | ||||
|                 BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream, "utf-8")); | ||||
|                 StringBuilder response = new StringBuilder(); | ||||
|                 String line; | ||||
| 
 | ||||
|                 while ((line = reader.readLine()) != null) { | ||||
|                     response.append(line.trim()); | ||||
|                 } | ||||
| 
 | ||||
|                 final String result = response.toString(); | ||||
|                 new Handler(Looper.getMainLooper()).post(() -> { | ||||
|                     if (responseCode >= 200 && responseCode < 300) { | ||||
|                         callback.onSuccess(result); | ||||
|                     } else { | ||||
|                         callback.onError("Error (" + responseCode + "): " + result); | ||||
|                     } | ||||
|                 }); | ||||
| 
 | ||||
|             } catch (Exception e) { | ||||
|                 new Handler(Looper.getMainLooper()).post(() -> | ||||
|                         callback.onError("Excepción: " + e.getMessage())); | ||||
|             } finally { | ||||
|                 if (connection != null) connection.disconnect(); | ||||
|             } | ||||
|         }).start(); | ||||
|     } | ||||
| 
 | ||||
|     public static void downloadImage(Context context, String imageUrl, ImageCallback callback) { | ||||
|         new Thread(() -> { | ||||
|             try { | ||||
|                 URL url = new URL(BASE_URL + imageUrl); | ||||
|                 HttpURLConnection connection = (HttpURLConnection) url.openConnection(); | ||||
|                 connection.setRequestMethod("GET"); | ||||
| 
 | ||||
|                 // Headers | ||||
|                 MasterKey masterKey = new MasterKey.Builder(context) | ||||
|                         .setKeyScheme(MasterKey.KeyScheme.AES256_GCM) | ||||
|                         .build(); | ||||
| 
 | ||||
|                 SharedPreferences encryptedPrefs = EncryptedSharedPreferences.create( | ||||
|                         context, | ||||
|                         "secure_auth", | ||||
|                         masterKey, | ||||
|                         EncryptedSharedPreferences.PrefKeyEncryptionScheme.AES256_SIV, | ||||
|                         EncryptedSharedPreferences.PrefValueEncryptionScheme.AES256_GCM | ||||
|                 ); | ||||
| 
 | ||||
|                 String token = encryptedPrefs.getString("token", null); | ||||
|                 if (token != null) { | ||||
|                     connection.setRequestProperty("Authorization", "Bearer " + token); | ||||
|                 } | ||||
| 
 | ||||
|                 InputStream inputStream = connection.getInputStream(); | ||||
|                 Bitmap bitmap = BitmapFactory.decodeStream(inputStream); | ||||
| 
 | ||||
|                 new Handler(Looper.getMainLooper()).post(() -> callback.onSuccess(bitmap)); | ||||
| 
 | ||||
|             } catch (Exception e) { | ||||
|                 new Handler(Looper.getMainLooper()).post(() -> callback.onError(e.getMessage())); | ||||
|             } | ||||
|         }).start(); | ||||
|     } | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| } | ||||
| 
 | ||||
|  |  | |||
|  | @ -16,7 +16,13 @@ import com.andresgmoran.apptrabajadores.interfaces.IOChangeFragmentListener; | |||
| 
 | ||||
| public class AccountFragment extends Fragment { | ||||
|     private Button editAccountButton; | ||||
|     private Button logOutButton; | ||||
|     private IOChangeFragmentListener listener; | ||||
|     private IOAccountFragmentListener accountFragmentListener; | ||||
| 
 | ||||
|     public interface IOAccountFragmentListener { | ||||
|         void onLogOutButtonClicked(); | ||||
|     } | ||||
| 
 | ||||
|     @Nullable | ||||
|     @Override | ||||
|  | @ -28,17 +34,27 @@ public class AccountFragment extends Fragment { | |||
|     public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) { | ||||
|         super.onViewCreated(view, savedInstanceState); | ||||
|         editAccountButton = view.findViewById(R.id.edit_account_button); | ||||
|         logOutButton = view.findViewById(R.id.btn_cerrar_sesion); | ||||
|         editAccountButton.setOnClickListener(new View.OnClickListener() { | ||||
|             @Override | ||||
|             public void onClick(View v) { | ||||
|                 listener.changeFragment("AccountFragment"); | ||||
|             } | ||||
|         }); | ||||
|         logOutButton.setOnClickListener(new View.OnClickListener() { | ||||
|             @Override | ||||
|             public void onClick(View v) { | ||||
|                 accountFragmentListener.onLogOutButtonClicked(); | ||||
|             } | ||||
|         }); | ||||
| 
 | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public void onAttach(@NonNull Context context) { | ||||
|         super.onAttach(context); | ||||
|         listener = (IOChangeFragmentListener) context; | ||||
|         accountFragmentListener = (IOAccountFragmentListener) context; | ||||
| 
 | ||||
|     } | ||||
| } | ||||
|  |  | |||
|  | @ -9,10 +9,35 @@ import android.view.ViewGroup; | |||
| import androidx.annotation.NonNull; | ||||
| import androidx.annotation.Nullable; | ||||
| import androidx.fragment.app.Fragment; | ||||
| import androidx.recyclerview.widget.LinearLayoutManager; | ||||
| import androidx.recyclerview.widget.RecyclerView; | ||||
| import androidx.swiperefreshlayout.widget.SwipeRefreshLayout; | ||||
| 
 | ||||
| import com.andresgmoran.apptrabajadores.R; | ||||
| import com.andresgmoran.apptrabajadores.interfaces.IOChangeFragmentListener; | ||||
| import com.andresgmoran.apptrabajadores.interfaces.IOClickOnActivityListener; | ||||
| import com.andresgmoran.apptrabajadores.interfaces.IOnChageStateActivityListener; | ||||
| import com.andresgmoran.apptrabajadores.models.Activity; | ||||
| import com.andresgmoran.apptrabajadores.models.ActivityResident; | ||||
| import com.andresgmoran.apptrabajadores.models.Resident; | ||||
| import com.andresgmoran.apptrabajadores.models.adapters.ActivitiesAdapter; | ||||
| import com.andresgmoran.apptrabajadores.ui.MainActivity; | ||||
| import com.andresgmoran.apptrabajadores.ui.fragments.residentList.ResidentsListFragment; | ||||
| import com.google.android.material.floatingactionbutton.FloatingActionButton; | ||||
| 
 | ||||
| import java.util.List; | ||||
| 
 | ||||
| public class ActivitiesFragment extends Fragment { | ||||
| 
 | ||||
|     public interface IOOnAttachListener{ | ||||
|         List<Activity> getActivities(); | ||||
|         List<ActivityResident> getParticipants(); | ||||
|     } | ||||
| 
 | ||||
|     private IOChangeFragmentListener changeFragmentListener; | ||||
| 
 | ||||
|     private List<Activity> activities; | ||||
|     private List<ActivityResident> participants; | ||||
|     @Nullable | ||||
|     @Override | ||||
|     public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) { | ||||
|  | @ -22,10 +47,34 @@ public class ActivitiesFragment extends Fragment { | |||
|     @Override | ||||
|     public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) { | ||||
|         super.onViewCreated(view, savedInstanceState); | ||||
| 
 | ||||
|         SwipeRefreshLayout swipeRefreshLayout = view.findViewById(R.id.swipe_refresh_layout_activities_list); | ||||
| 
 | ||||
|         swipeRefreshLayout.setOnRefreshListener(() -> { | ||||
|             ((MainActivity) requireActivity()).refreshActivitiesAndReload(); | ||||
|             swipeRefreshLayout.setRefreshing(false); | ||||
|         }); | ||||
| 
 | ||||
|         ActivitiesAdapter activitiesAdapter = new ActivitiesAdapter(requireContext(), activities, participants, (IOClickOnActivityListener) requireActivity(), (IOnChageStateActivityListener) requireActivity()); | ||||
|         RecyclerView activitiesRecyclerView = view.findViewById(R.id.activities_recycleView); | ||||
|         activitiesRecyclerView.setAdapter(activitiesAdapter); | ||||
|         activitiesRecyclerView.setHasFixedSize(true); | ||||
|         activitiesRecyclerView.setLayoutManager(new LinearLayoutManager(getActivity(), LinearLayoutManager.VERTICAL, false)); | ||||
| 
 | ||||
|         FloatingActionButton fab = view.findViewById(R.id.add_activity_button); | ||||
|         fab.setOnClickListener(v -> { | ||||
|             if (changeFragmentListener != null) { | ||||
|                 changeFragmentListener.changeFragment("ActivitiesFragment"); | ||||
|             } | ||||
|         }); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public void onAttach(@NonNull Context context) { | ||||
|         super.onAttach(context); | ||||
|         changeFragmentListener = (IOChangeFragmentListener) context; | ||||
|         ActivitiesFragment.IOOnAttachListener attachListener = (ActivitiesFragment.IOOnAttachListener) context; | ||||
|         activities = attachListener.getActivities(); | ||||
|         participants = attachListener.getParticipants(); | ||||
|     } | ||||
| } | ||||
|  |  | |||
|  | @ -0,0 +1,216 @@ | |||
| package com.andresgmoran.apptrabajadores.ui.fragments.activities; | ||||
| 
 | ||||
| import android.content.Context; | ||||
| import android.graphics.Color; | ||||
| import android.os.Bundle; | ||||
| 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.annotation.Nullable; | ||||
| import androidx.core.content.ContextCompat; | ||||
| import androidx.core.widget.NestedScrollView; | ||||
| import androidx.fragment.app.Fragment; | ||||
| import androidx.recyclerview.widget.LinearLayoutManager; | ||||
| import androidx.recyclerview.widget.RecyclerView; | ||||
| import androidx.swiperefreshlayout.widget.SwipeRefreshLayout; | ||||
| 
 | ||||
| import com.andresgmoran.apptrabajadores.R; | ||||
| import com.andresgmoran.apptrabajadores.interfaces.IOClickOnAddParticipantListener; | ||||
| import com.andresgmoran.apptrabajadores.interfaces.IOClickOnParticipantListener; | ||||
| import com.andresgmoran.apptrabajadores.interfaces.IOnChageStateActivityListener; | ||||
| import com.andresgmoran.apptrabajadores.models.Activity; | ||||
| import com.andresgmoran.apptrabajadores.models.ActivityResident; | ||||
| import com.andresgmoran.apptrabajadores.models.ActivityState; | ||||
| import com.andresgmoran.apptrabajadores.models.Resident; | ||||
| import com.andresgmoran.apptrabajadores.models.adapters.ParticipantsAdapter; | ||||
| import com.andresgmoran.apptrabajadores.ui.MainActivity; | ||||
| import com.google.android.material.button.MaterialButton; | ||||
| 
 | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| 
 | ||||
| public class ActivityDetailFragment extends Fragment { | ||||
| 
 | ||||
|     private Activity activity; | ||||
|     private List<ActivityResident> participants; | ||||
|     private List<Resident> residents; | ||||
| 
 | ||||
|     private ImageButton backButton; | ||||
| 
 | ||||
|     public interface IOOnAttachListener { | ||||
|         List<Resident> getResidents(); | ||||
|     } | ||||
| 
 | ||||
|     private IOClickOnAddParticipantListener addParticipantListener; | ||||
|     private IOnChageStateActivityListener changeStateActivityListener; | ||||
| 
 | ||||
|     @Nullable | ||||
|     @Override | ||||
|     public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) { | ||||
|         return inflater.inflate(R.layout.fragment_activity_detail, container, false); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) { | ||||
|         super.onViewCreated(view, savedInstanceState); | ||||
| 
 | ||||
|         SwipeRefreshLayout swipeRefreshLayout = view.findViewById(R.id.swipe_refresh_activity_detail); | ||||
| 
 | ||||
|         swipeRefreshLayout.setOnRefreshListener(() -> { | ||||
|             ((MainActivity) requireActivity()).refreshActivitiesAndReload(); | ||||
|             swipeRefreshLayout.setRefreshing(false); | ||||
|         }); | ||||
| 
 | ||||
|         backButton = view.findViewById(R.id.back_button); | ||||
|         backButton.setOnClickListener(v -> { | ||||
|             requireActivity().getSupportFragmentManager().popBackStack(); | ||||
|         }); | ||||
| 
 | ||||
|         TextView activityName = view.findViewById(R.id.activitie_detail_name); | ||||
| 
 | ||||
|         MaterialButton startEndButton = view.findViewById(R.id.btn_start_end_activity); | ||||
| 
 | ||||
|         if (activity.getState() == ActivityState.ABIERTO){ | ||||
|             startEndButton.setVisibility(View.GONE); | ||||
|         } else { | ||||
|             startEndButton.setVisibility(View.VISIBLE); | ||||
| 
 | ||||
|             ActivityState state = activity.getState(); | ||||
| 
 | ||||
|             if (state == ActivityState.CERRADO) { | ||||
|                 startEndButton.setText("Iniciar actividad"); | ||||
|                 startEndButton.setBackgroundTintList(ContextCompat.getColorStateList(requireContext(), R.color.purple_200)); | ||||
| 
 | ||||
|                 startEndButton.setOnClickListener(v -> { | ||||
|                     changeStateActivityListener.onChangeStateActivity(activity, ActivityState.EN_CURSO); | ||||
|                 }); | ||||
| 
 | ||||
|             } else if (state == ActivityState.EN_CURSO) { | ||||
|                 startEndButton.setText("Finalizar actividad"); | ||||
|                 startEndButton.setBackgroundTintList(ContextCompat.getColorStateList(requireContext(), R.color.teal_200)); | ||||
| 
 | ||||
|                 startEndButton.setOnClickListener(v -> { | ||||
|                     changeStateActivityListener.onChangeStateActivity(activity, ActivityState.FINALIZADA); | ||||
|                 }); | ||||
| 
 | ||||
|             } else if (state == ActivityState.FINALIZADA) { | ||||
|                 startEndButton.setText("Actividad finalizada"); | ||||
|                 startEndButton.setBackgroundTintList(ContextCompat.getColorStateList(requireContext(), R.color.teal_200)); | ||||
|                 startEndButton.setEnabled(false); | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
| 
 | ||||
|         TextView description = view.findViewById(R.id.tv_description_activity_detail); | ||||
|         description.setText(activity.getDescription()); | ||||
|         TextView activityDate = view.findViewById(R.id.tv_date_activity_detail); | ||||
|         TextView activityExitTime = view.findViewById(R.id.tv_exit_time_activity_detail); | ||||
|         activityName.setText(activity.getName()); | ||||
|         activityDate.setText(activity.getDate().toLocalDate().toString()); | ||||
|         activityExitTime.setText(activity.getDate().toLocalTime().toString()); | ||||
| 
 | ||||
|         List<ActivityResident> activityParticipants = new ArrayList<>(); | ||||
|         for (ActivityResident p : participants) { | ||||
|             if (p.getActivityId() == activity.getId()) { | ||||
|                 activityParticipants.add(p); | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         TextView totalParticipants = view.findViewById(R.id.tv_total_participants_activity_detail); | ||||
|         totalParticipants.setText(String.valueOf(activityParticipants.size())); | ||||
| 
 | ||||
|         TextView confirmedParticipants = view.findViewById(R.id.tv_confirmed_residents); | ||||
|         TextView notGoingParticipants = view.findViewById(R.id.tv_unconfirmed_residents); | ||||
|         int goingCount = 0; | ||||
|         int notGoingCount = 0; | ||||
|         for (ActivityResident p : activityParticipants) { | ||||
|             if (p.isAssistance()) { | ||||
|                 goingCount++; | ||||
|             } else { | ||||
|                 notGoingCount++; | ||||
|             } | ||||
|         } | ||||
|         confirmedParticipants.setText(String.valueOf(goingCount)); | ||||
|         notGoingParticipants.setText(String.valueOf(notGoingCount)); | ||||
| 
 | ||||
|         TextView humanAssistance = view.findViewById(R.id.tv_required_human_attendance); | ||||
|         TextView materialAssistance = view.findViewById(R.id.tv_required_material_attendance); | ||||
| 
 | ||||
|         int humanAssistanceCount = 0; | ||||
|         int materialAssistanceCount = 0; | ||||
|         for (ActivityResident p : activityParticipants) { | ||||
|             if (p.isHumanHelp()) { | ||||
|                 humanAssistanceCount++; | ||||
|             } else if (p.isMaterialHelp()) { | ||||
|                 materialAssistanceCount++; | ||||
|             } | ||||
|         } | ||||
|         humanAssistance.setText(String.valueOf(humanAssistanceCount)); | ||||
|         materialAssistance.setText(String.valueOf(materialAssistanceCount)); | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
|         RecyclerView recyclerView = view.findViewById(R.id.rv_participants); | ||||
|         ParticipantsAdapter participantsAdapter = new ParticipantsAdapter(requireContext() , activity, participants, residents, (IOClickOnParticipantListener) requireActivity()); | ||||
|         recyclerView.setAdapter(participantsAdapter); | ||||
|         recyclerView.setHasFixedSize(true); | ||||
|         recyclerView.setLayoutManager(new LinearLayoutManager(getActivity(), LinearLayoutManager.VERTICAL, false)); | ||||
| 
 | ||||
|         MaterialButton fab = view.findViewById(R.id.add_participant_button); | ||||
|         if (activity.getState() == ActivityState.CERRADO) { | ||||
|             fab.setVisibility(View.GONE); | ||||
|         } else { | ||||
|             fab.setOnClickListener(v -> | ||||
|                     addParticipantListener.onClickOnAddParticipant(activity, participants, residents) | ||||
|             ); | ||||
| 
 | ||||
|             NestedScrollView scrollView = view.findViewById(R.id.nested_scroll_game_detail); | ||||
| 
 | ||||
|             scrollView.setOnScrollChangeListener(new View.OnScrollChangeListener() { | ||||
|                 private int lastScrollY = 0; | ||||
|                 private boolean isButtonVisible = true; | ||||
| 
 | ||||
|                 @Override | ||||
|                 public void onScrollChange(View v, int scrollX, int scrollY, int oldScrollX, int oldScrollY) { | ||||
|                     if (scrollY > lastScrollY + 10 && isButtonVisible) { | ||||
|                         // Scroll hacia abajo: ocultar con animación | ||||
|                         fab.animate().translationY(fab.getHeight() + 50).alpha(0.0f).setDuration(200).withEndAction(() -> fab.setVisibility(View.GONE)); | ||||
|                         isButtonVisible = false; | ||||
|                     } else if (scrollY < lastScrollY - 10 && !isButtonVisible) { | ||||
|                         // Scroll hacia arriba: mostrar con animación | ||||
|                         fab.setVisibility(View.VISIBLE); | ||||
|                         fab.setAlpha(0f); | ||||
|                         fab.setTranslationY(fab.getHeight() + 50); | ||||
|                         fab.animate().translationY(0).alpha(1.0f).setDuration(200); | ||||
|                         isButtonVisible = true; | ||||
|                     } | ||||
|                     lastScrollY = scrollY; | ||||
|                 } | ||||
|             }); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public void onAttach(@NonNull Context context) { | ||||
|         if (getArguments() != null) { | ||||
|             activity = (Activity) getArguments().getSerializable("activity"); | ||||
|             participants = (List<ActivityResident>) getArguments().getSerializable("participants"); | ||||
|         } | ||||
| 
 | ||||
|         if (context instanceof IOOnAttachListener) { | ||||
|             residents = ((IOOnAttachListener) context).getResidents(); | ||||
|         } else { | ||||
|             throw new RuntimeException(context.toString() + " must implement IOOnAttachListener"); | ||||
|         } | ||||
| 
 | ||||
|         addParticipantListener = (IOClickOnAddParticipantListener) context; | ||||
|         changeStateActivityListener = (IOnChageStateActivityListener) context; | ||||
|         super.onAttach(context); | ||||
|     } | ||||
| } | ||||
|  | @ -0,0 +1,91 @@ | |||
| package com.andresgmoran.apptrabajadores.ui.fragments.activities; | ||||
| 
 | ||||
| import android.app.DatePickerDialog; | ||||
| import android.content.Context; | ||||
| import android.os.Bundle; | ||||
| import android.view.LayoutInflater; | ||||
| import android.view.View; | ||||
| import android.view.ViewGroup; | ||||
| import android.widget.Button; | ||||
| import android.widget.DatePicker; | ||||
| import android.widget.TextView; | ||||
| 
 | ||||
| import androidx.annotation.NonNull; | ||||
| import androidx.annotation.Nullable; | ||||
| import androidx.fragment.app.Fragment; | ||||
| 
 | ||||
| import com.andresgmoran.apptrabajadores.R; | ||||
| 
 | ||||
| import java.time.LocalDate; | ||||
| import java.time.LocalDateTime; | ||||
| import java.util.Calendar; | ||||
| 
 | ||||
| public class AddActivityFragment extends Fragment { | ||||
| 
 | ||||
|     public interface IOnAddActivity { | ||||
|         void onAddActivity(String activityName, String activityDescription, LocalDateTime date); | ||||
|     } | ||||
| 
 | ||||
|     private IOnAddActivity listener; | ||||
| 
 | ||||
|     private Button buttonFecha; | ||||
|     private LocalDateTime selectedDate; | ||||
| 
 | ||||
|     @Nullable | ||||
|     @Override | ||||
|     public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) { | ||||
|         return inflater.inflate(R.layout.fragment_add_activity, container, false); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) { | ||||
|         super.onViewCreated(view, savedInstanceState); | ||||
| 
 | ||||
|         TextView textViewNombre = view.findViewById(R.id.add_activity_name_input); | ||||
|         TextView textViewDescripcion = view.findViewById(R.id.add_activity_description_input); | ||||
|         buttonFecha = view.findViewById(R.id.buttonFecha); | ||||
|         Button buttonAgregar = view.findViewById(R.id.save_new_activity_button); | ||||
|         buttonFecha.setOnClickListener(v -> showDatePickerDialog()); | ||||
| 
 | ||||
|         buttonAgregar.setOnClickListener(v -> { | ||||
|             String activityName = textViewNombre.getText().toString(); | ||||
|             String activityDescription = textViewDescripcion.getText().toString(); | ||||
| 
 | ||||
|             if (activityName.isEmpty() || activityDescription.isEmpty() || selectedDate == null) { | ||||
|                 return; | ||||
|             } | ||||
| 
 | ||||
|             if (listener != null) { | ||||
|                 listener.onAddActivity(activityName, activityDescription, selectedDate); | ||||
|             } | ||||
|         }); | ||||
|     } | ||||
| 
 | ||||
|     private void showDatePickerDialog() { | ||||
|         final Calendar calendar = Calendar.getInstance(); | ||||
|         int year = calendar.get(Calendar.YEAR); | ||||
|         int month = calendar.get(Calendar.MONTH); | ||||
|         int day = calendar.get(Calendar.DAY_OF_MONTH); | ||||
| 
 | ||||
|         DatePickerDialog datePickerDialog = new DatePickerDialog( | ||||
|                 requireContext(), | ||||
|                 (DatePicker view, int selectedYear, int selectedMonth, int selectedDayOfMonth) -> { | ||||
|                     selectedDate = LocalDateTime.of(selectedYear, selectedMonth + 1, selectedDayOfMonth, 0, 0); | ||||
|                     buttonFecha.setText(selectedDate.toLocalDate().toString()); | ||||
|                 }, | ||||
|                 year, month, day | ||||
|         ); | ||||
| 
 | ||||
|         datePickerDialog.show(); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public void onAttach(@NonNull Context context) { | ||||
|         super.onAttach(context); | ||||
|         if (context instanceof IOnAddActivity) { | ||||
|             listener = (IOnAddActivity) context; | ||||
|         } else { | ||||
|             throw new RuntimeException(context.toString() + " must implement IOnAddActivity"); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | @ -0,0 +1,55 @@ | |||
| package com.andresgmoran.apptrabajadores.ui.fragments.activities; | ||||
| 
 | ||||
| import android.app.Activity; | ||||
| import android.content.Context; | ||||
| import android.os.Bundle; | ||||
| import android.view.LayoutInflater; | ||||
| import android.view.View; | ||||
| import android.view.ViewGroup; | ||||
| import android.widget.Button; | ||||
| import android.widget.EditText; | ||||
| 
 | ||||
| import androidx.annotation.NonNull; | ||||
| import androidx.annotation.Nullable; | ||||
| import androidx.fragment.app.Fragment; | ||||
| 
 | ||||
| import com.andresgmoran.apptrabajadores.R; | ||||
| import com.andresgmoran.apptrabajadores.models.ActivityResident; | ||||
| 
 | ||||
| public class OpinionFragment extends Fragment { | ||||
|     private ActivityResident participant; | ||||
|     private boolean isPreOpinion; | ||||
| 
 | ||||
|     public interface OnAddOpinionListener { | ||||
|         void onAddOpinion(ActivityResident participant, boolean isPreOpinion, String opinion); | ||||
|     } | ||||
|     private OnAddOpinionListener listener; | ||||
| 
 | ||||
|     @Nullable | ||||
|     @Override | ||||
|     public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) { | ||||
|         return inflater.inflate(R.layout.fragment_add_opinion, container, false);    } | ||||
| 
 | ||||
|     @Override | ||||
|     public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) { | ||||
|         super.onViewCreated(view, savedInstanceState); | ||||
|         EditText opinionEditText = view.findViewById(R.id.opinion_edit_text); | ||||
|         Button addOpinionButton = view.findViewById(R.id.add_opinion_button); | ||||
|         addOpinionButton.setOnClickListener(v -> { | ||||
|             String opinion = opinionEditText.getText().toString(); | ||||
|             requireActivity().getSupportFragmentManager().popBackStack(); | ||||
|             listener.onAddOpinion(participant, isPreOpinion, opinion); | ||||
|         }); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public void onAttach(@NonNull Context context) { | ||||
|         super.onAttach(context); | ||||
|         listener = (OnAddOpinionListener) context; | ||||
|         if (getArguments() != null) { | ||||
|             participant = (ActivityResident) getArguments().getSerializable("participant"); | ||||
|             isPreOpinion = getArguments().getBoolean("isPreOpinion"); | ||||
|         } | ||||
| 
 | ||||
|     } | ||||
| } | ||||
|  | @ -0,0 +1,77 @@ | |||
| package com.andresgmoran.apptrabajadores.ui.fragments.activities; | ||||
| 
 | ||||
| import android.app.AlertDialog; | ||||
| import android.app.Dialog; | ||||
| import android.content.Context; | ||||
| import android.os.Bundle; | ||||
| 
 | ||||
| import androidx.annotation.NonNull; | ||||
| import androidx.annotation.Nullable; | ||||
| import androidx.fragment.app.DialogFragment; | ||||
| 
 | ||||
| import com.andresgmoran.apptrabajadores.models.Activity; | ||||
| import com.andresgmoran.apptrabajadores.models.ActivityResident; | ||||
| import com.andresgmoran.apptrabajadores.models.Resident; | ||||
| 
 | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| 
 | ||||
| public class ParticipantSelectionDialogFragment extends DialogFragment { | ||||
|     public interface OnParticipantSelectedListener { | ||||
|         void onParticipantSelected(Activity activity, List<ActivityResident> participants , Resident selectedResident); | ||||
|     } | ||||
| 
 | ||||
|     private OnParticipantSelectedListener listener; | ||||
|     private Activity activityId; | ||||
|     private List<ActivityResident> participants; | ||||
|     private List<Resident> residentList; | ||||
| 
 | ||||
|     public ParticipantSelectionDialogFragment(Activity activityId, List<ActivityResident> participants, List<Resident> residents) { | ||||
|         this.residentList = residents; | ||||
|         this.participants = participants; | ||||
|         this.activityId = activityId; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public Dialog onCreateDialog(Bundle savedInstanceState) { | ||||
|         List<Resident> filteredResidents = new ArrayList<>(); | ||||
| 
 | ||||
|         for (Resident resident : residentList) { | ||||
|             boolean alreadyParticipant = false; | ||||
|             for (ActivityResident activityResident : participants) { | ||||
|                 if (activityResident.getIdResident().equals(resident.getId())) { | ||||
|                     alreadyParticipant = true; | ||||
|                     break; | ||||
|                 } | ||||
|             } | ||||
|             if (!alreadyParticipant) { | ||||
|                 filteredResidents.add(resident); | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         String[] names = new String[filteredResidents.size()]; | ||||
|         for (int i = 0; i < filteredResidents.size(); i++) { | ||||
|             Resident r = filteredResidents.get(i); | ||||
|             names[i] = r.getName() + " " + r.getSurnames(); | ||||
|         } | ||||
| 
 | ||||
|         return new AlertDialog.Builder(getActivity()) | ||||
|                 .setTitle("Selecciona un residente") | ||||
|                 .setItems(names, (dialog, which) -> { | ||||
|                     if (listener != null && which >= 0 && which < filteredResidents.size()) { | ||||
|                         listener.onParticipantSelected(activityId, participants, filteredResidents.get(which)); | ||||
|                     } | ||||
|                 }) | ||||
|                 .create(); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public void onAttach(@NonNull Context context) { | ||||
|         super.onAttach(context); | ||||
|         if (context instanceof OnParticipantSelectedListener) { | ||||
|             listener = (OnParticipantSelectedListener) context; | ||||
|         } else { | ||||
|             throw new RuntimeException(context.toString() + " must implement OnParticipantSelectedListener"); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | @ -1,4 +1,4 @@ | |||
| package com.andresgmoran.apptrabajadores.ui.fragments.logIn; | ||||
| package com.andresgmoran.apptrabajadores.ui.fragments.authentication; | ||||
| 
 | ||||
| import android.content.Context; | ||||
| import android.os.Bundle; | ||||
|  | @ -7,6 +7,7 @@ import android.view.LayoutInflater; | |||
| import android.view.View; | ||||
| import android.view.ViewGroup; | ||||
| import android.widget.Button; | ||||
| import android.widget.CheckBox; | ||||
| import android.widget.EditText; | ||||
| import android.widget.Toast; | ||||
| 
 | ||||
|  | @ -24,13 +25,12 @@ public class LoginFragment extends Fragment { | |||
|     private OnLoginListener loginListener; | ||||
| 
 | ||||
|     public interface OnLoginListener { | ||||
|         void onLoginSuccess(String email); | ||||
|         List<User> getUsers(); | ||||
|         void onLogin(String email, String password, boolean rememberPassword); | ||||
|     } | ||||
| 
 | ||||
|     private EditText emailInput, passwordInput; | ||||
|     private Button loginButton; | ||||
|     private List<User> users; | ||||
|     private CheckBox rememberPasswordCheckBox; | ||||
| 
 | ||||
|     @Nullable | ||||
|     @Override | ||||
|  | @ -43,38 +43,30 @@ public class LoginFragment extends Fragment { | |||
|     public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) { | ||||
|         super.onViewCreated(view, savedInstanceState); | ||||
| 
 | ||||
|         emailInput = view.findViewById(R.id.email_input); | ||||
|         passwordInput = view.findViewById(R.id.password_input); | ||||
|         emailInput = view.findViewById(R.id.email_input_login); | ||||
|         passwordInput = view.findViewById(R.id.password_input_login); | ||||
|         rememberPasswordCheckBox = view.findViewById(R.id.remember_password_checkBox); | ||||
|         loginButton = view.findViewById(R.id.login_button); | ||||
| 
 | ||||
|         loginButton.setOnClickListener(v -> { | ||||
|             String email = emailInput.getText().toString().trim(); | ||||
|             String password = passwordInput.getText().toString(); | ||||
|         loginButton.setOnClickListener(new View.OnClickListener() { | ||||
|             @Override | ||||
|             public void onClick(View v) { | ||||
|                 String email = emailInput.getText().toString().trim(); | ||||
|                 String password = passwordInput.getText().toString(); | ||||
| 
 | ||||
|             if (TextUtils.isEmpty(email) || TextUtils.isEmpty(password)) { | ||||
|                 Toast.makeText(getActivity(), "Completa todos los campos", Toast.LENGTH_SHORT).show(); | ||||
|                 return; | ||||
|             } | ||||
| 
 | ||||
| 
 | ||||
|             for (User user : users) { | ||||
|                 if (user.getEmail().equals(email)) { | ||||
|                     loginListener.onLoginSuccess(email); | ||||
|                 if (TextUtils.isEmpty(email) || TextUtils.isEmpty(password)) { | ||||
|                     Toast.makeText(getActivity(), "Completa todos los campos", Toast.LENGTH_SHORT).show(); | ||||
|                     return; | ||||
|                 } | ||||
| 
 | ||||
|                 loginListener.onLogin(email, password, rememberPasswordCheckBox.isChecked()); | ||||
|             } | ||||
|             Toast.makeText(getActivity(), "Usuario o contraseña incorrectos", Toast.LENGTH_SHORT).show(); | ||||
|         }); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public void onAttach(@NonNull Context context) { | ||||
|         super.onAttach(context); | ||||
|         if (context instanceof OnLoginListener) { | ||||
|             users = ((OnLoginListener) context).getUsers(); | ||||
|         } else { | ||||
|             throw new RuntimeException(context + " must implement OnLoginListener"); | ||||
|         } | ||||
|         if (context instanceof OnLoginListener) { | ||||
|             loginListener = (OnLoginListener) context; | ||||
|         } else { | ||||
|  | @ -6,6 +6,7 @@ import android.os.Bundle; | |||
| import android.view.LayoutInflater; | ||||
| import android.view.View; | ||||
| import android.view.ViewGroup; | ||||
| import android.widget.ImageButton; | ||||
| import android.widget.TextView; | ||||
| 
 | ||||
| import androidx.annotation.NonNull; | ||||
|  | @ -13,20 +14,19 @@ import androidx.annotation.Nullable; | |||
| import androidx.fragment.app.Fragment; | ||||
| import androidx.recyclerview.widget.LinearLayoutManager; | ||||
| import androidx.recyclerview.widget.RecyclerView; | ||||
| import androidx.swiperefreshlayout.widget.SwipeRefreshLayout; | ||||
| 
 | ||||
| import com.andresgmoran.apptrabajadores.R; | ||||
| import com.andresgmoran.apptrabajadores.interfaces.IOClickOnGameStatsListener; | ||||
| import com.andresgmoran.apptrabajadores.models.Game; | ||||
| import com.andresgmoran.apptrabajadores.models.Observation; | ||||
| import com.andresgmoran.apptrabajadores.models.Resident; | ||||
| import com.andresgmoran.apptrabajadores.models.adapters.LastGamesAdapter; | ||||
| import com.andresgmoran.apptrabajadores.models.adapters.ObservationsAdapter; | ||||
| import com.andresgmoran.apptrabajadores.models.gameStats.GameStat; | ||||
| import com.andresgmoran.apptrabajadores.ui.MainActivity; | ||||
| 
 | ||||
| import java.time.LocalDateTime; | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
| 
 | ||||
| public class GameFragment extends Fragment { | ||||
|     private List<GameStat> gameStats; | ||||
|  | @ -37,7 +37,7 @@ public class GameFragment extends Fragment { | |||
|     private TextView numberOfGamesLastWeekTextView; | ||||
|     private TextView totalGamesPlayedTextView; | ||||
|     private TextView topPlayerTextView; | ||||
| 
 | ||||
|     private ImageButton backButton; | ||||
|     @Nullable | ||||
|     @Override | ||||
|     public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) { | ||||
|  | @ -52,6 +52,18 @@ public class GameFragment extends Fragment { | |||
|         totalGamesPlayedTextView = view.findViewById(R.id.tv_total_games_played); | ||||
|         topPlayerTextView = view.findViewById(R.id.tv_top_jugador); | ||||
| 
 | ||||
|         SwipeRefreshLayout swipeRefreshLayout = view.findViewById(R.id.swipe_refresh_game); | ||||
| 
 | ||||
|         swipeRefreshLayout.setOnRefreshListener(() -> { | ||||
|             ((MainActivity) requireActivity()).refreshGameStatsAndReload(); | ||||
|             swipeRefreshLayout.setRefreshing(false); | ||||
|         }); | ||||
| 
 | ||||
|         backButton = view.findViewById(R.id.back_button); | ||||
|         backButton.setOnClickListener(v -> { | ||||
|             requireActivity().getSupportFragmentManager().popBackStack(); | ||||
|         }); | ||||
| 
 | ||||
|         List<GameStat> allGameStats = new ArrayList<>(); | ||||
|         for (GameStat gs : gameStats){ | ||||
|             if (gs.getGameId() == game.getId()){ | ||||
|  | @ -59,20 +71,24 @@ public class GameFragment extends Fragment { | |||
|             } | ||||
|         } | ||||
| 
 | ||||
| 
 | ||||
|         gameNameTextView.setText(game.getName()); | ||||
| 
 | ||||
|         if (allGameStats.isEmpty()) { | ||||
|             numberOfGamesLastWeekTextView.setText("-"); | ||||
|             totalGamesPlayedTextView.setText("-"); | ||||
|             topPlayerTextView.setText("-"); | ||||
|             view.findViewById(R.id.rv_game_stats).setVisibility( View.GONE ); | ||||
|             view.findViewById(R.id.tv_lastgames_empty_game).setVisibility( View.VISIBLE ); | ||||
|             return; | ||||
|         } | ||||
| 
 | ||||
|         int numberOfGamesLastWeek = getGamesLastWeek(allGameStats); | ||||
|         numberOfGamesLastWeekTextView.setText(String.valueOf(numberOfGamesLastWeek)); | ||||
|         totalGamesPlayedTextView.setText(String.valueOf(allGameStats.size())); | ||||
|         Resident topResident = getTopResident(allGameStats, residents); | ||||
|         if (topResident != null) { | ||||
|             topPlayerTextView.setText(topResident.getName()); | ||||
|         } else { | ||||
|             topPlayerTextView.setText("No hay jugadores"); | ||||
|         } | ||||
|         topPlayerTextView.setText(topResident.getName()); | ||||
| 
 | ||||
| 
 | ||||
|         LastGamesAdapter lastGamesAdapter = new LastGamesAdapter(gameStats, game, residents, (IOClickOnGameStatsListener) requireActivity()); | ||||
|         LastGamesAdapter lastGamesAdapter = new LastGamesAdapter(allGameStats, game, residents, (IOClickOnGameStatsListener) requireActivity()); | ||||
|         RecyclerView recyclerView = view.findViewById(R.id.rv_game_stats); | ||||
|         recyclerView.setAdapter( lastGamesAdapter ); | ||||
|         recyclerView.setHasFixedSize(true); | ||||
|  |  | |||
|  | @ -5,38 +5,46 @@ import android.os.Bundle; | |||
| import android.view.LayoutInflater; | ||||
| import android.view.View; | ||||
| import android.view.ViewGroup; | ||||
| import android.widget.Button; | ||||
| import android.widget.EditText; | ||||
| import android.widget.ImageButton; | ||||
| import android.widget.TextView; | ||||
| 
 | ||||
| import androidx.annotation.NonNull; | ||||
| import androidx.annotation.Nullable; | ||||
| import androidx.fragment.app.Fragment; | ||||
| import androidx.recyclerview.widget.LinearLayoutManager; | ||||
| import androidx.recyclerview.widget.RecyclerView; | ||||
| import androidx.swiperefreshlayout.widget.SwipeRefreshLayout; | ||||
| 
 | ||||
| import com.andresgmoran.apptrabajadores.R; | ||||
| import com.andresgmoran.apptrabajadores.models.Game; | ||||
| import com.andresgmoran.apptrabajadores.models.Observation; | ||||
| import com.andresgmoran.apptrabajadores.models.Resident; | ||||
| import com.andresgmoran.apptrabajadores.models.adapters.LastGamesAdapter; | ||||
| import com.andresgmoran.apptrabajadores.models.adapters.ObservationsAdapter; | ||||
| import com.andresgmoran.apptrabajadores.models.User; | ||||
| import com.andresgmoran.apptrabajadores.models.gameStats.GameStat; | ||||
| import com.andresgmoran.apptrabajadores.ui.MainActivity; | ||||
| 
 | ||||
| import java.util.List; | ||||
| 
 | ||||
| public class GameDetailFragment extends Fragment { | ||||
|     public interface IOOnAttachListener{ | ||||
|         List<Observation> getObservations(); | ||||
| 
 | ||||
|     public interface IOnAddObservationListener { | ||||
|         void onAddObservation(String observation, long gameId, long gameStatId); | ||||
|         List<User> getUsers(); | ||||
|     } | ||||
|     private IOnAddObservationListener addObservationListener; | ||||
| 
 | ||||
|     private GameStat gameStat; | ||||
|     private Resident gameStatResident; | ||||
|     private Game gameStatGame; | ||||
|     private User actualUser; | ||||
|     private List<User> users; | ||||
| 
 | ||||
|     private TextView residentNameTextView; | ||||
|     private TextView gameNameTextView; | ||||
|     private TextView durationTextView; | ||||
|     private TextView failsTitle; | ||||
|     private TextView failsTextView; | ||||
|     List<Observation> observations; | ||||
|     private ImageButton backButton; | ||||
| 
 | ||||
|     @Nullable | ||||
|     @Override | ||||
|     public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) { | ||||
|  | @ -47,15 +55,73 @@ public class GameDetailFragment extends Fragment { | |||
|     public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) { | ||||
|         super.onViewCreated(view, savedInstanceState); | ||||
| 
 | ||||
|         SwipeRefreshLayout swipeRefreshLayout = view.findViewById(R.id.swipe_refresh_game_detail); | ||||
| 
 | ||||
|         swipeRefreshLayout.setOnRefreshListener(() -> { | ||||
|             ((MainActivity) requireActivity()).refreshGameStatsAndReload(); | ||||
|             swipeRefreshLayout.setRefreshing(false); | ||||
|         }); | ||||
| 
 | ||||
|         backButton = view.findViewById(R.id.back_button); | ||||
|         backButton.setOnClickListener(v -> { | ||||
|             requireActivity().getSupportFragmentManager().popBackStack(); | ||||
|         }); | ||||
| 
 | ||||
|         residentNameTextView = view.findViewById(R.id.banner_name_resident); | ||||
|         gameNameTextView = view.findViewById(R.id.game_stats_detail_game_name); | ||||
|         durationTextView = view.findViewById(R.id.tv_duration); | ||||
|         failsTitle = view.findViewById(R.id.label_fails); | ||||
|         failsTextView = view.findViewById(R.id.tv_fails); | ||||
| 
 | ||||
|         residentNameTextView.setText(gameStatResident.getName() + " " + gameStatResident.getSurnames()); | ||||
|         gameNameTextView.setText(gameStatGame.getName()); | ||||
|         durationTextView.setText(Math.round(gameStat.getDuration()) + " seg"); | ||||
|         failsTextView.setText(gameStat.getNum() + " fallos"); | ||||
| 
 | ||||
|         if (gameStatGame.getName().equals("bingo auditivo") || gameStatGame.getName().equals("flecha y reacciona")) { | ||||
|             failsTitle.setText("Resultado"); | ||||
|             if (gameStat.getNum() == 0) { | ||||
|                 failsTextView.setText("PERDIDO"); | ||||
|             } else { | ||||
|                 failsTextView.setText("GANADO"); | ||||
|             } | ||||
|         } else { | ||||
|             failsTextView.setText(gameStat.getNum() + " fallos"); | ||||
|         } | ||||
| 
 | ||||
| 
 | ||||
|         View observation = view.findViewById(R.id.observation); | ||||
|         TextView observationUserName = observation.findViewById(R.id.observation_user_name); | ||||
|         TextView observationUserRole = observation.findViewById(R.id.observation_person_role); | ||||
|         TextView observationText = observation.findViewById(R.id.observation_text); | ||||
| 
 | ||||
|         EditText observationEditText = observation.findViewById(R.id.observation_input); | ||||
|         Button addObservationButton = observation.findViewById(R.id.add_observation_button); | ||||
| 
 | ||||
|         for (User user : users) { | ||||
|             if (user.getId() == gameStat.getUserId()) { | ||||
|                 observationUserName.setText(actualUser.getName()); | ||||
|                 break; | ||||
|             } | ||||
|         } | ||||
|         if (gameStat.getObservation().isEmpty()) { | ||||
|             if (gameStat.getUserId() == actualUser.getId()) { | ||||
|                 observationEditText.setVisibility(View.VISIBLE); | ||||
|                 addObservationButton.setVisibility(View.VISIBLE); | ||||
| 
 | ||||
|                 addObservationButton.setOnClickListener(new View.OnClickListener() { | ||||
|                     @Override | ||||
|                     public void onClick(View v) { | ||||
|                         addObservationListener.onAddObservation(observationEditText.getText().toString(), gameStatGame.getId(), gameStat.getId()); | ||||
|                     } | ||||
|                 }); | ||||
|             } else { | ||||
|                 observationUserRole.setText("El usuario encargado no ha añadido ninguna observación"); | ||||
|             } | ||||
|         } else { | ||||
|             observationText.setVisibility(View.VISIBLE); | ||||
|             //Añadir rol usuario | ||||
|             observationText.setText(gameStat.getObservation()); | ||||
|         } | ||||
| 
 | ||||
|     } | ||||
| 
 | ||||
|  | @ -63,12 +129,19 @@ public class GameDetailFragment extends Fragment { | |||
|     public void onAttach(@NonNull Context context) { | ||||
|         super.onAttach(context); | ||||
| 
 | ||||
|         if (context instanceof IOnAddObservationListener) { | ||||
|             addObservationListener = (IOnAddObservationListener) context; | ||||
|         } else { | ||||
|             throw new RuntimeException(context.toString() + " must implement IOnAddObservationListener"); | ||||
|         } | ||||
| 
 | ||||
|         users = addObservationListener.getUsers(); | ||||
| 
 | ||||
|         if(getArguments() != null) { | ||||
|             gameStat = (GameStat) getArguments().getSerializable("gameStat"); | ||||
|             gameStatResident = (Resident) getArguments().getSerializable("gameStatResident"); | ||||
|             gameStatGame = (Game) getArguments().getSerializable("gameStatGame"); | ||||
|             actualUser = (User) getArguments().getSerializable("user"); | ||||
|         } | ||||
|         GameDetailFragment.IOOnAttachListener attachListener = (GameDetailFragment.IOOnAttachListener) context; | ||||
|         observations = attachListener.getObservations(); | ||||
|     } | ||||
| } | ||||
|  |  | |||
|  | @ -1,17 +1,22 @@ | |||
| package com.andresgmoran.apptrabajadores.ui.fragments.home; | ||||
| 
 | ||||
| import android.content.Context; | ||||
| import android.graphics.Bitmap; | ||||
| import android.os.Bundle; | ||||
| import android.util.Log; | ||||
| import android.view.LayoutInflater; | ||||
| import android.view.View; | ||||
| import android.view.ViewGroup; | ||||
| import android.widget.ImageView; | ||||
| import android.widget.TextView; | ||||
| import android.widget.Toast; | ||||
| 
 | ||||
| import androidx.annotation.NonNull; | ||||
| import androidx.annotation.Nullable; | ||||
| import androidx.fragment.app.Fragment; | ||||
| import androidx.recyclerview.widget.LinearLayoutManager; | ||||
| import androidx.recyclerview.widget.RecyclerView; | ||||
| import androidx.swiperefreshlayout.widget.SwipeRefreshLayout; | ||||
| 
 | ||||
| import com.andresgmoran.apptrabajadores.R; | ||||
| import com.andresgmoran.apptrabajadores.interfaces.IOClickOnGameListener; | ||||
|  | @ -24,7 +29,13 @@ import com.andresgmoran.apptrabajadores.models.adapters.GamesAdapter; | |||
| import com.andresgmoran.apptrabajadores.models.adapters.LastGamesAdapter; | ||||
| import com.andresgmoran.apptrabajadores.models.adapters.ResidentsAdapter; | ||||
| import com.andresgmoran.apptrabajadores.models.gameStats.GameStat; | ||||
| import com.andresgmoran.apptrabajadores.network.ApiClient; | ||||
| import com.andresgmoran.apptrabajadores.ui.MainActivity; | ||||
| import com.bumptech.glide.Glide; | ||||
| 
 | ||||
| import java.util.ArrayList; | ||||
| import java.util.Collections; | ||||
| import java.util.Comparator; | ||||
| import java.util.List; | ||||
| 
 | ||||
| public class HomeFragment extends Fragment { | ||||
|  | @ -32,24 +43,41 @@ public class HomeFragment extends Fragment { | |||
|     public interface IOOnAttachListenerResidents { | ||||
|         List<Resident> getResidents(); | ||||
|     } | ||||
| 
 | ||||
|     public interface IOOnAttachListenerGameStats { | ||||
|         List<GameStat> getGameStats(); | ||||
|     } | ||||
| 
 | ||||
|     public interface IOOnAttachListenerGames { | ||||
|         List<Game> getGames(); | ||||
|     } | ||||
|     public interface IOOnAttachListenerUser { | ||||
|         User getActualUser(); | ||||
|         Bitmap getActualUserImage(); | ||||
|     } | ||||
| 
 | ||||
|     private List<Resident> residents; | ||||
|     private List<GameStat> gameStats; | ||||
|     private List<Game> games; | ||||
|     private List<User> users; | ||||
|     private User user; | ||||
|     private Bitmap userImageBitmap; | ||||
| 
 | ||||
| 
 | ||||
|     private String userEmail; | ||||
|     private TextView userNameTextView; | ||||
|     private RecyclerView recyclerViewLastGames; | ||||
|     private TextView emptyLatestGamesText; | ||||
|     private RecyclerView recyclerViewGames; | ||||
|     private TextView emptyGamesText; | ||||
|     private RecyclerView recyclerViewResidents; | ||||
|     private TextView emptyResidentsText; | ||||
|     private ImageView userImage; | ||||
| 
 | ||||
|     private GamesAdapter gamesAdapter; | ||||
|     private ResidentsAdapter residentsAdapter; | ||||
| 
 | ||||
|     @Nullable | ||||
|     @Override | ||||
|     public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) { | ||||
| 
 | ||||
|         return inflater.inflate(R.layout.fragment_home, container, false); | ||||
|     } | ||||
| 
 | ||||
|  | @ -57,50 +85,162 @@ public class HomeFragment extends Fragment { | |||
|     public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) { | ||||
|         super.onViewCreated(view, savedInstanceState); | ||||
| 
 | ||||
|         userNameTextView = view.findViewById(R.id.home_user_name); | ||||
|         SwipeRefreshLayout swipeRefreshLayout = view.findViewById(R.id.swipe_refresh_home); | ||||
| 
 | ||||
|         for (User user : users) { | ||||
|             if (user.getEmail().equals(userEmail)) { | ||||
|                 String fullName = user.getName() + " " + user.getSurnames(); | ||||
|                 userNameTextView.setText(fullName); | ||||
|                 break; | ||||
|             } | ||||
|         } | ||||
|         swipeRefreshLayout.setOnRefreshListener(() -> { | ||||
|             ((MainActivity) requireActivity()).refreshGameStatsAndReload(); | ||||
|             swipeRefreshLayout.setRefreshing(false); | ||||
|         }); | ||||
| 
 | ||||
|         userNameTextView = view.findViewById(R.id.home_user_name); | ||||
|         String fullName = user.getName() + " " + user.getSurnames(); | ||||
|         userNameTextView.setText(fullName); | ||||
|         userImage = view.findViewById(R.id.user_image_home); | ||||
|         userImage.setImageBitmap(userImageBitmap); | ||||
| 
 | ||||
|         emptyLatestGamesText = view.findViewById(R.id.tv_lastgames_empty_home); | ||||
|         LastGamesAdapter lastGamesAdapter = new LastGamesAdapter(gameStats, games, residents, (IOClickOnGameStatsListener) requireActivity()); | ||||
|         RecyclerView recyclerViewLastGames = view.findViewById(R.id.latestGames_recycleView_home); | ||||
|         recyclerViewLastGames = view.findViewById(R.id.latestGames_recycleView_home); | ||||
|         recyclerViewLastGames.setAdapter(lastGamesAdapter); | ||||
|         recyclerViewLastGames.setHasFixedSize(true); | ||||
|         recyclerViewLastGames.setLayoutManager(new LinearLayoutManager(getActivity(), LinearLayoutManager.HORIZONTAL, false)); | ||||
| 
 | ||||
|         GamesAdapter gamesAdapter = new GamesAdapter(games, (IOClickOnGameListener) requireActivity()); | ||||
|         RecyclerView recyclerViewGames = view.findViewById(R.id.games_recycleView_home); | ||||
|         if (gameStats.isEmpty()) { | ||||
|             recyclerViewLastGames.setVisibility(View.GONE); | ||||
|             emptyLatestGamesText.setVisibility(View.VISIBLE); | ||||
|         } else { | ||||
|             recyclerViewLastGames.setVisibility(View.VISIBLE); | ||||
|             emptyLatestGamesText.setVisibility(View.GONE); | ||||
|         } | ||||
| 
 | ||||
|         emptyGamesText = view.findViewById(R.id.tv_games_empty_home); | ||||
|         gamesAdapter = new GamesAdapter(games, (IOClickOnGameListener) requireActivity()); | ||||
|         recyclerViewGames = view.findViewById(R.id.games_recycleView_home); | ||||
|         recyclerViewGames.setAdapter(gamesAdapter); | ||||
|         recyclerViewGames.setHasFixedSize(true); | ||||
|         recyclerViewGames.setLayoutManager(new LinearLayoutManager(getActivity(), LinearLayoutManager.VERTICAL, false)); | ||||
| 
 | ||||
|         ResidentsAdapter adapter = new ResidentsAdapter(residents, (IOClickOnResidentListener) requireActivity()); | ||||
|         RecyclerView recyclerView = view.findViewById(R.id.residents_recycleView_home); | ||||
|         recyclerView.setAdapter(adapter); | ||||
|         recyclerView.setHasFixedSize(true); | ||||
|         recyclerView.setLayoutManager(new LinearLayoutManager(getActivity(), LinearLayoutManager.VERTICAL, false)); | ||||
|         if (games.isEmpty()) { | ||||
|             recyclerViewGames.setVisibility(View.GONE); | ||||
|             emptyGamesText.setVisibility(View.VISIBLE); | ||||
|         } else { | ||||
|             recyclerViewGames.setVisibility(View.VISIBLE); | ||||
|             emptyGamesText.setVisibility(View.GONE); | ||||
|         } | ||||
| 
 | ||||
|         emptyResidentsText = view.findViewById(R.id.tv_residents_empty_home); | ||||
|         residentsAdapter = new ResidentsAdapter(residents, (IOClickOnResidentListener) requireActivity()); | ||||
|         recyclerViewResidents = view.findViewById(R.id.residents_recycleView_home); | ||||
|         recyclerViewResidents.setAdapter(residentsAdapter); | ||||
|         recyclerViewResidents.setHasFixedSize(true); | ||||
|         recyclerViewResidents.setLayoutManager(new LinearLayoutManager(getActivity(), LinearLayoutManager.VERTICAL, false)); | ||||
| 
 | ||||
|         if (residents.isEmpty()) { | ||||
|             recyclerViewResidents.setVisibility(View.GONE); | ||||
|             emptyResidentsText.setVisibility(View.VISIBLE); | ||||
|         } else { | ||||
|             recyclerViewResidents.setVisibility(View.VISIBLE); | ||||
|             emptyResidentsText.setVisibility(View.GONE); | ||||
|         } | ||||
| 
 | ||||
|         View filterGamesButton = view.findViewById(R.id.filter_games_list_button); | ||||
|         View filterResidentsButton = view.findViewById(R.id.filter_residents_list_button); | ||||
| 
 | ||||
|         filterGamesButton.setOnClickListener(v -> showGamesFilterDialog()); | ||||
|         filterResidentsButton.setOnClickListener(v -> showResidentsFilterDialog()); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public void onAttach(@NonNull Context context) { | ||||
|         super.onAttach(context); | ||||
| 
 | ||||
|         if (getArguments() != null) { | ||||
|             userEmail = getArguments().getString("email"); | ||||
|             users = (List<User>) getArguments().getSerializable("users"); | ||||
|         } | ||||
|         IOOnAttachListenerUser attachListenerActualUser = (IOOnAttachListenerUser) context; | ||||
|         user = attachListenerActualUser.getActualUser(); | ||||
|         userImageBitmap = attachListenerActualUser.getActualUserImage(); | ||||
| 
 | ||||
|         IOOnAttachListenerGameStats attachListenerLastGames = (IOOnAttachListenerGameStats) context; | ||||
|         gameStats = attachListenerLastGames.getGameStats(); | ||||
| 
 | ||||
|         HomeFragment.IOOnAttachListenerResidents attachListenerResidents = (HomeFragment.IOOnAttachListenerResidents) context; | ||||
|         IOOnAttachListenerResidents attachListenerResidents = (IOOnAttachListenerResidents) context; | ||||
|         residents = attachListenerResidents.getResidents(); | ||||
| 
 | ||||
|         IOOnAttachListenerGames attachListenerGames = (IOOnAttachListenerGames) context; | ||||
|         games = attachListenerGames.getGames(); | ||||
|     } | ||||
| 
 | ||||
|     private void showGamesFilterDialog() { | ||||
|         String[] options = {"A-Z", "Z-A", "Más jugados"}; | ||||
| 
 | ||||
|         new androidx.appcompat.app.AlertDialog.Builder(requireContext()) | ||||
|                 .setTitle("Filtrar juegos") | ||||
|                 .setItems(options, (dialog, which) -> { | ||||
|                     String selectedOption = options[which]; | ||||
|                     filterGamesList(selectedOption); | ||||
|                 }) | ||||
|                 .show(); | ||||
|     } | ||||
| 
 | ||||
|     private void showResidentsFilterDialog() { | ||||
|         String[] options = {"A-Z", "Z-A", "Fecha de nacimiento"}; | ||||
| 
 | ||||
|         new androidx.appcompat.app.AlertDialog.Builder(requireContext()) | ||||
|                 .setTitle("Filtrar residentes") | ||||
|                 .setItems(options, (dialog, which) -> { | ||||
|                     String selectedOption = options[which]; | ||||
|                     filterResidentsList(selectedOption); | ||||
|                 }) | ||||
|                 .show(); | ||||
|     } | ||||
| 
 | ||||
|     private void filterGamesList(String option) { | ||||
|         List<Game> filteredList = new ArrayList<>(games); | ||||
| 
 | ||||
|         switch (option) { | ||||
|             case "A-Z": | ||||
|                 Collections.sort(filteredList, Comparator.comparing(Game::getName)); | ||||
|                 break; | ||||
|             case "Z-A": | ||||
|                 Collections.sort(filteredList, (g1, g2) -> g2.getName().compareTo(g1.getName())); | ||||
|                 break; | ||||
|             case "Más jugados": | ||||
|                 Collections.sort(filteredList, (g1, g2) -> { | ||||
|                     int count1 = 0; | ||||
|                     int count2 = 0; | ||||
| 
 | ||||
|                     for (GameStat stat : gameStats) { | ||||
|                         if (stat.getGameId() == g1.getId()) { | ||||
|                             count1++; | ||||
|                         } | ||||
|                         if (stat.getGameId() == g2.getId()) { | ||||
|                             count2++; | ||||
|                         } | ||||
|                     } | ||||
| 
 | ||||
|                     return Integer.compare(count2, count1); // Mayor a menor | ||||
|                 }); | ||||
|                 break; | ||||
|         } | ||||
| 
 | ||||
|         gamesAdapter.updateData(filteredList); | ||||
|         recyclerViewGames.scrollToPosition(0); | ||||
|     } | ||||
| 
 | ||||
|     private void filterResidentsList(String option) { | ||||
|         List<Resident> filteredList = new ArrayList<>(residents); | ||||
| 
 | ||||
|         switch (option) { | ||||
|             case "A-Z": | ||||
|                 Collections.sort(filteredList, Comparator.comparing(Resident::getName)); | ||||
|                 break; | ||||
|             case "Z-A": | ||||
|                 Collections.sort(filteredList, (r1, r2) -> r2.getName().compareTo(r1.getName())); | ||||
|                 break; | ||||
|             case "Fecha de nacimiento": | ||||
|                 Collections.sort(filteredList, Comparator.comparing(Resident::getBirthDate)); | ||||
|                 break; | ||||
|         } | ||||
| 
 | ||||
|         residentsAdapter.updateData(filteredList); | ||||
|         recyclerViewResidents.scrollToPosition(0); | ||||
|     } | ||||
| } | ||||
|  | @ -6,6 +6,7 @@ import android.os.Bundle; | |||
| import android.view.LayoutInflater; | ||||
| import android.view.View; | ||||
| import android.view.ViewGroup; | ||||
| import android.widget.ImageButton; | ||||
| import android.widget.LinearLayout; | ||||
| import android.widget.TextView; | ||||
| 
 | ||||
|  | @ -15,6 +16,7 @@ import androidx.cardview.widget.CardView; | |||
| import androidx.fragment.app.Fragment; | ||||
| import androidx.recyclerview.widget.LinearLayoutManager; | ||||
| import androidx.recyclerview.widget.RecyclerView; | ||||
| import androidx.swiperefreshlayout.widget.SwipeRefreshLayout; | ||||
| 
 | ||||
| import com.andresgmoran.apptrabajadores.R; | ||||
| import com.andresgmoran.apptrabajadores.interfaces.IOClickOnGameStatsListener; | ||||
|  | @ -22,6 +24,7 @@ import com.andresgmoran.apptrabajadores.models.Game; | |||
| import com.andresgmoran.apptrabajadores.models.Resident; | ||||
| import com.andresgmoran.apptrabajadores.models.adapters.LastGamesAdapter; | ||||
| import com.andresgmoran.apptrabajadores.models.gameStats.GameStat; | ||||
| import com.andresgmoran.apptrabajadores.ui.MainActivity; | ||||
| import com.andresgmoran.apptrabajadores.ui.graphics.CustomBarChartView; | ||||
| import com.andresgmoran.apptrabajadores.ui.graphics.CustomPieChartView; | ||||
| import java.util.ArrayList; | ||||
|  | @ -35,6 +38,13 @@ public class ResidentFragment extends Fragment { | |||
|     private List<Game> games; | ||||
|     private List<GameStat> gameStats; | ||||
| 
 | ||||
|     private ImageButton backButton; | ||||
|     private TextView residentName; | ||||
| 
 | ||||
|     private CardView pieCard; | ||||
|     private TextView tvLastGamesEmpty; | ||||
|     private TextView tvStatsEmpty; | ||||
| 
 | ||||
|     @Nullable | ||||
|     @Override | ||||
|     public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) { | ||||
|  | @ -44,9 +54,26 @@ public class ResidentFragment extends Fragment { | |||
|     @Override | ||||
|     public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) { | ||||
|         super.onViewCreated(view, savedInstanceState); | ||||
| 
 | ||||
|         SwipeRefreshLayout swipeRefreshLayout = view.findViewById(R.id.swipe_refresh_resident); | ||||
| 
 | ||||
|         swipeRefreshLayout.setOnRefreshListener(() -> { | ||||
|             ((MainActivity) requireActivity()).refreshGameStatsAndReload(); | ||||
|             swipeRefreshLayout.setRefreshing(false); | ||||
|         }); | ||||
| 
 | ||||
|         pieCard = view.findViewById(R.id.pieCard); | ||||
|         tvLastGamesEmpty = view.findViewById(R.id.tv_lastgames_empty_resident); | ||||
|         tvStatsEmpty = view.findViewById(R.id.tv_stats_empty_resident); | ||||
| 
 | ||||
|         View banner = view.findViewById(R.id.resident_banner); | ||||
| 
 | ||||
|         TextView residentName = banner.findViewById(R.id.banner_name_game); | ||||
|         backButton = banner.findViewById(R.id.back_button); | ||||
|         backButton.setOnClickListener(v -> { | ||||
|             requireActivity().getSupportFragmentManager().popBackStack(); | ||||
|         }); | ||||
| 
 | ||||
|         residentName = banner.findViewById(R.id.banner_name_game); | ||||
|         String fullName = resident.getName() + " " + resident.getSurnames(); | ||||
|         residentName.setText(fullName); | ||||
| 
 | ||||
|  | @ -64,8 +91,20 @@ public class ResidentFragment extends Fragment { | |||
|         recyclerViewLastGames.setHasFixedSize(true); | ||||
|         recyclerViewLastGames.setLayoutManager(new LinearLayoutManager(getActivity(), LinearLayoutManager.HORIZONTAL, false)); | ||||
| 
 | ||||
|         pieChart(view); | ||||
|         generateBarCharts(view); | ||||
|         if (filteredStats.isEmpty()){ | ||||
|             pieCard.setVisibility(View.GONE); | ||||
|             recyclerViewLastGames.setVisibility(View.GONE); | ||||
|             tvLastGamesEmpty.setVisibility(View.VISIBLE); | ||||
|             tvStatsEmpty.setVisibility(View.VISIBLE); | ||||
|         } else { | ||||
|             pieCard.setVisibility(View.VISIBLE); | ||||
|             recyclerViewLastGames.setVisibility(View.VISIBLE); | ||||
|             tvLastGamesEmpty.setVisibility(View.GONE); | ||||
|             tvStatsEmpty.setVisibility(View.GONE); | ||||
| 
 | ||||
|             pieChart(view); | ||||
|             generateBarCharts(view); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     private void pieChart(View view){ | ||||
|  |  | |||
|  | @ -17,10 +17,16 @@ import com.andresgmoran.apptrabajadores.interfaces.IOClickOnResidentListener; | |||
| import com.andresgmoran.apptrabajadores.models.Resident; | ||||
| import com.andresgmoran.apptrabajadores.models.adapters.ResidentsAdapter; | ||||
| 
 | ||||
| import java.util.ArrayList; | ||||
| import java.util.Collections; | ||||
| import java.util.Comparator; | ||||
| import java.util.List; | ||||
| 
 | ||||
| public class ResidentsListFragment extends Fragment { | ||||
| 
 | ||||
|     private ResidentsAdapter residentsAdapter; | ||||
|     private RecyclerView recyclerViewResidents; | ||||
| 
 | ||||
|     public interface IOOnAttachListener{ | ||||
|         List<Resident> getResidents(); | ||||
|     } | ||||
|  | @ -35,11 +41,46 @@ public class ResidentsListFragment extends Fragment { | |||
|     @Override | ||||
|     public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) { | ||||
|         super.onViewCreated(view, savedInstanceState); | ||||
|         ResidentsAdapter adapter = new ResidentsAdapter(residents, (IOClickOnResidentListener) requireActivity()); | ||||
|         RecyclerView recyclerView = view.findViewById(R.id.residents_recycleView); | ||||
|         recyclerView.setAdapter(adapter); | ||||
|         recyclerView.setHasFixedSize(true); | ||||
|         recyclerView.setLayoutManager(new LinearLayoutManager(getActivity(), LinearLayoutManager.VERTICAL, false)); | ||||
|         residentsAdapter = new ResidentsAdapter(residents, (IOClickOnResidentListener) requireActivity()); | ||||
|         recyclerViewResidents = view.findViewById(R.id.residents_recycleView); | ||||
|         recyclerViewResidents.setAdapter(residentsAdapter); | ||||
|         recyclerViewResidents.setHasFixedSize(true); | ||||
|         recyclerViewResidents.setLayoutManager(new LinearLayoutManager(getActivity(), LinearLayoutManager.VERTICAL, false)); | ||||
| 
 | ||||
|         View filterResidentsButton = view.findViewById(R.id.filter_residents_list_button); | ||||
| 
 | ||||
|         filterResidentsButton.setOnClickListener(v -> showResidentsFilterDialog()); | ||||
|     } | ||||
| 
 | ||||
|     private void showResidentsFilterDialog() { | ||||
|         String[] options = {"A-Z", "Z-A", "Fecha de nacimiento"}; | ||||
| 
 | ||||
|         new androidx.appcompat.app.AlertDialog.Builder(requireContext()) | ||||
|                 .setTitle("Filtrar residentes") | ||||
|                 .setItems(options, (dialog, which) -> { | ||||
|                     String selectedOption = options[which]; | ||||
|                     filterResidentsList(selectedOption); | ||||
|                 }) | ||||
|                 .show(); | ||||
|     } | ||||
| 
 | ||||
|     private void filterResidentsList(String option) { | ||||
|         List<Resident> filteredList = new ArrayList<>(residents); | ||||
| 
 | ||||
|         switch (option) { | ||||
|             case "A-Z": | ||||
|                 Collections.sort(filteredList, Comparator.comparing(Resident::getName)); | ||||
|                 break; | ||||
|             case "Z-A": | ||||
|                 Collections.sort(filteredList, (r1, r2) -> r2.getName().compareTo(r1.getName())); | ||||
|                 break; | ||||
|             case "Fecha de nacimiento": | ||||
|                 Collections.sort(filteredList, Comparator.comparing(Resident::getBirthDate)); | ||||
|                 break; | ||||
|         } | ||||
| 
 | ||||
|         residentsAdapter.updateData(filteredList); | ||||
|         recyclerViewResidents.scrollToPosition(0); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|  |  | |||
|  | @ -0,0 +1,55 @@ | |||
| package com.andresgmoran.apptrabajadores.utils; | ||||
| 
 | ||||
| import android.content.Context; | ||||
| import android.content.SharedPreferences; | ||||
| import androidx.security.crypto.EncryptedSharedPreferences; | ||||
| import androidx.security.crypto.MasterKey; | ||||
| 
 | ||||
| public class SecurePreferencesUtil { | ||||
| 
 | ||||
|     private static final String PREF_NAME = "secure_auth"; | ||||
|     private static SharedPreferences encryptedPrefs = null; | ||||
| 
 | ||||
|     public static SharedPreferences getEncryptedPrefs(Context context) { | ||||
|         if (encryptedPrefs == null) { | ||||
|             try { | ||||
|                 MasterKey masterKey = new MasterKey.Builder(context) | ||||
|                         .setKeyScheme(MasterKey.KeyScheme.AES256_GCM) | ||||
|                         .build(); | ||||
| 
 | ||||
|                 encryptedPrefs = EncryptedSharedPreferences.create( | ||||
|                         context, | ||||
|                         PREF_NAME, | ||||
|                         masterKey, | ||||
|                         EncryptedSharedPreferences.PrefKeyEncryptionScheme.AES256_SIV, | ||||
|                         EncryptedSharedPreferences.PrefValueEncryptionScheme.AES256_GCM | ||||
|                 ); | ||||
|             } catch (Exception e) { | ||||
|                 throw new RuntimeException("Error al inicializar SharedPreferences encriptadas", e); | ||||
|             } | ||||
|         } | ||||
|         return encryptedPrefs; | ||||
|     } | ||||
| 
 | ||||
|     public static String getString(Context context, String key, String defaultValue) { | ||||
|         return getEncryptedPrefs(context).getString(key, defaultValue); | ||||
|     } | ||||
| 
 | ||||
|     public static boolean getBoolean(Context context, String key, boolean defaultValue) { | ||||
|         return getEncryptedPrefs(context).getBoolean(key, defaultValue); | ||||
|     } | ||||
| 
 | ||||
|     public static long getLong(Context context, String key, long defaultValue) { | ||||
|         return getEncryptedPrefs(context).getLong(key, defaultValue); | ||||
|     } | ||||
| 
 | ||||
|     public static void edit(Context context, SharedPreferences.Editor editor) { | ||||
|         editor.apply(); | ||||
|     } | ||||
| 
 | ||||
|     public static void clear(Context context) { | ||||
|         SharedPreferences.Editor editor = getEncryptedPrefs(context).edit(); | ||||
|         editor.clear(); | ||||
|         edit(context, editor); | ||||
|     } | ||||
| } | ||||
|  | @ -0,0 +1,4 @@ | |||
| <shape xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     android:shape="oval"> | ||||
|     <solid android:color="#59FF00" /> | ||||
| </shape> | ||||
| After Width: | Height: | Size: 1.0 KiB | 
| After Width: | Height: | Size: 298 B | 
| After Width: | Height: | Size: 62 KiB | 
| After Width: | Height: | Size: 44 KiB | 
| After Width: | Height: | Size: 35 KiB | 
| After Width: | Height: | Size: 31 KiB | 
| Before Width: | Height: | Size: 1.2 MiB After Width: | Height: | Size: 1.2 MiB | 
| After Width: | Height: | Size: 999 B | 
| After Width: | Height: | Size: 29 KiB | 
| After Width: | Height: | Size: 1.3 KiB | 
|  | @ -7,7 +7,6 @@ | |||
|     android:layout_width="match_parent" | ||||
|     android:layout_height="match_parent"> | ||||
| 
 | ||||
|     <!-- Incluye el contenido principal respetando el BottomNav --> | ||||
|     <include | ||||
|         android:id="@+id/content_main_include" | ||||
|         layout="@layout/content_main" | ||||
|  | @ -18,7 +17,6 @@ | |||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintTop_toTopOf="parent" /> | ||||
| 
 | ||||
|     <!-- Círculo de carga centrado --> | ||||
|     <ProgressBar | ||||
|         android:id="@+id/progress_loader" | ||||
|         style="?android:attr/progressBarStyleLarge" | ||||
|  | @ -31,14 +29,11 @@ | |||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintEnd_toEndOf="parent" /> | ||||
| 
 | ||||
|     <!-- Menú de navegación inferior anclado abajo --> | ||||
|     <com.google.android.material.bottomnavigation.BottomNavigationView | ||||
|         android:id="@+id/nav_view" | ||||
|         style="@style/RoundedBottomNav" | ||||
|         android:layout_width="407dp" | ||||
|         android:layout_height="56dp" | ||||
|         android:layout_marginHorizontal="30dp" | ||||
|         android:layout_marginBottom="40dp" | ||||
|         android:layout_width="0dp" | ||||
|         android:layout_height="wrap_content" | ||||
|         android:background="?android:attr/windowBackground" | ||||
|         app:layout_constraintBottom_toBottomOf="parent" | ||||
|         app:layout_constraintEnd_toEndOf="parent" | ||||
|  |  | |||
|  | @ -31,7 +31,7 @@ | |||
|         app:layout_constraintTop_toTopOf="parent" /> | ||||
| 
 | ||||
|     <TextView | ||||
|         android:id="@+id/textView2" | ||||
|         android:id="@+id/label_highlights" | ||||
|         android:layout_width="0dp" | ||||
|         android:layout_height="20dp" | ||||
|         android:layout_marginTop="20dp" | ||||
|  | @ -44,47 +44,53 @@ | |||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintTop_toBottomOf="@+id/edit_account_button" /> | ||||
| 
 | ||||
|     <TextView | ||||
|         android:id="@+id/textView3" | ||||
|     <Button | ||||
|         android:id="@+id/btn_favoritos" | ||||
|         android:layout_width="0dp" | ||||
|         android:layout_height="20dp" | ||||
|         android:layout_height="wrap_content" | ||||
|         android:layout_marginTop="5dp" | ||||
|         android:background="@android:color/transparent" | ||||
|         android:fontFamily="@font/assistant_bold" | ||||
|         android:paddingStart="20dp" | ||||
|         android:text="Favoritos" | ||||
|         android:textAllCaps="false" | ||||
|         android:textColor="#324F5E" | ||||
|         app:layout_constraintEnd_toEndOf="parent" | ||||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintTop_toBottomOf="@+id/textView2" /> | ||||
|         app:layout_constraintTop_toBottomOf="@+id/label_highlights" /> | ||||
| 
 | ||||
|     <TextView | ||||
|         android:id="@+id/textView4" | ||||
|     <Button | ||||
|         android:id="@+id/btn_residentes" | ||||
|         android:layout_width="0dp" | ||||
|         android:layout_height="20dp" | ||||
|         android:layout_height="wrap_content" | ||||
|         android:layout_marginTop="5dp" | ||||
|         android:background="@android:color/transparent" | ||||
|         android:fontFamily="@font/assistant_bold" | ||||
|         android:paddingStart="20dp" | ||||
|         android:text="Residentes" | ||||
|         android:textAllCaps="false" | ||||
|         android:textColor="#324F5E" | ||||
|         app:layout_constraintEnd_toEndOf="parent" | ||||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintTop_toBottomOf="@+id/textView3" /> | ||||
|         app:layout_constraintTop_toBottomOf="@+id/btn_favoritos" /> | ||||
| 
 | ||||
|     <TextView | ||||
|         android:id="@+id/textView5" | ||||
|     <Button | ||||
|         android:id="@+id/btn_juegos" | ||||
|         android:layout_width="0dp" | ||||
|         android:layout_height="20dp" | ||||
|         android:layout_height="wrap_content" | ||||
|         android:layout_marginTop="5dp" | ||||
|         android:background="@android:color/transparent" | ||||
|         android:fontFamily="@font/assistant_bold" | ||||
|         android:paddingStart="20dp" | ||||
|         android:text="Juegos" | ||||
|         android:textAllCaps="false" | ||||
|         android:textColor="#324F5E" | ||||
|         app:layout_constraintEnd_toEndOf="parent" | ||||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintTop_toBottomOf="@+id/textView4" /> | ||||
|         app:layout_constraintTop_toBottomOf="@+id/btn_residentes" /> | ||||
| 
 | ||||
|     <TextView | ||||
|         android:id="@+id/textView6" | ||||
|         android:id="@+id/label_preferences" | ||||
|         android:layout_width="0dp" | ||||
|         android:layout_height="20dp" | ||||
|         android:layout_marginTop="10dp" | ||||
|  | @ -95,45 +101,51 @@ | |||
|         android:textColor="#324F5E" | ||||
|         app:layout_constraintEnd_toEndOf="parent" | ||||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintTop_toBottomOf="@+id/textView5" /> | ||||
|         app:layout_constraintTop_toBottomOf="@+id/btn_juegos" /> | ||||
| 
 | ||||
|     <TextView | ||||
|         android:id="@+id/textView7" | ||||
|     <Button | ||||
|         android:id="@+id/btn_idioma" | ||||
|         android:layout_width="0dp" | ||||
|         android:layout_height="20dp" | ||||
|         android:layout_height="wrap_content" | ||||
|         android:layout_marginTop="5dp" | ||||
|         android:background="@android:color/transparent" | ||||
|         android:fontFamily="@font/assistant_bold" | ||||
|         android:paddingStart="20dp" | ||||
|         android:text="Idioma" | ||||
|         android:textAllCaps="false" | ||||
|         android:textColor="#324F5E" | ||||
|         app:layout_constraintEnd_toEndOf="parent" | ||||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintTop_toBottomOf="@+id/textView6" /> | ||||
|         app:layout_constraintTop_toBottomOf="@+id/label_preferences" /> | ||||
| 
 | ||||
|     <TextView | ||||
|         android:id="@+id/textView8" | ||||
|     <Button | ||||
|         android:id="@+id/btn_modo_oscuro" | ||||
|         android:layout_width="0dp" | ||||
|         android:layout_height="20dp" | ||||
|         android:layout_marginTop="5dp" | ||||
|         android:fontFamily="@font/assistant_bold" | ||||
|         android:paddingStart="20dp" | ||||
|         android:text="Modo oscuro" | ||||
|         android:textColor="#324F5E" | ||||
|         app:layout_constraintEnd_toEndOf="parent" | ||||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintTop_toBottomOf="@+id/textView7" /> | ||||
| 
 | ||||
|     <TextView | ||||
|         android:id="@+id/textView9" | ||||
|         android:layout_width="0dp" | ||||
|         android:layout_height="20dp" | ||||
|         android:layout_height="wrap_content" | ||||
|         android:layout_marginTop="5dp" | ||||
|         android:background="@android:color/transparent" | ||||
|         android:fontFamily="@font/assistant_bold" | ||||
|         android:paddingStart="20dp" | ||||
|         android:text="Ajustes" | ||||
|         android:textAllCaps="false" | ||||
|         android:textColor="#324F5E" | ||||
|         app:layout_constraintEnd_toEndOf="parent" | ||||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintTop_toBottomOf="@+id/textView8" /> | ||||
|         app:layout_constraintTop_toBottomOf="@+id/btn_idioma" /> | ||||
| 
 | ||||
|     <Button | ||||
|         android:id="@+id/btn_cerrar_sesion" | ||||
|         android:layout_width="0dp" | ||||
|         android:layout_height="wrap_content" | ||||
|         android:layout_marginTop="5dp" | ||||
|         android:background="@android:color/transparent" | ||||
|         android:fontFamily="@font/assistant_bold" | ||||
|         android:paddingStart="20dp" | ||||
|         android:text="Cerrar Sesión" | ||||
|         android:textAllCaps="false" | ||||
|         android:textColor="#324F5E" | ||||
|         app:layout_constraintEnd_toEndOf="parent" | ||||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintTop_toBottomOf="@+id/btn_modo_oscuro" /> | ||||
| 
 | ||||
| </androidx.constraintlayout.widget.ConstraintLayout> | ||||
|  | @ -1,18 +1,75 @@ | |||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android" | ||||
| <androidx.swiperefreshlayout.widget.SwipeRefreshLayout 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/swipe_refresh_layout_activities_list" | ||||
|     android:layout_width="match_parent" | ||||
|     android:layout_height="match_parent"> | ||||
| 
 | ||||
|     <TextView | ||||
|         android:id="@+id/textView11" | ||||
|         android:layout_width="wrap_content" | ||||
|         android:layout_height="wrap_content" | ||||
|         android:layout_marginTop="284dp" | ||||
|         android:text="Activities" | ||||
|         app:layout_constraintEnd_toEndOf="parent" | ||||
|         app:layout_constraintHorizontal_bias="0.498" | ||||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintTop_toTopOf="parent" /> | ||||
| </androidx.constraintlayout.widget.ConstraintLayout> | ||||
|     <androidx.core.widget.NestedScrollView | ||||
|         android:layout_width="match_parent" | ||||
|         android:layout_height="match_parent" | ||||
|         android:fillViewport="true"> | ||||
| 
 | ||||
|         <androidx.constraintlayout.widget.ConstraintLayout | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:background="#CCCCCC" | ||||
|             android:backgroundTint="#CCCCCC"> | ||||
| 
 | ||||
|             <LinearLayout | ||||
|                 android:id="@+id/linearLayout3" | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginHorizontal="20dp" | ||||
|                 android:orientation="horizontal" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toTopOf="parent"> | ||||
| 
 | ||||
|                 <TextView | ||||
|                     android:id="@+id/textView18" | ||||
|                     android:layout_width="0dp" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:layout_weight="1" | ||||
|                     android:fontFamily="@font/assistant_bold" | ||||
|                     android:text="Actividades" | ||||
|                     android:textColor="#324F5E" | ||||
|                     android:textSize="20sp" /> | ||||
| 
 | ||||
|                 <ImageButton | ||||
|                     android:id="@+id/filter_activities_list_button" | ||||
|                     android:layout_width="wrap_content" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:layout_gravity="center" | ||||
|                     android:background="@android:color/transparent" | ||||
|                     android:contentDescription="Filtro" | ||||
|                     app:srcCompat="@drawable/filter" /> | ||||
|             </LinearLayout> | ||||
| 
 | ||||
|             <androidx.recyclerview.widget.RecyclerView | ||||
|                 android:id="@+id/activities_recycleView" | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginTop="10dp" | ||||
|                 app:layout_constraintEnd_toEndOf="@+id/linearLayout3" | ||||
|                 app:layout_constraintStart_toStartOf="@+id/linearLayout3" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/linearLayout3" /> | ||||
| 
 | ||||
|             <com.google.android.material.floatingactionbutton.FloatingActionButton | ||||
|                 android:id="@+id/add_activity_button" | ||||
|                 android:layout_width="wrap_content" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_margin="16dp" | ||||
|                 android:contentDescription="Añadir actividad" | ||||
|                 app:backgroundTint="#324F5E" | ||||
|                 app:layout_constraintBottom_toBottomOf="parent" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:srcCompat="@android:drawable/ic_input_add" | ||||
|                 app:tint="@android:color/white" /> | ||||
| 
 | ||||
|         </androidx.constraintlayout.widget.ConstraintLayout> | ||||
| 
 | ||||
|     </androidx.core.widget.NestedScrollView> | ||||
| 
 | ||||
| </androidx.swiperefreshlayout.widget.SwipeRefreshLayout> | ||||
|  |  | |||
|  | @ -1,6 +1,345 @@ | |||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android" | ||||
| <androidx.swiperefreshlayout.widget.SwipeRefreshLayout xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||
|     android:id="@+id/swipe_refresh_activity_detail" | ||||
|     android:layout_width="match_parent" | ||||
|     android:layout_height="match_parent"> | ||||
| 
 | ||||
| </androidx.constraintlayout.widget.ConstraintLayout> | ||||
|     <FrameLayout | ||||
|         android:layout_width="match_parent" | ||||
|         android:layout_height="match_parent"> | ||||
| 
 | ||||
|         <androidx.core.widget.NestedScrollView | ||||
|             android:id="@+id/nested_scroll_game_detail" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="match_parent" | ||||
|             android:background="#CCCCCC" | ||||
|             android:fillViewport="true"> | ||||
| 
 | ||||
|             <androidx.constraintlayout.widget.ConstraintLayout | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content"> | ||||
| 
 | ||||
|                 <!-- Banner azul --> | ||||
|                 <androidx.constraintlayout.widget.ConstraintLayout | ||||
|                     android:id="@+id/background_banner" | ||||
|                     android:layout_width="0dp" | ||||
|                     android:layout_height="174dp" | ||||
|                     android:background="#0062FF" | ||||
|                     app:layout_constraintEnd_toEndOf="parent" | ||||
|                     app:layout_constraintStart_toStartOf="parent" | ||||
|                     app:layout_constraintTop_toTopOf="parent"> | ||||
| 
 | ||||
|                     <ImageButton | ||||
|                         android:id="@+id/back_button" | ||||
|                         android:layout_width="wrap_content" | ||||
|                         android:layout_height="wrap_content" | ||||
|                         android:layout_margin="8dp" | ||||
|                         android:background="@null" | ||||
|                         app:layout_constraintStart_toStartOf="parent" | ||||
|                         app:layout_constraintTop_toTopOf="parent" | ||||
|                         app:srcCompat="@drawable/arrow_left" /> | ||||
| 
 | ||||
|                     <TextView | ||||
|                         android:id="@+id/activitie_detail_name" | ||||
|                         android:layout_width="wrap_content" | ||||
|                         android:layout_height="wrap_content" | ||||
|                         android:fontFamily="@font/assistant_bold" | ||||
|                         android:text="TextView" | ||||
|                         android:textColor="#324F5E" | ||||
|                         android:textSize="24sp" | ||||
|                         app:layout_constraintBottom_toBottomOf="parent" | ||||
|                         app:layout_constraintEnd_toEndOf="parent" | ||||
|                         app:layout_constraintStart_toStartOf="parent" | ||||
|                         app:layout_constraintTop_toTopOf="parent" /> | ||||
| 
 | ||||
|                     <com.google.android.material.button.MaterialButton | ||||
|                         android:id="@+id/btn_start_end_activity" | ||||
|                         android:layout_width="wrap_content" | ||||
|                         android:layout_height="wrap_content" | ||||
|                         android:layout_marginTop="8dp" | ||||
|                         android:layout_marginBottom="8dp" | ||||
|                         android:text="Iniciar actividad" | ||||
|                         android:textColor="@android:color/white" | ||||
|                         app:backgroundTint="#28A745" | ||||
|                         app:iconTint="@android:color/white" | ||||
|                         app:iconPadding="8dp" | ||||
|                         app:iconGravity="textStart" | ||||
|                         app:layout_constraintTop_toBottomOf="@id/activitie_detail_name" | ||||
|                         app:layout_constraintStart_toStartOf="parent" | ||||
|                         app:layout_constraintEnd_toEndOf="parent" | ||||
|                     /> | ||||
|                 </androidx.constraintlayout.widget.ConstraintLayout> | ||||
| 
 | ||||
|                 <!-- Nueva sección Descripción --> | ||||
|                 <TextView | ||||
|                     android:id="@+id/section_description_title" | ||||
|                     android:layout_width="0dp" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:background="#FFFFFFFF" | ||||
|                     android:fontFamily="@font/assistant_semibold" | ||||
|                     android:gravity="center" | ||||
|                     android:text="Descripción" | ||||
|                     android:textColor="#324F5E" | ||||
|                     android:textSize="18sp" | ||||
|                     app:layout_constraintEnd_toEndOf="parent" | ||||
|                     app:layout_constraintStart_toStartOf="parent" | ||||
|                     app:layout_constraintTop_toBottomOf="@id/background_banner" /> | ||||
| 
 | ||||
|                 <TextView | ||||
|                     android:id="@+id/tv_description_activity_detail" | ||||
|                     android:layout_width="0dp" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:layout_marginTop="4dp" | ||||
|                     android:fontFamily="@font/assistant_semibold" | ||||
|                     android:textColor="#324F5E" | ||||
|                     android:textSize="16sp" | ||||
|                     android:padding="16dp" | ||||
|                     app:layout_constraintTop_toBottomOf="@id/section_description_title" | ||||
|                     app:layout_constraintStart_toStartOf="parent" | ||||
|                     app:layout_constraintEnd_toEndOf="parent" /> | ||||
| 
 | ||||
|                 <!-- Sección Información --> | ||||
|                 <TextView | ||||
|                     android:id="@+id/textView2" | ||||
|                     android:layout_width="0dp" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:background="#FFFFFF" | ||||
|                     android:fontFamily="@font/assistant_semibold" | ||||
|                     android:gravity="center" | ||||
|                     android:text="Información" | ||||
|                     android:textColor="#324F5E" | ||||
|                     android:textSize="18sp" | ||||
|                     app:layout_constraintEnd_toEndOf="parent" | ||||
|                     app:layout_constraintStart_toStartOf="parent" | ||||
|                     app:layout_constraintTop_toBottomOf="@id/tv_description_activity_detail" /> | ||||
| 
 | ||||
|                 <androidx.constraintlayout.widget.ConstraintLayout | ||||
|                     android:id="@+id/info_container_activity_detail" | ||||
|                     android:layout_width="0dp" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:background="#CCCCCC" | ||||
|                     android:padding="16dp" | ||||
|                     app:layout_constraintEnd_toEndOf="parent" | ||||
|                     app:layout_constraintStart_toStartOf="parent" | ||||
|                     app:layout_constraintTop_toBottomOf="@id/textView2"> | ||||
| 
 | ||||
|                     <!-- Fecha --> | ||||
|                     <TextView | ||||
|                         android:id="@+id/label_date_activity_detail" | ||||
|                         android:layout_width="0dp" | ||||
|                         android:layout_height="wrap_content" | ||||
|                         android:fontFamily="@font/assistant_semibold" | ||||
|                         android:text="Fecha" | ||||
|                         android:textColor="#2C3E50" | ||||
|                         android:textSize="16sp" | ||||
|                         app:layout_constraintStart_toStartOf="parent" | ||||
|                         app:layout_constraintTop_toTopOf="parent" | ||||
|                         app:layout_constraintEnd_toStartOf="@+id/tv_date_activity_detail" | ||||
|                         app:layout_constraintHorizontal_bias="0" /> | ||||
| 
 | ||||
|                     <TextView | ||||
|                         android:id="@+id/tv_date_activity_detail" | ||||
|                         android:layout_width="wrap_content" | ||||
|                         android:layout_height="wrap_content" | ||||
|                         android:fontFamily="@font/assistant_semibold" | ||||
|                         android:text="20/09/2025" | ||||
|                         android:textColor="#2C3E50" | ||||
|                         android:textSize="16sp" | ||||
|                         app:layout_constraintEnd_toEndOf="parent" | ||||
|                         app:layout_constraintTop_toTopOf="@id/label_date_activity_detail" /> | ||||
| 
 | ||||
|                     <!-- Hora de salida --> | ||||
|                     <TextView | ||||
|                         android:id="@+id/label_exit_time_activity_detail" | ||||
|                         android:layout_width="0dp" | ||||
|                         android:layout_height="wrap_content" | ||||
|                         android:layout_marginTop="8dp" | ||||
|                         android:fontFamily="@font/assistant_semibold" | ||||
|                         android:text="Hora de salida" | ||||
|                         android:textColor="#2C3E50" | ||||
|                         android:textSize="16sp" | ||||
|                         app:layout_constraintStart_toStartOf="parent" | ||||
|                         app:layout_constraintTop_toBottomOf="@id/label_date_activity_detail" | ||||
|                         app:layout_constraintEnd_toStartOf="@+id/tv_exit_time_activity_detail" /> | ||||
| 
 | ||||
|                     <TextView | ||||
|                         android:id="@+id/tv_exit_time_activity_detail" | ||||
|                         android:layout_width="wrap_content" | ||||
|                         android:layout_height="wrap_content" | ||||
|                         android:fontFamily="@font/assistant_semibold" | ||||
|                         android:text="10:00am" | ||||
|                         android:textColor="#2C3E50" | ||||
|                         android:textSize="16sp" | ||||
|                         app:layout_constraintEnd_toEndOf="parent" | ||||
|                         app:layout_constraintTop_toTopOf="@id/label_exit_time_activity_detail" /> | ||||
| 
 | ||||
|                     <!-- Hora de vuelta estimada --> | ||||
|                     <TextView | ||||
|                         android:id="@+id/label_total_residents_activity_detail" | ||||
|                         android:layout_width="0dp" | ||||
|                         android:layout_height="wrap_content" | ||||
|                         android:layout_marginTop="8dp" | ||||
|                         android:fontFamily="@font/assistant_semibold" | ||||
|                         android:text="Residentes totales" | ||||
|                         android:textColor="#2C3E50" | ||||
|                         android:textSize="16sp" | ||||
|                         app:layout_constraintEnd_toStartOf="@+id/tv_total_participants_activity_detail" | ||||
|                         app:layout_constraintStart_toStartOf="parent" | ||||
|                         app:layout_constraintTop_toBottomOf="@id/label_exit_time_activity_detail" /> | ||||
| 
 | ||||
|                     <TextView | ||||
|                         android:id="@+id/tv_total_participants_activity_detail" | ||||
|                         android:layout_width="wrap_content" | ||||
|                         android:layout_height="wrap_content" | ||||
|                         android:fontFamily="@font/assistant_semibold" | ||||
|                         android:textColor="#2C3E50" | ||||
|                         android:textSize="16sp" | ||||
|                         app:layout_constraintEnd_toEndOf="parent" | ||||
|                         app:layout_constraintTop_toTopOf="@id/label_total_residents_activity_detail" /> | ||||
| 
 | ||||
|                     <!-- Residentes confirmados --> | ||||
|                     <TextView | ||||
|                         android:id="@+id/label_confirmed_residents" | ||||
|                         android:layout_width="0dp" | ||||
|                         android:layout_height="wrap_content" | ||||
|                         android:layout_marginTop="8dp" | ||||
|                         android:fontFamily="@font/assistant_semibold" | ||||
|                         android:text="Residentes que si van" | ||||
|                         android:textColor="#2C3E50" | ||||
|                         android:textSize="16sp" | ||||
|                         app:layout_constraintEnd_toStartOf="@+id/tv_confirmed_residents" | ||||
|                         app:layout_constraintStart_toStartOf="parent" | ||||
|                         app:layout_constraintTop_toBottomOf="@id/label_total_residents_activity_detail" /> | ||||
| 
 | ||||
|                     <TextView | ||||
|                         android:id="@+id/tv_confirmed_residents" | ||||
|                         android:layout_width="wrap_content" | ||||
|                         android:layout_height="wrap_content" | ||||
|                         android:fontFamily="@font/assistant_semibold" | ||||
|                         android:text="20" | ||||
|                         android:textColor="#2C3E50" | ||||
|                         android:textSize="16sp" | ||||
|                         app:layout_constraintEnd_toEndOf="parent" | ||||
|                         app:layout_constraintTop_toTopOf="@id/label_confirmed_residents" /> | ||||
| 
 | ||||
|                     <!-- Residentes sin confirmar --> | ||||
|                     <TextView | ||||
|                         android:id="@+id/label_unconfirmed_residents" | ||||
|                         android:layout_width="0dp" | ||||
|                         android:layout_height="wrap_content" | ||||
|                         android:layout_marginTop="8dp" | ||||
|                         android:fontFamily="@font/assistant_semibold" | ||||
|                         android:text="Residentes que no van" | ||||
|                         android:textColor="#2C3E50" | ||||
|                         android:textSize="16sp" | ||||
|                         app:layout_constraintEnd_toStartOf="@+id/tv_unconfirmed_residents" | ||||
|                         app:layout_constraintStart_toStartOf="parent" | ||||
|                         app:layout_constraintTop_toBottomOf="@id/label_confirmed_residents" /> | ||||
| 
 | ||||
|                     <TextView | ||||
|                         android:id="@+id/tv_unconfirmed_residents" | ||||
|                         android:layout_width="wrap_content" | ||||
|                         android:layout_height="wrap_content" | ||||
|                         android:fontFamily="@font/assistant_semibold" | ||||
|                         android:text="11" | ||||
|                         android:textColor="#2C3E50" | ||||
|                         android:textSize="16sp" | ||||
|                         app:layout_constraintEnd_toEndOf="parent" | ||||
|                         app:layout_constraintTop_toTopOf="@id/label_unconfirmed_residents" /> | ||||
| 
 | ||||
|                     <!-- Asistencia necesaria --> | ||||
|                     <TextView | ||||
|                         android:id="@+id/label_required_human_attendance" | ||||
|                         android:layout_width="0dp" | ||||
|                         android:layout_height="wrap_content" | ||||
|                         android:layout_marginTop="8dp" | ||||
|                         android:fontFamily="@font/assistant_semibold" | ||||
|                         android:text="Asistencia humana necesaria" | ||||
|                         android:textColor="#2C3E50" | ||||
|                         android:textSize="16sp" | ||||
|                         app:layout_constraintEnd_toStartOf="@+id/tv_required_human_attendance" | ||||
|                         app:layout_constraintStart_toStartOf="parent" | ||||
|                         app:layout_constraintTop_toBottomOf="@id/label_unconfirmed_residents" /> | ||||
| 
 | ||||
|                     <TextView | ||||
|                         android:id="@+id/tv_required_human_attendance" | ||||
|                         android:layout_width="wrap_content" | ||||
|                         android:layout_height="wrap_content" | ||||
|                         android:fontFamily="@font/assistant_semibold" | ||||
|                         android:text="9" | ||||
|                         android:textColor="#2C3E50" | ||||
|                         android:textSize="16sp" | ||||
|                         app:layout_constraintEnd_toEndOf="parent" | ||||
|                         app:layout_constraintTop_toTopOf="@id/label_required_human_attendance" /> | ||||
| 
 | ||||
|                     <TextView | ||||
|                         android:id="@+id/label_required_material_attendance" | ||||
|                         android:layout_width="0dp" | ||||
|                         android:layout_height="wrap_content" | ||||
|                         android:layout_marginTop="8dp" | ||||
|                         android:fontFamily="@font/assistant_semibold" | ||||
|                         android:text="Asistencia material necesaria" | ||||
|                         android:textColor="#2C3E50" | ||||
|                         android:textSize="16sp" | ||||
|                         app:layout_constraintEnd_toStartOf="@+id/tv_required_material_attendance" | ||||
|                         app:layout_constraintStart_toStartOf="parent" | ||||
|                         app:layout_constraintTop_toBottomOf="@id/label_required_human_attendance" /> | ||||
| 
 | ||||
|                     <TextView | ||||
|                         android:id="@+id/tv_required_material_attendance" | ||||
|                         android:layout_width="wrap_content" | ||||
|                         android:layout_height="wrap_content" | ||||
|                         android:fontFamily="@font/assistant_semibold" | ||||
|                         android:text="9" | ||||
|                         android:textColor="#2C3E50" | ||||
|                         android:textSize="16sp" | ||||
|                         app:layout_constraintEnd_toEndOf="parent" | ||||
|                         app:layout_constraintTop_toTopOf="@id/label_required_material_attendance" /> | ||||
| 
 | ||||
| 
 | ||||
|                 </androidx.constraintlayout.widget.ConstraintLayout> | ||||
| 
 | ||||
|                 <!-- Sección Participantes --> | ||||
|                 <TextView | ||||
|                     android:id="@+id/textView6" | ||||
|                     android:layout_width="0dp" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:layout_marginTop="16dp" | ||||
|                     android:background="#FFFFFFFF" | ||||
|                     android:fontFamily="@font/assistant_semibold" | ||||
|                     android:gravity="center" | ||||
|                     android:text="Participantes" | ||||
|                     android:textColor="#324F5E" | ||||
|                     android:textSize="18sp" | ||||
|                     app:layout_constraintEnd_toEndOf="parent" | ||||
|                     app:layout_constraintStart_toStartOf="parent" | ||||
|                     app:layout_constraintTop_toBottomOf="@+id/info_container_activity_detail" /> | ||||
| 
 | ||||
|                 <androidx.recyclerview.widget.RecyclerView | ||||
|                     android:id="@+id/rv_participants" | ||||
|                     android:layout_width="match_parent" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     app:layout_constraintEnd_toEndOf="@+id/textView6" | ||||
|                     app:layout_constraintStart_toStartOf="@+id/textView6" | ||||
|                     app:layout_constraintTop_toBottomOf="@+id/textView6" /> | ||||
| 
 | ||||
|             </androidx.constraintlayout.widget.ConstraintLayout> | ||||
|         </androidx.core.widget.NestedScrollView> | ||||
| 
 | ||||
|         <com.google.android.material.button.MaterialButton | ||||
|             android:id="@+id/add_participant_button" | ||||
|             android:layout_width="wrap_content" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:layout_gravity="bottom|end" | ||||
|             android:layout_margin="16dp" | ||||
|             android:text="Añadir participante" | ||||
|             android:contentDescription="Añadir participante" | ||||
|             app:icon="@android:drawable/ic_input_add" | ||||
|             app:iconTint="@android:color/white" | ||||
|             app:iconPadding="8dp" | ||||
|             app:iconGravity="textStart" | ||||
|             app:backgroundTint="#324F5E" | ||||
|             android:textColor="@android:color/white"/> | ||||
|     </FrameLayout> | ||||
| </androidx.swiperefreshlayout.widget.SwipeRefreshLayout> | ||||
|  |  | |||
|  | @ -0,0 +1,62 @@ | |||
| <?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" | ||||
|     android:layout_width="match_parent" | ||||
|     android:layout_height="match_parent" | ||||
|     android:padding="20dp" | ||||
|     android:background="#CCCCCC"> | ||||
| 
 | ||||
|     <EditText | ||||
|         android:id="@+id/add_activity_name_input" | ||||
|         android:layout_width="0dp" | ||||
|         android:layout_height="wrap_content" | ||||
|         android:hint="Nombre" | ||||
|         android:inputType="textPersonName" | ||||
|         android:textColor="#324F5E" | ||||
|         android:textColorLink="#6CADFF" | ||||
|         app:layout_constraintEnd_toEndOf="parent" | ||||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintTop_toTopOf="parent" /> | ||||
| 
 | ||||
|     <EditText | ||||
|         android:id="@+id/add_activity_description_input" | ||||
|         android:layout_width="0dp" | ||||
|         android:layout_height="wrap_content" | ||||
|         android:layout_marginTop="16dp" | ||||
|         android:gravity="top|start" | ||||
|         android:hint="Descripción" | ||||
|         android:inputType="textMultiLine" | ||||
|         android:minLines="3" | ||||
|         android:textColor="#324F5E" | ||||
|         android:textColorLink="#6CADFF" | ||||
|         app:layout_constraintEnd_toEndOf="parent" | ||||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintTop_toBottomOf="@id/add_activity_name_input" /> | ||||
| 
 | ||||
|     <Button | ||||
|         android:id="@+id/buttonFecha" | ||||
|         android:layout_width="0dp" | ||||
|         android:layout_height="wrap_content" | ||||
|         android:layout_marginTop="16dp" | ||||
|         android:backgroundTint="#00AF4848" | ||||
|         android:text="Seleccionar fecha" | ||||
|         android:textAlignment="viewStart" | ||||
|         android:textColor="#324F5E" | ||||
|         android:textColorLink="#6CADFF" | ||||
|         app:layout_constraintEnd_toEndOf="parent" | ||||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintTop_toBottomOf="@id/add_activity_description_input" /> | ||||
| 
 | ||||
|     <Button | ||||
|         android:id="@+id/save_new_activity_button" | ||||
|         android:layout_width="wrap_content" | ||||
|         android:layout_height="wrap_content" | ||||
|         android:layout_marginTop="24dp" | ||||
|         android:backgroundTint="#324F5E" | ||||
|         android:text="Guardar" | ||||
|         app:layout_constraintEnd_toEndOf="parent" | ||||
|         app:layout_constraintStart_toStartOf="@+id/buttonFecha" | ||||
|         app:layout_constraintTop_toBottomOf="@id/buttonFecha" /> | ||||
| 
 | ||||
| </androidx.constraintlayout.widget.ConstraintLayout> | ||||
|  | @ -0,0 +1,36 @@ | |||
| <?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" | ||||
|     android:id="@+id/constraint_layout" | ||||
|     android:layout_width="match_parent" | ||||
|     android:layout_height="match_parent" | ||||
|     android:padding="16dp"> | ||||
| 
 | ||||
|     <!-- Cuadro de texto para escribir --> | ||||
|     <EditText | ||||
|         android:id="@+id/opinion_edit_text" | ||||
|         android:layout_width="0dp" | ||||
|         android:layout_height="wrap_content" | ||||
|         android:hint="Escribe algo aquí" | ||||
|         android:padding="12dp" | ||||
|         android:textColor="#000000" | ||||
|         android:textColorHint="#888888" | ||||
|         app:layout_constraintTop_toTopOf="parent" | ||||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintEnd_toEndOf="parent" /> | ||||
| 
 | ||||
|     <!-- Botón debajo del cuadro de texto --> | ||||
|     <Button | ||||
|         android:id="@+id/add_opinion_button" | ||||
|         android:layout_width="0dp" | ||||
|         android:layout_height="wrap_content" | ||||
|         android:text="Enviar" | ||||
|         android:textColor="#FFFFFF" | ||||
|         android:backgroundTint="#324F5E" | ||||
|         android:layout_marginTop="16dp" | ||||
|         app:layout_constraintTop_toBottomOf="@id/opinion_edit_text" | ||||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintEnd_toEndOf="parent" /> | ||||
| 
 | ||||
| </androidx.constraintlayout.widget.ConstraintLayout> | ||||
|  | @ -1,188 +1,207 @@ | |||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <androidx.core.widget.NestedScrollView | ||||
| <androidx.swiperefreshlayout.widget.SwipeRefreshLayout | ||||
|     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/swipe_refresh_game" | ||||
|     android:layout_width="match_parent" | ||||
|     android:layout_height="match_parent" | ||||
|     android:fillViewport="true" | ||||
|     android:background="#CCCCCC"> | ||||
|     android:layout_height="match_parent"> | ||||
| 
 | ||||
|     <androidx.constraintlayout.widget.ConstraintLayout | ||||
|     <androidx.core.widget.NestedScrollView | ||||
|         android:layout_width="match_parent" | ||||
|         android:layout_height="wrap_content"> | ||||
|         android:layout_height="match_parent" | ||||
|         android:fillViewport="true" | ||||
|         android:background="#CCCCCC"> | ||||
| 
 | ||||
|         <!-- Banner azul --> | ||||
|         <androidx.constraintlayout.widget.ConstraintLayout | ||||
|             android:id="@+id/background_banner" | ||||
|             android:layout_width="0dp" | ||||
|             android:layout_height="174dp" | ||||
|             android:background="#0062FF" | ||||
|             app:layout_constraintTop_toTopOf="parent" | ||||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintEnd_toEndOf="parent"> | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="wrap_content"> | ||||
| 
 | ||||
|             <!-- Botón superior izquierdo --> | ||||
|             <ImageButton | ||||
|                 android:id="@+id/imageButton2" | ||||
|                 android:layout_width="wrap_content" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_margin="8dp" | ||||
|                 android:background="@null" | ||||
|                 app:srcCompat="@drawable/arrow_left" | ||||
|             <!-- Banner azul --> | ||||
|             <androidx.constraintlayout.widget.ConstraintLayout | ||||
|                 android:id="@+id/background_banner" | ||||
|                 android:layout_width="0dp" | ||||
|                 android:layout_height="174dp" | ||||
|                 android:background="#0062FF" | ||||
|                 app:layout_constraintTop_toTopOf="parent" | ||||
|                 app:layout_constraintStart_toStartOf="parent" /> | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintEnd_toEndOf="parent"> | ||||
| 
 | ||||
|             <!-- Avatar + texto --> | ||||
|             <com.google.android.material.imageview.ShapeableImageView | ||||
|                 android:id="@+id/circularImageView" | ||||
|                 android:layout_width="80dp" | ||||
|                 android:layout_height="80dp" | ||||
|                 android:layout_marginStart="20dp" | ||||
|                 <!-- Botón superior izquierdo --> | ||||
|                 <ImageButton | ||||
|                     android:id="@+id/back_button" | ||||
|                     android:layout_width="wrap_content" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:layout_margin="8dp" | ||||
|                     android:background="@null" | ||||
|                     app:srcCompat="@drawable/arrow_left" | ||||
|                     app:layout_constraintTop_toTopOf="parent" | ||||
|                     app:layout_constraintStart_toStartOf="parent" /> | ||||
| 
 | ||||
|                 <!-- Avatar + texto --> | ||||
|                 <com.google.android.material.imageview.ShapeableImageView | ||||
|                     android:id="@+id/circularImageView" | ||||
|                     android:layout_width="80dp" | ||||
|                     android:layout_height="80dp" | ||||
|                     android:layout_marginStart="20dp" | ||||
|                     android:layout_marginTop="16dp" | ||||
|                     android:scaleType="centerCrop" | ||||
|                     android:src="@drawable/woman_avatar" | ||||
|                     app:layout_constraintStart_toStartOf="parent" | ||||
|                     app:layout_constraintTop_toBottomOf="@id/back_button" | ||||
|                     app:shapeAppearanceOverlay="@style/CircularShapeAppearance" /> | ||||
| 
 | ||||
|                 <TextView | ||||
|                     android:id="@+id/banner_name_game" | ||||
|                     android:layout_width="wrap_content" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:layout_marginStart="10dp" | ||||
|                     android:layout_marginTop="15dp" | ||||
|                     android:layout_marginBottom="33dp" | ||||
|                     android:fontFamily="@font/assistant_bold" | ||||
|                     android:textColor="@android:color/white" | ||||
|                     android:textSize="24sp" | ||||
|                     app:layout_constraintBottom_toBottomOf="@+id/circularImageView" | ||||
|                     app:layout_constraintStart_toEndOf="@+id/circularImageView" | ||||
|                     app:layout_constraintTop_toTopOf="@+id/circularImageView" /> | ||||
| 
 | ||||
|             </androidx.constraintlayout.widget.ConstraintLayout> | ||||
| 
 | ||||
|             <!-- Título Información --> | ||||
|             <TextView | ||||
|                 android:id="@+id/textView2" | ||||
|                 android:layout_width="0dp" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:background="#FFFFFF" | ||||
|                 android:fontFamily="@font/assistant_semibold" | ||||
|                 android:gravity="center" | ||||
|                 android:text="Información" | ||||
|                 android:textColor="#324F5E" | ||||
|                 android:textSize="18sp" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/background_banner" /> | ||||
| 
 | ||||
|             <!-- Contenedor de duración y fallos --> | ||||
|             <androidx.constraintlayout.widget.ConstraintLayout | ||||
|                 android:id="@+id/stats_container" | ||||
|                 android:layout_width="0dp" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:background="#CCCCCC" | ||||
|                 android:padding="16dp" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@id/textView2"> | ||||
| 
 | ||||
|                 <!-- Duración --> | ||||
|                 <TextView | ||||
|                     android:id="@+id/label_number_of_games_last_week" | ||||
|                     android:layout_width="0dp" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:fontFamily="@font/assistant_semibold" | ||||
|                     android:text="Partidas última semana" | ||||
|                     android:textColor="#2C3E50" | ||||
|                     android:textSize="16sp" | ||||
|                     app:layout_constraintEnd_toStartOf="@+id/tv_number_of_games_last_week" | ||||
|                     app:layout_constraintStart_toStartOf="parent" | ||||
|                     app:layout_constraintTop_toTopOf="parent" /> | ||||
| 
 | ||||
|                 <TextView | ||||
|                     android:id="@+id/tv_number_of_games_last_week" | ||||
|                     android:layout_width="wrap_content" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:fontFamily="@font/assistant_semibold" | ||||
|                     android:textColor="#2C3E50" | ||||
|                     android:textSize="16sp" | ||||
|                     app:layout_constraintEnd_toEndOf="parent" | ||||
|                     app:layout_constraintTop_toTopOf="@+id/label_number_of_games_last_week" /> | ||||
| 
 | ||||
|                 <!-- Fallos --> | ||||
|                 <TextView | ||||
|                     android:id="@+id/label_total_games_played" | ||||
|                     android:layout_width="0dp" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:layout_marginTop="8dp" | ||||
|                     android:fontFamily="@font/assistant_semibold" | ||||
|                     android:text="Partidas jugadas" | ||||
|                     android:textColor="#2C3E50" | ||||
|                     android:textSize="16sp" | ||||
|                     app:layout_constraintEnd_toStartOf="@id/tv_total_games_played" | ||||
|                     app:layout_constraintStart_toStartOf="parent" | ||||
|                     app:layout_constraintTop_toBottomOf="@id/label_number_of_games_last_week" /> | ||||
| 
 | ||||
|                 <TextView | ||||
|                     android:id="@+id/tv_total_games_played" | ||||
|                     android:layout_width="wrap_content" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:fontFamily="@font/assistant_semibold" | ||||
|                     android:textColor="#2C3E50" | ||||
|                     android:textSize="16sp" | ||||
|                     app:layout_constraintEnd_toEndOf="parent" | ||||
|                     app:layout_constraintTop_toTopOf="@id/label_total_games_played" /> | ||||
| 
 | ||||
|                 <TextView | ||||
|                     android:id="@+id/label_top_jugador" | ||||
|                     android:layout_width="0dp" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:layout_marginTop="8dp" | ||||
|                     android:fontFamily="@font/assistant_semibold" | ||||
|                     android:text="Top jugador" | ||||
|                     android:textColor="#2C3E50" | ||||
|                     android:textSize="16sp" | ||||
|                     app:layout_constraintEnd_toStartOf="@id/tv_total_games_played" | ||||
|                     app:layout_constraintStart_toStartOf="parent" | ||||
|                     app:layout_constraintTop_toBottomOf="@id/label_total_games_played" /> | ||||
| 
 | ||||
|                 <TextView | ||||
|                     android:id="@+id/tv_top_jugador" | ||||
|                     android:layout_width="wrap_content" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:fontFamily="@font/assistant_semibold" | ||||
|                     android:textColor="#2C3E50" | ||||
|                     android:textSize="16sp" | ||||
|                     app:layout_constraintEnd_toEndOf="parent" | ||||
|                     app:layout_constraintTop_toTopOf="@id/label_top_jugador" /> | ||||
| 
 | ||||
|             </androidx.constraintlayout.widget.ConstraintLayout> | ||||
| 
 | ||||
|             <TextView | ||||
|                 android:id="@+id/textView6" | ||||
|                 android:layout_width="0dp" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginTop="16dp" | ||||
|                 android:scaleType="centerCrop" | ||||
|                 android:src="@drawable/woman_avatar" | ||||
|                 android:background="#FFFFFFFF" | ||||
|                 android:fontFamily="@font/assistant_semibold" | ||||
|                 android:gravity="center" | ||||
|                 android:text="Partidas" | ||||
|                 android:textColor="#324F5E" | ||||
|                 android:textSize="18sp" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@id/imageButton2" | ||||
|                 app:shapeAppearanceOverlay="@style/CircularShapeAppearance" /> | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/stats_container" /> | ||||
| 
 | ||||
|             <androidx.recyclerview.widget.RecyclerView | ||||
|                 android:id="@+id/rv_game_stats" | ||||
|                 android:layout_width="0dp" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginTop="10dp" | ||||
|                 android:nestedScrollingEnabled="false" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@id/textView6" /> | ||||
| 
 | ||||
|             <TextView | ||||
|                 android:id="@+id/banner_name_game" | ||||
|                 android:id="@+id/tv_lastgames_empty_game" | ||||
|                 android:layout_width="wrap_content" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginStart="10dp" | ||||
|                 android:layout_marginTop="15dp" | ||||
|                 android:layout_marginBottom="33dp" | ||||
|                 android:fontFamily="@font/assistant_bold" | ||||
|                 android:textColor="@android:color/white" | ||||
|                 android:textSize="24sp" | ||||
|                 app:layout_constraintBottom_toBottomOf="@+id/circularImageView" | ||||
|                 app:layout_constraintStart_toEndOf="@+id/circularImageView" | ||||
|                 app:layout_constraintTop_toTopOf="@+id/circularImageView" /> | ||||
| 
 | ||||
|                 android:layout_marginTop="5dp" | ||||
|                 android:fontFamily="@font/assistant_semibold" | ||||
|                 android:text="No hay partidas disponibles" | ||||
|                 android:textColor="#96A7AF" | ||||
|                 android:visibility="gone" | ||||
|                 app:layout_constraintEnd_toEndOf="@+id/textView6" | ||||
|                 app:layout_constraintStart_toStartOf="@+id/textView6" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/textView6" /> | ||||
|         </androidx.constraintlayout.widget.ConstraintLayout> | ||||
| 
 | ||||
|         <!-- Título Información --> | ||||
|         <TextView | ||||
|             android:id="@+id/textView2" | ||||
|             android:layout_width="0dp" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:background="#FFFFFF" | ||||
|             android:fontFamily="@font/assistant_semibold" | ||||
|             android:gravity="center" | ||||
|             android:text="Información" | ||||
|             android:textColor="#324F5E" | ||||
|             android:textSize="18sp" | ||||
|             app:layout_constraintEnd_toEndOf="parent" | ||||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintTop_toBottomOf="@+id/background_banner" /> | ||||
| 
 | ||||
|         <!-- Contenedor de duración y fallos --> | ||||
|         <androidx.constraintlayout.widget.ConstraintLayout | ||||
|             android:id="@+id/stats_container" | ||||
|             android:layout_width="0dp" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:background="#CCCCCC" | ||||
|             android:padding="16dp" | ||||
|             app:layout_constraintEnd_toEndOf="parent" | ||||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintTop_toBottomOf="@id/textView2"> | ||||
| 
 | ||||
|             <!-- Duración --> | ||||
|             <TextView | ||||
|                 android:id="@+id/label_number_of_games_last_week" | ||||
|                 android:layout_width="0dp" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:fontFamily="@font/assistant_semibold" | ||||
|                 android:text="Partidas úlrima semana" | ||||
|                 android:textColor="#2C3E50" | ||||
|                 android:textSize="16sp" | ||||
|                 app:layout_constraintEnd_toStartOf="@+id/tv_number_of_games_last_week" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toTopOf="parent" /> | ||||
| 
 | ||||
|             <TextView | ||||
|                 android:id="@+id/tv_number_of_games_last_week" | ||||
|                 android:layout_width="wrap_content" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:fontFamily="@font/assistant_semibold" | ||||
|                 android:textColor="#2C3E50" | ||||
|                 android:textSize="16sp" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintTop_toTopOf="@+id/label_number_of_games_last_week" /> | ||||
| 
 | ||||
|             <!-- Fallos --> | ||||
|             <TextView | ||||
|                 android:id="@+id/label_total_games_played" | ||||
|                 android:layout_width="0dp" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginTop="8dp" | ||||
|                 android:fontFamily="@font/assistant_semibold" | ||||
|                 android:text="Partidas jugadas" | ||||
|                 android:textColor="#2C3E50" | ||||
|                 android:textSize="16sp" | ||||
|                 app:layout_constraintEnd_toStartOf="@id/tv_total_games_played" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@id/label_number_of_games_last_week" /> | ||||
| 
 | ||||
|             <TextView | ||||
|                 android:id="@+id/tv_total_games_played" | ||||
|                 android:layout_width="wrap_content" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:fontFamily="@font/assistant_semibold" | ||||
|                 android:textColor="#2C3E50" | ||||
|                 android:textSize="16sp" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintTop_toTopOf="@id/label_total_games_played" /> | ||||
| 
 | ||||
|             <TextView | ||||
|                 android:id="@+id/label_top_jugador" | ||||
|                 android:layout_width="0dp" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginTop="8dp" | ||||
|                 android:fontFamily="@font/assistant_semibold" | ||||
|                 android:text="Top jugador" | ||||
|                 android:textColor="#2C3E50" | ||||
|                 android:textSize="16sp" | ||||
|                 app:layout_constraintEnd_toStartOf="@id/tv_total_games_played" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@id/label_total_games_played" /> | ||||
| 
 | ||||
|             <TextView | ||||
|                 android:id="@+id/tv_top_jugador" | ||||
|                 android:layout_width="wrap_content" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:fontFamily="@font/assistant_semibold" | ||||
|                 android:textColor="#2C3E50" | ||||
|                 android:textSize="16sp" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintTop_toTopOf="@id/label_top_jugador" /> | ||||
| 
 | ||||
|         </androidx.constraintlayout.widget.ConstraintLayout> | ||||
| 
 | ||||
|         <TextView | ||||
|             android:id="@+id/textView6" | ||||
|             android:layout_width="0dp" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:layout_marginTop="16dp" | ||||
|             android:background="#FFFFFFFF" | ||||
|             android:fontFamily="@font/assistant_semibold" | ||||
|             android:gravity="center" | ||||
|             android:text="Partidas" | ||||
|             android:textColor="#324F5E" | ||||
|             android:textSize="18sp" | ||||
|             app:layout_constraintEnd_toEndOf="parent" | ||||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintTop_toBottomOf="@+id/stats_container" /> | ||||
| 
 | ||||
|         <androidx.recyclerview.widget.RecyclerView | ||||
|             android:id="@+id/rv_game_stats" | ||||
|             android:layout_width="0dp" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:layout_marginTop="10dp" | ||||
|             android:nestedScrollingEnabled="false" | ||||
|             app:layout_constraintEnd_toEndOf="parent" | ||||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintTop_toBottomOf="@id/textView6" /> | ||||
|     </androidx.constraintlayout.widget.ConstraintLayout> | ||||
| </androidx.core.widget.NestedScrollView> | ||||
|     </androidx.core.widget.NestedScrollView> | ||||
| </androidx.swiperefreshlayout.widget.SwipeRefreshLayout> | ||||
|  |  | |||
|  | @ -1,178 +1,182 @@ | |||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <androidx.core.widget.NestedScrollView | ||||
|     xmlns:android="http://schemas.android.com/apk/res/android" | ||||
| <androidx.swiperefreshlayout.widget.SwipeRefreshLayout xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||
|     android:id="@+id/swipe_refresh_game_detail" | ||||
|     android:layout_width="match_parent" | ||||
|     android:layout_height="match_parent" | ||||
|     android:fillViewport="true" | ||||
|     android:background="#CCCCCC"> | ||||
|     android:layout_height="match_parent"> | ||||
| 
 | ||||
|     <androidx.constraintlayout.widget.ConstraintLayout | ||||
|     <androidx.core.widget.NestedScrollView | ||||
|         android:layout_width="match_parent" | ||||
|         android:layout_height="wrap_content"> | ||||
|         android:layout_height="match_parent" | ||||
|         android:background="#CCCCCC" | ||||
|         android:fillViewport="true"> | ||||
| 
 | ||||
|         <!-- Banner azul --> | ||||
|         <androidx.constraintlayout.widget.ConstraintLayout | ||||
|             android:id="@+id/background_banner" | ||||
|             android:layout_width="0dp" | ||||
|             android:layout_height="174dp" | ||||
|             android:background="#0062FF" | ||||
|             app:layout_constraintTop_toTopOf="parent" | ||||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintEnd_toEndOf="parent"> | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="wrap_content"> | ||||
| 
 | ||||
|             <!-- Botón superior izquierdo --> | ||||
|             <ImageButton | ||||
|                 android:id="@+id/imageButton2" | ||||
|                 android:layout_width="wrap_content" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_margin="8dp" | ||||
|                 android:background="@null" | ||||
|                 app:srcCompat="@drawable/arrow_left" | ||||
|                 app:layout_constraintTop_toTopOf="parent" | ||||
|                 app:layout_constraintStart_toStartOf="parent" /> | ||||
| 
 | ||||
|             <!-- Avatar + texto --> | ||||
|             <com.google.android.material.imageview.ShapeableImageView | ||||
|                 android:id="@+id/circularImageView" | ||||
|                 android:layout_width="80dp" | ||||
|                 android:layout_height="80dp" | ||||
|                 android:scaleType="centerCrop" | ||||
|                 android:src="@drawable/woman_avatar" | ||||
|                 app:shapeAppearanceOverlay="@style/CircularShapeAppearance" | ||||
|                 app:layout_constraintTop_toBottomOf="@id/imageButton2" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 android:layout_marginTop="16dp" | ||||
|                 android:layout_marginStart="20dp" /> | ||||
| 
 | ||||
|             <LinearLayout | ||||
|                 android:id="@+id/user_info" | ||||
|             <!-- Banner azul --> | ||||
|             <androidx.constraintlayout.widget.ConstraintLayout | ||||
|                 android:id="@+id/background_banner" | ||||
|                 android:layout_width="0dp" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:orientation="vertical" | ||||
|                 app:layout_constraintTop_toTopOf="@id/circularImageView" | ||||
|                 app:layout_constraintBottom_toBottomOf="@id/circularImageView" | ||||
|                 app:layout_constraintStart_toEndOf="@id/circularImageView" | ||||
|                 android:layout_height="174dp" | ||||
|                 android:background="#0062FF" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 android:layout_marginStart="12dp" | ||||
|                 android:layout_marginEnd="12dp"> | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toTopOf="parent"> | ||||
| 
 | ||||
|                 <TextView | ||||
|                     android:id="@+id/banner_name_resident" | ||||
|                 <!-- Botón superior izquierdo --> | ||||
|                 <ImageButton | ||||
|                     android:id="@+id/back_button" | ||||
|                     android:layout_width="wrap_content" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:fontFamily="@font/assistant_bold" | ||||
|                     android:textColor="@android:color/white" | ||||
|                     android:textSize="24sp" /> | ||||
|                     android:layout_margin="8dp" | ||||
|                     android:background="@null" | ||||
|                     app:layout_constraintStart_toStartOf="parent" | ||||
|                     app:layout_constraintTop_toTopOf="parent" | ||||
|                     app:srcCompat="@drawable/arrow_left" /> | ||||
| 
 | ||||
|                 <!-- Avatar + texto --> | ||||
|                 <com.google.android.material.imageview.ShapeableImageView | ||||
|                     android:id="@+id/circularImageView" | ||||
|                     android:layout_width="80dp" | ||||
|                     android:layout_height="80dp" | ||||
|                     android:layout_marginStart="20dp" | ||||
|                     android:layout_marginTop="16dp" | ||||
|                     android:scaleType="centerCrop" | ||||
|                     android:src="@drawable/woman_avatar" | ||||
|                     app:layout_constraintStart_toStartOf="parent" | ||||
|                     app:layout_constraintTop_toBottomOf="@id/back_button" | ||||
|                     app:shapeAppearanceOverlay="@style/CircularShapeAppearance" /> | ||||
| 
 | ||||
|                 <LinearLayout | ||||
|                     android:id="@+id/user_info" | ||||
|                     android:layout_width="0dp" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:layout_marginStart="12dp" | ||||
|                     android:layout_marginEnd="12dp" | ||||
|                     android:orientation="vertical" | ||||
|                     app:layout_constraintBottom_toBottomOf="@id/circularImageView" | ||||
|                     app:layout_constraintEnd_toEndOf="parent" | ||||
|                     app:layout_constraintStart_toEndOf="@id/circularImageView" | ||||
|                     app:layout_constraintTop_toTopOf="@id/circularImageView"> | ||||
| 
 | ||||
|                     <TextView | ||||
|                         android:id="@+id/banner_name_resident" | ||||
|                         android:layout_width="wrap_content" | ||||
|                         android:layout_height="wrap_content" | ||||
|                         android:fontFamily="@font/assistant_bold" | ||||
|                         android:textColor="@android:color/white" | ||||
|                         android:textSize="24sp" /> | ||||
| 
 | ||||
|                     <TextView | ||||
|                         android:id="@+id/game_stats_detail_game_name" | ||||
|                         android:layout_width="wrap_content" | ||||
|                         android:layout_height="wrap_content" | ||||
|                         android:fontFamily="@font/assistant_semibold" | ||||
|                         android:textColor="@android:color/white" /> | ||||
|                 </LinearLayout> | ||||
|             </androidx.constraintlayout.widget.ConstraintLayout> | ||||
| 
 | ||||
|             <!-- Título Información --> | ||||
|             <TextView | ||||
|                 android:id="@+id/textView2" | ||||
|                 android:layout_width="0dp" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:background="#FFFFFF" | ||||
|                 android:fontFamily="@font/assistant_semibold" | ||||
|                 android:gravity="center" | ||||
|                 android:text="Información" | ||||
|                 android:textColor="#324F5E" | ||||
|                 android:textSize="18sp" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/background_banner" /> | ||||
| 
 | ||||
|             <!-- Contenedor de duración y fallos --> | ||||
|             <androidx.constraintlayout.widget.ConstraintLayout | ||||
|                 android:id="@+id/stats_container" | ||||
|                 android:layout_width="0dp" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:background="#CCCCCC" | ||||
|                 android:padding="16dp" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@id/textView2"> | ||||
| 
 | ||||
|                 <!-- Duración --> | ||||
|                 <TextView | ||||
|                     android:id="@+id/label_duration" | ||||
|                     android:layout_width="0dp" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:fontFamily="@font/assistant_semibold" | ||||
|                     android:text="Duración" | ||||
|                     android:textColor="#2C3E50" | ||||
|                     android:textSize="16sp" | ||||
|                     app:layout_constraintEnd_toStartOf="@+id/tv_duration" | ||||
|                     app:layout_constraintStart_toStartOf="parent" | ||||
|                     app:layout_constraintTop_toTopOf="parent" /> | ||||
| 
 | ||||
|                 <TextView | ||||
|                     android:id="@+id/game_stats_detail_game_name" | ||||
|                     android:id="@+id/tv_duration" | ||||
|                     android:layout_width="wrap_content" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:fontFamily="@font/assistant_semibold" | ||||
|                     android:textColor="@android:color/white" /> | ||||
|             </LinearLayout> | ||||
|         </androidx.constraintlayout.widget.ConstraintLayout> | ||||
|                     android:textColor="#2C3E50" | ||||
|                     android:textSize="16sp" | ||||
|                     app:layout_constraintEnd_toEndOf="parent" | ||||
|                     app:layout_constraintTop_toTopOf="@+id/label_duration" /> | ||||
| 
 | ||||
|         <!-- Título Información --> | ||||
|         <TextView | ||||
|             android:id="@+id/textView2" | ||||
|             android:layout_width="0dp" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:background="#FFFFFF" | ||||
|             android:fontFamily="@font/assistant_semibold" | ||||
|             android:gravity="center" | ||||
|             android:text="Información" | ||||
|             android:textColor="#324F5E" | ||||
|             android:textSize="18sp" | ||||
|             app:layout_constraintEnd_toEndOf="parent" | ||||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintTop_toBottomOf="@+id/background_banner" /> | ||||
|                 <!-- Fallos --> | ||||
|                 <TextView | ||||
|                     android:id="@+id/label_fails" | ||||
|                     android:layout_width="0dp" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:layout_marginTop="8dp" | ||||
|                     android:fontFamily="@font/assistant_semibold" | ||||
|                     android:text="Fallos" | ||||
|                     android:textColor="#2C3E50" | ||||
|                     android:textSize="16sp" | ||||
|                     app:layout_constraintEnd_toStartOf="@id/tv_fails" | ||||
|                     app:layout_constraintStart_toStartOf="parent" | ||||
|                     app:layout_constraintTop_toBottomOf="@id/label_duration" /> | ||||
| 
 | ||||
|         <!-- Contenedor de duración y fallos --> | ||||
|         <androidx.constraintlayout.widget.ConstraintLayout | ||||
|             android:id="@+id/stats_container" | ||||
|             android:layout_width="0dp" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:background="#CCCCCC" | ||||
|             android:padding="16dp" | ||||
|             app:layout_constraintEnd_toEndOf="parent" | ||||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintTop_toBottomOf="@id/textView2"> | ||||
|                 <TextView | ||||
|                     android:id="@+id/tv_fails" | ||||
|                     android:layout_width="wrap_content" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:fontFamily="@font/assistant_semibold" | ||||
|                     android:textColor="#2C3E50" | ||||
|                     android:textSize="16sp" | ||||
|                     app:layout_constraintEnd_toEndOf="parent" | ||||
|                     app:layout_constraintTop_toTopOf="@id/label_fails" /> | ||||
|             </androidx.constraintlayout.widget.ConstraintLayout> | ||||
| 
 | ||||
|             <!-- Duración --> | ||||
|             <!-- Observaciones --> | ||||
|             <TextView | ||||
|                 android:id="@+id/label_duration" | ||||
|                 android:id="@+id/textView6" | ||||
|                 android:layout_width="0dp" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginTop="16dp" | ||||
|                 android:background="#FFFFFFFF" | ||||
|                 android:fontFamily="@font/assistant_semibold" | ||||
|                 android:text="Duración" | ||||
|                 android:textColor="#2C3E50" | ||||
|                 android:textSize="16sp" | ||||
|                 app:layout_constraintEnd_toStartOf="@+id/tv_duration" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toTopOf="parent" /> | ||||
| 
 | ||||
|             <TextView | ||||
|                 android:id="@+id/tv_duration" | ||||
|                 android:layout_width="wrap_content" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:fontFamily="@font/assistant_semibold" | ||||
|                 android:textColor="#2C3E50" | ||||
|                 android:textSize="16sp" | ||||
|                 android:gravity="center" | ||||
|                 android:text="Observacion" | ||||
|                 android:textColor="#324F5E" | ||||
|                 android:textSize="18sp" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintTop_toTopOf="@+id/label_duration" /> | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/stats_container" /> | ||||
| 
 | ||||
|             <!-- Fallos --> | ||||
|             <TextView | ||||
|                 android:id="@+id/label_fails" | ||||
|             <include | ||||
|                 android:id="@+id/observation" | ||||
|                 layout="@layout/item_observation" | ||||
|                 android:layout_width="0dp" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginTop="8dp" | ||||
|                 android:fontFamily="@font/assistant_semibold" | ||||
|                 android:text="Fallos" | ||||
|                 android:textColor="#2C3E50" | ||||
|                 android:textSize="16sp" | ||||
|                 app:layout_constraintEnd_toStartOf="@id/tv_fails" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@id/label_duration" /> | ||||
| 
 | ||||
|             <TextView | ||||
|                 android:id="@+id/tv_fails" | ||||
|                 android:layout_width="wrap_content" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:fontFamily="@font/assistant_semibold" | ||||
|                 android:textColor="#2C3E50" | ||||
|                 android:textSize="16sp" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintTop_toTopOf="@id/label_fails" /> | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/textView6" /> | ||||
| 
 | ||||
|         </androidx.constraintlayout.widget.ConstraintLayout> | ||||
| 
 | ||||
|         <!-- Observaciones --> | ||||
|         <TextView | ||||
|             android:id="@+id/textView6" | ||||
|             android:layout_width="0dp" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:layout_marginTop="16dp" | ||||
|             android:background="#FFFFFFFF" | ||||
|             android:fontFamily="@font/assistant_semibold" | ||||
|             android:gravity="center" | ||||
|             android:text="Observaciones" | ||||
|             android:textColor="#324F5E" | ||||
|             android:textSize="18sp" | ||||
|             app:layout_constraintEnd_toEndOf="parent" | ||||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintTop_toBottomOf="@+id/stats_container" /> | ||||
| 
 | ||||
|         <androidx.recyclerview.widget.RecyclerView | ||||
|             android:id="@+id/last_games_recycleView" | ||||
|             android:layout_width="0dp" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:layout_marginTop="10dp" | ||||
|             android:nestedScrollingEnabled="false" | ||||
|             app:layout_constraintEnd_toEndOf="parent" | ||||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintTop_toBottomOf="@id/textView6" /> | ||||
|     </androidx.constraintlayout.widget.ConstraintLayout> | ||||
| </androidx.core.widget.NestedScrollView> | ||||
|     </androidx.core.widget.NestedScrollView> | ||||
| </androidx.swiperefreshlayout.widget.SwipeRefreshLayout> | ||||
|  |  | |||
|  | @ -1,194 +1,239 @@ | |||
| <androidx.core.widget.NestedScrollView | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <androidx.swiperefreshlayout.widget.SwipeRefreshLayout | ||||
|     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/swipe_refresh_home" | ||||
|     android:layout_width="match_parent" | ||||
|     android:layout_height="match_parent" | ||||
|     android:fillViewport="true"> | ||||
|     android:layout_height="match_parent"> | ||||
| 
 | ||||
|     <androidx.constraintlayout.widget.ConstraintLayout | ||||
|     <androidx.core.widget.NestedScrollView | ||||
|         android:layout_width="match_parent" | ||||
|         android:layout_height="wrap_content" | ||||
|         android:background="#CCCCCC" | ||||
|         tools:context=".ui.fragments.home.HomeFragment"> | ||||
|         android:layout_height="match_parent" | ||||
|         android:fillViewport="true"> | ||||
| 
 | ||||
|         <LinearLayout | ||||
|             android:id="@+id/linearLayout" | ||||
|         <androidx.constraintlayout.widget.ConstraintLayout | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:layout_margin="10dp" | ||||
|             android:orientation="horizontal" | ||||
|             app:layout_constraintEnd_toEndOf="parent" | ||||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintTop_toTopOf="parent"> | ||||
|             android:background="#CCCCCC" | ||||
|             tools:context=".ui.fragments.home.HomeFragment"> | ||||
| 
 | ||||
|             <LinearLayout | ||||
|                 android:layout_width="0dp" | ||||
|                 android:id="@+id/linearLayout" | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_weight="1" | ||||
|                 android:gravity="center" | ||||
|                 android:orientation="vertical"> | ||||
|                 android:layout_margin="10dp" | ||||
|                 android:orientation="horizontal" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toTopOf="parent"> | ||||
| 
 | ||||
|                 <TextView | ||||
|                     android:id="@+id/home_user_name" | ||||
|                     android:layout_width="match_parent" | ||||
|                 <LinearLayout | ||||
|                     android:layout_width="0dp" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:layout_marginStart="5dp" | ||||
|                     android:fontFamily="@font/assistant_bold" | ||||
|                     android:text="Maria Cartes Sanchez" | ||||
|                     android:textColor="#324F5E" | ||||
|                     android:textSize="24sp" /> | ||||
|                     android:layout_weight="1" | ||||
|                     android:gravity="center" | ||||
|                     android:orientation="vertical"> | ||||
| 
 | ||||
|                 <TextView | ||||
|                     android:id="@+id/observation_person_role" | ||||
|                     android:layout_width="match_parent" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:layout_marginStart="5dp" | ||||
|                     android:fontFamily="@font/assistant_semibold" | ||||
|                     android:text="Psicologa" | ||||
|                     android:textColor="#96A7AF" | ||||
|                     android:textSize="16sp" /> | ||||
|                     <TextView | ||||
|                         android:id="@+id/home_user_name" | ||||
|                         android:layout_width="match_parent" | ||||
|                         android:layout_height="wrap_content" | ||||
|                         android:layout_marginStart="5dp" | ||||
|                         android:fontFamily="@font/assistant_bold" | ||||
|                         android:text="Maria Cartes Sanchez" | ||||
|                         android:textColor="#324F5E" | ||||
|                         android:textSize="24sp" /> | ||||
| 
 | ||||
|                     <TextView | ||||
|                         android:id="@+id/observation_person_role" | ||||
|                         android:layout_width="match_parent" | ||||
|                         android:layout_height="wrap_content" | ||||
|                         android:layout_marginStart="5dp" | ||||
|                         android:fontFamily="@font/assistant_semibold" | ||||
|                         android:text="Psicologa" | ||||
|                         android:textColor="#96A7AF" | ||||
|                         android:textSize="16sp" /> | ||||
|                 </LinearLayout> | ||||
| 
 | ||||
|                 <com.google.android.material.imageview.ShapeableImageView | ||||
|                     android:id="@+id/user_image_home" | ||||
|                     android:layout_width="55dp" | ||||
|                     android:layout_height="match_parent" | ||||
|                     android:scaleType="centerCrop" | ||||
|                     app:shapeAppearanceOverlay="@style/CircularShapeAppearance" | ||||
|                     app:srcCompat="@drawable/woman_avatar" /> | ||||
|             </LinearLayout> | ||||
| 
 | ||||
|             <com.google.android.material.imageview.ShapeableImageView | ||||
|                 android:id="@+id/circularImageView3" | ||||
|                 android:layout_width="55dp" | ||||
|                 android:layout_height="match_parent" | ||||
|                 android:scaleType="centerCrop" | ||||
|                 app:shapeAppearanceOverlay="@style/CircularShapeAppearance" | ||||
|                 app:srcCompat="@drawable/woman_avatar" /> | ||||
|         </LinearLayout> | ||||
|             <LinearLayout | ||||
|                 android:id="@+id/linearLayout2" | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginHorizontal="20dp" | ||||
|                 android:layout_marginTop="30dp" | ||||
|                 android:orientation="horizontal" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@id/linearLayout"> | ||||
| 
 | ||||
|         <LinearLayout | ||||
|             android:id="@+id/linearLayout2" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:layout_marginHorizontal="20dp" | ||||
|             android:layout_marginTop="30dp" | ||||
|             android:orientation="horizontal" | ||||
|             app:layout_constraintEnd_toEndOf="parent" | ||||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintTop_toBottomOf="@id/linearLayout"> | ||||
|                 <TextView | ||||
|                     android:id="@+id/textView15" | ||||
|                     android:layout_width="0dp" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:layout_weight="1" | ||||
|                     android:fontFamily="@font/assistant_bold" | ||||
|                     android:text="Últimas partidas" | ||||
|                     android:textColor="#324F5E" | ||||
|                     android:textSize="20sp" /> | ||||
| 
 | ||||
|                 <TextView | ||||
|                     android:id="@+id/textView17" | ||||
|                     android:layout_width="0dp" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:layout_weight="1" | ||||
|                     android:fontFamily="@font/assistant_semibold" | ||||
|                     android:gravity="right" | ||||
|                     android:text="Ver más" | ||||
|                     android:textColor="#0062FF" | ||||
|                     android:textSize="14sp" /> | ||||
|             </LinearLayout> | ||||
| 
 | ||||
|             <androidx.recyclerview.widget.RecyclerView | ||||
|                 android:id="@+id/latestGames_recycleView_home" | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginHorizontal="10dp" | ||||
|                 android:layout_marginTop="10dp" | ||||
|                 android:nestedScrollingEnabled="false" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@id/linearLayout2" /> | ||||
| 
 | ||||
|             <TextView | ||||
|                 android:id="@+id/textView15" | ||||
|                 android:layout_width="0dp" | ||||
|                 android:id="@+id/tv_lastgames_empty_home" | ||||
|                 android:layout_width="wrap_content" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_weight="1" | ||||
|                 android:fontFamily="@font/assistant_bold" | ||||
|                 android:text="Últimas partidas" | ||||
|                 android:textColor="#324F5E" | ||||
|                 android:textSize="20sp" /> | ||||
| 
 | ||||
|             <TextView | ||||
|                 android:id="@+id/textView17" | ||||
|                 android:layout_width="0dp" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_weight="1" | ||||
|                 android:layout_marginTop="10dp" | ||||
|                 android:fontFamily="@font/assistant_semibold" | ||||
|                 android:gravity="right" | ||||
|                 android:text="Ver más" | ||||
|                 android:textColor="#0062FF" | ||||
|                 android:textSize="14sp" /> | ||||
|         </LinearLayout> | ||||
|                 android:text="No hay partidas disponibles" | ||||
|                 android:textColor="#96A7AF" | ||||
|                 android:visibility="gone" | ||||
|                 app:layout_constraintEnd_toEndOf="@+id/linearLayout2" | ||||
|                 app:layout_constraintStart_toStartOf="@+id/linearLayout2" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/linearLayout2" /> | ||||
| 
 | ||||
|         <androidx.recyclerview.widget.RecyclerView | ||||
|             android:id="@+id/latestGames_recycleView_home" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:layout_marginHorizontal="10dp" | ||||
|             android:layout_marginTop="10dp" | ||||
|             android:nestedScrollingEnabled="false" | ||||
|             app:layout_constraintEnd_toEndOf="parent" | ||||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintTop_toBottomOf="@id/linearLayout2" /> | ||||
|             <LinearLayout | ||||
|                 android:id="@+id/linearLayout4" | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginHorizontal="20dp" | ||||
|                 android:layout_marginTop="20dp" | ||||
|                 android:orientation="horizontal" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@id/latestGames_recycleView_home"> | ||||
| 
 | ||||
|         <LinearLayout | ||||
|             android:id="@+id/linearLayout4" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:layout_marginHorizontal="20dp" | ||||
|             android:layout_marginTop="20dp" | ||||
|             android:orientation="horizontal" | ||||
|             app:layout_constraintEnd_toEndOf="parent" | ||||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintTop_toBottomOf="@id/latestGames_recycleView_home"> | ||||
|                 <TextView | ||||
|                     android:id="@+id/textView19" | ||||
|                     android:layout_width="0dp" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:layout_weight="1" | ||||
|                     android:fontFamily="@font/assistant_bold" | ||||
|                     android:text="Juegos" | ||||
|                     android:textColor="#324F5E" | ||||
|                     android:textSize="20sp" /> | ||||
| 
 | ||||
|                 <ImageButton | ||||
|                     android:id="@+id/filter_games_list_button" | ||||
|                     android:layout_width="wrap_content" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:layout_gravity="center" | ||||
|                     android:background="@android:color/transparent" | ||||
|                     android:contentDescription="Filtro" | ||||
|                     app:srcCompat="@drawable/filter" /> | ||||
|             </LinearLayout> | ||||
| 
 | ||||
|             <androidx.recyclerview.widget.RecyclerView | ||||
|                 android:id="@+id/games_recycleView_home" | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginHorizontal="10dp" | ||||
|                 android:layout_marginTop="10dp" | ||||
|                 android:nestedScrollingEnabled="false" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@id/linearLayout4" /> | ||||
| 
 | ||||
|             <TextView | ||||
|                 android:id="@+id/textView19" | ||||
|                 android:layout_width="0dp" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_weight="1" | ||||
|                 android:fontFamily="@font/assistant_bold" | ||||
|                 android:text="Juegos" | ||||
|                 android:textColor="#324F5E" | ||||
|                 android:textSize="20sp" /> | ||||
| 
 | ||||
|             <ImageButton | ||||
|                 android:id="@+id/filter_games_list_button" | ||||
|                 android:id="@+id/tv_games_empty_home" | ||||
|                 android:layout_width="wrap_content" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_gravity="center" | ||||
|                 android:background="@android:color/transparent" | ||||
|                 android:contentDescription="Filtro" | ||||
|                 app:srcCompat="@drawable/filter" /> | ||||
|                 android:layout_marginTop="10dp" | ||||
|                 android:fontFamily="@font/assistant_semibold" | ||||
|                 android:text="No hay juegos disponibles" | ||||
|                 android:textColor="#96A7AF" | ||||
|                 android:visibility="gone" | ||||
|                 app:layout_constraintEnd_toEndOf="@+id/linearLayout4" | ||||
|                 app:layout_constraintStart_toStartOf="@+id/linearLayout4" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/linearLayout4" /> | ||||
| 
 | ||||
|         </LinearLayout> | ||||
|             <LinearLayout | ||||
|                 android:id="@+id/linearLayout3" | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginHorizontal="20dp" | ||||
|                 android:layout_marginTop="20dp" | ||||
|                 android:orientation="horizontal" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@id/games_recycleView_home"> | ||||
| 
 | ||||
|         <androidx.recyclerview.widget.RecyclerView | ||||
|             android:id="@+id/games_recycleView_home" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:layout_marginHorizontal="10dp" | ||||
|             android:layout_marginTop="10dp" | ||||
|             android:nestedScrollingEnabled="false" | ||||
|             app:layout_constraintEnd_toEndOf="parent" | ||||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintTop_toBottomOf="@id/linearLayout4" /> | ||||
|                 <TextView | ||||
|                     android:id="@+id/textView18" | ||||
|                     android:layout_width="0dp" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:layout_weight="1" | ||||
|                     android:fontFamily="@font/assistant_bold" | ||||
|                     android:text="Residentes" | ||||
|                     android:textColor="#324F5E" | ||||
|                     android:textSize="20sp" /> | ||||
| 
 | ||||
|         <LinearLayout | ||||
|             android:id="@+id/linearLayout3" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:layout_marginHorizontal="20dp" | ||||
|             android:layout_marginTop="20dp" | ||||
|             android:orientation="horizontal" | ||||
|             app:layout_constraintEnd_toEndOf="parent" | ||||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintTop_toBottomOf="@id/games_recycleView_home"> | ||||
|                 <ImageButton | ||||
|                     android:id="@+id/filter_residents_list_button" | ||||
|                     android:layout_width="wrap_content" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:layout_gravity="center" | ||||
|                     android:background="@android:color/transparent" | ||||
|                     android:contentDescription="Filtro" | ||||
|                     app:srcCompat="@drawable/filter" /> | ||||
|             </LinearLayout> | ||||
| 
 | ||||
|             <androidx.recyclerview.widget.RecyclerView | ||||
|                 android:id="@+id/residents_recycleView_home" | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginHorizontal="10dp" | ||||
|                 android:layout_marginTop="10dp" | ||||
|                 android:nestedScrollingEnabled="false" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@id/linearLayout3" /> | ||||
| 
 | ||||
|             <TextView | ||||
|                 android:id="@+id/textView18" | ||||
|                 android:layout_width="0dp" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_weight="1" | ||||
|                 android:fontFamily="@font/assistant_bold" | ||||
|                 android:text="Residentes" | ||||
|                 android:textColor="#324F5E" | ||||
|                 android:textSize="20sp" /> | ||||
| 
 | ||||
|             <ImageButton | ||||
|                 android:id="@+id/filter_residents_list_button" | ||||
|                 android:id="@+id/tv_residents_empty_home" | ||||
|                 android:layout_width="wrap_content" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_gravity="center" | ||||
|                 android:background="@android:color/transparent" | ||||
|                 android:contentDescription="Filtro" | ||||
|                 app:srcCompat="@drawable/filter" /> | ||||
|                 android:layout_marginTop="10dp" | ||||
|                 android:fontFamily="@font/assistant_semibold" | ||||
|                 android:text="No hay residentes disponibles" | ||||
|                 android:textColor="#96A7AF" | ||||
|                 android:visibility="gone" | ||||
|                 app:layout_constraintEnd_toEndOf="@+id/linearLayout3" | ||||
|                 app:layout_constraintStart_toStartOf="@+id/linearLayout3" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/linearLayout3" /> | ||||
| 
 | ||||
|         </LinearLayout> | ||||
|         </androidx.constraintlayout.widget.ConstraintLayout> | ||||
| 
 | ||||
|         <androidx.recyclerview.widget.RecyclerView | ||||
|             android:id="@+id/residents_recycleView_home" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:layout_marginHorizontal="10dp" | ||||
|             android:layout_marginTop="10dp" | ||||
|             android:nestedScrollingEnabled="false" | ||||
|             app:layout_constraintEnd_toEndOf="parent" | ||||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintTop_toBottomOf="@id/linearLayout3" /> | ||||
|     </androidx.core.widget.NestedScrollView> | ||||
| 
 | ||||
|     </androidx.constraintlayout.widget.ConstraintLayout> | ||||
| 
 | ||||
| </androidx.core.widget.NestedScrollView> | ||||
| </androidx.swiperefreshlayout.widget.SwipeRefreshLayout> | ||||
|  |  | |||
|  | @ -1,6 +1,7 @@ | |||
| <?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:layout_width="match_parent" | ||||
|     android:layout_height="match_parent" | ||||
|     android:padding="24dp"> | ||||
|  | @ -15,13 +16,13 @@ | |||
|         android:textColor="#324F5E" | ||||
|         android:textSize="24sp" | ||||
|         android:textStyle="bold" | ||||
|         app:layout_constraintBottom_toTopOf="@+id/email_input" | ||||
|         app:layout_constraintBottom_toTopOf="@+id/email_input_login" | ||||
|         app:layout_constraintEnd_toEndOf="parent" | ||||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintTop_toTopOf="parent" /> | ||||
| 
 | ||||
|     <EditText | ||||
|         android:id="@+id/email_input" | ||||
|         android:id="@+id/email_input_login" | ||||
|         android:layout_width="0dp" | ||||
|         android:layout_height="wrap_content" | ||||
|         android:layout_marginTop="32dp" | ||||
|  | @ -33,7 +34,7 @@ | |||
|         app:layout_constraintTop_toBottomOf="@id/login_title" /> | ||||
| 
 | ||||
|     <EditText | ||||
|         android:id="@+id/password_input" | ||||
|         android:id="@+id/password_input_login" | ||||
|         android:layout_width="0dp" | ||||
|         android:layout_height="wrap_content" | ||||
|         android:layout_marginTop="16dp" | ||||
|  | @ -42,17 +43,25 @@ | |||
|         android:textColorLink="#324F5E" | ||||
|         app:layout_constraintEnd_toEndOf="parent" | ||||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintTop_toBottomOf="@id/email_input" /> | ||||
|         app:layout_constraintTop_toBottomOf="@id/email_input_login" /> | ||||
| 
 | ||||
|     <CheckBox | ||||
|         android:id="@+id/remember_password_checkBox" | ||||
|         android:layout_width="wrap_content" | ||||
|         android:layout_height="wrap_content" | ||||
|         android:layout_marginTop="8dp" | ||||
|         android:text="Recordar contraseña" | ||||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintTop_toBottomOf="@+id/password_input_login" /> | ||||
| 
 | ||||
|     <Button | ||||
|         android:id="@+id/login_button" | ||||
|         android:layout_width="wrap_content" | ||||
|         android:layout_height="wrap_content" | ||||
|         android:text="Iniciar sesión" | ||||
|         android:backgroundTint="#324F5E" | ||||
|         app:layout_constraintTop_toBottomOf="@id/password_input" | ||||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         android:text="Iniciar sesión" | ||||
|         app:layout_constraintEnd_toEndOf="parent" | ||||
|         android:layout_marginTop="24dp" /> | ||||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintTop_toBottomOf="@+id/remember_password_checkBox" /> | ||||
| 
 | ||||
| </androidx.constraintlayout.widget.ConstraintLayout> | ||||
|  |  | |||
|  | @ -1,169 +1,207 @@ | |||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <androidx.core.widget.NestedScrollView | ||||
| <androidx.swiperefreshlayout.widget.SwipeRefreshLayout | ||||
|     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/swipe_refresh_resident" | ||||
|     android:layout_width="match_parent" | ||||
|     android:layout_height="match_parent" | ||||
|     android:background="#CCCCCC" | ||||
|     android:fillViewport="true"> | ||||
|     android:layout_height="match_parent"> | ||||
| 
 | ||||
|     <androidx.constraintlayout.widget.ConstraintLayout | ||||
|         android:id="@+id/resident_scroll_container" | ||||
|     <androidx.core.widget.NestedScrollView | ||||
|         android:layout_width="match_parent" | ||||
|         android:layout_height="wrap_content"> | ||||
|         android:layout_height="match_parent" | ||||
|         android:background="#CCCCCC" | ||||
|         android:fillViewport="true"> | ||||
| 
 | ||||
|         <!-- Banner superior --> | ||||
|         <include | ||||
|             android:id="@+id/resident_banner" | ||||
|             layout="@layout/item_person_banner" | ||||
|             android:layout_width="0dp" | ||||
|             android:layout_height="wrap_content" | ||||
|             app:layout_constraintTop_toTopOf="parent" | ||||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintEnd_toEndOf="parent" /> | ||||
| 
 | ||||
|         <LinearLayout | ||||
|             android:id="@+id/linearLayout2" | ||||
|         <androidx.constraintlayout.widget.ConstraintLayout | ||||
|             android:id="@+id/resident_scroll_container" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:layout_marginHorizontal="20dp" | ||||
|             android:layout_marginTop="20dp" | ||||
|             android:orientation="horizontal" | ||||
|             app:layout_constraintEnd_toEndOf="parent" | ||||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintTop_toBottomOf="@id/resident_banner"> | ||||
|             android:layout_height="wrap_content"> | ||||
| 
 | ||||
|             <TextView | ||||
|                 android:id="@+id/textView15" | ||||
|             <!-- Banner superior --> | ||||
|             <include | ||||
|                 android:id="@+id/resident_banner" | ||||
|                 layout="@layout/item_person_banner" | ||||
|                 android:layout_width="0dp" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_weight="1" | ||||
|                 android:fontFamily="@font/assistant_bold" | ||||
|                 android:text="Últimas partidas" | ||||
|                 android:textColor="#324F5E" | ||||
|                 android:textSize="20sp" /> | ||||
| 
 | ||||
|             <TextView | ||||
|                 android:id="@+id/textView17" | ||||
|                 android:layout_width="0dp" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_weight="1" | ||||
|                 android:fontFamily="@font/assistant_semibold" | ||||
|                 android:gravity="right" | ||||
|                 android:text="Ver más" | ||||
|                 android:textColor="#0062FF" /> | ||||
|         </LinearLayout> | ||||
| 
 | ||||
|         <androidx.recyclerview.widget.RecyclerView | ||||
|             android:id="@+id/latestGames_recycleView_resident" | ||||
|             android:layout_width="match_parent" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:layout_marginHorizontal="10dp" | ||||
|             android:layout_marginTop="10dp" | ||||
|             android:nestedScrollingEnabled="false" | ||||
|             app:layout_constraintEnd_toEndOf="parent" | ||||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintTop_toBottomOf="@id/linearLayout2" /> | ||||
| 
 | ||||
|         <TextView | ||||
|             android:id="@+id/textView16" | ||||
|             android:layout_width="0dp" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:layout_marginHorizontal="20dp" | ||||
|             android:layout_marginTop="20dp" | ||||
|             android:fontFamily="@font/assistant_bold" | ||||
|             android:text="Estadísticas" | ||||
|             android:textColor="#324F5E" | ||||
|             android:textSize="20sp" | ||||
|             app:layout_constraintEnd_toEndOf="parent" | ||||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintTop_toBottomOf="@id/latestGames_recycleView_resident" /> | ||||
| 
 | ||||
|         <!-- Card con gráfico de dona --> | ||||
|         <androidx.cardview.widget.CardView | ||||
|             android:id="@+id/pieCard" | ||||
|             android:layout_width="0dp" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:layout_margin="16dp" | ||||
|             android:layout_marginTop="20dp" | ||||
|             app:cardCornerRadius="16dp" | ||||
|             app:cardElevation="6dp" | ||||
|             app:layout_constraintTop_toBottomOf="@+id/textView16" | ||||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintEnd_toEndOf="parent"> | ||||
|                 app:layout_constraintTop_toTopOf="parent" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" /> | ||||
| 
 | ||||
|             <LinearLayout | ||||
|                 android:id="@+id/linearLayout2" | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:gravity="center" | ||||
|                 android:layout_marginHorizontal="20dp" | ||||
|                 android:layout_marginTop="20dp" | ||||
|                 android:orientation="horizontal" | ||||
|                 android:padding="16dp"> | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@id/resident_banner"> | ||||
| 
 | ||||
|                 <com.andresgmoran.apptrabajadores.ui.graphics.CustomPieChartView | ||||
|                     android:id="@+id/pieChart" | ||||
|                     android:layout_width="150dp" | ||||
|                     android:layout_height="150dp" /> | ||||
| 
 | ||||
|                 <LinearLayout | ||||
|                 <TextView | ||||
|                     android:id="@+id/textView15" | ||||
|                     android:layout_width="0dp" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:layout_weight="1" | ||||
|                     android:orientation="vertical" | ||||
|                     android:paddingStart="16dp"> | ||||
|                     android:fontFamily="@font/assistant_bold" | ||||
|                     android:text="Últimas partidas" | ||||
|                     android:textColor="#324F5E" | ||||
|                     android:textSize="20sp" /> | ||||
| 
 | ||||
|                     <TextView | ||||
|                         android:id="@+id/pie_game_1_name" | ||||
|                         android:layout_width="wrap_content" | ||||
|                         android:layout_height="wrap_content" | ||||
|                         android:layout_marginBottom="5dp" | ||||
|                         android:background="#FFFF00" | ||||
|                         android:text="Sombras - 45%" | ||||
|                         android:textColor="#000000" /> | ||||
| 
 | ||||
|                     <TextView | ||||
|                         android:id="@+id/pie_game_2_name" | ||||
|                         android:layout_width="wrap_content" | ||||
|                         android:layout_height="wrap_content" | ||||
|                         android:layout_marginBottom="5dp" | ||||
|                         android:background="#FF6666" | ||||
|                         android:text="Reacción auditiva - 30%" | ||||
|                         android:textColor="#000000" /> | ||||
| 
 | ||||
|                     <TextView | ||||
|                         android:id="@+id/pie_game_3_name" | ||||
|                         android:layout_width="wrap_content" | ||||
|                         android:layout_height="wrap_content" | ||||
|                         android:layout_marginBottom="5dp" | ||||
|                         android:background="#FF00FF" | ||||
|                         android:text="Memoria - 15%" | ||||
|                         android:textColor="#000000" /> | ||||
| 
 | ||||
|                     <TextView | ||||
|                         android:id="@+id/pie_game_4_name" | ||||
|                         android:layout_width="wrap_content" | ||||
|                         android:layout_height="wrap_content" | ||||
|                         android:layout_marginBottom="5dp" | ||||
|                         android:background="#00FF00" | ||||
|                         android:text="Laberinto - 10%" | ||||
|                         android:textColor="#000000" /> | ||||
|                     <TextView android:id="@+id/pie_game_5_name" android:layout_width="wrap_content" android:layout_height="wrap_content" android:background="#00FF00" android:text="Laberinto - 10%" android:textColor="#000000"/> | ||||
|                 </LinearLayout> | ||||
|                 <TextView | ||||
|                     android:id="@+id/textView17" | ||||
|                     android:layout_width="0dp" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:layout_weight="1" | ||||
|                     android:fontFamily="@font/assistant_semibold" | ||||
|                     android:gravity="right" | ||||
|                     android:text="Ver más" | ||||
|                     android:textColor="#0062FF" /> | ||||
|             </LinearLayout> | ||||
|         </androidx.cardview.widget.CardView> | ||||
| 
 | ||||
|         <!-- Contenedor de gráficos de barras --> | ||||
|         <LinearLayout | ||||
|             android:id="@+id/barChartsContainer" | ||||
|             android:layout_width="0dp" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:orientation="vertical" | ||||
|             android:gravity="center" | ||||
|             android:layout_margin="16dp" | ||||
|             app:layout_constraintTop_toBottomOf="@id/pieCard" | ||||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintEnd_toEndOf="parent" /> | ||||
|             <androidx.recyclerview.widget.RecyclerView | ||||
|                 android:id="@+id/latestGames_recycleView_resident" | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginHorizontal="10dp" | ||||
|                 android:layout_marginTop="10dp" | ||||
|                 android:nestedScrollingEnabled="false" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@id/linearLayout2" /> | ||||
| 
 | ||||
|     </androidx.constraintlayout.widget.ConstraintLayout> | ||||
|             <TextView | ||||
|                 android:id="@+id/tv_lastgames_empty_resident" | ||||
|                 android:layout_width="wrap_content" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginTop="5dp" | ||||
|                 android:fontFamily="@font/assistant_semibold" | ||||
|                 android:text="No hay partidas disponibles" | ||||
|                 android:textColor="#96A7AF" | ||||
|                 android:visibility="gone" | ||||
|                 app:layout_constraintEnd_toEndOf="@+id/linearLayout2" | ||||
|                 app:layout_constraintStart_toStartOf="@+id/linearLayout2" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/linearLayout2" /> | ||||
| 
 | ||||
| </androidx.core.widget.NestedScrollView> | ||||
|             <TextView | ||||
|                 android:id="@+id/textView16" | ||||
|                 android:layout_width="0dp" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginHorizontal="20dp" | ||||
|                 android:layout_marginTop="20dp" | ||||
|                 android:fontFamily="@font/assistant_bold" | ||||
|                 android:text="Estadísticas" | ||||
|                 android:textColor="#324F5E" | ||||
|                 android:textSize="20sp" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintTop_toBottomOf="@id/latestGames_recycleView_resident" /> | ||||
| 
 | ||||
|             <!-- Card con gráfico de dona --> | ||||
|             <TextView | ||||
|                 android:id="@+id/tv_stats_empty_resident" | ||||
|                 android:layout_width="wrap_content" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_marginTop="5dp" | ||||
|                 android:fontFamily="@font/assistant_semibold" | ||||
|                 android:text="No hay estadisticas disponibles" | ||||
|                 android:textColor="#96A7AF" | ||||
|                 android:visibility="gone" | ||||
|                 app:layout_constraintEnd_toEndOf="@+id/textView16" | ||||
|                 app:layout_constraintStart_toStartOf="@+id/textView16" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/textView16" /> | ||||
| 
 | ||||
|             <androidx.cardview.widget.CardView | ||||
|                 android:id="@+id/pieCard" | ||||
|                 android:layout_width="0dp" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:layout_margin="16dp" | ||||
|                 android:layout_marginTop="20dp" | ||||
|                 app:cardCornerRadius="16dp" | ||||
|                 app:cardElevation="6dp" | ||||
|                 app:layout_constraintTop_toBottomOf="@+id/textView16" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintEnd_toEndOf="parent"> | ||||
| 
 | ||||
|                 <LinearLayout | ||||
|                     android:layout_width="match_parent" | ||||
|                     android:layout_height="wrap_content" | ||||
|                     android:gravity="center" | ||||
|                     android:orientation="horizontal" | ||||
|                     android:padding="16dp"> | ||||
| 
 | ||||
|                     <com.andresgmoran.apptrabajadores.ui.graphics.CustomPieChartView | ||||
|                         android:id="@+id/pieChart" | ||||
|                         android:layout_width="150dp" | ||||
|                         android:layout_height="150dp" /> | ||||
| 
 | ||||
|                     <LinearLayout | ||||
|                         android:layout_width="0dp" | ||||
|                         android:layout_height="wrap_content" | ||||
|                         android:layout_weight="1" | ||||
|                         android:orientation="vertical" | ||||
|                         android:paddingStart="16dp"> | ||||
| 
 | ||||
|                         <TextView | ||||
|                             android:id="@+id/pie_game_1_name" | ||||
|                             android:layout_width="wrap_content" | ||||
|                             android:layout_height="wrap_content" | ||||
|                             android:layout_marginBottom="5dp" | ||||
|                             android:background="#FFFF00" | ||||
|                             android:text="Sombras - 45%" | ||||
|                             android:textColor="#000000" /> | ||||
| 
 | ||||
|                         <TextView | ||||
|                             android:id="@+id/pie_game_2_name" | ||||
|                             android:layout_width="wrap_content" | ||||
|                             android:layout_height="wrap_content" | ||||
|                             android:layout_marginBottom="5dp" | ||||
|                             android:background="#FF6666" | ||||
|                             android:text="Reacción auditiva - 30%" | ||||
|                             android:textColor="#000000" /> | ||||
| 
 | ||||
|                         <TextView | ||||
|                             android:id="@+id/pie_game_3_name" | ||||
|                             android:layout_width="wrap_content" | ||||
|                             android:layout_height="wrap_content" | ||||
|                             android:layout_marginBottom="5dp" | ||||
|                             android:background="#FF00FF" | ||||
|                             android:text="Memoria - 15%" | ||||
|                             android:textColor="#000000" /> | ||||
| 
 | ||||
|                         <TextView | ||||
|                             android:id="@+id/pie_game_4_name" | ||||
|                             android:layout_width="wrap_content" | ||||
|                             android:layout_height="wrap_content" | ||||
|                             android:layout_marginBottom="5dp" | ||||
|                             android:background="#00FF00" | ||||
|                             android:text="Laberinto - 10%" | ||||
|                             android:textColor="#000000" /> | ||||
| 
 | ||||
|                         <TextView | ||||
|                             android:id="@+id/pie_game_5_name" | ||||
|                             android:layout_width="wrap_content" | ||||
|                             android:layout_height="wrap_content" | ||||
|                             android:background="#00FF00" | ||||
|                             android:text="Laberinto - 10%" | ||||
|                             android:textColor="#000000"/> | ||||
|                     </LinearLayout> | ||||
|                 </LinearLayout> | ||||
|             </androidx.cardview.widget.CardView> | ||||
| 
 | ||||
|             <!-- Contenedor de gráficos de barras --> | ||||
|             <LinearLayout | ||||
|                 android:id="@+id/barChartsContainer" | ||||
|                 android:layout_width="0dp" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:orientation="vertical" | ||||
|                 android:gravity="center" | ||||
|                 android:layout_margin="16dp" | ||||
|                 app:layout_constraintTop_toBottomOf="@id/pieCard" | ||||
|                 app:layout_constraintStart_toStartOf="parent" | ||||
|                 app:layout_constraintEnd_toEndOf="parent" /> | ||||
| 
 | ||||
|         </androidx.constraintlayout.widget.ConstraintLayout> | ||||
|     </androidx.core.widget.NestedScrollView> | ||||
| </androidx.swiperefreshlayout.widget.SwipeRefreshLayout> | ||||
|  |  | |||
|  | @ -14,7 +14,7 @@ | |||
|         android:orientation="horizontal" | ||||
|         app:layout_constraintEnd_toEndOf="parent" | ||||
|         app:layout_constraintStart_toStartOf="parent" | ||||
|         app:layout_constraintTop_toBottomOf="@id/latestGames_recycleView_home"> | ||||
|         app:layout_constraintTop_toTopOf="parent"> | ||||
| 
 | ||||
|         <TextView | ||||
|             android:id="@+id/textView18" | ||||
|  |  | |||
|  | @ -36,11 +36,14 @@ | |||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintTop_toBottomOf="@+id/textview_salida" /> | ||||
| 
 | ||||
|         <ImageView | ||||
|             android:id="@+id/more_options_activityItem" | ||||
|         <ImageButton | ||||
|             android:id="@+id/more_options_activity_item" | ||||
|             android:layout_width="24dp" | ||||
|             android:layout_height="24dp" | ||||
|             android:src="@drawable/woman_avatar" | ||||
|             android:background="@null" | ||||
|             android:padding="0dp" | ||||
|             android:scaleType="centerInside" | ||||
|             android:src="@drawable/elipsis" | ||||
|             android:tint="#FFFFFF" | ||||
|             app:layout_constraintEnd_toEndOf="parent" | ||||
|             app:layout_constraintTop_toTopOf="parent" /> | ||||
|  | @ -67,5 +70,17 @@ | |||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintTop_toBottomOf="@id/date_text_activityItem" /> | ||||
| 
 | ||||
|         <ImageButton | ||||
|             android:id="@+id/activity_status_button" | ||||
|             android:layout_width="40dp" | ||||
|             android:layout_height="40dp" | ||||
|             android:background="@android:color/transparent" | ||||
|             android:src="@drawable/open_activity_status" | ||||
|             android:scaleType="fitCenter" | ||||
|             android:adjustViewBounds="true" | ||||
|             app:layout_constraintBottom_toBottomOf="parent" | ||||
|             app:layout_constraintEnd_toEndOf="parent" /> | ||||
| 
 | ||||
| 
 | ||||
|     </androidx.constraintlayout.widget.ConstraintLayout> | ||||
| </androidx.cardview.widget.CardView> | ||||
|  |  | |||
|  | @ -47,11 +47,14 @@ | |||
|             app:layout_constraintStart_toEndOf="@id/profile_image_latestGameItem" | ||||
|             app:layout_constraintTop_toBottomOf="@id/name_text_latestGameItem" /> | ||||
| 
 | ||||
|         <ImageView | ||||
|         <ImageButton | ||||
|             android:id="@+id/more_options_latestGameItem" | ||||
|             android:layout_width="24dp" | ||||
|             android:layout_height="24dp" | ||||
|             android:src="@drawable/woman_avatar" | ||||
|             android:background="@null" | ||||
|             android:padding="0dp" | ||||
|             android:scaleType="centerInside" | ||||
|             android:src="@drawable/elipsis" | ||||
|             android:tint="#FFFFFF" | ||||
|             app:layout_constraintEnd_toEndOf="parent" | ||||
|             app:layout_constraintTop_toTopOf="@id/profile_image_latestGameItem" /> | ||||
|  |  | |||
|  | @ -15,7 +15,7 @@ | |||
|         android:padding="10dp"> | ||||
| 
 | ||||
|         <com.google.android.material.imageview.ShapeableImageView | ||||
|             android:id="@+id/circularImageView2" | ||||
|             android:id="@+id/item_list_image" | ||||
|             android:layout_width="41dp" | ||||
|             android:layout_height="41dp" | ||||
|             android:scaleType="centerCrop" | ||||
|  | @ -45,11 +45,11 @@ | |||
|         </LinearLayout> | ||||
| 
 | ||||
|         <ImageButton | ||||
|             android:id="@+id/imageButton3" | ||||
|             android:layout_width="wrap_content" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:id="@+id/more_options_item_list" | ||||
|             android:layout_width="24dp" | ||||
|             android:layout_height="24dp" | ||||
|             android:background="@null" | ||||
|             app:srcCompat="@drawable/ic_dashboard_black_24dp" /> | ||||
|             app:srcCompat="@drawable/elipsis" /> | ||||
| 
 | ||||
|     </LinearLayout> | ||||
| </androidx.cardview.widget.CardView> | ||||
|  |  | |||
|  | @ -13,8 +13,8 @@ | |||
| 
 | ||||
|         <com.google.android.material.imageview.ShapeableImageView | ||||
|             android:id="@+id/circularImageView3" | ||||
|             android:layout_width="72dp" | ||||
|             android:layout_height="72dp" | ||||
|             android:layout_width="52dp" | ||||
|             android:layout_height="51dp" | ||||
|             android:scaleType="centerCrop" | ||||
|             app:shapeAppearanceOverlay="@style/CircularShapeAppearance" | ||||
|             app:srcCompat="@drawable/woman_avatar" /> | ||||
|  | @ -28,7 +28,7 @@ | |||
|             android:layout_marginStart="10dp"> | ||||
| 
 | ||||
|             <TextView | ||||
|                 android:id="@+id/home_user_name" | ||||
|                 android:id="@+id/observation_user_name" | ||||
|                 android:layout_width="match_parent" | ||||
|                 android:layout_height="wrap_content" | ||||
|                 android:text="Person Name" | ||||
|  | @ -42,17 +42,31 @@ | |||
|         </LinearLayout> | ||||
|     </LinearLayout> | ||||
| 
 | ||||
|     <!-- Texto de la observación existente --> | ||||
|     <TextView | ||||
|         android:id="@+id/observation_text" | ||||
|         android:layout_width="match_parent" | ||||
|         android:layout_height="wrap_content" | ||||
|         android:layout_marginTop="5dp" | ||||
|         android:text="Observation text" /> | ||||
|         android:layout_marginTop="10dp" | ||||
|         android:text="Observation text" | ||||
|         android:visibility="gone" /> | ||||
| 
 | ||||
|     <!-- Línea de separación --> | ||||
|     <View | ||||
|     <!-- Campo de texto si no hay observación --> | ||||
|     <EditText | ||||
|         android:id="@+id/observation_input" | ||||
|         android:layout_width="match_parent" | ||||
|         android:layout_height="1dp" | ||||
|         android:layout_height="wrap_content" | ||||
|         android:layout_marginTop="10dp" | ||||
|         android:hint="Escribe una observación..." | ||||
|         android:inputType="textMultiLine" | ||||
|         android:visibility="invisible" /> | ||||
| 
 | ||||
|     <Button | ||||
|         android:id="@+id/add_observation_button" | ||||
|         android:layout_width="wrap_content" | ||||
|         android:layout_height="wrap_content" | ||||
|         android:layout_marginTop="8dp" | ||||
|         android:background="@android:color/darker_gray" /> | ||||
|         android:backgroundTint="#324F5E" | ||||
|         android:text="Subir observación" | ||||
|         android:visibility="invisible" /> | ||||
| </LinearLayout> | ||||
|  |  | |||
|  | @ -30,7 +30,7 @@ | |||
|                     android:orientation="horizontal"> | ||||
| 
 | ||||
|                     <ImageButton | ||||
|                         android:id="@+id/imageButton2" | ||||
|                         android:id="@+id/back_button" | ||||
|                         android:layout_width="wrap_content" | ||||
|                         android:layout_height="wrap_content" | ||||
|                         android:layout_weight="1" | ||||
|  |  | |||
|  | @ -14,75 +14,92 @@ | |||
|         android:padding="12dp"> | ||||
| 
 | ||||
|         <!-- Imagen circular --> | ||||
|         <ImageView | ||||
|         <com.google.android.material.imageview.ShapeableImageView | ||||
|             android:id="@+id/profile_image" | ||||
|             android:layout_width="40dp" | ||||
|             android:layout_height="40dp" | ||||
|             android:scaleType="centerInside" | ||||
|             android:scaleType="centerCrop" | ||||
|             android:src="@drawable/woman_avatar" | ||||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintTop_toTopOf="parent"/> | ||||
|             app:layout_constraintTop_toTopOf="parent" | ||||
|             app:shapeAppearanceOverlay="@style/CircularShapeAppearance" /> | ||||
| 
 | ||||
|         <!-- Nombre --> | ||||
| 
 | ||||
|         <TextView | ||||
|             android:id="@+id/tv_name" | ||||
|             android:id="@+id/tv_name_participant_item" | ||||
|             android:layout_width="0dp" | ||||
|             android:layout_height="wrap_content" | ||||
|             android:layout_marginEnd="156dp" | ||||
|             android:text="Jose Manuel Carrasco" | ||||
|             android:textColor="#324F5E" | ||||
|             android:textStyle="bold" | ||||
|             android:textSize="16sp" | ||||
|             android:layout_marginStart="12dp" | ||||
|             android:textStyle="bold" | ||||
|             app:layout_constraintEnd_toEndOf="parent" | ||||
|             app:layout_constraintStart_toEndOf="@id/profile_image" | ||||
|             app:layout_constraintTop_toTopOf="@id/profile_image" | ||||
|             app:layout_constraintEnd_toEndOf="parent"/> | ||||
|             app:layout_constraintTop_toTopOf="@id/profile_image" /> | ||||
| 
 | ||||
|         <!-- Primer botón --> | ||||
|         <ImageButton | ||||
|             android:id="@+id/btn_first" | ||||
|             android:id="@+id/participant_asistencia_button" | ||||
|             android:layout_width="48dp" | ||||
|             android:layout_height="48dp" | ||||
|             android:backgroundTint="#6CADFF" | ||||
|             android:src="@drawable/logo_confirmation" | ||||
|             android:layout_marginStart="16dp" | ||||
|             android:layout_marginTop="12dp" | ||||
|             android:background="@drawable/activity_button_unpressed_background" | ||||
|             android:scaleType="centerInside" | ||||
|             android:padding="8dp" | ||||
|             app:layout_constraintTop_toBottomOf="@id/tv_name" | ||||
|             android:scaleType="centerInside" | ||||
|             android:src="@drawable/logo_confirmation" | ||||
|             app:layout_constraintEnd_toStartOf="@id/participant_opinion_button" | ||||
|             app:layout_constraintStart_toStartOf="parent" | ||||
|             app:layout_constraintEnd_toStartOf="@id/btn_second" | ||||
|             android:layout_marginTop="12dp"/> | ||||
|             app:layout_constraintTop_toBottomOf="@id/tv_name_participant_item" /> | ||||
| 
 | ||||
|         <!-- Segundo botón --> | ||||
|         <ImageButton | ||||
|             android:id="@+id/btn_second" | ||||
|             android:id="@+id/participant_opinion_button" | ||||
|             android:layout_width="48dp" | ||||
|             android:layout_height="48dp" | ||||
|             android:backgroundTint="#6CADFF" | ||||
|             android:src="@drawable/logo_text" | ||||
|             android:background="@drawable/activity_button_unpressed_background" | ||||
|             android:scaleType="centerInside" | ||||
|             android:padding="8dp" | ||||
|             app:layout_constraintTop_toBottomOf="@id/tv_name" | ||||
|             app:layout_constraintStart_toEndOf="@id/btn_first" | ||||
|             app:layout_constraintEnd_toStartOf="@id/btn_third" | ||||
|             app:layout_constraintTop_toBottomOf="@id/tv_name_participant_item" | ||||
|             app:layout_constraintStart_toEndOf="@id/participant_asistencia_button" | ||||
|             app:layout_constraintEnd_toStartOf="@id/participant_material_help_button" | ||||
|             android:layout_marginStart="8dp" | ||||
|             android:layout_marginTop="12dp"/> | ||||
| 
 | ||||
|         <!-- Nuevo botón --> | ||||
|         <ImageButton | ||||
|             android:id="@+id/participant_material_help_button" | ||||
|             android:layout_width="48dp" | ||||
|             android:layout_height="48dp" | ||||
|             android:background="@drawable/activity_button_unpressed_background" | ||||
|             android:src="@drawable/logo_material_help" | ||||
|             android:scaleType="centerInside" | ||||
|             android:padding="8dp" | ||||
|             app:layout_constraintTop_toBottomOf="@id/tv_name_participant_item" | ||||
|             app:layout_constraintStart_toEndOf="@id/participant_opinion_button" | ||||
|             app:layout_constraintEnd_toStartOf="@id/participant_human_help_button" | ||||
|             android:layout_marginStart="8dp" | ||||
|             android:layout_marginTop="12dp"/> | ||||
| 
 | ||||
|         <!-- Tercer botón --> | ||||
|         <ImageButton | ||||
|             android:id="@+id/btn_third" | ||||
|             android:id="@+id/participant_human_help_button" | ||||
|             android:layout_width="48dp" | ||||
|             android:layout_height="48dp" | ||||
|             android:background="@drawable/activity_button_unpressed_background" | ||||
|             android:src="@drawable/logo_help" | ||||
|             android:src="@drawable/logo_human_help" | ||||
|             app:shapeAppearanceOverlay="@style/CircularShapeAppearance" | ||||
|             android:scaleType="centerInside" | ||||
|             android:padding="8dp" | ||||
|             app:layout_constraintTop_toBottomOf="@id/tv_name" | ||||
|             app:layout_constraintStart_toEndOf="@id/btn_second" | ||||
|             app:layout_constraintTop_toBottomOf="@id/tv_name_participant_item" | ||||
|             app:layout_constraintStart_toEndOf="@id/participant_material_help_button" | ||||
|             app:layout_constraintEnd_toEndOf="parent" | ||||
|             android:layout_marginStart="8dp" | ||||
|             android:layout_marginTop="12dp"/> | ||||
| 
 | ||||
| 
 | ||||
|     </androidx.constraintlayout.widget.ConstraintLayout> | ||||
| 
 | ||||
| </androidx.cardview.widget.CardView> | ||||
|  |  | |||
|  | @ -0,0 +1,5 @@ | |||
| <menu xmlns:android="http://schemas.android.com/apk/res/android"> | ||||
|     <item | ||||
|         android:id="@+id/action_delete" | ||||
|         android:title="Eliminar" /> | ||||
| </menu> | ||||
|  | @ -1,5 +0,0 @@ | |||
| [folding] | ||||
| 
 | ||||
| node_unfolds=[NodePath("."), PackedStringArray("Layout"), NodePath("MarginContainer"), PackedStringArray("Layout", "Theme Overrides", "Theme Overrides/constants"), NodePath("MarginContainer/VBoxContainer"), PackedStringArray("Layout", "Theme Overrides", "Theme Overrides/constants"), NodePath("MarginContainer/VBoxContainer/Label"), PackedStringArray("Theme Overrides", "Theme Overrides/font_sizes", "Theme Overrides/colors"), NodePath("MarginContainer/VBoxContainer/Label2"), PackedStringArray("Theme Overrides", "Theme Overrides/font_sizes", "Theme Overrides/colors"), NodePath("MarginContainer/VBoxContainer/MenuButton"), PackedStringArray("Layout", "Layout/Container Sizing", "Theme Overrides", "theme_override_styles/normal", "Theme Overrides/colors"), NodePath("MarginContainer/VBoxContainer/Label3"), PackedStringArray("Theme Overrides", "Theme Overrides/font_sizes", "Layout", "Theme Overrides/colors"), NodePath("MarginContainer/VBoxContainer/PlayButton"), PackedStringArray("Theme Overrides", "Layout", "Layout/Container Sizing", "Theme Overrides/styles", "theme_override_styles/normal"), NodePath("Bloqueador"), PackedStringArray("Visibility", "Layout", "Mouse"), NodePath("Warning"), PackedStringArray("Flags"), NodePath("Warning/Panel"), PackedStringArray("Theme Overrides", "Layout", "Theme Overrides/styles", "theme_override_styles/panel"), NodePath("Window/ColorRect"), PackedStringArray("Layout"), NodePath("Window/VBoxContainer"), PackedStringArray("Layout", "Theme Overrides", "Theme Overrides/constants", "Layout/Transform"), NodePath("Window/VBoxContainer/Label"), PackedStringArray("Layout", "Layout/Container Sizing", "Theme Overrides", "Theme Overrides/colors"), NodePath("Window/VBoxContainer/EmailInput"), PackedStringArray("Layout", "Layout/Container Sizing", "Theme Overrides", "Theme Overrides/styles", "theme_override_styles/normal"), NodePath("Window/VBoxContainer/Label2"), PackedStringArray("Layout", "Layout/Container Sizing", "Theme Overrides", "Theme Overrides/colors"), NodePath("Window/VBoxContainer/PasswordInput"), PackedStringArray("Layout", "Layout/Container Sizing", "Theme Overrides", "Theme Overrides/styles", "theme_override_styles/normal"), NodePath("Window/VBoxContainer/LogInButton"), PackedStringArray("Layout", "Layout/Container Sizing", "Theme Overrides", "Theme Overrides/styles", "theme_override_styles/normal")] | ||||
| resource_unfolds=["res://scenes/configuration.tscn::StyleBoxFlat_kporg", PackedStringArray(), "res://scenes/configuration.tscn::StyleBoxFlat_etp6l", PackedStringArray(), "res://scenes/configuration.tscn::StyleBoxFlat_sb302", PackedStringArray(), "res://scenes/configuration.tscn::StyleBoxFlat_hc4xr", PackedStringArray(), "res://scenes/configuration.tscn::StyleBoxFlat_7wcdq", PackedStringArray(), "res://scenes/configuration.tscn::StyleBoxFlat_lditi", PackedStringArray()] | ||||
| nodes_folded=[NodePath("MarginContainer"), NodePath("MarginContainer/VBoxContainer"), NodePath("Warning"), NodePath("Window")] | ||||
|  | @ -1,63 +0,0 @@ | |||
| [docks] | ||||
| 
 | ||||
| dock_3_selected_tab_idx=0 | ||||
| dock_4_selected_tab_idx=0 | ||||
| dock_5_selected_tab_idx=0 | ||||
| dock_floating={} | ||||
| dock_filesystem_h_split_offset=240 | ||||
| dock_filesystem_v_split_offset=0 | ||||
| dock_filesystem_display_mode=0 | ||||
| dock_filesystem_file_sort=0 | ||||
| dock_filesystem_file_list_display_mode=1 | ||||
| dock_filesystem_selected_paths=PackedStringArray("res://audio/wolf.mp3") | ||||
| dock_filesystem_uncollapsed_paths=PackedStringArray("Favorites", "res://", "res://audio/") | ||||
| dock_node_current_tab=0 | ||||
| dock_history_include_scene=true | ||||
| dock_history_include_global=true | ||||
| dock_bottom=[] | ||||
| dock_closed=[] | ||||
| dock_split_2=0 | ||||
| dock_split_3=0 | ||||
| dock_hsplit_1=0 | ||||
| dock_hsplit_2=270 | ||||
| dock_hsplit_3=-270 | ||||
| dock_hsplit_4=0 | ||||
| dock_3="Scene,Import" | ||||
| dock_4="FileSystem" | ||||
| dock_5="Inspector,Node,History" | ||||
| 
 | ||||
| [EditorNode] | ||||
| 
 | ||||
| open_scenes=PackedStringArray("res://scenes/game.tscn", "res://scenes/configuration.tscn") | ||||
| current_scene="res://scenes/game.tscn" | ||||
| center_split_offset=0 | ||||
| selected_default_debugger_tab_idx=0 | ||||
| selected_main_editor_idx=2 | ||||
| selected_bottom_panel_item=0 | ||||
| 
 | ||||
| [EditorWindow] | ||||
| 
 | ||||
| screen=0 | ||||
| mode="maximized" | ||||
| position=Vector2i(0, 23) | ||||
| 
 | ||||
| [ScriptEditor] | ||||
| 
 | ||||
| open_scripts=["res://scripts/configuration.gd", "res://scripts/game.gd", "res://scripts/game_data.gd", "res://scripts/screen_manager.gd"] | ||||
| selected_script="res://scripts/game.gd" | ||||
| open_help=[] | ||||
| script_split_offset=200 | ||||
| list_split_offset=0 | ||||
| zoom_factor=1.0 | ||||
| 
 | ||||
| [GameView] | ||||
| 
 | ||||
| floating_window_rect=Rect2i(0, 0, 1932, 1127) | ||||
| floating_window_screen=0 | ||||
| 
 | ||||
| [ShaderEditor] | ||||
| 
 | ||||
| open_shaders=[] | ||||
| split_offset=200 | ||||
| selected_shader="" | ||||
| text_shader_zoom_factor=1.0 | ||||
|  | @ -1,27 +0,0 @@ | |||
| ea4bc82a6ad023ab7ee23ee620429895 | ||||
| ::res://::1745432009 | ||||
| icon.svg::CompressedTexture2D/CompressedTexture2D::5044074093411398912::1744633584::1744633593::1::::<><><>0<>0<>f9adb246a70a7d9e0b44b9e0dce472bb<>res://.godot/imported/icon.svg-218a8f2b3041327d8a5756f3a245f83b.ctex:: | ||||
| ::res://audio/::1745432009 | ||||
| asno.mp3::AudioStreamMP3::5781012165743321088::1745431872::1745432009::1::::<><><>0<>0<>04849c7d6b682c88a72746e86174f73b<>res://.godot/imported/asno.mp3-c8f31127f4b178b8b2b8415bad169e07.mp3str:: | ||||
| caballo.mp3::AudioStreamMP3/AudioStreamMP3::4827901584278378597::1745431804::1745431804::1::::<><><>0<>0<>6e43501dc2941926b422a069a37c5b89<>res://.godot/imported/caballo.mp3-7b2055e87f29328701251f905f0ed53e.mp3str:: | ||||
| cuervo.mp3::AudioStreamMP3/AudioStreamMP3::4812865155172360579::1745431882::1745431965::1::::<><><>0<>0<>18f7a5517aeb2c4811e41ee28fbdbcbb<>res://.godot/imported/cuervo.mp3-699bfa229f8863609609060333d7eb09.mp3str:: | ||||
| Elefant.mp3::AudioStreamMP3/AudioStreamMP3::1108392689846586874::1744635741::1744635741::1::::<><><>0<>0<>009805eb2c7d9c7cd23bd1fcf3eb7bf8<>res://.godot/imported/Elefant.mp3-e3e5c8488a65256c22d6f79b59243fe7.mp3str:: | ||||
| gallo.mp3::AudioStreamMP3/AudioStreamMP3::2685061553279347517::1745431722::1745431729::1::::<><><>0<>0<>e12bc0d6db956a381fb428c74ffa81da<>res://.godot/imported/gallo.mp3-1271e286f267d5e31c477f20fe593d30.mp3str:: | ||||
| gato.mp3::AudioStreamMP3/AudioStreamMP3::5660098724051234660::1745431771::1745431930::1::::<><><>0<>0<>c8f2a66613b407ab6ee342c4cf2b03b7<>res://.godot/imported/gato.mp3-c61b66ed22aa208d5f0638a947cee007.mp3str:: | ||||
| gaviota.mp3::AudioStreamMP3/AudioStreamMP3::6630534147415936020::1745431879::1745431925::1::::<><><>0<>0<>1d8f8d40adc742a900f128a403114939<>res://.godot/imported/gaviota.mp3-b98dceded202a98d930f5d8b44e91302.mp3str:: | ||||
| oveja.mp3::AudioStreamMP3/AudioStreamMP3::6892221899961415453::1745431841::1745431936::1::::<><><>0<>0<>6b9875df10bf634bc5185ea4479c2ce1<>res://.godot/imported/oveja.mp3-548c0afa06640f39be929ba57d20c25d.mp3str:: | ||||
| perro.mp3::AudioStreamMP3/AudioStreamMP3::2363096142831056952::1745431668::1745431676::1::::<><><>0<>0<>6c8a300409787c83d6c5fda9997fc150<>res://.godot/imported/perro.mp3-70940168b9c50fc40928609c0024fb67.mp3str:: | ||||
| vaca.mp3::AudioStreamMP3/AudioStreamMP3::2596745121890943594::1745431616::1745431617::1::::<><><>0<>0<>dc233f99ee2d0d4acfba4fac41e69ab3<>res://.godot/imported/vaca.mp3-6fdbb0ce8242846d3ac977450b7ac193.mp3str:: | ||||
| wolf.mp3::AudioStreamMP3/AudioStreamMP3::4476753813126878356::1745431572::1745431573::1::::<><><>0<>0<>fe6fb338f4476d030d4e8e5517df5fa5<>res://.godot/imported/wolf.mp3-5b3e376d644832f42cc7647c484fe190.mp3str:: | ||||
| ::res://images/::1745428753 | ||||
| background.png::CompressedTexture2D/CompressedTexture2D::6822232804109568982::1745428752::1745428753::1::::<><><>0<>0<>5a780c38dac7e0dc6cc39dd8f5632c97<>res://.godot/imported/background.png-23e4ff24ed3606e496e606d42ded004c.ctex:: | ||||
| Fondo.png::CompressedTexture2D/CompressedTexture2D::6448476263866600585::1745260713::1745260728::1::::<><><>0<>0<>a5e23edebe29509b5d3e01644f8185aa<>res://.godot/imported/Fondo.png-ba7036b5421ed7e1312c2cbc3c746cc2.ctex:: | ||||
| menu_logo.png::CompressedTexture2D/CompressedTexture2D::5689433748730304837::1745260946::1745260946::1::::<><><>0<>0<>aa8e65da18c85d772f423418a1d50520<>res://.godot/imported/menu_logo.png-326be22817043ae765a9aaa1fb96c9c5.ctex:: | ||||
| ::res://scenes/::1745431061 | ||||
| configuration.tscn::PackedScene/PackedScene::1018116344755398623::1745431061::0::1::::<><><>0<>0<><>::uid://e41ws6j00o5::::res://scripts/configuration.gd | ||||
| game.tscn::PackedScene/PackedScene::6874229073676243903::1745431061::0::1::::<><><>0<>0<><>::uid://dhqnf0xm2mwwm::::res://scripts/game.gd<>uid://c4g0gbifdm5fe::::res://images/background.png<>uid://cncae337c0ws8::::res://images/menu_logo.png | ||||
| ::res://scripts/::1745431061 | ||||
| configuration.gd::GDScript/GDScript::10065919189713686::1745414906::0::1::::<>Control<><>0<>0<><>:: | ||||
| game.gd::GDScript/GDScript::7684413599682033952::1745431061::0::1::::<>Control<><>0<>0<><>:: | ||||
| game_data.gd::GDScript/GDScript::3070763797095879381::1745431026::0::1::::<>Node<><>0<>0<><>:: | ||||
| screen_manager.gd::GDScript/GDScript::1421320781459715397::1744728053::0::1::::<>Node<><>0<>0<><>:: | ||||
|  | @ -1,4 +0,0 @@ | |||
| res://scenes/game.tscn | ||||
| res://scenes/configuration.tscn | ||||
| res://scripts/game_data.gd | ||||
| res://scripts/game.gd | ||||
|  | @ -1,5 +0,0 @@ | |||
| [folding] | ||||
| 
 | ||||
| node_unfolds=[] | ||||
| resource_unfolds=[] | ||||
| nodes_folded=[] | ||||
|  | @ -1,5 +0,0 @@ | |||
| [folding] | ||||
| 
 | ||||
| node_unfolds=[NodePath("."), PackedStringArray("Layout"), NodePath("TextureRect"), PackedStringArray("Layout", "Layout/Transform", "Layout/Container Sizing"), NodePath("CenterContainer"), PackedStringArray("Layout"), NodePath("CenterContainer/VBoxContainer"), PackedStringArray("Layout", "Layout/Transform", "Layout/Container Sizing", "Layout/Grow Direction", "Layout/Anchor Points", "Layout/Anchor Offsets"), NodePath("CenterContainer/VBoxContainer/NextSoundButton"), PackedStringArray("Layout", "Theme Overrides/font_sizes", "Theme Overrides", "Theme Overrides/styles", "theme_override_styles/normal", "Layout/Container Sizing", "theme_override_styles/pressed", "Focus"), NodePath("CenterContainer/VBoxContainer/HBoxContainer"), PackedStringArray("Layout", "Layout/Container Sizing"), NodePath("CenterContainer/VBoxContainer/HBoxContainer/RepeatSoundButton"), PackedStringArray("Layout", "Layout/Container Sizing", "theme_override_styles/normal", "Theme Overrides/font_sizes", "Focus"), NodePath("CenterContainer/VBoxContainer/HBoxContainer/EndOfGameButton"), PackedStringArray("Layout", "Layout/Container Sizing", "Theme Overrides", "Theme Overrides/styles", "Theme Overrides/font_sizes", "theme_override_styles/normal", "Focus"), NodePath("WinnerSelector"), PackedStringArray("Theme Overrides", "theme_override_styles/embedded_unfocused_border", "theme_override_styles/embedded_border"), NodePath("WinnerSelector/ColorRect"), PackedStringArray("Layout"), NodePath("WinnerSelector/VBoxContainer"), PackedStringArray("Layout", "Layout/Container Sizing", "Theme Overrides", "Theme Overrides/constants", "Layout/Transform"), NodePath("WinnerSelector/VBoxContainer/Label"), PackedStringArray("Layout/Container Sizing", "Theme Overrides", "Theme Overrides/colors", "Layout"), NodePath("WinnerSelector/VBoxContainer/MenuPlayers"), PackedStringArray("Layout", "Layout/Container Sizing", "Theme Overrides", "Theme Overrides/styles", "theme_override_styles/normal", "Theme Overrides/colors"), NodePath("WinnerSelector/VBoxContainer/Button"), PackedStringArray("Layout", "Layout/Container Sizing", "Theme Overrides", "Theme Overrides/styles", "theme_override_styles/normal"), NodePath("Bloqueador"), PackedStringArray("Layout", "Layout/Transform"), NodePath("ExitButtonWindow/ColorRect"), PackedStringArray("Layout", "Layout/Container Sizing"), NodePath("ExitButtonWindow/VBoxContainer"), PackedStringArray("Layout", "Theme Overrides", "Theme Overrides/constants"), NodePath("ExitButtonWindow/VBoxContainer/Label"), PackedStringArray("Theme Overrides", "Theme Overrides/colors", "Theme Overrides/font_sizes"), NodePath("ExitButtonWindow/VBoxContainer/HBoxContainer/ConfirmExitButton"), PackedStringArray("Layout", "Layout/Container Sizing", "Theme Overrides", "Theme Overrides/styles", "theme_override_styles/normal"), NodePath("ExitButtonWindow/VBoxContainer/HBoxContainer/CancelExitButton"), PackedStringArray("Layout", "Layout/Container Sizing", "Theme Overrides", "Theme Overrides/styles", "theme_override_styles/normal")] | ||||
| resource_unfolds=["res://scenes/game.tscn::StyleBoxFlat_0tnpc", PackedStringArray("Border", "Corner Radius"), "res://scenes/game.tscn::StyleBoxFlat_yqjtg", PackedStringArray("Corner Radius", "Border", "Border Width"), "res://scenes/game.tscn::StyleBoxFlat_lnu2h", PackedStringArray("Corner Radius", "Border", "Border Width"), "res://scenes/game.tscn::StyleBoxFlat_lbhrr", PackedStringArray("Corner Radius", "Border", "Border Width"), "res://scenes/game.tscn::StyleBoxFlat_iywne", PackedStringArray(), "res://scenes/game.tscn::StyleBoxFlat_p57ef", PackedStringArray(), "res://scenes/game.tscn::StyleBoxFlat_u5sy4", PackedStringArray(), "res://scenes/game.tscn::StyleBoxFlat_gee14", PackedStringArray()] | ||||
| nodes_folded=[NodePath("WinnerSelector"), NodePath("WinnerSelector/VBoxContainer"), NodePath("ExitButtonWindow"), NodePath("ExitButtonWindow/VBoxContainer"), NodePath("ExitButtonWindow/VBoxContainer/HBoxContainer")] | ||||
|  | @ -1,27 +0,0 @@ | |||
| [editor_metadata] | ||||
| 
 | ||||
| executable_path="C:/Users/moran/Downloads/Godot_v4.4.1-stable_win64.exe/Godot_v4.4.1-stable_win64.exe" | ||||
| use_advanced_connections=false | ||||
| 
 | ||||
| [dialog_bounds] | ||||
| 
 | ||||
| project_settings=Rect2(360, 190, 1200, 700) | ||||
| create_new_node=Rect2(510, 190, 900, 700) | ||||
| 
 | ||||
| [script_setup] | ||||
| 
 | ||||
| last_selected_language="GDScript" | ||||
| 
 | ||||
| [recent_files] | ||||
| 
 | ||||
| scripts=["res://scripts/game_data.gd", "res://scripts/screen_manager.gd", "res://scenes/configuration.gd", "res://game.gd"] | ||||
| scenes=["res://scenes/configuration.tscn", "res://scenes/game.tscn", "res://game.tscn"] | ||||
| 
 | ||||
| [color_picker] | ||||
| 
 | ||||
| picker_shape=3 | ||||
| recent_presets=PackedColorArray(0.588235, 0.654902, 0.686275, 1, 0.588235, 0.654902, 0.686275, 1, 0.8, 0.8, 0.8, 1, 0.196078, 0.309804, 0.368627, 1, 0.0156863, 0.8, 0.8, 1, 0.0156863, 0.247059, 0.8, 1, 0.92549, 0.376471, 0.615686, 1, 0.92549, 0.376471, 0.615686, 0.788235, 0, 0, 0, 1) | ||||
| 
 | ||||
| [quick_open_dialog] | ||||
| 
 | ||||
| last_mode=1 | ||||
|  | @ -1,2 +0,0 @@ | |||
| res://scripts | ||||
| res:// | ||||
|  | @ -1,55 +0,0 @@ | |||
| [res://scripts/game.gd] | ||||
| 
 | ||||
| state={ | ||||
| "bookmarks": PackedInt32Array(), | ||||
| "breakpoints": PackedInt32Array(), | ||||
| "column": 0, | ||||
| "folded_lines": Array[int]([]), | ||||
| "h_scroll_position": 0, | ||||
| "row": 11, | ||||
| "scroll_position": 0.0, | ||||
| "selection": false, | ||||
| "syntax_highlighter": "GDScript" | ||||
| } | ||||
| 
 | ||||
| [res://scripts/configuration.gd] | ||||
| 
 | ||||
| state={ | ||||
| "bookmarks": PackedInt32Array(), | ||||
| "breakpoints": PackedInt32Array(), | ||||
| "column": 35, | ||||
| "folded_lines": Array[int]([]), | ||||
| "h_scroll_position": 0, | ||||
| "row": 98, | ||||
| "scroll_position": 0.0, | ||||
| "selection": false, | ||||
| "syntax_highlighter": "GDScript" | ||||
| } | ||||
| 
 | ||||
| [res://scripts/screen_manager.gd] | ||||
| 
 | ||||
| state={ | ||||
| "bookmarks": PackedInt32Array(), | ||||
| "breakpoints": PackedInt32Array(), | ||||
| "column": 0, | ||||
| "folded_lines": Array[int]([]), | ||||
| "h_scroll_position": 0, | ||||
| "row": 22, | ||||
| "scroll_position": 0.0, | ||||
| "selection": false, | ||||
| "syntax_highlighter": "GDScript" | ||||
| } | ||||
| 
 | ||||
| [res://scripts/game_data.gd] | ||||
| 
 | ||||
| state={ | ||||
| "bookmarks": PackedInt32Array(), | ||||
| "breakpoints": PackedInt32Array(), | ||||
| "column": 25, | ||||
| "folded_lines": Array[int]([]), | ||||
| "h_scroll_position": 0, | ||||
| "row": 65, | ||||
| "scroll_position": 49.0, | ||||
| "selection": false, | ||||
| "syntax_highlighter": "GDScript" | ||||
| } | ||||
|  | @ -1 +0,0 @@ | |||
| list=[] | ||||
|  | @ -1,3 +0,0 @@ | |||
| source_md5="d0a2e8efb08934c125002ca2342b0b1b" | ||||
| dest_md5="5eac8ba711953b444769459c636fa80b" | ||||
| 
 | ||||
|  | @ -1,3 +0,0 @@ | |||
| source_md5="78687f61f7636839a990dd7241418511" | ||||
| dest_md5="31495e9ef1c8480db9407b663c9641c9" | ||||
| 
 | ||||
|  | @ -1,3 +0,0 @@ | |||
| source_md5="b7ed27b8733ac3938ea97b6f62f03fe2" | ||||
| dest_md5="078c2aa3879e5d8fb05a4a95d08f28cb" | ||||
| 
 | ||||
|  | @ -1,3 +0,0 @@ | |||
| source_md5="075ad337ca02a6bc710d719cbd92152d" | ||||
| dest_md5="b4db58192c0182707fd3e8d35c0876fc" | ||||
| 
 | ||||
|  | @ -1,3 +0,0 @@ | |||
| source_md5="075ad337ca02a6bc710d719cbd92152d" | ||||
| dest_md5="b4db58192c0182707fd3e8d35c0876fc" | ||||
| 
 | ||||
|  | @ -1,3 +0,0 @@ | |||
| source_md5="8dbbd7f9330263827d347335fd9dad42" | ||||
| dest_md5="5eb839a7b20831690790941de63c628e" | ||||
| 
 | ||||
|  | @ -1,3 +0,0 @@ | |||
| source_md5="74669865a41a88e37013718177f1f241" | ||||
| dest_md5="0801e44c630f05fe51ab6cb331943b60" | ||||
| 
 | ||||
|  | @ -1,3 +0,0 @@ | |||
| source_md5="13fa9399ff066d872ef7b61ab16b2e73" | ||||
| dest_md5="3bf30b72728e17e5b7b4f06796930ccd" | ||||
| 
 | ||||
|  | @ -1,3 +0,0 @@ | |||
| source_md5="d5bfa1139ddeff9aa03f46281442a4c0" | ||||
| dest_md5="7cb96bbbcffc716ecb3f9ec84e6f9752" | ||||
| 
 | ||||