feat-WIP: Style updates, new form inputs.

This commit is contained in:
2026-01-12 22:49:58 -05:00
parent fa9e8cffb0
commit 432533c940
13 changed files with 249 additions and 134 deletions

View File

@@ -20,6 +20,8 @@
--spacing: 0.25rem; --spacing: 0.25rem;
--text-sm: 0.875rem; --text-sm: 0.875rem;
--text-sm--line-height: calc(1.25 / 0.875); --text-sm--line-height: calc(1.25 / 0.875);
--text-base: 1rem;
--text-base--line-height: calc(1.5 / 1);
--text-lg: 1.125rem; --text-lg: 1.125rem;
--text-lg--line-height: calc(1.75 / 1.125); --text-lg--line-height: calc(1.75 / 1.125);
--text-xl: 1.25rem; --text-xl: 1.25rem;
@@ -5366,6 +5368,12 @@
} }
} }
} }
.-my-2 {
margin-block: calc(var(--spacing) * -2);
}
.-my-4 {
margin-block: calc(var(--spacing) * -4);
}
.my-1 { .my-1 {
margin-block: calc(var(--spacing) * 1); margin-block: calc(var(--spacing) * 1);
} }
@@ -5581,6 +5589,12 @@
border-width: var(--border, 1px) 0 var(--border, 1px) var(--border, 1px); border-width: var(--border, 1px) 0 var(--border, 1px) var(--border, 1px);
} }
} }
.me-4 {
margin-inline-end: calc(var(--spacing) * 4);
}
.me-6 {
margin-inline-end: calc(var(--spacing) * 6);
}
.modal-action { .modal-action {
@layer daisyui.l1.l2.l3 { @layer daisyui.l1.l2.l3 {
margin-top: calc(0.25rem * 6); margin-top: calc(0.25rem * 6);
@@ -5620,6 +5634,12 @@
} }
} }
} }
.mt-1 {
margin-top: calc(var(--spacing) * 1);
}
.mt-2 {
margin-top: calc(var(--spacing) * 2);
}
.mt-4 { .mt-4 {
margin-top: calc(var(--spacing) * 4); margin-top: calc(var(--spacing) * 4);
} }
@@ -6768,6 +6788,9 @@
.grid-cols-2 { .grid-cols-2 {
grid-template-columns: repeat(2, minmax(0, 1fr)); grid-template-columns: repeat(2, minmax(0, 1fr));
} }
.grid-cols-3 {
grid-template-columns: repeat(3, minmax(0, 1fr));
}
.grid-cols-5 { .grid-cols-5 {
grid-template-columns: repeat(5, minmax(0, 1fr)); grid-template-columns: repeat(5, minmax(0, 1fr));
} }
@@ -7922,6 +7945,10 @@
font-size: var(--text-4xl); font-size: var(--text-4xl);
line-height: var(--tw-leading, var(--text-4xl--line-height)); line-height: var(--tw-leading, var(--text-4xl--line-height));
} }
.text-base {
font-size: var(--text-base);
line-height: var(--tw-leading, var(--text-base--line-height));
}
.text-lg { .text-lg {
font-size: var(--text-lg); font-size: var(--text-lg);
line-height: var(--tw-leading, var(--text-lg--line-height)); line-height: var(--tw-leading, var(--text-lg--line-height));

View File

@@ -15,6 +15,7 @@ public enum Theme: String, CaseIterable, Codable, Equatable, Sendable {
public static let darkThemes = [ public static let darkThemes = [
Self.aqua, Self.aqua,
Self.cyberpunk,
Self.dark, Self.dark,
Self.dracula, Self.dracula,
Self.night, Self.night,
@@ -23,7 +24,6 @@ public enum Theme: String, CaseIterable, Codable, Equatable, Sendable {
public static let lightThemes = [ public static let lightThemes = [
Self.cupcake, Self.cupcake,
Self.cyberpunk,
Self.light, Self.light,
Self.nord, Self.nord,
Self.retro, Self.retro,

View File

@@ -13,7 +13,7 @@ public struct Label: HTML, Sendable {
} }
public var body: some HTML<HTMLTag.span> { public var body: some HTML<HTMLTag.span> {
span(.class("text-xl text-secondary font-bold")) { span(.class("text-lg text-secondary font-bold")) {
title title
} }
} }

View File

@@ -0,0 +1,18 @@
import Elementary
public struct PageTitle: HTML, Sendable {
let title: String
public init(_ title: String) {
self.title = title
}
public init(_ title: () -> String) {
self.title = title()
}
public var body: some HTML<HTMLTag.h1> {
h1(.class("text-3xl font-bold")) { title }
}
}

View File

@@ -14,7 +14,7 @@ struct DuctSizingView: HTML, Sendable {
var body: some HTML { var body: some HTML {
div { div {
h1(.class("text-2xl py-4")) { "Duct Sizes" } PageTitle { "Duct Sizes" }
if rooms.count == 0 { if rooms.count == 0 {
p(.class("text-error italic")) { p(.class("text-error italic")) {
"Must complete all the previous sections to display duct sizing calculations." "Must complete all the previous sections to display duct sizing calculations."

View File

@@ -23,11 +23,9 @@ struct EffectiveLengthsView: HTML, Sendable {
} }
var body: some HTML { var body: some HTML {
div( div(.class("space-y-4")) {
.class("m-4 space-y-4")
) {
Row { Row {
h1(.class("text-2xl font-bold")) { "Equivalent Lengths" } PageTitle { "Equivalent Lengths" }
PlusButton() PlusButton()
.attributes( .attributes(
.class("btn-ghost"), .class("btn-ghost"),
@@ -40,6 +38,8 @@ struct EffectiveLengthsView: HTML, Sendable {
div { div {
h2(.class("text-xl font-bold pb-4")) { "Supplies" } h2(.class("text-xl font-bold pb-4")) { "Supplies" }
.attributes(.class("hidden"), when: supplies.count == 0)
div(.class("grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4")) { div(.class("grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4")) {
for (n, row) in supplies.enumerated() { for (n, row) in supplies.enumerated() {
EffectiveLengthView(effectiveLength: row) EffectiveLengthView(effectiveLength: row)
@@ -50,6 +50,7 @@ struct EffectiveLengthsView: HTML, Sendable {
div { div {
h2(.class("text-xl font-bold pb-4")) { "Returns" } h2(.class("text-xl font-bold pb-4")) { "Returns" }
.attributes(.class("hidden"), when: returns.count == 0)
div(.class("grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 space-x-4 space-y-4")) { div(.class("grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 space-x-4 space-y-4")) {
for (n, row) in returns.enumerated() { for (n, row) in returns.enumerated() {
EffectiveLengthView(effectiveLength: row) EffectiveLengthView(effectiveLength: row)

View File

@@ -29,7 +29,7 @@ struct EquipmentInfoForm: HTML, Sendable {
ModalForm(id: Self.id, dismiss: dismiss) { ModalForm(id: Self.id, dismiss: dismiss) {
h1(.class("text-3xl font-bold pb-6 ps-2")) { "Equipment Info" } h1(.class("text-3xl font-bold pb-6 ps-2")) { "Equipment Info" }
form( form(
.class("space-y-4 p-4"), .class("grid grid-cols-1 gap-4"),
equipmentInfo != nil equipmentInfo != nil
? .hx.patch(route) ? .hx.patch(route)
: .hx.post(route), : .hx.post(route),
@@ -42,27 +42,40 @@ struct EquipmentInfoForm: HTML, Sendable {
input(.class("hidden"), .name("id"), .value("\(equipmentInfo.id)")) input(.class("hidden"), .name("id"), .value("\(equipmentInfo.id)"))
} }
div { LabeledInput(
label(.for("staticPressure")) { "Static Pressure" } "Static Pressure",
Input(id: "staticPressure", placeholder: "Static pressure") .name("staticPressure"),
.attributes( .type(.number),
.type(.number), .value(staticPressure), .min("0"), .max("1.0"), .step("0.1") .value(staticPressure),
) .min("0"),
} .max("1.0"),
div { .step("0.1"),
label(.for("heatingCFM")) { "Heating CFM" } .required
Input(id: "heatingCFM", placeholder: "CFM") )
.attributes(.type(.number), .min("0"), .value(equipmentInfo?.heatingCFM))
} LabeledInput(
div { "Heating CFM",
label(.for("coolingCFM")) { "Cooling CFM" } .name("heatingCFM"),
Input(id: "coolingCFM", placeholder: "CFM") .type(.number),
.attributes(.type(.number), .min("0"), .value(equipmentInfo?.coolingCFM)) .value(equipmentInfo?.heatingCFM),
} .placeholder("1000"),
div { .min("0"),
SubmitButton(title: "Save") .required,
.attributes(.class("btn-block")) .autofocus
} )
LabeledInput(
"Cooling CFM",
.name("coolingCFM"),
.type(.number),
.value(equipmentInfo?.coolingCFM),
.placeholder("1000"),
.min("0"),
.required
)
SubmitButton(title: "Save")
.attributes(.class("btn-block my-6"))
} }
} }
} }

View File

@@ -8,12 +8,12 @@ struct EquipmentInfoView: HTML, Sendable {
var body: some HTML { var body: some HTML {
div( div(
.class("space-y-4 p-4"), .class("space-y-4"),
.id("equipmentInfo") .id("equipmentInfo")
) { ) {
Row { Row {
h1(.class("text-2xl font-bold")) { "Equipment Info" } PageTitle { "Equipment Info" }
Tooltip("Edit equipment info") { Tooltip("Edit equipment info") {
EditButton() EditButton()
@@ -29,16 +29,28 @@ struct EquipmentInfoView: HTML, Sendable {
table(.class("table table-zebra")) { table(.class("table table-zebra")) {
tbody(.class("text-lg")) { tbody(.class("text-lg")) {
tr { tr {
td { Label("Static Pressure") } td { span { "Static Pressure" } }
td { Number(equipmentInfo.staticPressure) } td {
div(.class("flex justify-end")) {
Number(equipmentInfo.staticPressure)
}
}
} }
tr { tr {
td { Label("Heating CFM") } td { span { "Heating CFM" } }
td { Number(equipmentInfo.heatingCFM) } td {
div(.class("flex justify-end")) {
Number(equipmentInfo.heatingCFM)
}
}
} }
tr { tr {
td { Label("Cooling CFM") } td { span { "Cooling CFM" } }
td { Number(equipmentInfo.coolingCFM) } td {
div(.class("flex justify-end")) {
Number(equipmentInfo.coolingCFM)
}
}
} }
} }
} }

View File

@@ -44,7 +44,7 @@ struct FrictionRateView: HTML, Sendable {
div(.class("space-y-6")) { div(.class("space-y-6")) {
div(.class("grid grid-cols-2 px-4")) { div(.class("grid grid-cols-2 px-4")) {
h1(.class("text-4xl font-bold items-end my-auto")) { "Friction Rate" } PageTitle { "Friction Rate" }
div(.class("space-y-4 justify-end")) { div(.class("space-y-4 justify-end")) {

View File

@@ -9,7 +9,7 @@ struct ProjectDetail: HTML, Sendable {
var body: some HTML { var body: some HTML {
div { div {
Row { Row {
h1(.class("text-2xl font-bold")) { "Project" } h1(.class("text-3xl font-bold")) { "Project" }
EditButton() EditButton()
.attributes( .attributes(
.class("btn-ghost"), .class("btn-ghost"),
@@ -21,24 +21,44 @@ struct ProjectDetail: HTML, Sendable {
table(.class("table table-zebra text-lg")) { table(.class("table table-zebra text-lg")) {
tbody { tbody {
tr { tr {
td { Label("Name") } td { "Name" }
td { project.name } td {
div(.class("flex justify-end")) {
project.name
}
}
} }
tr { tr {
td { Label("Street Address") } td { "Street Address" }
td { project.streetAddress } td {
div(.class("flex justify-end")) {
project.streetAddress
}
}
} }
tr { tr {
td { Label("City") } td { "City" }
td { project.city } td {
div(.class("flex justify-end")) {
project.city
}
}
} }
tr { tr {
td { Label("State") } td { "State" }
td { project.state } td {
div(.class("flex justify-end")) {
project.state
}
}
} }
tr { tr {
td { Label("Zip") } td { "Zip" }
td { project.zipCode } td {
div(.class("flex justify-end")) {
project.zipCode
}
}
} }
} }
} }

View File

@@ -20,7 +20,7 @@ struct ProjectsTable: HTML, Sendable {
Navbar(sidebarToggle: false) Navbar(sidebarToggle: false)
div(.class("m-6")) { div(.class("m-6")) {
Row { Row {
h1(.class("text-2xl font-bold")) { "Projects" } PageTitle { "Projects" }
Tooltip("Add project") { Tooltip("Add project") {
PlusButton() PlusButton()
.attributes( .attributes(
@@ -31,7 +31,7 @@ struct ProjectsTable: HTML, Sendable {
} }
.attributes(.class("pb-6")) .attributes(.class("pb-6"))
div(.class("overflow-x-auto rounded-box border")) { div(.class("overflow-x-auto")) {
table(.class("table table-zebra")) { table(.class("table table-zebra")) {
thead { thead {
tr { tr {

View File

@@ -39,8 +39,7 @@ struct RoomForm: HTML, Sendable {
ModalForm(id: Self.id(room), dismiss: dismiss) { ModalForm(id: Self.id(room), dismiss: dismiss) {
h1(.class("text-3xl font-bold pb-6")) { "Room" } h1(.class("text-3xl font-bold pb-6")) { "Room" }
form( form(
.class("modal-backdrop"), .class("grid grid-cols-1 gap-4"),
.init(name: "method", value: "dialog"),
room == nil room == nil
? .hx.post(route) ? .hx.post(route)
: .hx.patch(route), : .hx.patch(route),
@@ -54,34 +53,54 @@ struct RoomForm: HTML, Sendable {
input(.class("hidden"), .name("id"), .value("\(id)")) input(.class("hidden"), .name("id"), .value("\(id)"))
} }
div { LabeledInput(
label(.for("name")) { "Name:" } "Name",
Input(id: "name", placeholder: "Room Name") .name("name"),
.attributes(.type(.text), .required, .autofocus, .value(room?.name)) .type(.text),
} .placeholder("Name"),
div { .required,
label(.for("heatingLoad")) { "Heating Load:" } .autofocus,
Input(id: "heatingLoad", placeholder: "Heating Load") .value(room?.name)
.attributes(.type(.number), .required, .min("0"), .value(room?.heatingLoad)) )
}
div { LabeledInput(
label(.for("coolingTotal")) { "Cooling Total:" } "Heating Load",
Input(id: "coolingTotal", placeholder: "Cooling Total") .name("heatingLoad"),
.attributes(.type(.number), .required, .min("0"), .value(room?.coolingTotal)) .type(.number),
} .placeholder("1234"),
div { .required,
label(.for("coolingSensible")) { "Cooling Sensible:" } .min("0"),
Input(id: "coolingSensible", placeholder: "Cooling Sensible (Optional)") .value(room?.heatingLoad)
.attributes(.type(.number), .min("0"), .value(room?.coolingSensible)) )
}
div(.class("pb-6")) { LabeledInput(
label(.for("registerCount")) { "Registers:" } "Cooling Total",
Input(id: "registerCount", placeholder: "Register Count") .name("coolingTotal"),
.attributes( .type(.number),
.type(.number), .required, .min("0"), .placeholder("1234"),
.value("\(room != nil ? room!.registerCount : 1)"), .required,
) .min("0"),
} .value(room?.coolingTotal)
)
LabeledInput(
"Cooling Sensible",
.name("coolingSensible"),
.type(.number),
.placeholder("1234 (Optional)"),
.min("0"),
.value(room?.coolingSensible)
)
LabeledInput(
"Registers",
.name("registerCount"),
.type(.number),
.min("1"),
.required,
.value(room?.registerCount ?? 1)
)
SubmitButton() SubmitButton()
.attributes(.class("btn-block")) .attributes(.class("btn-block"))
} }

View File

@@ -14,61 +14,84 @@ struct RoomsView: HTML, Sendable {
var body: some HTML { var body: some HTML {
div(.class("flex w-full flex-col")) { div(.class("flex w-full flex-col")) {
Row { Row {
h1( PageTitle { "Room Loads" }
.class("flex flex-row text-2xl font-bold pb-6 h-full items-center")
) { "Room Loads" }
div(.class("flex justify-end")) { div(.class("flex justify-end items-end -my-2")) {
Tooltip("Project wide sensible heat ratio", position: .left) { Tooltip("Project wide sensible heat ratio", position: .left) {
button( button(
.class( .class(
""" """
grid grid-cols-1 gap-2 p-4 justify-end justify-end items-end p-4
hover:bg-neutral hover:text-white hover:rounded-lg hover:bg-neutral hover:text-white hover:rounded-lg
""" """
), ),
.showModal(id: SHRForm.id) .showModal(id: SHRForm.id)
) { ) {
LabeledContent("Sensible Heat Ratio") { LabeledContent {
div(.class("flex justify-end items-end space-x-4")) {
// SVG(.squarePen)
span(.class("font-bold")) {
"Sensible Heat Ratio"
}
}
} content: {
if let sensibleHeatRatio { if let sensibleHeatRatio {
Badge(number: sensibleHeatRatio) Badge(number: sensibleHeatRatio)
} }
} }
div(.class("flex justify-end")) {
SVG(.squarePen)
}
} }
} }
} }
} }
div(.class("flex flex-wrap justify-between mt-6")) {
div(.class("flex items-end space-x-4")) {
span(.class("font-bold")) { "Heating Total" }
Badge(number: rooms.heatingTotal, digits: 0)
.attributes(.class("badge-error"))
}
div(.class("flex items-end space-x-4")) {
span(.class("font-bold")) { "Cooling Total" }
Badge(number: rooms.coolingTotal, digits: 0)
.attributes(.class("badge-success"))
}
div(.class("flex justify-end items-end space-x-4 me-4")) {
span(.class("font-bold")) { "Cooling Sensible" }
Badge(number: rooms.coolingSensible(shr: sensibleHeatRatio), digits: 0)
.attributes(.class("badge-info"))
}
}
// .attributes(.class("mt-6 me-4"))
div(.class("divider")) {} div(.class("divider")) {}
SHRForm(projectID: projectID, sensibleHeatRatio: sensibleHeatRatio) SHRForm(projectID: projectID, sensibleHeatRatio: sensibleHeatRatio)
div(.class("overflow-x-auto")) { div(.class("overflow-x-auto")) {
table(.class("table table-zebra"), .id("roomsTable")) { table(.class("table table-zebra text-lg"), .id("roomsTable")) {
thead { thead {
tr { tr(.class("text-lg font-bold")) {
th { Label("Name") } th { "Name" }
th { th {
div(.class("flex justify-center")) { div(.class("flex justify-center")) {
Label("Heating Load") "Heating Load"
} }
} }
th { th {
div(.class("flex justify-center")) { div(.class("flex justify-center")) {
Label("Cooling Total") "Cooling Total"
} }
} }
th { th {
div(.class("flex justify-center")) { div(.class("flex justify-center")) {
Label("Cooling Sensible") "Cooling Sensible"
} }
} }
th { th {
div(.class("flex justify-center")) { div(.class("flex justify-center")) {
Label("Register Count") "Register Count"
} }
} }
th { th {
@@ -89,30 +112,6 @@ struct RoomsView: HTML, Sendable {
for room in rooms { for room in rooms {
RoomRow(room: room, shr: sensibleHeatRatio) RoomRow(room: room, shr: sensibleHeatRatio)
} }
// TOTALS
tr(.class("font-bold text-xl")) {
td { Label("Total") }
td {
div(.class("flex justify-center")) {
Badge(number: rooms.heatingTotal)
.attributes(.class("badge-error badge-xl"))
}
}
td {
div(.class("flex justify-center")) {
Badge(number: rooms.coolingTotal, digits: 0)
.attributes(.class("badge-success badge-xl"))
}
}
td {
div(.class("flex justify-center")) {
Badge(number: rooms.coolingSensible(shr: sensibleHeatRatio), digits: 0)
.attributes(.class("badge-info badge-xl"))
}
}
td {}
td {}
}
} }
} }
} }
@@ -143,19 +142,19 @@ struct RoomsView: HTML, Sendable {
td { td {
div(.class("flex justify-center")) { div(.class("flex justify-center")) {
Number(room.heatingLoad, digits: 0) Number(room.heatingLoad, digits: 0)
.attributes(.class("text-error")) // .attributes(.class("text-error"))
} }
} }
td { td {
div(.class("flex justify-center")) { div(.class("flex justify-center")) {
Number(room.coolingTotal, digits: 0) Number(room.coolingTotal, digits: 0)
.attributes(.class("text-success")) // .attributes(.class("text-success"))
} }
} }
td { td {
div(.class("flex justify-center")) { div(.class("flex justify-center")) {
Number(coolingSensible, digits: 0) Number(coolingSensible, digits: 0)
.attributes(.class("text-info")) // .attributes(.class("text-info"))
} }
} }
td { td {
@@ -204,22 +203,28 @@ struct RoomsView: HTML, Sendable {
var body: some HTML { var body: some HTML {
ModalForm(id: Self.id, dismiss: true) { ModalForm(id: Self.id, dismiss: true) {
h1(.class("text-xl font-bold mb-6")) {
"Sensible Heat Ratio"
}
form( form(
.class("space-y-6"), .class("grid grid-cols-1 gap-4"),
.hx.patch("/projects/\(projectID)/rooms/update-shr"), .hx.patch("/projects/\(projectID)/rooms/update-shr"),
.hx.target("body"), .hx.target("body"),
.hx.swap(.outerHTML) .hx.swap(.outerHTML)
) { ) {
input(.class("hidden"), .name("projectID"), .value("\(projectID)")) input(.class("hidden"), .name("projectID"), .value("\(projectID)"))
div { LabeledInput(
label(.for("sensibleHeatRatio")) { "Sensible Heat Ratio" } "SHR",
Input(id: "sensibleHeatRatio", placeholder: "Sensible Heat Ratio") .type(.number),
.attributes(.min("0"), .max("1"), .step("0.01"), .value(sensibleHeatRatio)) .placeholder("0.83"),
} .min("0"),
div { .max("1"),
SubmitButton() .step("0.01"),
.attributes(.class("btn-block")) .value(sensibleHeatRatio),
} .autofocus
)
SubmitButton()
.attributes(.class("btn-block my-6"))
} }
} }
} }