268 lines
7.6 KiB
Swift
268 lines
7.6 KiB
Swift
import DatabaseClient
|
|
import Dependencies
|
|
import Elementary
|
|
import Foundation
|
|
import ManualDCore
|
|
|
|
extension ViewController.Request {
|
|
|
|
func render() async throws -> AnySendableHTML {
|
|
|
|
@Dependency(\.database) var database
|
|
|
|
switch route {
|
|
case .login(let route):
|
|
switch route {
|
|
case .index(let next):
|
|
return view {
|
|
LoginForm(next: next)
|
|
}
|
|
case .submit(let login):
|
|
let _ = try await authenticate(login)
|
|
return view {
|
|
LoggedIn(next: login.next)
|
|
}
|
|
}
|
|
case .signup(let route):
|
|
switch route {
|
|
case .index:
|
|
return view {
|
|
LoginForm(style: .signup)
|
|
}
|
|
case .submit(let request):
|
|
// Create a new user and log them in.
|
|
let user = try await createAndAuthenticate(request)
|
|
let projects = try await database.projects.fetch(user.id, .init(page: 1, per: 25))
|
|
return view {
|
|
ProjectsTable(userID: user.id, projects: projects)
|
|
}
|
|
}
|
|
case .project(let route):
|
|
return try await route.renderView(on: self)
|
|
case .effectiveLength(let route):
|
|
return try await route.renderView(isHtmxRequest: isHtmxRequest)
|
|
// case .user(let route):
|
|
// return try await route.renderView(isHtmxRequest: isHtmxRequest)
|
|
default:
|
|
// FIX: FIX
|
|
return _render(isHtmxRequest: false) {
|
|
div { "Fix me!" }
|
|
}
|
|
}
|
|
}
|
|
|
|
func view<C: HTML>(
|
|
@HTMLBuilder inner: () -> C
|
|
) -> AnySendableHTML where C: Sendable {
|
|
_render(isHtmxRequest: isHtmxRequest, showSidebar: showSidebar) {
|
|
inner()
|
|
}
|
|
}
|
|
|
|
var showSidebar: Bool {
|
|
switch route {
|
|
case .login, .signup, .project(.page):
|
|
return false
|
|
default:
|
|
return true
|
|
}
|
|
}
|
|
}
|
|
|
|
extension SiteRoute.View.ProjectRoute {
|
|
|
|
func renderView(on request: ViewController.Request) async throws -> AnySendableHTML {
|
|
@Dependency(\.database) var database
|
|
let user = try request.currentUser()
|
|
|
|
switch self {
|
|
case .index:
|
|
let projects = try await database.projects.fetchPage(userID: user.id)
|
|
return request.view {
|
|
ProjectsTable(userID: user.id, projects: projects)
|
|
}
|
|
case .page(let page):
|
|
let projects = try await database.projects.fetch(user.id, page)
|
|
return ProjectsTable(userID: user.id, projects: projects)
|
|
|
|
case .form(let dismiss):
|
|
return ProjectForm(dismiss: dismiss)
|
|
|
|
case .create(let form):
|
|
let project = try await database.projects.create(user.id, form)
|
|
try await database.componentLoss.createDefaults(projectID: project.id)
|
|
return ProjectView(projectID: project.id, activeTab: .rooms)
|
|
|
|
case .delete(let id):
|
|
try await database.projects.delete(id)
|
|
return EmptyHTML()
|
|
|
|
case .detail(let projectID, let route):
|
|
switch route {
|
|
case .index(let tab):
|
|
return ProjectView(projectID: projectID, activeTab: tab)
|
|
// return try await defaultDetailView(projectID: projectID, activeTab: tab)
|
|
case .equipment(let route):
|
|
return try await route.renderView(on: request, projectID: projectID)
|
|
case .frictionRate(let route):
|
|
return try await route.renderView(on: request, projectID: projectID)
|
|
case .rooms(let route):
|
|
return try await route.renderView(on: request, projectID: projectID)
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
extension SiteRoute.View.ProjectRoute.EquipmentInfoRoute {
|
|
func renderView(
|
|
on request: ViewController.Request,
|
|
projectID: Project.ID
|
|
) async throws -> AnySendableHTML {
|
|
@Dependency(\.database) var database
|
|
|
|
switch self {
|
|
case .index:
|
|
let equipment = try await database.equipment.fetch(projectID)
|
|
return EquipmentInfoView(equipmentInfo: equipment, projectID: projectID)
|
|
case .form(let dismiss):
|
|
let equipment = try await database.equipment.fetch(projectID)
|
|
return EquipmentInfoForm(dismiss: dismiss, projectID: projectID, equipmentInfo: equipment)
|
|
case .submit(let form):
|
|
let equipment = try await database.equipment.create(form)
|
|
return EquipmentInfoView(equipmentInfo: equipment, projectID: projectID)
|
|
case .update(let updates):
|
|
let equipment = try await database.equipment.update(updates)
|
|
return EquipmentInfoView(equipmentInfo: equipment, projectID: projectID)
|
|
}
|
|
}
|
|
}
|
|
|
|
extension SiteRoute.View.ProjectRoute.RoomRoute {
|
|
func renderView(
|
|
on request: ViewController.Request,
|
|
projectID: Project.ID
|
|
) async throws -> AnySendableHTML {
|
|
@Dependency(\.database) var database
|
|
|
|
switch self {
|
|
|
|
case .delete(let id):
|
|
try await database.rooms.delete(id)
|
|
return EmptyHTML()
|
|
|
|
case .form(let id, let dismiss):
|
|
var room: Room? = nil
|
|
if let id, dismiss == false {
|
|
room = try await database.rooms.get(id)
|
|
}
|
|
return RoomForm(dismiss: dismiss, projectID: projectID, room: room)
|
|
|
|
case .index:
|
|
return request.view {
|
|
ProjectView(projectID: projectID, activeTab: .rooms)
|
|
}
|
|
|
|
case .submit(let form):
|
|
request.logger.debug("New room form submitted.")
|
|
let _ = try await database.rooms.create(.init(form: form, projectID: projectID))
|
|
return request.view {
|
|
ProjectView(projectID: projectID, activeTab: .rooms)
|
|
}
|
|
|
|
case .update(let form):
|
|
_ = try await database.rooms.update(form)
|
|
return ProjectView(projectID: projectID, activeTab: .rooms)
|
|
}
|
|
}
|
|
}
|
|
|
|
extension SiteRoute.View.ProjectRoute.FrictionRateRoute {
|
|
func renderView(on request: ViewController.Request, projectID: Project.ID) async throws
|
|
-> AnySendableHTML
|
|
{
|
|
@Dependency(\.database) var database
|
|
|
|
switch self {
|
|
case .index:
|
|
let equipment = try await database.equipment.fetch(projectID)
|
|
let componentLosses = try await database.componentLoss.fetch(projectID)
|
|
|
|
return request.view {
|
|
ProjectView(projectID: projectID, activeTab: .frictionRate)
|
|
}
|
|
|
|
case .form(let type, let dismiss):
|
|
// FIX: Forms need to reference existing items.
|
|
switch type {
|
|
case .equipmentInfo:
|
|
return div { "REMOVE ME!" }
|
|
// return EquipmentForm(dismiss: dismiss, projectID: projectID)
|
|
case .componentPressureLoss:
|
|
return ComponentLossForm(dismiss: dismiss, projectID: projectID)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
extension SiteRoute.View.ProjectRoute.FrictionRateRoute.FormType {
|
|
var id: String {
|
|
switch self {
|
|
case .equipmentInfo:
|
|
return "equipmentForm"
|
|
case .componentPressureLoss:
|
|
return "componentLossForm"
|
|
}
|
|
}
|
|
}
|
|
|
|
extension SiteRoute.View.EffectiveLengthRoute {
|
|
|
|
func renderView(isHtmxRequest: Bool) async throws -> AnySendableHTML {
|
|
switch self {
|
|
case .index:
|
|
return _render(isHtmxRequest: isHtmxRequest, active: .effectiveLength) {
|
|
EffectiveLengthsView(effectiveLengths: EffectiveLength.mocks)
|
|
}
|
|
case .form(let dismiss):
|
|
return EffectiveLengthForm(dismiss: dismiss)
|
|
|
|
case .field(let type, let style):
|
|
switch type {
|
|
case .straightLength:
|
|
return StraightLengthField()
|
|
case .group:
|
|
// FIX:
|
|
return GroupField(style: style ?? .supply)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private func _render<C: HTML>(
|
|
isHtmxRequest: Bool,
|
|
active activeTab: SiteRoute.View.ProjectRoute.DetailRoute.Tab = .rooms,
|
|
showSidebar: Bool = true,
|
|
@HTMLBuilder inner: () async throws -> C
|
|
) async throws -> AnySendableHTML where C: Sendable {
|
|
let inner = try await inner()
|
|
if isHtmxRequest {
|
|
return inner
|
|
}
|
|
return MainPage { inner }
|
|
}
|
|
|
|
private func _render<C: HTML>(
|
|
isHtmxRequest: Bool,
|
|
active activeTab: SiteRoute.View.ProjectRoute.DetailRoute.Tab = .rooms,
|
|
showSidebar: Bool = true,
|
|
@HTMLBuilder inner: () -> C
|
|
) -> AnySendableHTML where C: Sendable {
|
|
let inner = inner()
|
|
if isHtmxRequest {
|
|
return inner
|
|
}
|
|
return MainPage { inner }
|
|
}
|