This commit is contained in:
2026-04-14 17:13:52 +01:00
parent fa9ba84d5c
commit 740680e9fb
13 changed files with 605 additions and 58 deletions

View File

@@ -4,6 +4,14 @@
<selectionStates>
<SelectionState runConfigName="app">
<option name="selectionMode" value="DROPDOWN" />
<DropdownSelection timestamp="2026-04-14T16:06:26.670067Z">
<Target type="DEFAULT_BOOT">
<handle>
<DeviceId pluginId="PhysicalDevice" identifier="serial=b93659d0e5dd" />
</handle>
</Target>
</DropdownSelection>
<DialogSelection />
</SelectionState>
</selectionStates>
</component>

View File

@@ -22,13 +22,26 @@ import com.google.firebase.database.ValueEventListener;
import java.util.HashMap;
import java.util.Map;
import android.net.Uri;
import android.content.Intent;
import android.widget.ImageView;
import com.bumptech.glide.Glide;
import com.google.firebase.storage.FirebaseStorage;
import com.google.firebase.storage.StorageReference;
import java.util.UUID;
import androidx.activity.result.ActivityResultLauncher;
import androidx.activity.result.contract.ActivityResultContracts;
public class DefinicoesAdminActivity extends AppCompatActivity {
private EditText inputRadius, inputAddress;
private android.widget.Spinner spinnerCategory;
private ImageView imgLogo;
private DatabaseReference databaseReference;
private String documentId;
private String[] categories = {"Carnes", "Massas", "Sushi", "Pizzas", "Sobremesas"};
private String photoUrl;
private ActivityResultLauncher<Intent> imagePickerLauncher;
private String[] categories = {"Carnes", "Massas", "Sushi", "Pizzas", "Sobremesas", "Italiana", "Moderna"};
@Override
protected void onCreate(Bundle savedInstanceState) {
@@ -51,6 +64,7 @@ public class DefinicoesAdminActivity extends AppCompatActivity {
inputRadius = findViewById(R.id.inputRadius);
inputAddress = findViewById(R.id.inputAddress);
spinnerCategory = findViewById(R.id.spinnerCategory);
imgLogo = findViewById(R.id.imgRestaurantLogo);
android.widget.ArrayAdapter<String> adapter = new android.widget.ArrayAdapter<>(this,
android.R.layout.simple_spinner_item, categories);
@@ -59,16 +73,79 @@ public class DefinicoesAdminActivity extends AppCompatActivity {
Button btnSave = findViewById(R.id.btnSaveSettings);
Button btnBack = findViewById(R.id.btnVoltar);
Button btnChangeLogo = findViewById(R.id.btnChangeLogo);
if (btnBack != null) {
btnBack.setOnClickListener(v -> finish());
}
imagePickerLauncher = registerForActivityResult(
new ActivityResultContracts.StartActivityForResult(),
result -> {
if (result.getResultCode() == RESULT_OK && result.getData() != null) {
Uri imageUri = result.getData().getData();
if (imageUri != null) {
uploadImageToFirebase(imageUri);
}
}
});
btnChangeLogo.setOnClickListener(v -> {
String[] options = {"Galeria", "URL da Imagem"};
new androidx.appcompat.app.AlertDialog.Builder(this)
.setTitle("Escolher Logótipo")
.setItems(options, (dialog, which) -> {
if (which == 0) {
Intent intent = new Intent(Intent.ACTION_PICK);
intent.setType("image/*");
imagePickerLauncher.launch(intent);
} else {
showUrlInputDialog();
}
}).show();
});
loadCurrentSettings();
btnSave.setOnClickListener(v -> saveSettings());
}
private void showUrlInputDialog() {
androidx.appcompat.app.AlertDialog.Builder builder = new androidx.appcompat.app.AlertDialog.Builder(this);
builder.setTitle("Inserir URL da Imagem");
final EditText input = new EditText(this);
input.setInputType(android.text.InputType.TYPE_TEXT_VARIATION_URI);
builder.setView(input);
builder.setPositiveButton("Confirmar", (dialog, which) -> {
String url = input.getText().toString().trim();
if (!TextUtils.isEmpty(url)) {
this.photoUrl = url;
Glide.with(this).load(photoUrl).circleCrop().into(imgLogo);
Toast.makeText(this, "Logótipo definido!", Toast.LENGTH_SHORT).show();
}
});
builder.setNegativeButton("Cancelar", (dialog, which) -> dialog.cancel());
builder.show();
}
private void uploadImageToFirebase(Uri imageUri) {
if (documentId == null) return;
StorageReference storageRef = FirebaseStorage.getInstance().getReference()
.child("restaurant_logos/" + UUID.randomUUID().toString());
storageRef.putFile(imageUri).addOnSuccessListener(taskSnapshot -> {
storageRef.getDownloadUrl().addOnSuccessListener(uri -> {
photoUrl = uri.toString();
Glide.with(this).load(photoUrl).circleCrop().into(imgLogo);
});
}).addOnFailureListener(e -> {
Toast.makeText(this, "Falha no upload: " + e.getMessage(), Toast.LENGTH_LONG).show();
});
}
private void loadCurrentSettings() {
if (documentId == null)
return;
@@ -96,6 +173,12 @@ public class DefinicoesAdminActivity extends AppCompatActivity {
}
}
}
if (snapshot.hasChild("logoUrl")) {
photoUrl = snapshot.child("logoUrl").getValue(String.class);
if (photoUrl != null && !photoUrl.isEmpty()) {
Glide.with(DefinicoesAdminActivity.this).load(photoUrl).circleCrop().into(imgLogo);
}
}
}
}
@@ -126,6 +209,9 @@ public class DefinicoesAdminActivity extends AppCompatActivity {
updates.put("securityDistance", radius);
updates.put("address", addressStr);
updates.put("category", selectedCategory);
if (photoUrl != null) {
updates.put("logoUrl", photoUrl);
}
databaseReference.child(documentId).updateChildren(updates)
.addOnSuccessListener(aVoid -> {

View File

@@ -68,5 +68,77 @@ public class EstablishmentDashboardActivity extends AppCompatActivity {
btnBack.setOnClickListener(v -> finish());
}
loadProximasReservas();
}
private void loadProximasReservas() {
android.widget.LinearLayout llProximasReservas = findViewById(R.id.llProximasReservas);
String email = getIntent().getStringExtra(MainActivity.EXTRA_EMAIL);
if (email == null) {
if (com.google.firebase.auth.FirebaseAuth.getInstance().getCurrentUser() != null) {
email = com.google.firebase.auth.FirebaseAuth.getInstance().getCurrentUser().getEmail();
} else {
return;
}
}
com.google.firebase.database.FirebaseDatabase.getInstance().getReference("reservas")
.orderByChild("restauranteEmail").equalTo(email)
.addValueEventListener(new com.google.firebase.database.ValueEventListener() {
@Override
public void onDataChange(@androidx.annotation.NonNull com.google.firebase.database.DataSnapshot snapshot) {
llProximasReservas.removeAllViews();
boolean found = false;
for (com.google.firebase.database.DataSnapshot ds : snapshot.getChildren()) {
com.example.pap_teste.models.Reserva r = ds.getValue(com.example.pap_teste.models.Reserva.class);
if (r != null && "Confirmada".equals(r.getEstado())) {
found = true;
addReservaView(llProximasReservas, r);
}
}
if (!found) {
TextView empty = new TextView(EstablishmentDashboardActivity.this);
empty.setText("Não há reservas confirmadas para hoje.");
empty.setTextColor(android.graphics.Color.parseColor("#5F5F5F"));
empty.setTextSize(14f);
llProximasReservas.addView(empty);
}
}
@Override
public void onCancelled(@androidx.annotation.NonNull com.google.firebase.database.DatabaseError error) {
}
});
}
private void addReservaView(android.widget.LinearLayout container, com.example.pap_teste.models.Reserva r) {
TextView title = new TextView(this);
title.setText(String.format("%s - %s", r.getHora(), r.getData()));
title.setTextColor(android.graphics.Color.BLACK);
title.setTextSize(16f);
title.setTypeface(null, android.graphics.Typeface.BOLD);
TextView subtitle = new TextView(this);
subtitle.setText(String.format("Cliente: %s • %d pessoas", r.getClienteEmail(), r.getPessoas()));
subtitle.setTextColor(android.graphics.Color.parseColor("#5F5F5F"));
subtitle.setTextSize(14f);
android.widget.LinearLayout.LayoutParams params = new android.widget.LinearLayout.LayoutParams(
android.widget.LinearLayout.LayoutParams.MATCH_PARENT, android.widget.LinearLayout.LayoutParams.WRAP_CONTENT);
subtitle.setLayoutParams(params);
android.view.View divider = new android.view.View(this);
android.widget.LinearLayout.LayoutParams divParams = new android.widget.LinearLayout.LayoutParams(
android.widget.LinearLayout.LayoutParams.MATCH_PARENT, 2);
divParams.setMargins(0, 24, 0, 24);
divider.setLayoutParams(divParams);
divider.setBackgroundColor(android.graphics.Color.parseColor("#EEEEEE"));
container.addView(title);
container.addView(subtitle);
container.addView(divider);
}
}

View File

@@ -97,11 +97,10 @@ public class ExplorarRestaurantesActivity extends AppCompatActivity {
if (name == null) name = ds.child("displayName").getValue(String.class);
String email = ds.child("email").getValue(String.class);
String cat = ds.child("category").getValue(String.class);
String logoUrl = ds.child("logoUrl").getValue(String.class);
// If no category filter in query, we might need a client-side filter if the index isn't used
// but here we use the query. Actually if filter is null we get all, but we need to ensure they are ESTAB
if (name != null && email != null) {
restaurantsList.add(new com.example.pap_teste.models.Restaurant(name, cat, email, false));
restaurantsList.add(new com.example.pap_teste.models.Restaurant(name, cat, email, false, logoUrl));
}
}
}
@@ -120,9 +119,13 @@ public class ExplorarRestaurantesActivity extends AppCompatActivity {
}
});
}
private androidx.activity.result.ActivityResultLauncher<android.content.Intent> photoPickerLauncher;
private void setupReservationOptions() {
android.widget.Button btnDate = findViewById(R.id.btnSelectDate);
android.widget.Button btnTime = findViewById(R.id.btnSelectTime);
android.widget.Button btnVerAvaliacoes = findViewById(R.id.btnVerAvaliacoes);
android.widget.Button btnUploadFoto = findViewById(R.id.btnUploadFoto);
btnDate.setOnClickListener(v -> {
java.util.Calendar cal = java.util.Calendar.getInstance();
@@ -141,6 +144,119 @@ public class ExplorarRestaurantesActivity extends AppCompatActivity {
});
findViewById(R.id.btnConfirmarReserva).setOnClickListener(v -> saveReservation());
btnVerAvaliacoes.setOnClickListener(v -> showReviewsDialog());
if (photoPickerLauncher == null) {
photoPickerLauncher = registerForActivityResult(
new androidx.activity.result.contract.ActivityResultContracts.StartActivityForResult(),
result -> {
if (result.getResultCode() == RESULT_OK && result.getData() != null) {
android.net.Uri imageUri = result.getData().getData();
if (imageUri != null) {
uploadRestaurantPhoto(imageUri);
}
}
});
}
btnUploadFoto.setOnClickListener(v -> {
android.content.Intent intent = new android.content.Intent(android.content.Intent.ACTION_PICK);
intent.setType("image/*");
photoPickerLauncher.launch(intent);
});
}
private void uploadRestaurantPhoto(android.net.Uri imageUri) {
if (selectedRestaurant == null || selectedRestaurant.getEmail() == null) return;
String encodedEmail = selectedRestaurant.getEmail().replace(".", "_").replace("@", "_at_");
com.google.firebase.storage.StorageReference storageRef = com.google.firebase.storage.FirebaseStorage.getInstance().getReference()
.child("restaurant_photos/" + encodedEmail + "/" + java.util.UUID.randomUUID().toString());
storageRef.putFile(imageUri).addOnSuccessListener(taskSnapshot -> {
storageRef.getDownloadUrl().addOnSuccessListener(uri -> {
String photoUrl = uri.toString();
com.google.firebase.database.FirebaseDatabase.getInstance().getReference("photos").child(encodedEmail)
.push().child("url").setValue(photoUrl).addOnCompleteListener(task -> {
if (task.isSuccessful()) {
android.widget.Toast.makeText(this, "Foto adicionada com sucesso!", android.widget.Toast.LENGTH_SHORT).show();
}
});
});
}).addOnFailureListener(e -> {
android.widget.Toast.makeText(this, "Falha no upload: " + e.getMessage(), android.widget.Toast.LENGTH_LONG).show();
});
}
private void showReviewsDialog() {
if (selectedRestaurant == null || selectedRestaurant.getEmail() == null) return;
String encodedEmail = selectedRestaurant.getEmail().replace(".", "_").replace("@", "_at_");
com.google.firebase.database.DatabaseReference reviewsRef = com.google.firebase.database.FirebaseDatabase.getInstance()
.getReference("reviews").child(encodedEmail);
reviewsRef.addListenerForSingleValueEvent(new com.google.firebase.database.ValueEventListener() {
@Override
public void onDataChange(@androidx.annotation.NonNull com.google.firebase.database.DataSnapshot snapshot) {
java.util.List<String> reviewsList = new java.util.ArrayList<>();
for (com.google.firebase.database.DataSnapshot dst : snapshot.getChildren()) {
String author = dst.child("author").getValue(String.class);
String text = dst.child("text").getValue(String.class);
if (author != null && text != null) {
reviewsList.add(author + "\n" + text);
}
}
String[] reviewsArray = reviewsList.toArray(new String[0]);
androidx.appcompat.app.AlertDialog.Builder builder = new androidx.appcompat.app.AlertDialog.Builder(ExplorarRestaurantesActivity.this)
.setTitle("Avaliações")
.setItems(reviewsArray, null)
.setPositiveButton("Fechar", null)
.setNeutralButton("Adicionar", (dialog, which) -> addReviewDialog());
if (reviewsList.isEmpty()) {
builder.setMessage("Ainda sem avaliações.");
}
builder.show();
}
@Override
public void onCancelled(@androidx.annotation.NonNull com.google.firebase.database.DatabaseError error) {
android.widget.Toast.makeText(ExplorarRestaurantesActivity.this, "Erro ao carregar avaliações.", android.widget.Toast.LENGTH_SHORT).show();
}
});
}
private void addReviewDialog() {
if (selectedRestaurant == null || selectedRestaurant.getEmail() == null) return;
android.widget.EditText input = new android.widget.EditText(this);
input.setHint("Escreva a sua avaliação aqui...");
new androidx.appcompat.app.AlertDialog.Builder(this)
.setTitle("Adicionar Avaliação")
.setView(input)
.setPositiveButton("Enviar", (dialog, which) -> {
String revText = input.getText().toString().trim();
if (!revText.isEmpty()) {
String userEmail = com.google.firebase.auth.FirebaseAuth.getInstance().getCurrentUser() != null
? com.google.firebase.auth.FirebaseAuth.getInstance().getCurrentUser().getEmail()
: "Visitante";
String encodedEmail = selectedRestaurant.getEmail().replace(".", "_").replace("@", "_at_");
java.util.Map<String, Object> review = new java.util.HashMap<>();
review.put("author", userEmail);
review.put("text", revText);
com.google.firebase.database.FirebaseDatabase.getInstance().getReference("reviews")
.child(encodedEmail).push().setValue(review).addOnCompleteListener(task -> {
if (task.isSuccessful()) {
android.widget.Toast.makeText(this, "Avaliação enviada!", android.widget.Toast.LENGTH_SHORT).show();
}
});
}
})
.setNegativeButton("Cancelar", null)
.show();
}
private void saveReservation() {

View File

@@ -10,8 +10,32 @@ import androidx.core.view.WindowInsetsCompat;
import android.widget.Button;
import android.widget.ArrayAdapter;
import android.widget.ListView;
import android.widget.TextView;
import android.widget.Toast;
import com.example.pap_teste.models.Reserva;
import com.google.firebase.database.DataSnapshot;
import com.google.firebase.database.DatabaseError;
import com.google.firebase.database.DatabaseReference;
import com.google.firebase.database.FirebaseDatabase;
import com.google.firebase.database.ValueEventListener;
import java.util.ArrayList;
import java.util.List;
public class ListaEsperaActivity extends AppCompatActivity {
private final List<Reserva> reservasPendentes = new ArrayList<>();
private ArrayAdapter<String> adapter;
private ListView listReservas;
private TextView txtInfo, txtNotas, txtMensagem;
private Button btnConfirmar, btnRecusar;
private int selectedIndex = -1;
private String restaurantEmail;
private DatabaseReference databaseReference;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
@@ -23,11 +47,130 @@ public class ListaEsperaActivity extends AppCompatActivity {
return insets;
});
restaurantEmail = getIntent().getStringExtra(MainActivity.EXTRA_EMAIL);
if (restaurantEmail == null) {
// Se o extra não chegou, falha segura
if (com.google.firebase.auth.FirebaseAuth.getInstance().getCurrentUser() != null) {
restaurantEmail = com.google.firebase.auth.FirebaseAuth.getInstance().getCurrentUser().getEmail();
} else {
restaurantEmail = "sabor_arte@restaurante.com";
}
}
bindViews();
setupList();
setupActions();
loadReservasPendentes();
}
private void bindViews() {
listReservas = findViewById(R.id.listReservasP);
txtInfo = findViewById(R.id.txtReservaInfoP);
txtNotas = findViewById(R.id.txtReservaNotasP);
txtMensagem = findViewById(R.id.txtMensagemReservaP);
btnConfirmar = findViewById(R.id.btnConfirmarReservaP);
btnRecusar = findViewById(R.id.btnRecusarReservaP);
Button back = findViewById(R.id.btnVoltar);
if (back != null) {
back.setOnClickListener(v -> finish());
}
}
private void setupList() {
adapter = new ArrayAdapter<>(this, android.R.layout.simple_list_item_activated_1);
listReservas.setAdapter(adapter);
listReservas.setOnItemClickListener((parent, view, position, id) -> {
selectedIndex = position;
mostrarDetalhe(reservasPendentes.get(position));
});
}
private void setupActions() {
btnConfirmar.setOnClickListener(v -> atualizarEstadoSelecionado("Confirmada"));
btnRecusar.setOnClickListener(v -> showRecusarDialog());
}
private void showRecusarDialog() {
if (selectedIndex < 0) return;
String[] motivos = { "Sem espaço no restaurante", "Fora de horas", "Reserva duplicada", "Outro" };
new androidx.appcompat.app.AlertDialog.Builder(this)
.setTitle("Motivo da Recusa")
.setItems(motivos, (dialog, which) -> {
atualizarEstadoSelecionado("Recusada (" + motivos[which] + ")");
})
.setNegativeButton("Voltar", null)
.show();
}
private void atualizarEstadoSelecionado(String novoEstado) {
if (selectedIndex < 0 || selectedIndex >= reservasPendentes.size()) {
Toast.makeText(this, "Selecione uma reserva para avaliar.", Toast.LENGTH_SHORT).show();
return;
}
Reserva item = reservasPendentes.get(selectedIndex);
databaseReference.child(item.getId()).child("estado").setValue(novoEstado).addOnCompleteListener(task -> {
if (task.isSuccessful()) {
Toast.makeText(this, "Reserva avaliada com sucesso.", Toast.LENGTH_SHORT).show();
selectedIndex = -1;
toggleButtons(null);
} else {
Toast.makeText(this, "Erro ao alterar estado da reserva.", Toast.LENGTH_SHORT).show();
}
});
}
private void mostrarDetalhe(Reserva item) {
txtInfo.setText(String.format("%s • %s", item.getClienteEmail(), item.getHora()));
txtNotas.setText(String.format("Data: %s • Pessoas: %d", item.getData(), item.getPessoas()));
toggleButtons(item);
}
private void toggleButtons(Reserva item) {
if (item == null) {
btnConfirmar.setVisibility(android.view.View.GONE);
btnRecusar.setVisibility(android.view.View.GONE);
txtInfo.setText("Selecione uma reserva");
txtNotas.setText("");
return;
}
btnConfirmar.setVisibility(android.view.View.VISIBLE);
btnRecusar.setVisibility(android.view.View.VISIBLE);
}
private void loadReservasPendentes() {
databaseReference = FirebaseDatabase.getInstance().getReference("reservas");
databaseReference.orderByChild("restauranteEmail").equalTo(restaurantEmail)
.addValueEventListener(new ValueEventListener() {
@Override
public void onDataChange(@androidx.annotation.NonNull DataSnapshot snapshot) {
reservasPendentes.clear();
for (DataSnapshot data : snapshot.getChildren()) {
Reserva r = data.getValue(Reserva.class);
if (r != null && "Pendente".equals(r.getEstado())) {
reservasPendentes.add(r);
}
}
refreshList();
}
@Override
public void onCancelled(@androidx.annotation.NonNull DatabaseError error) {
Toast.makeText(ListaEsperaActivity.this, "Erro ao carregar lista de espera.", Toast.LENGTH_SHORT).show();
}
});
}
private void refreshList() {
adapter.clear();
for (Reserva item : reservasPendentes) {
adapter.add(String.format("%s - %dp • %s", item.getHora(), item.getPessoas(), item.getClienteEmail()));
}
adapter.notifyDataSetChanged();
}
}

View File

@@ -46,6 +46,15 @@ public class RestaurantAdapter extends RecyclerView.Adapter<RestaurantAdapter.Vi
holder.text1.setText(restaurant.getName());
holder.text2.setText(restaurant.getCategory() + (restaurant.isAvailable() ? " - Disponível" : " - Indisponível"));
if (restaurant.getLogoUrl() != null && !restaurant.getLogoUrl().isEmpty()) {
com.bumptech.glide.Glide.with(holder.itemView.getContext())
.load(restaurant.getLogoUrl())
.circleCrop()
.into(holder.imgThumb);
} else {
holder.imgThumb.setImageResource(R.mipmap.ic_launcher);
}
holder.itemView.setOnClickListener(v -> {
if (listener != null) {
listener.onRestaurantClick(restaurant);
@@ -99,11 +108,13 @@ public class RestaurantAdapter extends RecyclerView.Adapter<RestaurantAdapter.Vi
public static class ViewHolder extends RecyclerView.ViewHolder {
TextView text1, text2;
ImageButton btnFavorite;
android.widget.ImageView imgThumb;
public ViewHolder(@NonNull View itemView) {
super(itemView);
text1 = itemView.findViewById(R.id.txtRestaurantName);
text2 = itemView.findViewById(R.id.txtRestaurantCategory);
btnFavorite = itemView.findViewById(R.id.btnFavorite);
imgThumb = itemView.findViewById(R.id.imgRestaurantThumb);
}
}
}

View File

@@ -5,6 +5,7 @@ public class Restaurant {
private String category;
private String email;
private boolean available;
private String logoUrl;
// No-argument constructor required for Firebase
public Restaurant() {
@@ -17,13 +18,23 @@ public class Restaurant {
this.available = available;
}
public Restaurant(String name, String category, String email, boolean available, String logoUrl) {
this.name = name;
this.category = category;
this.email = email;
this.available = available;
this.logoUrl = logoUrl;
}
public String getName() { return name; }
public String getCategory() { return category; }
public String getEmail() { return email; }
public boolean isAvailable() { return available; }
public String getLogoUrl() { return logoUrl; }
public void setName(String name) { this.name = name; }
public void setCategory(String category) { this.category = category; }
public void setEmail(String email) { this.email = email; }
public void setAvailable(boolean available) { this.available = available; }
public void setLogoUrl(String logoUrl) { this.logoUrl = logoUrl; }
}

View File

@@ -43,6 +43,26 @@
android:textSize="24sp"
android:textStyle="bold" />
<ImageView
android:id="@+id/imgRestaurantLogo"
android:layout_width="120dp"
android:layout_height="120dp"
android:layout_gravity="center_horizontal"
android:layout_marginTop="24dp"
android:src="@android:drawable/ic_menu_gallery"
android:background="#E0E0E0" />
<Button
android:id="@+id/btnChangeLogo"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center_horizontal"
android:layout_marginTop="12dp"
android:text="Alterar Logótipo"
android:textAllCaps="false"
android:backgroundTint="#2E7D32"
android:textColor="#FFFFFF" />
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"

View File

@@ -191,6 +191,7 @@
app:cardElevation="3dp">
<LinearLayout
android:id="@+id/llProximasReservas"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical"
@@ -199,38 +200,7 @@
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="20h00 - Mesa 14"
android:textColor="#000"
android:textSize="16sp"
android:textStyle="bold" />
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="4dp"
android:text="Cliente: Ana Ribeiro • 4 pessoas"
android:textColor="#5F5F5F"
android:textSize="14sp" />
<View
android:layout_width="match_parent"
android:layout_height="1dp"
android:layout_marginVertical="12dp"
android:background="#EEEEEE" />
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="21h15 - Mesa 03"
android:textColor="#000"
android:textSize="16sp"
android:textStyle="bold" />
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="4dp"
android:text="Cliente: Bruno Costa • 2 pessoas"
android:text="A procurar próximas reservas..."
android:textColor="#5F5F5F"
android:textSize="14sp" />
</LinearLayout>

View File

@@ -65,6 +65,40 @@
android:orientation="vertical"
android:paddingBottom="32dp">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:paddingHorizontal="24dp"
android:layout_marginTop="16dp">
<Button
android:id="@+id/btnVerAvaliacoes"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:layout_marginEnd="8dp"
android:text="Avaliações"
android:textAllCaps="false"
android:backgroundTint="#FFFFFF"
android:textColor="#1976D2"
app:strokeColor="#1976D2"
app:strokeWidth="1dp" />
<Button
android:id="@+id/btnUploadFoto"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:layout_marginStart="8dp"
android:text="Adicionar Foto"
android:textAllCaps="false"
android:backgroundTint="#FFFFFF"
android:textColor="#E53935"
app:strokeColor="#E53935"
app:strokeWidth="1dp" />
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"

View File

@@ -45,6 +45,92 @@
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintEnd_toEndOf="parent" />
<ListView
android:id="@+id/listReservasP"
android:layout_width="match_parent"
android:layout_height="0dp"
android:layout_margin="16dp"
android:background="#FFFFFF"
android:elevation="2dp"
app:layout_constraintTop_toBottomOf="@id/txtDescricaoListaEspera"
app:layout_constraintBottom_toTopOf="@id/detalhesCardP" />
<androidx.cardview.widget.CardView
android:id="@+id/detalhesCardP"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_margin="16dp"
app:cardCornerRadius="12dp"
app:cardElevation="4dp"
app:layout_constraintBottom_toBottomOf="parent">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical"
android:padding="16dp">
<TextView
android:id="@+id/txtReservaInfoP"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Selecione uma reserva"
android:textColor="#000"
android:textSize="16sp"
android:textStyle="bold" />
<TextView
android:id="@+id/txtReservaNotasP"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="4dp"
android:text=""
android:textColor="#5F5F5F"
android:textSize="14sp" />
<TextView
android:id="@+id/txtMensagemReservaP"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="8dp"
android:text=""
android:textColor="#1976D2"
android:textSize="14sp"
android:textStyle="italic" />
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="16dp"
android:orientation="horizontal">
<Button
android:id="@+id/btnConfirmarReservaP"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_marginEnd="8dp"
android:layout_weight="1"
android:backgroundTint="#4CAF50"
android:text="Aceitar"
android:textAllCaps="false"
android:textColor="#FFF"
android:visibility="gone" />
<Button
android:id="@+id/btnRecusarReservaP"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_marginStart="8dp"
android:layout_weight="1"
android:backgroundTint="#E53935"
android:text="Recusar"
android:textAllCaps="false"
android:textColor="#FFF"
android:visibility="gone" />
</LinearLayout>
</LinearLayout>
</androidx.cardview.widget.CardView>
</androidx.constraintlayout.widget.ConstraintLayout>

View File

@@ -1,21 +0,0 @@
<?xml version="1.0" encoding="utf-8"?>
<androidx.cardview.widget.CardView xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:id="@+id/cardOption"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_margin="8dp"
app:cardCornerRadius="12dp"
app:cardElevation="2dp"
app:cardBackgroundColor="#FFFFFF">
<TextView
android:id="@+id/txtOptionValue"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:paddingHorizontal="24dp"
android:paddingVertical="12dp"
android:text="Option"
android:textColor="#000"
android:textSize="14sp" />
</androidx.cardview.widget.CardView>

View File

@@ -13,11 +13,22 @@
android:layout_height="wrap_content"
android:padding="16dp">
<ImageView
android:id="@+id/imgRestaurantThumb"
android:layout_width="60dp"
android:layout_height="60dp"
android:layout_alignParentStart="true"
android:layout_centerVertical="true"
android:layout_marginEnd="12dp"
android:scaleType="centerCrop"
android:src="@mipmap/ic_launcher" />
<LinearLayout
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentStart="true"
android:layout_toEndOf="@id/imgRestaurantThumb"
android:layout_toStartOf="@id/btnFavorite"
android:layout_centerVertical="true"
android:orientation="vertical">
<TextView