feat: Working on route and id helpers for views.

This commit is contained in:
2025-01-17 23:50:04 -05:00
parent 531a385dba
commit d8328314ed
21 changed files with 585 additions and 255 deletions

View File

@@ -0,0 +1,84 @@
import Dependencies
import Elementary
import Fluent
import SharedModels
import Vapor
import VaporElementary
struct PurchaseOrderSearchViewController: RouteCollection {
@Dependency(\.database.employees) var employees
@Dependency(\.database.vendorBranches) var vendorBranches
@Dependency(\.database.purchaseOrders) var purchaseOrders
func boot(routes: any RoutesBuilder) throws {
let route = routes.protected.grouped("purchase-orders", "search")
route.get(use: index)
// route.get("form", use: form)
route.post(use: post)
}
@Sendable
func index(req: Request) async throws -> HTMLResponse {
let query = try? req.query.decode(FormQuery.self)
let html = PurchaseOrderSearch(context: query?.context)
guard req.isHtmxRequest else {
return await req.render { mainPage(search: html) }
}
return await req.render { html }
}
@Sendable
func post(req: Request) async throws -> HTMLResponse {
let context = try req.content.decode(PurchaseOrderSearchContent.self)
let results = try await purchaseOrders.search(context.toDatabaseQuery(), .init(page: 1, per: 25))
return await req.render { PurchaseOrderTable(page: results, context: .search, searchContext: nil) }
}
//
// @Sendable
// func form(req: Request) async throws -> HTMLResponse {
// let query = try req.query.decode(FormQuery.self)
// let html = PurchaseOrderSearch(context: query.context)
// guard req.isHtmxRequest else {
// return await req.render { mainPage(search: html) }
// }
// return await req.render { PurchaseOrderSearch(context: query.context) }
// }
func mainPage(search: PurchaseOrderSearch = .init()) -> some SendableHTMLDocument {
MainPage(displayNav: true, route: .purchaseOrders) {
div(.class("container"), .id("purchase-order-content")) {
search
PurchaseOrderTable(page: .init(items: [], metadata: .init(page: 0, per: 50, total: 0)))
}
}
}
}
extension PurchaseOrderSearchContent {
func toDatabaseQuery() throws -> PurchaseOrder.SearchContext {
switch context {
case .employee:
guard let createdForID else {
throw Abort(.badRequest, reason: "Employee id not provided")
}
return .employee(createdForID)
case .customer:
guard let search, !search.isEmpty else {
throw Abort(.badRequest, reason: "Customer search string is empty.")
}
return .customer(search)
case .vendor:
guard let vendorBranchID else {
throw Abort(.badRequest, reason: "Vendor branch id not provided.")
}
return .vendor(vendorBranchID)
}
}
}
private struct FormQuery: Content {
let context: PurchaseOrderSearchContext
}

View File

@@ -1,5 +1,6 @@
import Dependencies
import Elementary
import Fluent
import SharedModels
import Vapor
import VaporElementary
@@ -14,12 +15,10 @@ struct PurchaseOrderViewController: RouteCollection {
route.get(use: index)
route.get("next", use: nextPage)
route.post(use: create(req:))
route.post("search", use: postSearch)
route.get("search", use: getSearch)
// route.post("search", use: postSearch)
// route.get("search", use: getSearch)
route.group("create") {
$0.get(use: form)
$0.get("vendor-branch-select", use: vendorBranchSelect(req:))
$0.get("employee-select", use: employeeSelect(req:))
}
route.group(":id") {
$0.get(use: get)
@@ -49,18 +48,6 @@ struct PurchaseOrderViewController: RouteCollection {
return await req.render { PurchaseOrderForm(shouldShow: true) }
}
@Sendable
func vendorBranchSelect(req: Request) async throws -> HTMLResponse {
let branches = try await vendorBranches.fetchAllWithDetail()
return await req.render { PurchaseOrderForm.VendorSelect(vendorBranches: branches) }
}
@Sendable
func employeeSelect(req: Request) async throws -> HTMLResponse {
let employees = try await self.employees.fetchAll()
return await req.render { PurchaseOrderForm.EmployeeSelect(employees: employees) }
}
@Sendable
func get(req: Request) async throws -> HTMLResponse {
let purchaseOrder = try await purchaseOrders.get(req.ensureIDPathComponent(as: Int.self))
@@ -83,18 +70,23 @@ struct PurchaseOrderViewController: RouteCollection {
@Sendable
func postSearch(req: Request) async throws -> HTMLResponse {
let query = try req.content.decode([String: String].self)
req.logger.info("query: \(query)")
let context = try req.content.decode(SearchContext.self).toSearch()
let purchaseOrders = try await purchaseOrders.search(context)
req.logger.info("\(purchaseOrders)")
return await req.render { PurchaseOrderTable.Rows(page: purchaseOrders) }
let context = try req.content.decode(PurchaseOrderSearchContent.self)
let results = try await purchaseOrders.search(context.toDatabaseQuery(), .init(page: 1, per: 25))
return await req.render { PurchaseOrderTable(page: results, context: .search, searchContext: nil) }
}
// Show the form to generate a search query.
@Sendable
func getSearch(req: Request) async throws -> HTMLResponse {
let context = try req.query.decode(SearchQuery.self).toSearchContext()
return await req.render { PurchaseOrderSearch(context: context) }
// TODO: Need to handle updating the form.
return await req.render {
MainPage(displayNav: true, route: .purchaseOrders) {
div(.class("container"), .id("purchase-order-content")) {
PurchaseOrderSearch()
PurchaseOrderTable(page: .init(items: [], metadata: .init(page: 0, per: 50, total: 0)))
}
}
}
}
private func mainPage<C: HTML>(
@@ -103,9 +95,8 @@ struct PurchaseOrderViewController: RouteCollection {
) async throws -> some SendableHTMLDocument where C: Sendable {
let page = try await purchaseOrders.fetchPage(.init(page: page.page, per: page.limit))
return MainPage(displayNav: true, route: .purchaseOrders) {
div(.class("container")) {
div(.class("container"), .id("purchase-order-content")) {
html
PurchaseOrderSearch()
PurchaseOrderTable(page: page)
}
}
@@ -142,32 +133,3 @@ private struct CreateContext: Content {
)
}
}
private struct SearchContext: Content {
let context: String
let search: String
func toSearch() throws -> PurchaseOrder.SearchContext {
switch context {
case "employee":
return .employee(search)
case "customer":
return .customer(search)
case "vendor":
return .vendor(search)
default:
throw Abort(.badRequest, reason: "Invalid search context.")
}
}
}
struct SearchQuery: Content {
let context: String
func toSearchContext() throws -> PurchaseOrderSearchContext {
guard let context = PurchaseOrderSearchContext(rawValue: context) else {
throw Abort(.badRequest, reason: "Invalid context.")
}
return context
}
}

View File

@@ -11,28 +11,44 @@ struct UtilsViewController: RouteCollection {
let route = routes.protected
route.group("select") {
$0.get("employee", use: employeeSelect(req:))
$0.get("vendor-branches", use: vendorBranchSelect(req:))
}
}
@Sendable
func employeeSelect(req: Request) async throws -> HTMLResponse {
let context = try req.query.decode(EmployeeSelectContext.self)
let context = try req.query.decode(SelectQueryContext.self)
let employees = try await database.employees.fetchAll()
return await req.render { context.toHTML(employees: employees) }
}
@Sendable
func vendorBranchSelect(req: Request) async throws -> HTMLResponse {
let context = try req.query.decode(SelectQueryContext.self)
let branches = try await database.vendorBranches.fetchAllWithDetail()
return await req.render { context.toHTML(branches: branches) }
}
}
private struct EmployeeSelectContext: Content {
private struct SelectQueryContext: Content {
let context: EmployeeSelect.Context
let context: SelectContext
func toHTML(employees: [Employee]) -> EmployeeSelect {
switch context {
case .form:
case .purchaseOrderForm:
return .purchaseOrderForm(employees: employees)
case .search:
case .purchaseOrderSearch:
return .purchaseOrderSearch(employees: employees)
}
}
func toHTML(branches: [VendorBranch.Detail]) -> VendorBranchSelect {
switch context {
case .purchaseOrderForm:
return .purchaseOrderForm(branches: branches)
case .purchaseOrderSearch:
return .purchaseOrderSearch(branches: branches)
}
}
}