feat: Begins vendor views, working form, and table. Styles need some updates.
This commit is contained in:
@@ -59,7 +59,8 @@ nav {
|
|||||||
}
|
}
|
||||||
|
|
||||||
form {
|
form {
|
||||||
padding: 50px 0;
|
padding: 80px;
|
||||||
|
width: 100%;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -69,6 +70,7 @@ form label {
|
|||||||
|
|
||||||
form input {
|
form input {
|
||||||
margin-bottom: 15px;
|
margin-bottom: 15px;
|
||||||
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.login-form {
|
.login-form {
|
||||||
@@ -204,3 +206,8 @@ input[type=text]:focus, input[type=password]:focus, input[type=email]:focus {
|
|||||||
.show {
|
.show {
|
||||||
display: block;
|
display: block;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
tr.htmx-swapping td {
|
||||||
|
opacity: 0;
|
||||||
|
transition: opacity 0.5s ease-out;
|
||||||
|
}
|
||||||
|
|||||||
@@ -31,12 +31,21 @@
|
|||||||
Employees
|
Employees
|
||||||
</a>
|
</a>
|
||||||
</li>
|
</li>
|
||||||
|
<li>
|
||||||
|
<a hx-get="/vendors?branches=true"
|
||||||
|
hx-target="body"
|
||||||
|
hx-push-url="true"
|
||||||
|
>
|
||||||
|
Vendors
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
</ul>
|
</ul>
|
||||||
</nav>
|
</nav>
|
||||||
</div>
|
</div>
|
||||||
<div id="home-content" class="container">
|
#import("homeContent")
|
||||||
<p>We're in!</p>
|
<!-- <div id="home-content" class="container"> -->
|
||||||
</div>
|
<!-- <p>We're in!</p> -->
|
||||||
|
<!-- </div> -->
|
||||||
</section>
|
</section>
|
||||||
</div>
|
</div>
|
||||||
#endexport
|
#endexport
|
||||||
|
|||||||
@@ -4,7 +4,7 @@
|
|||||||
#endif
|
#endif
|
||||||
#if(htmxPostTargetUrl):
|
#if(htmxPostTargetUrl):
|
||||||
hx-post="#(htmxPostTargetUrl)"
|
hx-post="#(htmxPostTargetUrl)"
|
||||||
#else:
|
#elseif(htmxPutTargetUrl):
|
||||||
hx-put="#(htmxPutTargetUrl)"
|
hx-put="#(htmxPutTargetUrl)"
|
||||||
#endif
|
#endif
|
||||||
hx-target="#(htmxTarget)"
|
hx-target="#(htmxTarget)"
|
||||||
|
|||||||
@@ -1,9 +1,11 @@
|
|||||||
<form hx-post="/vendors">
|
#extend("htmx-form", htmxForm):
|
||||||
|
#export("formBody"):
|
||||||
<input type="hidden"
|
<input type="hidden"
|
||||||
id="vendorId"
|
id="vendorId"
|
||||||
name="id"
|
name="id"
|
||||||
#if(vendor.id):
|
#if(context.vendor.id):
|
||||||
value="#(vendor.id)"
|
value="#(context.vendor.id)"
|
||||||
|
#endif
|
||||||
>
|
>
|
||||||
|
|
||||||
<input type="text"
|
<input type="text"
|
||||||
@@ -12,9 +14,22 @@
|
|||||||
placeholder="Vendor Name"
|
placeholder="Vendor Name"
|
||||||
autofocus
|
autofocus
|
||||||
required
|
required
|
||||||
#if(vendor.name):
|
#if(context.vendor.name):
|
||||||
value="#(vendor.name)"
|
value="#(context.vendor.name)"
|
||||||
#endif
|
#endif
|
||||||
>
|
>
|
||||||
<input type="submit" value="#(buttonLabel)">
|
<br>
|
||||||
</form>
|
<input type="text"
|
||||||
|
id="branches"
|
||||||
|
name="branches"
|
||||||
|
#if(context.branches):
|
||||||
|
value="#(context.branches)"
|
||||||
|
#endif
|
||||||
|
placeholder="Monroe, Dayton..."
|
||||||
|
>
|
||||||
|
<br>
|
||||||
|
<input type="submit" value="#(context.buttonLabel)">
|
||||||
|
<br>
|
||||||
|
<p><small>Mutliple branches can be specified separated by a comma.</small></p>
|
||||||
|
#endexport
|
||||||
|
#endextend
|
||||||
|
|||||||
34
Resources/Views/vendor-table.leaf
Normal file
34
Resources/Views/vendor-table.leaf
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
<table id="vendor-table">
|
||||||
|
<tr>
|
||||||
|
<th>Name</th>
|
||||||
|
<th>Branches</th>
|
||||||
|
<th></th>
|
||||||
|
</tr>
|
||||||
|
<tbody id="vendor-table-body">
|
||||||
|
#for(vendor in vendors):
|
||||||
|
<tr id="vendor_#(vendor.id)">
|
||||||
|
<td>#capitalized(vendor.name)</td>
|
||||||
|
<td>
|
||||||
|
#if(vendor.branches):
|
||||||
|
<ul>
|
||||||
|
#for(branch in vendor.branches):
|
||||||
|
<li>#capitalized(branch.name)</li>
|
||||||
|
#endfor
|
||||||
|
</ul>
|
||||||
|
#endif
|
||||||
|
</td>
|
||||||
|
<!-- TODO: Add edit button -->
|
||||||
|
<td>
|
||||||
|
<a class="btn btn-delete"
|
||||||
|
hx-delete="/vendors/#(vendor.id)"
|
||||||
|
hx-target="closest tr"
|
||||||
|
hx-swap="outerHTML swap:0.5s"
|
||||||
|
hx-confirm="Are you sure you want to delete this vendor?"
|
||||||
|
>
|
||||||
|
<img src="images/trash-can.svg" alt="Delete">
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
#endfor
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
14
Resources/Views/vendors.leaf
Normal file
14
Resources/Views/vendors.leaf
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
#extend("home"):
|
||||||
|
#export("homeContent"):
|
||||||
|
<div id="home-content" class="container">
|
||||||
|
<div class="container">
|
||||||
|
<h1>Vendors</h1>
|
||||||
|
<br>
|
||||||
|
<p>Vendors are who purchase orders can be issued for, they consist of multiple branches / locations.</p>
|
||||||
|
<br>
|
||||||
|
#extend("vendor-form", form)
|
||||||
|
#extend("vendor-table")
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
#endexport
|
||||||
|
#endextend
|
||||||
@@ -2,7 +2,7 @@ import Vapor
|
|||||||
|
|
||||||
/// Represents a generic form context that is used to generate form templates
|
/// Represents a generic form context that is used to generate form templates
|
||||||
/// that are handled by htmx.
|
/// that are handled by htmx.
|
||||||
struct HtmxFormCTX<C: Content>: Content {
|
struct HtmxFormCTX<Context: Content>: Content {
|
||||||
let formClass: String?
|
let formClass: String?
|
||||||
let formId: String
|
let formId: String
|
||||||
let htmxPostTargetUrl: String?
|
let htmxPostTargetUrl: String?
|
||||||
@@ -12,7 +12,7 @@ struct HtmxFormCTX<C: Content>: Content {
|
|||||||
let htmxResetAfterRequest: Bool
|
let htmxResetAfterRequest: Bool
|
||||||
let htmxSwapOob: String?
|
let htmxSwapOob: String?
|
||||||
let htmxSwap: String?
|
let htmxSwap: String?
|
||||||
let context: C
|
let context: Context
|
||||||
|
|
||||||
init(
|
init(
|
||||||
formClass: String? = nil,
|
formClass: String? = nil,
|
||||||
@@ -23,7 +23,7 @@ struct HtmxFormCTX<C: Content>: Content {
|
|||||||
htmxResetAfterRequest: Bool = true,
|
htmxResetAfterRequest: Bool = true,
|
||||||
htmxSwapOob: HtmxSwap? = nil,
|
htmxSwapOob: HtmxSwap? = nil,
|
||||||
htmxSwap: HtmxSwap? = nil,
|
htmxSwap: HtmxSwap? = nil,
|
||||||
context: C
|
context: Context
|
||||||
) {
|
) {
|
||||||
self.formClass = formClass
|
self.formClass = formClass
|
||||||
self.formId = formId
|
self.formId = formId
|
||||||
@@ -65,3 +65,33 @@ struct HtmxFormCTX<C: Content>: Content {
|
|||||||
}
|
}
|
||||||
|
|
||||||
struct EmptyContent: Content {}
|
struct EmptyContent: Content {}
|
||||||
|
|
||||||
|
struct ButtonLabelContext: Content {
|
||||||
|
let buttonLabel: String
|
||||||
|
}
|
||||||
|
|
||||||
|
extension HtmxFormCTX where Context == ButtonLabelContext {
|
||||||
|
init(
|
||||||
|
formClass: String? = nil,
|
||||||
|
formId: String,
|
||||||
|
htmxTargetUrl: TargetUrl,
|
||||||
|
htmxTarget: String,
|
||||||
|
htmxPushUrl: Bool,
|
||||||
|
htmxResetAfterRequest: Bool = true,
|
||||||
|
htmxSwapOob: HtmxSwap? = nil,
|
||||||
|
htmxSwap: HtmxSwap? = nil,
|
||||||
|
buttonLabel: String
|
||||||
|
) {
|
||||||
|
self.init(
|
||||||
|
formClass: formClass,
|
||||||
|
formId: formId,
|
||||||
|
htmxTargetUrl: htmxTargetUrl,
|
||||||
|
htmxTarget: htmxTarget,
|
||||||
|
htmxPushUrl: htmxPushUrl,
|
||||||
|
htmxResetAfterRequest: htmxResetAfterRequest,
|
||||||
|
htmxSwapOob: htmxSwapOob,
|
||||||
|
htmxSwap: htmxSwapOob,
|
||||||
|
context: .init(buttonLabel: buttonLabel)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -31,7 +31,7 @@ struct ApiController: RouteCollection {
|
|||||||
purchaseOrders.post(use: createPurchaseOrder(req:))
|
purchaseOrders.post(use: createPurchaseOrder(req:))
|
||||||
|
|
||||||
users.get(use: usersIndex(req:))
|
users.get(use: usersIndex(req:))
|
||||||
users.post(use: createUser(req:))
|
api.post("users", use: createUser(req:))
|
||||||
users.group("login") {
|
users.group("login") {
|
||||||
$0.get(use: self.login(req:))
|
$0.get(use: self.login(req:))
|
||||||
}
|
}
|
||||||
@@ -83,7 +83,7 @@ struct ApiController: RouteCollection {
|
|||||||
throw Abort(.notFound)
|
throw Abort(.notFound)
|
||||||
}
|
}
|
||||||
try await employee.delete(on: req.db)
|
try await employee.delete(on: req.db)
|
||||||
return .noContent
|
return .ok
|
||||||
}
|
}
|
||||||
|
|
||||||
@Sendable
|
@Sendable
|
||||||
@@ -150,6 +150,13 @@ struct ApiController: RouteCollection {
|
|||||||
|
|
||||||
@Sendable
|
@Sendable
|
||||||
func createUser(req: Request) async throws -> User.DTO {
|
func createUser(req: Request) async throws -> User.DTO {
|
||||||
|
let count = try await User.query(on: req.db).count()
|
||||||
|
if count > 0 {
|
||||||
|
guard req.auth.get(User.self) != nil else {
|
||||||
|
throw Abort(.unauthorized)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
try User.Create.validate(content: req)
|
try User.Create.validate(content: req)
|
||||||
let create = try req.content.decode(User.Create.self)
|
let create = try req.content.decode(User.Create.self)
|
||||||
guard create.password == create.confirmPassword else {
|
guard create.password == create.confirmPassword else {
|
||||||
@@ -184,7 +191,7 @@ struct ApiController: RouteCollection {
|
|||||||
}
|
}
|
||||||
|
|
||||||
try await user.delete(on: req.db)
|
try await user.delete(on: req.db)
|
||||||
return .noContent
|
return .ok
|
||||||
}
|
}
|
||||||
|
|
||||||
// MARK: - Vendors
|
// MARK: - Vendors
|
||||||
@@ -198,7 +205,10 @@ struct ApiController: RouteCollection {
|
|||||||
dbQuery = dbQuery.with(\.$branches)
|
dbQuery = dbQuery.with(\.$branches)
|
||||||
}
|
}
|
||||||
|
|
||||||
return try await dbQuery.all().map { $0.toDTO(includeBranches: params.branches) }
|
return try await dbQuery
|
||||||
|
.sort(\.$name, .ascending)
|
||||||
|
.all()
|
||||||
|
.map { $0.toDTO(includeBranches: params.branches) }
|
||||||
}
|
}
|
||||||
|
|
||||||
@Sendable
|
@Sendable
|
||||||
@@ -228,7 +238,7 @@ struct ApiController: RouteCollection {
|
|||||||
}
|
}
|
||||||
|
|
||||||
try await vendor.delete(on: req.db)
|
try await vendor.delete(on: req.db)
|
||||||
return .noContent
|
return .ok
|
||||||
}
|
}
|
||||||
|
|
||||||
// MARK: - VendorBranch
|
// MARK: - VendorBranch
|
||||||
@@ -258,7 +268,7 @@ struct ApiController: RouteCollection {
|
|||||||
throw Abort(.notFound)
|
throw Abort(.notFound)
|
||||||
}
|
}
|
||||||
try await branch.delete(on: req.db)
|
try await branch.delete(on: req.db)
|
||||||
return .noContent
|
return .ok
|
||||||
}
|
}
|
||||||
|
|
||||||
@Sendable
|
@Sendable
|
||||||
|
|||||||
@@ -5,16 +5,105 @@ struct VendorViewController: RouteCollection {
|
|||||||
private let api = ApiController()
|
private let api = ApiController()
|
||||||
|
|
||||||
func boot(routes: any RoutesBuilder) throws {
|
func boot(routes: any RoutesBuilder) throws {
|
||||||
// Do something.
|
let vendors = routes.protected.grouped("vendors")
|
||||||
|
|
||||||
|
vendors.get(use: index(req:))
|
||||||
|
vendors.post(use: create(req:))
|
||||||
|
vendors.group(":vendorID") {
|
||||||
|
$0.delete(use: delete(req:))
|
||||||
|
$0.put(use: update(req:))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Sendable
|
||||||
|
func index(req: Request) async throws -> View {
|
||||||
|
return try await req.view.render("vendors", makeCtx(req: req))
|
||||||
|
}
|
||||||
|
|
||||||
|
@Sendable
|
||||||
|
func create(req: Request) async throws -> View {
|
||||||
|
let ctx = try req.content.decode(CreateVendorCTX.self)
|
||||||
|
req.logger.info("CTX: \(ctx)")
|
||||||
|
let vendor = Vendor.Create(name: ctx.name).toModel()
|
||||||
|
try await vendor.save(on: req.db)
|
||||||
|
|
||||||
|
if let branchString = ctx.branches {
|
||||||
|
let branches = branchString.split(separator: ",")
|
||||||
|
.map { $0.trimmingCharacters(in: .whitespacesAndNewlines) }
|
||||||
|
|
||||||
|
for branch in branches {
|
||||||
|
try await vendor.$branches.create(
|
||||||
|
VendorBranch(name: String(branch), vendorId: vendor.requireID()),
|
||||||
|
on: req.db
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// _ = try await api.createVendor(req: req)
|
||||||
|
return try await req.view.render("vendor-table", makeCtx(req: req))
|
||||||
|
}
|
||||||
|
|
||||||
|
@Sendable
|
||||||
|
func delete(req: Request) async throws -> HTTPStatus {
|
||||||
|
try await api.deleteVendor(req: req)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Sendable
|
||||||
|
func update(req: Request) async throws -> View {
|
||||||
|
_ = try await api.updateVendor(req: req)
|
||||||
|
return try await req.view.render("vendor-table", makeCtx(req: req, oob: true))
|
||||||
|
}
|
||||||
|
|
||||||
|
private func makeCtx(req: Request, vendor: Vendor? = nil, oob: Bool = false) async throws -> VendorsCTX {
|
||||||
|
let vendors = try await Vendor.query(on: req.db)
|
||||||
|
.with(\.$branches)
|
||||||
|
.sort(\.$name, .ascending)
|
||||||
|
.all()
|
||||||
|
.map { $0.toDTO() }
|
||||||
|
|
||||||
|
return .init(
|
||||||
|
vendors: vendors,
|
||||||
|
form: .init(vendor: vendor, oob: oob)
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct VendorFormCTX: Content {
|
struct VendorFormCTX: Content {
|
||||||
|
let htmxForm: HtmxFormCTX<Context>
|
||||||
|
|
||||||
|
init(vendor: Vendor? = nil, oob: Bool = false) {
|
||||||
|
self.htmxForm = .init(
|
||||||
|
formClass: "vendor-form",
|
||||||
|
formId: "vendor-form",
|
||||||
|
htmxTargetUrl: vendor == nil ? .post("/vendors") : .put("/vendors"),
|
||||||
|
htmxTarget: "#vendor-table",
|
||||||
|
htmxPushUrl: false,
|
||||||
|
htmxResetAfterRequest: true,
|
||||||
|
htmxSwapOob: oob ? .outerHTML : nil,
|
||||||
|
htmxSwap: oob ? nil : .outerHTML,
|
||||||
|
context: .init(vendor: vendor)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
struct Context: Content {
|
||||||
let vendor: Vendor?
|
let vendor: Vendor?
|
||||||
|
let branches: String?
|
||||||
let buttonLabel: String
|
let buttonLabel: String
|
||||||
|
|
||||||
init(vendor: Vendor? = nil, buttonLabel: String = "Create") {
|
init(vendor: Vendor? = nil) {
|
||||||
self.vendor = vendor
|
self.vendor = vendor
|
||||||
self.buttonLabel = buttonLabel
|
self.branches = vendor?.branches.map(\.name).joined(separator: ", ")
|
||||||
|
self.buttonLabel = vendor == nil ? "Create" : "Update"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private struct VendorsCTX: Content {
|
||||||
|
let vendors: [Vendor.DTO]
|
||||||
|
let form: VendorFormCTX
|
||||||
|
}
|
||||||
|
|
||||||
|
private struct CreateVendorCTX: Content {
|
||||||
|
let name: String
|
||||||
|
let branches: String?
|
||||||
|
}
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ struct ViewController: RouteCollection {
|
|||||||
private let api = ApiController()
|
private let api = ApiController()
|
||||||
private let employees = EmployeeViewController()
|
private let employees = EmployeeViewController()
|
||||||
private let users = UserViewController()
|
private let users = UserViewController()
|
||||||
|
private let vendors = VendorViewController()
|
||||||
|
|
||||||
func boot(routes: any RoutesBuilder) throws {
|
func boot(routes: any RoutesBuilder) throws {
|
||||||
let protected = routes.protected
|
let protected = routes.protected
|
||||||
@@ -25,6 +26,7 @@ struct ViewController: RouteCollection {
|
|||||||
// protected.get("users", use: users(req:))
|
// protected.get("users", use: users(req:))
|
||||||
try routes.register(collection: employees)
|
try routes.register(collection: employees)
|
||||||
try routes.register(collection: users)
|
try routes.register(collection: users)
|
||||||
|
try routes.register(collection: vendors)
|
||||||
}
|
}
|
||||||
|
|
||||||
@Sendable
|
@Sendable
|
||||||
@@ -99,6 +101,7 @@ private struct UserForm: Content {
|
|||||||
enum HomeRoute: String, Content {
|
enum HomeRoute: String, Content {
|
||||||
case employees
|
case employees
|
||||||
case users
|
case users
|
||||||
|
case vendors
|
||||||
}
|
}
|
||||||
|
|
||||||
struct HomeCTX: Content {
|
struct HomeCTX: Content {
|
||||||
|
|||||||
@@ -79,7 +79,6 @@ extension Vendor {
|
|||||||
func revert(on database: Database) async throws {
|
func revert(on database: Database) async throws {
|
||||||
try await database.schema(Vendor.schema).delete()
|
try await database.schema(Vendor.schema).delete()
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
struct Update: Content {
|
struct Update: Content {
|
||||||
|
|||||||
@@ -73,7 +73,8 @@ extension VendorBranch {
|
|||||||
try await database.schema(VendorBranch.schema)
|
try await database.schema(VendorBranch.schema)
|
||||||
.id()
|
.id()
|
||||||
.field("name", .string, .required)
|
.field("name", .string, .required)
|
||||||
.field("vendor_id", .uuid, .required, .references("vendor", "id"))
|
.field("vendor_id", .uuid, .required)
|
||||||
|
.foreignKey("vendor_id", references: Vendor.schema, "id", onDelete: .cascade)
|
||||||
.create()
|
.create()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user