feat: features 025-032 - favoritos, contatos, trabalhe-conosco, area-cliente, navbar, hero-light-dark, performance-homepage
- 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:
parent
6ef5a7a17e
commit
cf5603243c
106 changed files with 11927 additions and 1367 deletions
|
|
@ -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
|
||||
|
|
|
|||
43
backend/app/schemas/contact_config.py
Normal file
43
backend/app/schemas/contact_config.py
Normal 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
|
||||
|
|
@ -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
|
||||
|
|
|
|||
59
backend/app/schemas/job_application.py
Normal file
59
backend/app/schemas/job_application.py
Normal 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
|
||||
|
|
@ -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
|
||||
|
|
|
|||
|
|
@ -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):
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue