feat: Moves employee views to their own controller, updates css, and employee table view.

This commit is contained in:
2025-01-07 17:07:37 -05:00
parent 08a0a8e1a3
commit 6eb723a7cf
11 changed files with 215 additions and 81 deletions

View File

@@ -0,0 +1,111 @@
import Fluent
import Leaf
import Vapor
struct EmployeeViewController: RouteCollection {
private let api = ApiController()
func boot(routes: any RoutesBuilder) throws {
let employees = routes.protected.grouped("employees")
// MARK: Protected routes.
employees.get(use: employees(req:))
employees.get("form", use: employeeForm(req:))
employees.post(use: postEmployeeForm(req:))
employees.group(":employeeID") {
$0.get(use: editEmployee(req:))
$0.delete(use: deleteEmployee(req:))
$0.put(use: updateEmployee(req:))
$0.post("toggle-active", use: toggleActiveEmployee(req:))
}
}
@Sendable
func employees(req: Request) async throws -> View {
return try await req.view.render("employees", EmployeesCTX(api: api, req: req))
}
@Sendable
func postEmployeeForm(req: Request) async throws -> View {
_ = try await api.createEmployee(req: req)
let employees = try await api.getSortedEmployees(req: req)
return try await req.view.render("employee-table", ["employees": employees])
}
@Sendable
func toggleActiveEmployee(req: Request) async throws -> View {
guard let employee = try await Employee.find(req.parameters.get("employeeID"), on: req.db) else {
throw Abort(.notFound)
}
employee.active.toggle()
try await employee.save(on: req.db)
let employees = try await api.getSortedEmployees(req: req)
return try await req.view.render("employee-table", ["employees": employees])
}
@Sendable
func deleteEmployee(req: Request) async throws -> View {
_ = try await api.deleteEmployee(req: req)
let employees = try await api.getSortedEmployees(req: req)
return try await req.view.render("employee-table", ["employees": employees])
}
@Sendable
func editEmployee(req: Request) async throws -> View {
guard let employee = try await Employee.find(req.parameters.get("employeeID"), on: req.db) else {
throw Abort(.notFound)
}
return try await req.view.render("employee-form", EmployeeFormCTX(employee: employee.toDTO()))
}
@Sendable
func updateEmployee(req: Request) async throws -> View {
_ = try await api.updateEmployee(req: req)
return try await req.view.render("employees", EmployeesCTX(oob: true, api: api, req: req))
}
@Sendable
func employeeForm(req: Request) async throws -> View {
try await req.view.render("employee-form", EmployeeFormCTX())
}
}
private struct EmployeesCTX: Content {
let oob: Bool
let employees: [Employee.DTO]
let form: EmployeeFormCTX
init(
oob: Bool = false,
employee: Employee? = nil,
api: ApiController,
req: Request
) async throws {
self.oob = oob
self.employees = try await api.getSortedEmployees(req: req)
self.form = .init(employee: employee.map { $0.toDTO() })
}
}
private struct EmployeeFormCTX: Content {
let employee: Employee.DTO?
let oob: Bool
init(employee: Employee.DTO? = nil) {
self.employee = employee
self.oob = employee != nil
}
}
private extension ApiController {
func getSortedEmployees(req: Request) async throws -> [Employee.DTO] {
var employees = try await employeesIndex(req: req)
employees.sort { ($0.active ?? false) && !($1.active ?? false) }
employees.sort { ($0.lastName ?? "") < ($1.lastName ?? "") }
return employees
}
}

View File

@@ -7,9 +7,8 @@ struct ViewController: RouteCollection {
private let api = ApiController()
func boot(routes: any RoutesBuilder) throws {
let protected = routes.grouped(User.credentialsAuthenticator(), User.redirectMiddleware(path: "login"))
let protected = routes.protected
let login = routes.grouped("login")
let employees = protected.grouped("employees")
// MARK: - Non-protected routes.
@@ -23,12 +22,7 @@ struct ViewController: RouteCollection {
protected.get("home", use: home(req:))
protected.get("users", use: users(req:))
employees.get(use: employees(req:))
employees.post(use: postEmployeeForm(req:))
employees.group(":employeeID") {
$0.delete(use: deleteEmployee(req:))
$0.post("toggle-active", use: toggleActiveEmployee(req:))
}
try routes.register(collection: EmployeeViewController())
}
@Sendable
@@ -66,6 +60,8 @@ struct ViewController: RouteCollection {
return try await req.view.render("login")
}
// TODO: Add route parameters for active route / tab.
@Sendable
func home(req: Request) async throws -> View {
try await req.view.render("home")
@@ -77,50 +73,9 @@ struct ViewController: RouteCollection {
return try await req.view.render("users", ["users": users])
}
@Sendable
func employees(req: Request) async throws -> View {
let employees = try await api.getSortedEmployees(req: req)
return try await req.view.render("employees", ["employees": employees])
}
@Sendable
func postEmployeeForm(req: Request) async throws -> View {
_ = try await api.createEmployee(req: req)
let employees = try await api.getSortedEmployees(req: req)
return try await req.view.render("employee-table", ["employees": employees])
}
@Sendable
func toggleActiveEmployee(req: Request) async throws -> View {
guard let employee = try await Employee.find(req.parameters.get("employeeID"), on: req.db) else {
throw Abort(.notFound)
}
employee.active.toggle()
try await employee.save(on: req.db)
let employees = try await api.getSortedEmployees(req: req)
return try await req.view.render("employee-table", ["employees": employees])
}
@Sendable
func deleteEmployee(req: Request) async throws -> View {
_ = try await api.deleteEmployee(req: req)
let employees = try await api.getSortedEmployees(req: req)
return try await req.view.render("employee-table", ["employees": employees])
}
}
private struct UserForm: Content {
let username: String
let password: String
}
private extension ApiController {
func getSortedEmployees(req: Request) async throws -> [Employee.DTO] {
var employees = try await employeesIndex(req: req)
employees.sort { ($0.active ?? false) && !($1.active ?? false) }
employees.sort { ($0.lastName ?? "") < ($1.lastName ?? "") }
return employees
}
}

View File

@@ -0,0 +1,10 @@
import Vapor
extension RoutesBuilder {
// Used to ensure views are protected, redirects users to the login page if they're
// not authenticated.
var protected: any RoutesBuilder {
grouped(User.credentialsAuthenticator(), User.redirectMiddleware(path: "login"))
}
}

View File

@@ -2,6 +2,8 @@ import Fluent
import struct Foundation.UUID
import Vapor
// TODO: Add soft-delete??
/// The employee database model.
///
/// An employee is someone that PO's can be generated for. They can be either a field