feat: features 025-032 - favoritos, contatos, trabalhe-conosco, area-cliente, navbar, hero-light-dark, performance-homepage
Some checks failed
CI/CD → Deploy via SSH / Build & Push Docker Images (push) Successful in 1m0s
CI/CD → Deploy via SSH / Deploy via SSH (push) Successful in 4m35s
CI/CD → Deploy via SSH / Validate HTTPS & Endpoints (push) Failing after 46s

- feat(025): favoritos locais com FavoritesContext, HeartButton, PublicFavoritesPage
- feat(026): central de contatos admin (leads/contatos unificados)
- feat(027): configuração da página de contato via admin
- feat(028): trabalhe conosco - candidaturas com upload e admin
- feat(029): UX área do cliente - visitas, comparação, perfil
- feat(030): navbar UX - menu mobile, ThemeToggle, useFavorites
- feat(031): hero light/dark - imagens separadas por tema, upload, preview, seed
- feat(032): performance homepage - Promise.all parallel fetches, sessionStorage cache,
  preload hero image, loading=lazy nos cards, useInView hook, will-change carrossel,
  keyframes em index.css, AgentsCarousel e HomeScrollScene via props
- fix: light mode HomeScrollScene - gradiente, cores de texto, scroll hint

migrations: g1h2i3j4k5l6 (source em leads), h1i2j3k4l5m6 (contact_config),
            i1j2k3l4m5n6 (job_applications), j2k3l4m5n6o7 (hero theme images)
This commit is contained in:
MatheusAlves96 2026-04-22 22:35:17 -03:00
parent 6ef5a7a17e
commit cf5603243c
106 changed files with 11927 additions and 1367 deletions

View file

@ -13,14 +13,51 @@ class PropertyBrief(BaseModel):
model_config = {"from_attributes": True}
class PropertyCard(BaseModel):
id: str
title: str
slug: str
price: Optional[Decimal] = None
city: Optional[str] = None
neighborhood: Optional[str] = None
cover_photo_url: Optional[str] = None
model_config = {"from_attributes": True}
@classmethod
def from_property(cls, prop) -> "PropertyCard":
cover = prop.photos[0].url if prop.photos else None
city = prop.city.name if prop.city else None
neighborhood = prop.neighborhood.name if prop.neighborhood else None
return cls(
id=str(prop.id),
title=prop.title,
slug=prop.slug,
price=prop.price,
city=city,
neighborhood=neighborhood,
cover_photo_url=cover,
)
class SavedPropertyOut(BaseModel):
id: str
property_id: Optional[str]
property: Optional[PropertyBrief]
property: Optional[PropertyCard]
created_at: datetime
model_config = {"from_attributes": True}
@classmethod
def from_saved(cls, saved) -> "SavedPropertyOut":
prop_card = PropertyCard.from_property(saved.property) if saved.property else None
return cls(
id=str(saved.id),
property_id=str(saved.property_id) if saved.property_id else None,
property=prop_card,
created_at=saved.created_at,
)
class FavoriteIn(BaseModel):
property_id: str
@ -93,3 +130,34 @@ class BoletoCreateIn(BaseModel):
if v <= 0:
raise ValueError("Valor deve ser positivo")
return v
class UpdateProfileIn(BaseModel):
name: str
@field_validator("name")
@classmethod
def name_not_empty(cls, v: str) -> str:
if not v.strip():
raise ValueError("Nome não pode ser vazio")
return v.strip()
class UpdateProfileOut(BaseModel):
id: str
name: str
email: str
model_config = {"from_attributes": True}
class UpdatePasswordIn(BaseModel):
current_password: str
new_password: str
@field_validator("new_password")
@classmethod
def min_length(cls, v: str) -> str:
if len(v) < 8:
raise ValueError("A nova senha deve ter pelo menos 8 caracteres")
return v

View file

@ -0,0 +1,43 @@
from __future__ import annotations
from pydantic import BaseModel, ConfigDict, field_validator
class ContactConfigOut(BaseModel):
model_config = ConfigDict(from_attributes=True)
address_street: str | None = None
address_neighborhood_city: str | None = None
address_zip: str | None = None
phone: str | None = None
email: str | None = None
business_hours: str | None = None
class ContactConfigIn(BaseModel):
address_street: str | None = None
address_neighborhood_city: str | None = None
address_zip: str | None = None
phone: str | None = None
email: str | None = None
business_hours: str | None = None
@field_validator("email")
@classmethod
def validate_email(cls, v: str | None) -> str | None:
if v is None:
return v
import re
v = v.strip().lower()
if v and not re.match(r"^[^@\s]+@[^@\s]+\.[^@\s]+$", v):
raise ValueError("E-mail inválido.")
return v or None
@field_validator("phone")
@classmethod
def validate_phone(cls, v: str | None) -> str | None:
if v is None:
return v
v = v.strip()
return v or None

View file

@ -12,6 +12,8 @@ class HomepageConfigOut(BaseModel):
hero_cta_url: str
featured_properties_limit: int
hero_image_url: str | None = None
hero_image_light_url: str | None = None
hero_image_dark_url: str | None = None
class HomepageConfigIn(BaseModel):
@ -21,6 +23,8 @@ class HomepageConfigIn(BaseModel):
hero_cta_url: str = "/imoveis"
featured_properties_limit: int = 6
hero_image_url: str | None = None
hero_image_light_url: str | None = None
hero_image_dark_url: str | None = None
@field_validator("hero_headline")
@classmethod
@ -35,3 +39,17 @@ class HomepageConfigIn(BaseModel):
if not (1 <= v <= 12):
raise ValueError("featured_properties_limit deve estar entre 1 e 12")
return v
class HomepageHeroImagesIn(BaseModel):
hero_image_url: str | None = None
hero_image_light_url: str | None = None
hero_image_dark_url: str | None = None
@field_validator("hero_image_url", "hero_image_light_url", "hero_image_dark_url")
@classmethod
def normalize_empty_to_none(cls, v: str | None) -> str | None:
if v is None:
return None
trimmed = v.strip()
return trimmed or None

View file

@ -0,0 +1,59 @@
from datetime import datetime
from typing import Optional
from pydantic import BaseModel, ConfigDict, EmailStr, field_validator
ROLE_INTEREST_OPTIONS = [
"Corretor(a)",
"Assistente Administrativo",
"Estagiário(a)",
"Outro",
]
class JobApplicationIn(BaseModel):
name: str
email: EmailStr
phone: Optional[str] = None
role_interest: str
message: str
file_name: Optional[str] = None
@field_validator("name")
@classmethod
def name_not_empty(cls, v: str) -> str:
v = v.strip()
if not v:
raise ValueError("Nome não pode ser vazio")
return v
@field_validator("role_interest")
@classmethod
def valid_role(cls, v: str) -> str:
if v not in ROLE_INTEREST_OPTIONS:
raise ValueError(f"Cargo inválido. Opções: {ROLE_INTEREST_OPTIONS}")
return v
@field_validator("message")
@classmethod
def message_not_empty(cls, v: str) -> str:
v = v.strip()
if not v:
raise ValueError("Mensagem não pode ser vazia")
if len(v) > 5000:
raise ValueError("Mensagem não pode ultrapassar 5000 caracteres")
return v
class JobApplicationOut(BaseModel):
model_config = ConfigDict(from_attributes=True)
id: int
name: str
email: str
phone: Optional[str]
role_interest: str
message: str
file_name: Optional[str]
status: str
created_at: datetime

View file

@ -13,6 +13,8 @@ class ContactLeadIn(BaseModel):
email: str
phone: str | None = None
message: str
source: str | None = None
source_detail: str | None = None
@field_validator("name")
@classmethod

View file

@ -7,7 +7,13 @@ from typing import Literal
from pydantic import BaseModel, ConfigDict
from app.schemas.catalog import AmenityOut, ImobiliariaOut, PropertyTypeOut, CityOut, NeighborhoodOut
from app.schemas.catalog import (
AmenityOut,
ImobiliariaOut,
PropertyTypeOut,
CityOut,
NeighborhoodOut,
)
class PropertyPhotoOut(BaseModel):