- 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)
59 lines
1.4 KiB
Python
59 lines
1.4 KiB
Python
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
|