- Backend: Migrations para tabelas 'escalas' e 'logistica' (transporte) - Backend: Handlers e Services completos para gestão de escalas e logística - Backend: Suporte a auth vinculado a perfil profissional - Frontend: Nova página de Detalhes Operacionais (/agenda/:id) - Frontend: Componente EventScheduler com verificação robusta de conflitos - Frontend: Componente EventLogistics para gestão de motoristas e caronas - Frontend: Modal de Detalhes de Profissional unificado (Admin + Self-view) - Frontend: Dashboard com modal de gestão de equipe e filtros avançados - Fix: Correção crítica de timezone (UTC) em horários de agendamento - Fix: Tratamento de URLs no campo de local do evento - Fix: Filtros de profissional com carro na logística
158 lines
4.3 KiB
Go
158 lines
4.3 KiB
Go
package escalas
|
|
|
|
import (
|
|
"net/http"
|
|
|
|
"github.com/gin-gonic/gin"
|
|
"github.com/google/uuid"
|
|
)
|
|
|
|
type Handler struct {
|
|
service *Service
|
|
}
|
|
|
|
func NewHandler(service *Service) *Handler {
|
|
return &Handler{service: service}
|
|
}
|
|
|
|
// Create godoc
|
|
// @Summary Create a schedule entry
|
|
// @Description Assign a professional to a time block in an event
|
|
// @Tags escalas
|
|
// @Accept json
|
|
// @Produce json
|
|
// @Param request body CreateEscalaInput true "Create Escala"
|
|
// @Success 201 {object} map[string]string
|
|
// @Router /api/escalas [post]
|
|
func (h *Handler) Create(c *gin.Context) {
|
|
var req CreateEscalaInput
|
|
if err := c.ShouldBindJSON(&req); err != nil {
|
|
c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
|
|
return
|
|
}
|
|
|
|
escala, err := h.service.Create(c.Request.Context(), req)
|
|
if err != nil {
|
|
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
|
|
return
|
|
}
|
|
|
|
c.JSON(http.StatusCreated, gin.H{"id": uuid.UUID(escala.ID.Bytes).String()})
|
|
}
|
|
|
|
// ListByAgenda godoc
|
|
// @Summary List schedules for an event
|
|
// @Description Get all schedule entries for a specific agenda_id
|
|
// @Tags escalas
|
|
// @Accept json
|
|
// @Produce json
|
|
// @Param agenda_id query string true "Agenda ID"
|
|
// @Success 200 {array} map[string]interface{}
|
|
// @Router /api/escalas [get]
|
|
func (h *Handler) ListByAgenda(c *gin.Context) {
|
|
agendaID := c.Query("agenda_id")
|
|
if agendaID == "" {
|
|
c.JSON(http.StatusBadRequest, gin.H{"error": "agenda_id is required"})
|
|
return
|
|
}
|
|
|
|
escalas, err := h.service.ListByAgenda(c.Request.Context(), agendaID)
|
|
if err != nil {
|
|
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
|
|
return
|
|
}
|
|
|
|
resp := make([]map[string]interface{}, len(escalas))
|
|
for i, e := range escalas {
|
|
var funcao string
|
|
if e.FuncaoEspecifica.Valid {
|
|
funcao = e.FuncaoEspecifica.String
|
|
}
|
|
var profNome string
|
|
profNome = e.ProfissionalNome
|
|
var avatar string
|
|
if e.AvatarUrl.Valid {
|
|
avatar = e.AvatarUrl.String
|
|
}
|
|
var phone string
|
|
if e.Whatsapp.Valid {
|
|
phone = e.Whatsapp.String
|
|
}
|
|
var profFuncao string
|
|
if e.FuncaoNome.Valid {
|
|
profFuncao = e.FuncaoNome.String
|
|
}
|
|
|
|
resp[i] = map[string]interface{}{
|
|
"id": uuid.UUID(e.ID.Bytes).String(),
|
|
"agenda_id": uuid.UUID(e.AgendaID.Bytes).String(),
|
|
"profissional_id": uuid.UUID(e.ProfissionalID.Bytes).String(),
|
|
"profissional_nome": profNome,
|
|
"avatar_url": avatar,
|
|
"phone": phone,
|
|
"profissional_role": profFuncao,
|
|
"start": e.DataHoraInicio.Time,
|
|
"end": e.DataHoraFim.Time,
|
|
"role": funcao, // Specific role in this slot
|
|
}
|
|
}
|
|
|
|
c.JSON(http.StatusOK, resp)
|
|
}
|
|
|
|
// Delete godoc
|
|
// @Summary Delete a schedule entry
|
|
// @Description Remove a professional from a time block
|
|
// @Tags escalas
|
|
// @Accept json
|
|
// @Produce json
|
|
// @Param id path string true "Escala ID"
|
|
// @Success 200 {object} map[string]string
|
|
// @Router /api/escalas/{id} [delete]
|
|
func (h *Handler) Delete(c *gin.Context) {
|
|
id := c.Param("id")
|
|
if id == "" {
|
|
c.JSON(http.StatusBadRequest, gin.H{"error": "id required"})
|
|
return
|
|
}
|
|
|
|
err := h.service.Delete(c.Request.Context(), id)
|
|
if err != nil {
|
|
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
|
|
return
|
|
}
|
|
|
|
c.JSON(http.StatusOK, gin.H{"message": "deleted"})
|
|
}
|
|
|
|
// Update godoc
|
|
// @Summary Update a schedule entry
|
|
// @Description Update time or professional for a slot
|
|
// @Tags escalas
|
|
// @Accept json
|
|
// @Produce json
|
|
// @Param id path string true "Escala ID"
|
|
// @Param request body UpdateEscalaInput true "Update Escala"
|
|
// @Success 200 {object} map[string]string
|
|
// @Router /api/escalas/{id} [put]
|
|
func (h *Handler) Update(c *gin.Context) {
|
|
id := c.Param("id")
|
|
if id == "" {
|
|
c.JSON(http.StatusBadRequest, gin.H{"error": "id required"})
|
|
return
|
|
}
|
|
|
|
var req UpdateEscalaInput
|
|
if err := c.ShouldBindJSON(&req); err != nil {
|
|
c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
|
|
return
|
|
}
|
|
|
|
_, err := h.service.Update(c.Request.Context(), id, req)
|
|
if err != nil {
|
|
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
|
|
return
|
|
}
|
|
|
|
c.JSON(http.StatusOK, gin.H{"message": "updated"})
|
|
}
|