|
|
|
import Elementary
|
|
|
|
import ElementaryHTMX
|
|
|
|
import ElementaryHTMXSSE
|
|
|
|
import ElementaryHTMXWS
|
|
|
|
import Fluent
|
|
|
|
import Vapor
|
|
|
|
import VaporElementary
|
|
|
|
|
|
|
|
struct TodoController: RouteCollection {
|
|
|
|
func boot(routes: RoutesBuilder) throws {
|
|
|
|
routes.group("todos") { todos in
|
|
|
|
todos.get(use: index)
|
|
|
|
|
|
|
|
todos.post(use: create)
|
|
|
|
|
|
|
|
todos.group(":id") { todo in
|
|
|
|
todo.delete(use: delete)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Sendable
|
|
|
|
func index(req: Request) async throws -> HTMLResponse {
|
|
|
|
let todos = try await todos(db: req.db)
|
|
|
|
|
|
|
|
return HTMLResponse {
|
|
|
|
MainLayout(title: "Todos") {
|
|
|
|
TodosTableComponent(name: "todos", todos: todos)
|
|
|
|
TodosFormComponent(name: "todos-form", target: "todos")
|
|
|
|
|
|
|
|
button(.class("btn btn-primary"), .type(.button), .hx.get("/test/toast")) { "Toast" }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Sendable
|
|
|
|
func create(req: Request) async throws -> HTMLResponse {
|
|
|
|
do {
|
|
|
|
try TodoDTO.validate(content: req)
|
|
|
|
}
|
|
|
|
catch let error as ValidationsError {
|
|
|
|
return HTMLResponse {
|
|
|
|
TodosFormComponent(name: "todos-form", target: "todos", errors: ["title": error.description])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let todo = try req.content.decode(TodoDTO.self).toModel()
|
|
|
|
try await todo.save(on: req.db)
|
|
|
|
|
|
|
|
let todos = try await todos(db: req.db)
|
|
|
|
|
|
|
|
return HTMLResponse {
|
|
|
|
// Return the empty form
|
|
|
|
TodosFormComponent(name: "todos-form", target: "todos")
|
|
|
|
|
|
|
|
// Also update the todos table
|
|
|
|
TodosTableComponent(name: "todos", todos: todos, refresh: true) // TODO: Put component names inside variables
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Sendable
|
|
|
|
func delete(req: Request) async throws -> HTMLResponse {
|
|
|
|
guard let uuid = hexToUUID(hex: req.parameters.get("id")!),
|
|
|
|
let todo = try await Todo.find(uuid, on: req.db) else {
|
|
|
|
throw Abort(.notFound)
|
|
|
|
}
|
|
|
|
|
|
|
|
try await todo.delete(on: req.db)
|
|
|
|
|
|
|
|
return HTMLResponse {} // TODO: Return 204 No Content
|
|
|
|
}
|
|
|
|
|
|
|
|
// -------------------------------------
|
|
|
|
|
|
|
|
@Sendable
|
|
|
|
private func todos(db: any Database) async throws -> [TodoDTO] {
|
|
|
|
try await Todo.query(on: db).all().map { $0.toDTO() }
|
|
|
|
}
|
|
|
|
|
|
|
|
private func hexToUUID(hex: String) -> UUID? {
|
|
|
|
var uuid: String = hex.replacingOccurrences(of: "-", with: "")
|
|
|
|
|
|
|
|
guard uuid.count == 32 else { return nil }
|
|
|
|
|
|
|
|
uuid.insert("-", at: uuid.index(uuid.startIndex, offsetBy: 8))
|
|
|
|
uuid.insert("-", at: uuid.index(uuid.startIndex, offsetBy: 12 + 1))
|
|
|
|
uuid.insert("-", at: uuid.index(uuid.startIndex, offsetBy: 16 + 2))
|
|
|
|
uuid.insert("-", at: uuid.index(uuid.startIndex, offsetBy: 20 + 3))
|
|
|
|
|
|
|
|
return UUID(uuidString: uuid)
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|