diff --git a/.goreleaser.yaml b/.goreleaser.yaml
index 9a396098..ff1880d5 100644
--- a/.goreleaser.yaml
+++ b/.goreleaser.yaml
@@ -1,20 +1,22 @@
# Grendel goreleaser configs
# See here: https://goreleaser.com
-
-version: 1
-
before:
hooks:
- go mod tidy
builds:
- env:
- - CGO_ENABLED=0
+ - CGO_ENABLED=1
goarch:
- amd64
goos:
- linux
ldflags:
- -s -w -X github.com/ubccr/grendel/api.Version={{.Version}}
+ - -extldflags=-static
+ tags:
+ - sqlite_omit_load_extension
+ - osusergo
+ - netgo
archives:
- format: tar.gz
wrap_in_directory: true
@@ -64,7 +66,7 @@ nfpms:
- src: ./scripts/nfpm/grendel.service
dst: /usr/lib/systemd/system/grendel.service
checksum:
- name_template: 'checksums.txt'
+ name_template: "checksums.txt"
snapshot:
name_template: "{{ incpatch .Version }}-SNAPSHOT-{{.ShortCommit}}"
changelog:
@@ -73,12 +75,12 @@ changelog:
- title: Features
regexp: "^.*feat[(\\w)]*:+.*$"
order: 0
- - title: 'Bug fixes'
+ - title: "Bug fixes"
regexp: "^.*fix[(\\w)]*:+.*$"
order: 1
- title: Other
order: 999
filters:
exclude:
- - '^docs:'
- - 'typo'
+ - "^docs:"
+ - "typo"
diff --git a/CHANGELOG.md b/CHANGELOG.md
index 5ccefc8b..a42d88af 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -1,5 +1,17 @@
# Grendel Changelog
+## [0.0.12] - 2024-08-05
+
+- Update echo, fiber deps
+- Fix provision - tftp segfault when starting with bind port permission error
+- Feat: provision - added "proxmox" tag DHCP 250 response code for Proxmox Automated Installs
+- Fix: frontend - segfault when non nodeset hostname is added
+- Fix: frontend - rack selection checkboxes not preserving state after submitting an action
+- Fix: frontend - sqlite session storage
+- Feat: fronend - added Nodes page
+- Feat: frontend - added Bond support in Host page
+- Feat: frontend - added status page
+
## [0.0.11] - 2024-02-27
- Add prometheus service discovery.
@@ -130,4 +142,5 @@
[0.0.9]: https://github.com/ubccr/grendel/releases/tag/v0.0.9
[0.0.10]: https://github.com/ubccr/grendel/releases/tag/v0.0.10
[0.0.11]: https://github.com/ubccr/grendel/releases/tag/v0.0.11
-[Unreleased]: https://github.com/ubccr/grendel/compare/v0.0.11...HEAD
+[0.0.12]: https://github.com/ubccr/grendel/releases/tag/v0.0.12
+[Unreleased]: https://github.com/ubccr/grendel/compare/v0.0.12...HEAD
diff --git a/dhcp/static.go b/dhcp/static.go
index 954913f4..6f93501e 100644
--- a/dhcp/static.go
+++ b/dhcp/static.go
@@ -69,6 +69,23 @@ func (s *Server) setZTD(host *model.Host, nic *model.NetInterface, serverIP net.
resp.UpdateOption(dhcpv4.Option{Code: dhcpv4.GenericOptionCode(240), Value: dhcpv4.String(provisionURL)})
}
+ if host.HasTags("proxmox") {
+ // Proxmox Automated Installation
+ // See: https://pve.proxmox.com/wiki/Automated_Installation
+
+ log.WithFields(logrus.Fields{
+ "ip": nic.AddrString(),
+ "name": host.Name,
+ }).Info("Host tagged with proxmox. Setting automated install answer file url")
+
+ token, _ := model.NewBootToken(host.ID.String(), nic.MAC.String())
+ endpoints := model.NewEndpoints(serverIP.String(), token)
+
+ proxmoxURL := endpoints.ProxmoxURL()
+ log.Debugf("Proxmox Answer file url: %s", proxmoxURL)
+ resp.UpdateOption(dhcpv4.Option{Code: dhcpv4.GenericOptionCode(250), Value: dhcpv4.String(proxmoxURL)})
+ }
+
if req.ClassIdentifier() == "NVIDIA" || req.ClassIdentifier() == "Mellanox" {
// MLNXOS ZTP
// See: https://docs.nvidia.com/networking/display/MLNXOSv3103100/Getting+Started#heading-Zero-touchProvisioning
diff --git a/frontend/api.go b/frontend/api.go
index c87b4ce0..a323c5af 100644
--- a/frontend/api.go
+++ b/frontend/api.go
@@ -144,6 +144,7 @@ type FormData struct {
BootImage string `form:"BootImage"`
Tags string `form:"Tags"`
Interfaces string `form:"Interfaces"`
+ Bonds string `form:"Bonds"`
}
type InterfacesString struct {
FQDN string `json:"Fqdn"`
@@ -154,6 +155,10 @@ type InterfacesString struct {
VLAN string `json:"Vlan"`
MTU string `json:"Mtu"`
}
+type BondsString struct {
+ InterfacesString
+ Peers string `json:"Peers"`
+}
func (h *Handler) EditHost(f *fiber.Ctx) error {
formHost := new(FormData)
@@ -172,8 +177,11 @@ func (h *Handler) EditHost(f *fiber.Ctx) error {
var ifaces []InterfacesString
json.Unmarshal([]byte(formHost.Interfaces), &ifaces)
+ var bondsStr []BondsString
+ json.Unmarshal([]byte(formHost.Bonds), &bondsStr)
var interfaces []*model.NetInterface
+ var bonds []*model.Bond
for i, iface := range ifaces {
mac, _ := net.ParseMAC(iface.MAC)
@@ -197,6 +205,31 @@ func (h *Handler) EditHost(f *fiber.Ctx) error {
MTU: uint16(mtu),
})
}
+ for i, bond := range bondsStr {
+ mac, _ := net.ParseMAC(bond.MAC)
+ ip, _ := netip.ParsePrefix(bond.IP)
+ bmc, err := strconv.ParseBool(bond.BMC)
+ if err != nil {
+ return ToastError(f, err, fmt.Sprintf("Failed to parse BMC boolean on interface %d", i))
+ }
+ mtu, err := strconv.Atoi(bond.MTU)
+ if err != nil {
+ return ToastError(f, err, fmt.Sprintf("Failed to parse MTU on interface %d", i))
+ }
+
+ bonds = append(bonds, &model.Bond{
+ NetInterface: model.NetInterface{
+ Name: bond.Name,
+ FQDN: bond.FQDN,
+ MAC: mac,
+ IP: ip,
+ BMC: bmc,
+ VLAN: bond.VLAN,
+ MTU: uint16(mtu),
+ },
+ Peers: strings.Split(bond.Peers, ","),
+ })
+ }
newHost := model.Host{
ID: id,
@@ -206,6 +239,7 @@ func (h *Handler) EditHost(f *fiber.Ctx) error {
BootImage: formHost.BootImage,
Tags: strings.Split(formHost.Tags, ","),
Interfaces: interfaces,
+ Bonds: bonds,
}
err = h.DB.StoreHost(&newHost)
diff --git a/frontend/client.go b/frontend/client.go
index 9e382ecb..52bfff3f 100644
--- a/frontend/client.go
+++ b/frontend/client.go
@@ -2,8 +2,10 @@ package frontend
import (
"fmt"
+ "os"
"github.com/gofiber/fiber/v2"
+ "github.com/ubccr/grendel/api"
)
func (h *Handler) Index(f *fiber.Ctx) error {
@@ -38,6 +40,12 @@ func (h *Handler) Rack(f *fiber.Ctx) error {
})
}
+func (h *Handler) nodes(f *fiber.Ctx) error {
+ return f.Render("nodes", fiber.Map{
+ "Title": "Grendel - Nodes",
+ })
+}
+
func (h *Handler) Host(f *fiber.Ctx) error {
host := f.Params("host")
return f.Render("host", fiber.Map{
@@ -51,3 +59,26 @@ func (h *Handler) Users(f *fiber.Ctx) error {
"Title": "Grendel - Users",
})
}
+
+func (h *Handler) status(f *fiber.Ctx) error {
+ hosts, err := h.DB.Hosts()
+ if err != nil {
+ return err
+ }
+ b, err := h.DB.BootImages()
+ if err != nil {
+ return err
+ }
+ hostName, err := os.Hostname()
+ if err != nil {
+ return err
+ }
+
+ return f.Render("status", fiber.Map{
+ "Title": "Grendel - Status",
+ "HostName": hostName,
+ "Version": api.Version,
+ "Hosts": len(hosts),
+ "BootImages": len(b),
+ })
+}
diff --git a/frontend/fragments.go b/frontend/fragments.go
index 861db38a..4a607acb 100644
--- a/frontend/fragments.go
+++ b/frontend/fragments.go
@@ -3,6 +3,9 @@ package frontend
import (
"encoding/json"
"fmt"
+ "math"
+ "regexp"
+ "sort"
"strconv"
"strings"
@@ -32,6 +35,16 @@ func (h *Handler) hostForm(f *fiber.Ctx) error {
VLAN string
MTU string
}
+ type BondStrings struct {
+ FQDN string
+ MAC string
+ IP string
+ Name string
+ BMC string
+ VLAN string
+ MTU string
+ Peers []string
+ }
Interfaces := make([]IfaceStrings, len(host[0].Interfaces))
for i, iface := range host[0].Interfaces {
@@ -43,12 +56,25 @@ func (h *Handler) hostForm(f *fiber.Ctx) error {
Interfaces[i].VLAN = iface.VLAN
Interfaces[i].MTU = strconv.FormatUint(uint64(iface.MTU), 10)
}
+ Bonds := make([]BondStrings, len(host[0].Bonds))
+
+ for i, bond := range host[0].Bonds {
+ Bonds[i].FQDN = bond.FQDN
+ Bonds[i].MAC = bond.MAC.String()
+ Bonds[i].IP = bond.IP.String()
+ Bonds[i].Name = bond.Name
+ Bonds[i].BMC = strconv.FormatBool(bond.BMC)
+ Bonds[i].VLAN = bond.VLAN
+ Bonds[i].MTU = strconv.FormatUint(uint64(bond.MTU), 10)
+ Bonds[i].Peers = bond.Peers
+ }
return f.Render("fragments/host/form", fiber.Map{
"Host": host[0],
"BootImages": h.getBootImages(),
"Firmwares": h.getFirmware(),
"Interfaces": Interfaces,
+ "Bonds": Bonds,
}, "")
}
@@ -106,7 +132,7 @@ func (h *Handler) rackTable(f *fiber.Ctx) error {
}, "")
}
-func (h *Handler) rackActions(f *fiber.Ctx) error {
+func (h *Handler) actions(f *fiber.Ctx) error {
hosts := f.FormValue("hosts")
ns, err := nodeset.NewNodeSet(hosts)
if err != nil {
@@ -114,7 +140,7 @@ func (h *Handler) rackActions(f *fiber.Ctx) error {
}
nodeset := ns.String()
- return f.Render("fragments/rack/actions", fiber.Map{
+ return f.Render("fragments/actions", fiber.Map{
"Hosts": nodeset,
"BootImages": h.getBootImages(),
}, "")
@@ -290,6 +316,136 @@ func (h *Handler) rackAddTable(f *fiber.Ctx) error {
}, "")
}
+func (h *Handler) nodesTable(f *fiber.Ctx) error {
+ hosts, err := h.DB.Hosts()
+ if err != nil {
+ return ToastError(f, err, "Error getting hosts from DB")
+ }
+ qp := f.Queries()
+
+ matches := []string{}
+
+ // Filter Regex:
+ nameRegex, err := regexp.Compile(qp["Name"])
+ if err != nil {
+ return ToastError(f, err, "Invalid Name Regex")
+ }
+ provisionRegex, err := regexp.Compile(qp["Provision"])
+ if err != nil {
+ return ToastError(f, err, "Invalid Boot Image Regex")
+ }
+ firmwareRegex, err := regexp.Compile(qp["Firmware"])
+ if err != nil {
+ return ToastError(f, err, "Invalid Boot Image Regex")
+ }
+ bootImageRegex, err := regexp.Compile(qp["BootImage"])
+ if err != nil {
+ return ToastError(f, err, "Invalid Boot Image Regex")
+ }
+ tagsRegex, err := regexp.Compile(qp["Tags"])
+ if err != nil {
+ return ToastError(f, err, "Invalid Tag Regex")
+ }
+
+ for _, h := range hosts {
+ nameMatch := false
+ provisionMatch := false
+ firmwareMatch := false
+ bootImageMatch := false
+ tagsMatch := false
+
+ if nameRegex.MatchString(h.Name) {
+ nameMatch = true
+ }
+ if provisionRegex.MatchString(strconv.FormatBool(h.Provision)) {
+ provisionMatch = true
+ }
+ if firmwareRegex.MatchString(h.Firmware.String()) {
+ firmwareMatch = true
+ }
+ if bootImageRegex.MatchString(h.BootImage) {
+ bootImageMatch = true
+ }
+ for _, tag := range h.Tags {
+ if tagsRegex.MatchString(tag) {
+ tagsMatch = true
+ break
+ }
+ }
+
+ if nameMatch && provisionMatch && firmwareMatch && bootImageMatch && tagsMatch {
+ matches = append(matches, h.Name)
+ }
+
+ }
+
+ ns, err := nodeset.NewNodeSet(strings.Join(matches, ","))
+ if err != nil {
+ return ToastError(f, err, "Error filtering hosts")
+ }
+ filtered, err := h.DB.FindHosts(ns)
+ if err != nil {
+ return ToastError(f, err, "Error filtering hosts")
+ }
+ sort.Slice(filtered, func(i, j int) bool {
+ o := strings.Compare(filtered[i].Name, filtered[j].Name)
+
+ return o == -1
+ })
+ allHosts, err := filtered.ToNodeSet()
+ if err != nil {
+ return ToastError(f, err, "Error filtering hosts")
+ }
+
+ // Pagination:
+ pageSize, err := strconv.Atoi(qp["PageSize"])
+ if err != nil {
+ return ToastError(f, err, "Error calculating pagination")
+ }
+ filteredLen := len(filtered)
+ numPages := filteredLen / pageSize
+ if math.Remainder(float64(filteredLen), float64(pageSize)) != 0 {
+ numPages++
+ }
+
+ currentPage := 1
+ if qp["CurrentPage"] != "" {
+ currentPage, err = strconv.Atoi(qp["CurrentPage"])
+ if err != nil {
+ return ToastError(f, err, "Error calculating pagination")
+ }
+ }
+
+ start := (currentPage - 1) * pageSize
+
+ if start > filteredLen {
+ start = filteredLen
+ }
+
+ end := start + pageSize
+ if end > filteredLen {
+ end = filteredLen
+ }
+
+ filteredList := filtered[start:end]
+
+ checkAll := false
+ if qp["CheckAll"] == "on" {
+ checkAll = true
+ }
+
+ return f.Render("fragments/nodes/table", fiber.Map{
+ "Hosts": filteredList,
+ "AllHosts": allHosts,
+ "CheckAll": checkAll,
+ "Pagination": fiber.Map{
+ "CurrentPage": currentPage,
+ "PageSize": pageSize,
+ "NumPages": numPages,
+ },
+ }, "")
+}
+
func (h *Handler) usersTable(f *fiber.Ctx) error {
users, err := h.DB.GetUsers()
if err != nil {
@@ -305,7 +461,11 @@ func (h *Handler) floorplanTable(f *fiber.Ctx) error {
hosts, _ := h.DB.Hosts()
racks := map[string]int{}
for _, host := range hosts {
- rack := strings.Split(host.Name, "-")[1]
+ name := strings.Split(host.Name, "-")
+ if len(name) < 2 {
+ continue
+ }
+ rack := name[1]
racks[rack] += 1
}
@@ -351,6 +511,14 @@ func (h *Handler) interfaces(f *fiber.Ctx) error {
}, "")
}
+func (h *Handler) bonds(f *fiber.Ctx) error {
+ id := f.Query("ID", "0")
+
+ return f.Render("fragments/bonds", fiber.Map{
+ "ID": id,
+ }, "")
+}
+
func (h *Handler) events(f *fiber.Ctx) error {
return f.Render("fragments/events/table", fiber.Map{
"Events": h.Events,
diff --git a/frontend/handler.go b/frontend/handler.go
index 466f5ad2..fd2d8f17 100644
--- a/frontend/handler.go
+++ b/frontend/handler.go
@@ -110,6 +110,7 @@ func (h *Handler) SetupRoutes(app *fiber.App) {
api.Post("/host/import", auth, h.importHost)
fragment.Get("/interfaces", auth, h.interfaces)
+ fragment.Get("/bonds", auth, h.bonds)
app.Get("/floorplan", auth, h.Floorplan)
fragment.Get("/floorplan/table", auth, h.floorplanTable)
@@ -117,10 +118,13 @@ func (h *Handler) SetupRoutes(app *fiber.App) {
app.Get("/rack/:rack", auth, h.Rack)
fragment.Get("/rack/:rack/table", auth, h.rackTable)
- fragment.Get("/rack/:rack/actions", auth, h.rackActions)
fragment.Get("/rack/:rack/add/modal", auth, h.rackAddModal)
fragment.Post("/rack/:rack/add/table", auth, h.rackAddTable)
+ app.Get("/nodes", auth, h.nodes)
+ fragment.Get("/nodes", auth, h.nodesTable)
+
+ fragment.Put("/actions", auth, h.actions)
api.Post("/bulkHostAdd", auth, h.bulkHostAdd)
api.Patch("/hosts/provision", auth, h.provisionHosts)
@@ -137,6 +141,8 @@ func (h *Handler) SetupRoutes(app *fiber.App) {
fragment.Get("/events", auth, h.events)
+ app.Get("/status", h.status)
+
api.Post("/bmc/powerCycle", auth, h.bmcPowerCycle)
api.Post("/bmc/powerCycleBmc", auth, h.bmcPowerCycleBmc)
api.Post("/bmc/clearSel", auth, h.bmcClearSel)
diff --git a/frontend/public/htmx.min.js b/frontend/public/htmx.min.js
index c86e83c5..6eeb64f5 100644
--- a/frontend/public/htmx.min.js
+++ b/frontend/public/htmx.min.js
@@ -1,3150 +1 @@
-(function (e, t) {
- if (typeof define === "function" && define.amd) {
- define([], t);
- } else if (typeof module === "object" && module.exports) {
- module.exports = t();
- } else {
- e.htmx = e.htmx || t();
- }
-})(typeof self !== "undefined" ? self : this, function () {
- return (function () {
- "use strict";
- var Y = {
- onLoad: t,
- process: Pt,
- on: Z,
- off: K,
- trigger: fe,
- ajax: wr,
- find: E,
- findAll: f,
- closest: v,
- values: function (e, t) {
- var r = nr(e, t || "post");
- return r.values;
- },
- remove: U,
- addClass: B,
- removeClass: n,
- toggleClass: V,
- takeClass: j,
- defineExtension: qr,
- removeExtension: Hr,
- logAll: X,
- logNone: F,
- logger: null,
- config: {
- historyEnabled: true,
- historyCacheSize: 10,
- refreshOnHistoryMiss: false,
- defaultSwapStyle: "innerHTML",
- defaultSwapDelay: 0,
- defaultSettleDelay: 20,
- includeIndicatorStyles: true,
- indicatorClass: "htmx-indicator",
- requestClass: "htmx-request",
- addedClass: "htmx-added",
- settlingClass: "htmx-settling",
- swappingClass: "htmx-swapping",
- allowEval: true,
- allowScriptTags: true,
- inlineScriptNonce: "",
- attributesToSettle: ["class", "style", "width", "height"],
- withCredentials: false,
- timeout: 0,
- wsReconnectDelay: "full-jitter",
- wsBinaryType: "blob",
- disableSelector: "[hx-disable], [data-hx-disable]",
- useTemplateFragments: false,
- scrollBehavior: "smooth",
- defaultFocusScroll: false,
- getCacheBusterParam: false,
- globalViewTransitions: false,
- methodsThatUseUrlParams: ["get"],
- selfRequestsOnly: false,
- },
- parseInterval: d,
- _: e,
- createEventSource: function (e) {
- return new EventSource(e, { withCredentials: true });
- },
- createWebSocket: function (e) {
- var t = new WebSocket(e, []);
- t.binaryType = Y.config.wsBinaryType;
- return t;
- },
- version: "1.9.6",
- };
- var r = {
- addTriggerHandler: St,
- bodyContains: oe,
- canAccessLocalStorage: M,
- findThisElement: de,
- filterValues: lr,
- hasAttribute: o,
- getAttributeValue: ee,
- getClosestAttributeValue: re,
- getClosestMatch: c,
- getExpressionVars: xr,
- getHeaders: sr,
- getInputValues: nr,
- getInternalData: ie,
- getSwapSpecification: fr,
- getTriggerSpecs: Ze,
- getTarget: ge,
- makeFragment: l,
- mergeObjects: se,
- makeSettleInfo: T,
- oobSwap: ye,
- querySelectorExt: le,
- selectAndSwap: Fe,
- settleImmediately: Wt,
- shouldCancel: tt,
- triggerEvent: fe,
- triggerErrorEvent: ue,
- withExtensions: C,
- };
- var b = ["get", "post", "put", "delete", "patch"];
- var w = b
- .map(function (e) {
- return "[hx-" + e + "], [data-hx-" + e + "]";
- })
- .join(", ");
- function d(e) {
- if (e == undefined) {
- return undefined;
- }
- if (e.slice(-2) == "ms") {
- return parseFloat(e.slice(0, -2)) || undefined;
- }
- if (e.slice(-1) == "s") {
- return parseFloat(e.slice(0, -1)) * 1e3 || undefined;
- }
- if (e.slice(-1) == "m") {
- return parseFloat(e.slice(0, -1)) * 1e3 * 60 || undefined;
- }
- return parseFloat(e) || undefined;
- }
- function Q(e, t) {
- return e.getAttribute && e.getAttribute(t);
- }
- function o(e, t) {
- return e.hasAttribute && (e.hasAttribute(t) || e.hasAttribute("data-" + t));
- }
- function ee(e, t) {
- return Q(e, t) || Q(e, "data-" + t);
- }
- function u(e) {
- return e.parentElement;
- }
- function te() {
- return document;
- }
- function c(e, t) {
- while (e && !t(e)) {
- e = u(e);
- }
- return e ? e : null;
- }
- function O(e, t, r) {
- var n = ee(t, r);
- var i = ee(t, "hx-disinherit");
- if (e !== t && i && (i === "*" || i.split(" ").indexOf(r) >= 0)) {
- return "unset";
- } else {
- return n;
- }
- }
- function re(t, r) {
- var n = null;
- c(t, function (e) {
- return (n = O(t, e, r));
- });
- if (n !== "unset") {
- return n;
- }
- }
- function h(e, t) {
- var r =
- e.matches ||
- e.matchesSelector ||
- e.msMatchesSelector ||
- e.mozMatchesSelector ||
- e.webkitMatchesSelector ||
- e.oMatchesSelector;
- return r && r.call(e, t);
- }
- function q(e) {
- var t = /<([a-z][^\/\0>\x20\t\r\n\f]*)/i;
- var r = t.exec(e);
- if (r) {
- return r[1].toLowerCase();
- } else {
- return "";
- }
- }
- function i(e, t) {
- var r = new DOMParser();
- var n = r.parseFromString(e, "text/html");
- var i = n.body;
- while (t > 0) {
- t--;
- i = i.firstChild;
- }
- if (i == null) {
- i = te().createDocumentFragment();
- }
- return i;
- }
- function H(e) {
- return e.match(/
" + e + " ", 0);
- return r.querySelector("template").content;
- } else {
- var n = q(e);
- switch (n) {
- case "thead":
- case "tbody":
- case "tfoot":
- case "colgroup":
- case "caption":
- return i("", 1);
- case "col":
- return i("", 2);
- case "tr":
- return i("", 2);
- case "td":
- case "th":
- return i("", 3);
- case "script":
- case "style":
- return i("" + e + "
", 1);
- default:
- return i(e, 0);
- }
- }
- }
- function ne(e) {
- if (e) {
- e();
- }
- }
- function L(e, t) {
- return Object.prototype.toString.call(e) === "[object " + t + "]";
- }
- function A(e) {
- return L(e, "Function");
- }
- function N(e) {
- return L(e, "Object");
- }
- function ie(e) {
- var t = "htmx-internal-data";
- var r = e[t];
- if (!r) {
- r = e[t] = {};
- }
- return r;
- }
- function I(e) {
- var t = [];
- if (e) {
- for (var r = 0; r < e.length; r++) {
- t.push(e[r]);
- }
- }
- return t;
- }
- function ae(e, t) {
- if (e) {
- for (var r = 0; r < e.length; r++) {
- t(e[r]);
- }
- }
- }
- function P(e) {
- var t = e.getBoundingClientRect();
- var r = t.top;
- var n = t.bottom;
- return r < window.innerHeight && n >= 0;
- }
- function oe(e) {
- if (e.getRootNode && e.getRootNode() instanceof window.ShadowRoot) {
- return te().body.contains(e.getRootNode().host);
- } else {
- return te().body.contains(e);
- }
- }
- function k(e) {
- return e.trim().split(/\s+/);
- }
- function se(e, t) {
- for (var r in t) {
- if (t.hasOwnProperty(r)) {
- e[r] = t[r];
- }
- }
- return e;
- }
- function S(e) {
- try {
- return JSON.parse(e);
- } catch (e) {
- y(e);
- return null;
- }
- }
- function M() {
- var e = "htmx:localStorageTest";
- try {
- localStorage.setItem(e, e);
- localStorage.removeItem(e);
- return true;
- } catch (e) {
- return false;
- }
- }
- function D(t) {
- try {
- var e = new URL(t);
- if (e) {
- t = e.pathname + e.search;
- }
- if (!t.match("^/$")) {
- t = t.replace(/\/+$/, "");
- }
- return t;
- } catch (e) {
- return t;
- }
- }
- function e(e) {
- return gr(te().body, function () {
- return eval(e);
- });
- }
- function t(t) {
- var e = Y.on("htmx:load", function (e) {
- t(e.detail.elt);
- });
- return e;
- }
- function X() {
- Y.logger = function (e, t, r) {
- if (console) {
- console.log(t, e, r);
- }
- };
- }
- function F() {
- Y.logger = null;
- }
- function E(e, t) {
- if (t) {
- return e.querySelector(t);
- } else {
- return E(te(), e);
- }
- }
- function f(e, t) {
- if (t) {
- return e.querySelectorAll(t);
- } else {
- return f(te(), e);
- }
- }
- function U(e, t) {
- e = s(e);
- if (t) {
- setTimeout(function () {
- U(e);
- e = null;
- }, t);
- } else {
- e.parentElement.removeChild(e);
- }
- }
- function B(e, t, r) {
- e = s(e);
- if (r) {
- setTimeout(function () {
- B(e, t);
- e = null;
- }, r);
- } else {
- e.classList && e.classList.add(t);
- }
- }
- function n(e, t, r) {
- e = s(e);
- if (r) {
- setTimeout(function () {
- n(e, t);
- e = null;
- }, r);
- } else {
- if (e.classList) {
- e.classList.remove(t);
- if (e.classList.length === 0) {
- e.removeAttribute("class");
- }
- }
- }
- }
- function V(e, t) {
- e = s(e);
- e.classList.toggle(t);
- }
- function j(e, t) {
- e = s(e);
- ae(e.parentElement.children, function (e) {
- n(e, t);
- });
- B(e, t);
- }
- function v(e, t) {
- e = s(e);
- if (e.closest) {
- return e.closest(t);
- } else {
- do {
- if (e == null || h(e, t)) {
- return e;
- }
- } while ((e = e && u(e)));
- return null;
- }
- }
- function g(e, t) {
- return e.substring(0, t.length) === t;
- }
- function _(e, t) {
- return e.substring(e.length - t.length) === t;
- }
- function z(e) {
- var t = e.trim();
- if (g(t, "<") && _(t, "/>")) {
- return t.substring(1, t.length - 2);
- } else {
- return t;
- }
- }
- function W(e, t) {
- if (t.indexOf("closest ") === 0) {
- return [v(e, z(t.substr(8)))];
- } else if (t.indexOf("find ") === 0) {
- return [E(e, z(t.substr(5)))];
- } else if (t.indexOf("next ") === 0) {
- return [$(e, z(t.substr(5)))];
- } else if (t.indexOf("previous ") === 0) {
- return [G(e, z(t.substr(9)))];
- } else if (t === "document") {
- return [document];
- } else if (t === "window") {
- return [window];
- } else if (t === "body") {
- return [document.body];
- } else {
- return te().querySelectorAll(z(t));
- }
- }
- var $ = function (e, t) {
- var r = te().querySelectorAll(t);
- for (var n = 0; n < r.length; n++) {
- var i = r[n];
- if (i.compareDocumentPosition(e) === Node.DOCUMENT_POSITION_PRECEDING) {
- return i;
- }
- }
- };
- var G = function (e, t) {
- var r = te().querySelectorAll(t);
- for (var n = r.length - 1; n >= 0; n--) {
- var i = r[n];
- if (i.compareDocumentPosition(e) === Node.DOCUMENT_POSITION_FOLLOWING) {
- return i;
- }
- }
- };
- function le(e, t) {
- if (t) {
- return W(e, t)[0];
- } else {
- return W(te().body, e)[0];
- }
- }
- function s(e) {
- if (L(e, "String")) {
- return E(e);
- } else {
- return e;
- }
- }
- function J(e, t, r) {
- if (A(t)) {
- return { target: te().body, event: e, listener: t };
- } else {
- return { target: s(e), event: t, listener: r };
- }
- }
- function Z(t, r, n) {
- Nr(function () {
- var e = J(t, r, n);
- e.target.addEventListener(e.event, e.listener);
- });
- var e = A(r);
- return e ? r : n;
- }
- function K(t, r, n) {
- Nr(function () {
- var e = J(t, r, n);
- e.target.removeEventListener(e.event, e.listener);
- });
- return A(r) ? r : n;
- }
- var he = te().createElement("output");
- function ve(e, t) {
- var r = re(e, t);
- if (r) {
- if (r === "this") {
- return [de(e, t)];
- } else {
- var n = W(e, r);
- if (n.length === 0) {
- y('The selector "' + r + '" on ' + t + " returned no matches!");
- return [he];
- } else {
- return n;
- }
- }
- }
- }
- function de(e, t) {
- return c(e, function (e) {
- return ee(e, t) != null;
- });
- }
- function ge(e) {
- var t = re(e, "hx-target");
- if (t) {
- if (t === "this") {
- return de(e, "hx-target");
- } else {
- return le(e, t);
- }
- } else {
- var r = ie(e);
- if (r.boosted) {
- return te().body;
- } else {
- return e;
- }
- }
- }
- function me(e) {
- var t = Y.config.attributesToSettle;
- for (var r = 0; r < t.length; r++) {
- if (e === t[r]) {
- return true;
- }
- }
- return false;
- }
- function pe(t, r) {
- ae(t.attributes, function (e) {
- if (!r.hasAttribute(e.name) && me(e.name)) {
- t.removeAttribute(e.name);
- }
- });
- ae(r.attributes, function (e) {
- if (me(e.name)) {
- t.setAttribute(e.name, e.value);
- }
- });
- }
- function xe(e, t) {
- var r = Lr(t);
- for (var n = 0; n < r.length; n++) {
- var i = r[n];
- try {
- if (i.isInlineSwap(e)) {
- return true;
- }
- } catch (e) {
- y(e);
- }
- }
- return e === "outerHTML";
- }
- function ye(e, i, a) {
- var t = "#" + Q(i, "id");
- var o = "outerHTML";
- if (e === "true") {
- } else if (e.indexOf(":") > 0) {
- o = e.substr(0, e.indexOf(":"));
- t = e.substr(e.indexOf(":") + 1, e.length);
- } else {
- o = e;
- }
- var r = te().querySelectorAll(t);
- if (r) {
- ae(r, function (e) {
- var t;
- var r = i.cloneNode(true);
- t = te().createDocumentFragment();
- t.appendChild(r);
- if (!xe(o, e)) {
- t = r;
- }
- var n = { shouldSwap: true, target: e, fragment: t };
- if (!fe(e, "htmx:oobBeforeSwap", n)) return;
- e = n.target;
- if (n["shouldSwap"]) {
- De(o, e, e, t, a);
- }
- ae(a.elts, function (e) {
- fe(e, "htmx:oobAfterSwap", n);
- });
- });
- i.parentNode.removeChild(i);
- } else {
- i.parentNode.removeChild(i);
- ue(te().body, "htmx:oobErrorNoTarget", { content: i });
- }
- return e;
- }
- function be(e, t, r) {
- var n = re(e, "hx-select-oob");
- if (n) {
- var i = n.split(",");
- for (let e = 0; e < i.length; e++) {
- var a = i[e].split(":", 2);
- var o = a[0].trim();
- if (o.indexOf("#") === 0) {
- o = o.substring(1);
- }
- var s = a[1] || "true";
- var l = t.querySelector("#" + o);
- if (l) {
- ye(s, l, r);
- }
- }
- }
- ae(f(t, "[hx-swap-oob], [data-hx-swap-oob]"), function (e) {
- var t = ee(e, "hx-swap-oob");
- if (t != null) {
- ye(t, e, r);
- }
- });
- }
- function we(e) {
- ae(f(e, "[hx-preserve], [data-hx-preserve]"), function (e) {
- var t = ee(e, "id");
- var r = te().getElementById(t);
- if (r != null) {
- e.parentNode.replaceChild(r, e);
- }
- });
- }
- function Se(o, e, s) {
- ae(e.querySelectorAll("[id]"), function (e) {
- var t = Q(e, "id");
- if (t && t.length > 0) {
- var r = t.replace("'", "\\'");
- var n = e.tagName.replace(":", "\\:");
- var i = o.querySelector(n + "[id='" + r + "']");
- if (i && i !== o) {
- var a = e.cloneNode();
- pe(e, i);
- s.tasks.push(function () {
- pe(e, a);
- });
- }
- }
- });
- }
- function Ee(e) {
- return function () {
- n(e, Y.config.addedClass);
- Pt(e);
- Ct(e);
- Ce(e);
- fe(e, "htmx:load");
- };
- }
- function Ce(e) {
- var t = "[autofocus]";
- var r = h(e, t) ? e : e.querySelector(t);
- if (r != null) {
- r.focus();
- }
- }
- function a(e, t, r, n) {
- Se(e, r, n);
- while (r.childNodes.length > 0) {
- var i = r.firstChild;
- B(i, Y.config.addedClass);
- e.insertBefore(i, t);
- if (i.nodeType !== Node.TEXT_NODE && i.nodeType !== Node.COMMENT_NODE) {
- n.tasks.push(Ee(i));
- }
- }
- }
- function Te(e, t) {
- var r = 0;
- while (r < e.length) {
- t = ((t << 5) - t + e.charCodeAt(r++)) | 0;
- }
- return t;
- }
- function Re(e) {
- var t = 0;
- if (e.attributes) {
- for (var r = 0; r < e.attributes.length; r++) {
- var n = e.attributes[r];
- if (n.value) {
- t = Te(n.name, t);
- t = Te(n.value, t);
- }
- }
- }
- return t;
- }
- function Oe(t) {
- var r = ie(t);
- if (r.onHandlers) {
- for (let e = 0; e < r.onHandlers.length; e++) {
- const n = r.onHandlers[e];
- t.removeEventListener(n.event, n.listener);
- }
- delete r.onHandlers;
- }
- }
- function qe(e) {
- var t = ie(e);
- if (t.timeout) {
- clearTimeout(t.timeout);
- }
- if (t.webSocket) {
- t.webSocket.close();
- }
- if (t.sseEventSource) {
- t.sseEventSource.close();
- }
- if (t.listenerInfos) {
- ae(t.listenerInfos, function (e) {
- if (e.on) {
- e.on.removeEventListener(e.trigger, e.listener);
- }
- });
- }
- if (t.initHash) {
- t.initHash = null;
- }
- Oe(e);
- }
- function m(e) {
- fe(e, "htmx:beforeCleanupElement");
- qe(e);
- if (e.children) {
- ae(e.children, function (e) {
- m(e);
- });
- }
- }
- function He(t, e, r) {
- if (t.tagName === "BODY") {
- return ke(t, e, r);
- } else {
- var n;
- var i = t.previousSibling;
- a(u(t), t, e, r);
- if (i == null) {
- n = u(t).firstChild;
- } else {
- n = i.nextSibling;
- }
- ie(t).replacedWith = n;
- r.elts = r.elts.filter(function (e) {
- return e != t;
- });
- while (n && n !== t) {
- if (n.nodeType === Node.ELEMENT_NODE) {
- r.elts.push(n);
- }
- n = n.nextElementSibling;
- }
- m(t);
- u(t).removeChild(t);
- }
- }
- function Le(e, t, r) {
- return a(e, e.firstChild, t, r);
- }
- function Ae(e, t, r) {
- return a(u(e), e, t, r);
- }
- function Ne(e, t, r) {
- return a(e, null, t, r);
- }
- function Ie(e, t, r) {
- return a(u(e), e.nextSibling, t, r);
- }
- function Pe(e, t, r) {
- m(e);
- return u(e).removeChild(e);
- }
- function ke(e, t, r) {
- var n = e.firstChild;
- a(e, n, t, r);
- if (n) {
- while (n.nextSibling) {
- m(n.nextSibling);
- e.removeChild(n.nextSibling);
- }
- m(n);
- e.removeChild(n);
- }
- }
- function Me(e, t, r) {
- var n = r || re(e, "hx-select");
- if (n) {
- var i = te().createDocumentFragment();
- ae(t.querySelectorAll(n), function (e) {
- i.appendChild(e);
- });
- t = i;
- }
- return t;
- }
- function De(e, t, r, n, i) {
- switch (e) {
- case "none":
- return;
- case "outerHTML":
- He(r, n, i);
- return;
- case "afterbegin":
- Le(r, n, i);
- return;
- case "beforebegin":
- Ae(r, n, i);
- return;
- case "beforeend":
- Ne(r, n, i);
- return;
- case "afterend":
- Ie(r, n, i);
- return;
- case "delete":
- Pe(r, n, i);
- return;
- default:
- var a = Lr(t);
- for (var o = 0; o < a.length; o++) {
- var s = a[o];
- try {
- var l = s.handleSwap(e, r, n, i);
- if (l) {
- if (typeof l.length !== "undefined") {
- for (var u = 0; u < l.length; u++) {
- var f = l[u];
- if (f.nodeType !== Node.TEXT_NODE && f.nodeType !== Node.COMMENT_NODE) {
- i.tasks.push(Ee(f));
- }
- }
- }
- return;
- }
- } catch (e) {
- y(e);
- }
- }
- if (e === "innerHTML") {
- ke(r, n, i);
- } else {
- De(Y.config.defaultSwapStyle, t, r, n, i);
- }
- }
- }
- function Xe(e) {
- if (e.indexOf(" -1) {
- var t = e.replace(/]*>|>)([\s\S]*?)<\/svg>/gim, "");
- var r = t.match(/]*>|>)([\s\S]*?)<\/title>/im);
- if (r) {
- return r[2];
- }
- }
- }
- function Fe(e, t, r, n, i, a) {
- i.title = Xe(n);
- var o = l(n);
- if (o) {
- be(r, o, i);
- o = Me(r, o, a);
- we(o);
- return De(e, r, t, o, i);
- }
- }
- function Ue(e, t, r) {
- var n = e.getResponseHeader(t);
- if (n.indexOf("{") === 0) {
- var i = S(n);
- for (var a in i) {
- if (i.hasOwnProperty(a)) {
- var o = i[a];
- if (!N(o)) {
- o = { value: o };
- }
- fe(r, a, o);
- }
- }
- } else {
- var s = n.split(",");
- for (var l = 0; l < s.length; l++) {
- fe(r, s[l].trim(), []);
- }
- }
- }
- var Be = /\s/;
- var p = /[\s,]/;
- var Ve = /[_$a-zA-Z]/;
- var je = /[_$a-zA-Z0-9]/;
- var _e = ['"', "'", "/"];
- var ze = /[^\s]/;
- function We(e) {
- var t = [];
- var r = 0;
- while (r < e.length) {
- if (Ve.exec(e.charAt(r))) {
- var n = r;
- while (je.exec(e.charAt(r + 1))) {
- r++;
- }
- t.push(e.substr(n, r - n + 1));
- } else if (_e.indexOf(e.charAt(r)) !== -1) {
- var i = e.charAt(r);
- var n = r;
- r++;
- while (r < e.length && e.charAt(r) !== i) {
- if (e.charAt(r) === "\\") {
- r++;
- }
- r++;
- }
- t.push(e.substr(n, r - n + 1));
- } else {
- var a = e.charAt(r);
- t.push(a);
- }
- r++;
- }
- return t;
- }
- function $e(e, t, r) {
- return Ve.exec(e.charAt(0)) && e !== "true" && e !== "false" && e !== "this" && e !== r && t !== ".";
- }
- function Ge(e, t, r) {
- if (t[0] === "[") {
- t.shift();
- var n = 1;
- var i = " return (function(" + r + "){ return (";
- var a = null;
- while (t.length > 0) {
- var o = t[0];
- if (o === "]") {
- n--;
- if (n === 0) {
- if (a === null) {
- i = i + "true";
- }
- t.shift();
- i += ")})";
- try {
- var s = gr(
- e,
- function () {
- return Function(i)();
- },
- function () {
- return true;
- },
- );
- s.source = i;
- return s;
- } catch (e) {
- ue(te().body, "htmx:syntax:error", { error: e, source: i });
- return null;
- }
- }
- } else if (o === "[") {
- n++;
- }
- if ($e(o, a, r)) {
- i += "((" + r + "." + o + ") ? (" + r + "." + o + ") : (window." + o + "))";
- } else {
- i = i + o;
- }
- a = t.shift();
- }
- }
- }
- function x(e, t) {
- var r = "";
- while (e.length > 0 && !e[0].match(t)) {
- r += e.shift();
- }
- return r;
- }
- var Je = "input, textarea, select";
- function Ze(e) {
- var t = ee(e, "hx-trigger");
- var r = [];
- if (t) {
- var n = We(t);
- do {
- x(n, ze);
- var i = n.length;
- var a = x(n, /[,\[\s]/);
- if (a !== "") {
- if (a === "every") {
- var o = { trigger: "every" };
- x(n, ze);
- o.pollInterval = d(x(n, /[,\[\s]/));
- x(n, ze);
- var s = Ge(e, n, "event");
- if (s) {
- o.eventFilter = s;
- }
- r.push(o);
- } else if (a.indexOf("sse:") === 0) {
- r.push({ trigger: "sse", sseEvent: a.substr(4) });
- } else {
- var l = { trigger: a };
- var s = Ge(e, n, "event");
- if (s) {
- l.eventFilter = s;
- }
- while (n.length > 0 && n[0] !== ",") {
- x(n, ze);
- var u = n.shift();
- if (u === "changed") {
- l.changed = true;
- } else if (u === "once") {
- l.once = true;
- } else if (u === "consume") {
- l.consume = true;
- } else if (u === "delay" && n[0] === ":") {
- n.shift();
- l.delay = d(x(n, p));
- } else if (u === "from" && n[0] === ":") {
- n.shift();
- var f = x(n, p);
- if (f === "closest" || f === "find" || f === "next" || f === "previous") {
- n.shift();
- f += " " + x(n, p);
- }
- l.from = f;
- } else if (u === "target" && n[0] === ":") {
- n.shift();
- l.target = x(n, p);
- } else if (u === "throttle" && n[0] === ":") {
- n.shift();
- l.throttle = d(x(n, p));
- } else if (u === "queue" && n[0] === ":") {
- n.shift();
- l.queue = x(n, p);
- } else if ((u === "root" || u === "threshold") && n[0] === ":") {
- n.shift();
- l[u] = x(n, p);
- } else {
- ue(e, "htmx:syntax:error", { token: n.shift() });
- }
- }
- r.push(l);
- }
- }
- if (n.length === i) {
- ue(e, "htmx:syntax:error", { token: n.shift() });
- }
- x(n, ze);
- } while (n[0] === "," && n.shift());
- }
- if (r.length > 0) {
- return r;
- } else if (h(e, "form")) {
- return [{ trigger: "submit" }];
- } else if (h(e, 'input[type="button"], input[type="submit"]')) {
- return [{ trigger: "click" }];
- } else if (h(e, Je)) {
- return [{ trigger: "change" }];
- } else {
- return [{ trigger: "click" }];
- }
- }
- function Ke(e) {
- ie(e).cancelled = true;
- }
- function Ye(e, t, r) {
- var n = ie(e);
- n.timeout = setTimeout(function () {
- if (oe(e) && n.cancelled !== true) {
- if (!nt(r, e, Mt("hx:poll:trigger", { triggerSpec: r, target: e }))) {
- t(e);
- }
- Ye(e, t, r);
- }
- }, r.pollInterval);
- }
- function Qe(e) {
- return location.hostname === e.hostname && Q(e, "href") && Q(e, "href").indexOf("#") !== 0;
- }
- function et(t, r, e) {
- if ((t.tagName === "A" && Qe(t) && (t.target === "" || t.target === "_self")) || t.tagName === "FORM") {
- r.boosted = true;
- var n, i;
- if (t.tagName === "A") {
- n = "get";
- i = Q(t, "href");
- } else {
- var a = Q(t, "method");
- n = a ? a.toLowerCase() : "get";
- if (n === "get") {
- }
- i = Q(t, "action");
- }
- e.forEach(function (e) {
- it(
- t,
- function (e, t) {
- if (v(e, Y.config.disableSelector)) {
- m(e);
- return;
- }
- ce(n, i, e, t);
- },
- r,
- e,
- true,
- );
- });
- }
- }
- function tt(e, t) {
- if (e.type === "submit" || e.type === "click") {
- if (t.tagName === "FORM") {
- return true;
- }
- if (h(t, 'input[type="submit"], button') && v(t, "form") !== null) {
- return true;
- }
- if (
- t.tagName === "A" &&
- t.href &&
- (t.getAttribute("href") === "#" || t.getAttribute("href").indexOf("#") !== 0)
- ) {
- return true;
- }
- }
- return false;
- }
- function rt(e, t) {
- return ie(e).boosted && e.tagName === "A" && t.type === "click" && (t.ctrlKey || t.metaKey);
- }
- function nt(e, t, r) {
- var n = e.eventFilter;
- if (n) {
- try {
- return n.call(t, r) !== true;
- } catch (e) {
- ue(te().body, "htmx:eventFilter:error", { error: e, source: n.source });
- return true;
- }
- }
- return false;
- }
- function it(a, o, e, s, l) {
- var u = ie(a);
- var t;
- if (s.from) {
- t = W(a, s.from);
- } else {
- t = [a];
- }
- if (s.changed) {
- t.forEach(function (e) {
- var t = ie(e);
- t.lastValue = e.value;
- });
- }
- ae(t, function (n) {
- var i = function (e) {
- if (!oe(a)) {
- n.removeEventListener(s.trigger, i);
- return;
- }
- if (rt(a, e)) {
- return;
- }
- if (l || tt(e, a)) {
- e.preventDefault();
- }
- if (nt(s, a, e)) {
- return;
- }
- var t = ie(e);
- t.triggerSpec = s;
- if (t.handledFor == null) {
- t.handledFor = [];
- }
- if (t.handledFor.indexOf(a) < 0) {
- t.handledFor.push(a);
- if (s.consume) {
- e.stopPropagation();
- }
- if (s.target && e.target) {
- if (!h(e.target, s.target)) {
- return;
- }
- }
- if (s.once) {
- if (u.triggeredOnce) {
- return;
- } else {
- u.triggeredOnce = true;
- }
- }
- if (s.changed) {
- var r = ie(n);
- if (r.lastValue === n.value) {
- return;
- }
- r.lastValue = n.value;
- }
- if (u.delayed) {
- clearTimeout(u.delayed);
- }
- if (u.throttle) {
- return;
- }
- if (s.throttle) {
- if (!u.throttle) {
- o(a, e);
- u.throttle = setTimeout(function () {
- u.throttle = null;
- }, s.throttle);
- }
- } else if (s.delay) {
- u.delayed = setTimeout(function () {
- o(a, e);
- }, s.delay);
- } else {
- fe(a, "htmx:trigger");
- o(a, e);
- }
- }
- };
- if (e.listenerInfos == null) {
- e.listenerInfos = [];
- }
- e.listenerInfos.push({ trigger: s.trigger, listener: i, on: n });
- n.addEventListener(s.trigger, i);
- });
- }
- var at = false;
- var ot = null;
- function st() {
- if (!ot) {
- ot = function () {
- at = true;
- };
- window.addEventListener("scroll", ot);
- setInterval(function () {
- if (at) {
- at = false;
- ae(te().querySelectorAll("[hx-trigger='revealed'],[data-hx-trigger='revealed']"), function (e) {
- lt(e);
- });
- }
- }, 200);
- }
- }
- function lt(t) {
- if (!o(t, "data-hx-revealed") && P(t)) {
- t.setAttribute("data-hx-revealed", "true");
- var e = ie(t);
- if (e.initHash) {
- fe(t, "revealed");
- } else {
- t.addEventListener(
- "htmx:afterProcessNode",
- function (e) {
- fe(t, "revealed");
- },
- { once: true },
- );
- }
- }
- }
- function ut(e, t, r) {
- var n = k(r);
- for (var i = 0; i < n.length; i++) {
- var a = n[i].split(/:(.+)/);
- if (a[0] === "connect") {
- ft(e, a[1], 0);
- }
- if (a[0] === "send") {
- ht(e);
- }
- }
- }
- function ft(s, r, n) {
- if (!oe(s)) {
- return;
- }
- if (r.indexOf("/") == 0) {
- var e = location.hostname + (location.port ? ":" + location.port : "");
- if (location.protocol == "https:") {
- r = "wss://" + e + r;
- } else if (location.protocol == "http:") {
- r = "ws://" + e + r;
- }
- }
- var t = Y.createWebSocket(r);
- t.onerror = function (e) {
- ue(s, "htmx:wsError", { error: e, socket: t });
- ct(s);
- };
- t.onclose = function (e) {
- if ([1006, 1012, 1013].indexOf(e.code) >= 0) {
- var t = vt(n);
- setTimeout(function () {
- ft(s, r, n + 1);
- }, t);
- }
- };
- t.onopen = function (e) {
- n = 0;
- };
- ie(s).webSocket = t;
- t.addEventListener("message", function (e) {
- if (ct(s)) {
- return;
- }
- var t = e.data;
- C(s, function (e) {
- t = e.transformResponse(t, null, s);
- });
- var r = T(s);
- var n = l(t);
- var i = I(n.children);
- for (var a = 0; a < i.length; a++) {
- var o = i[a];
- ye(ee(o, "hx-swap-oob") || "true", o, r);
- }
- Wt(r.tasks);
- });
- }
- function ct(e) {
- if (!oe(e)) {
- ie(e).webSocket.close();
- return true;
- }
- }
- function ht(u) {
- var f = c(u, function (e) {
- return ie(e).webSocket != null;
- });
- if (f) {
- u.addEventListener(Ze(u)[0].trigger, function (e) {
- var t = ie(f).webSocket;
- var r = sr(u, f);
- var n = nr(u, "post");
- var i = n.errors;
- var a = n.values;
- var o = xr(u);
- var s = se(a, o);
- var l = lr(s, u);
- l["HEADERS"] = r;
- if (i && i.length > 0) {
- fe(u, "htmx:validation:halted", i);
- return;
- }
- t.send(JSON.stringify(l));
- if (tt(e, u)) {
- e.preventDefault();
- }
- });
- } else {
- ue(u, "htmx:noWebSocketSourceError");
- }
- }
- function vt(e) {
- var t = Y.config.wsReconnectDelay;
- if (typeof t === "function") {
- return t(e);
- }
- if (t === "full-jitter") {
- var r = Math.min(e, 6);
- var n = 1e3 * Math.pow(2, r);
- return n * Math.random();
- }
- y('htmx.config.wsReconnectDelay must either be a function or the string "full-jitter"');
- }
- function dt(e, t, r) {
- var n = k(r);
- for (var i = 0; i < n.length; i++) {
- var a = n[i].split(/:(.+)/);
- if (a[0] === "connect") {
- gt(e, a[1]);
- }
- if (a[0] === "swap") {
- mt(e, a[1]);
- }
- }
- }
- function gt(t, e) {
- var r = Y.createEventSource(e);
- r.onerror = function (e) {
- ue(t, "htmx:sseError", { error: e, source: r });
- xt(t);
- };
- ie(t).sseEventSource = r;
- }
- function mt(a, o) {
- var s = c(a, yt);
- if (s) {
- var l = ie(s).sseEventSource;
- var u = function (e) {
- if (xt(s)) {
- return;
- }
- if (!oe(a)) {
- l.removeEventListener(o, u);
- return;
- }
- var t = e.data;
- C(a, function (e) {
- t = e.transformResponse(t, null, a);
- });
- var r = fr(a);
- var n = ge(a);
- var i = T(a);
- Fe(r.swapStyle, n, a, t, i);
- Wt(i.tasks);
- fe(a, "htmx:sseMessage", e);
- };
- ie(a).sseListener = u;
- l.addEventListener(o, u);
- } else {
- ue(a, "htmx:noSSESourceError");
- }
- }
- function pt(e, t, r) {
- var n = c(e, yt);
- if (n) {
- var i = ie(n).sseEventSource;
- var a = function () {
- if (!xt(n)) {
- if (oe(e)) {
- t(e);
- } else {
- i.removeEventListener(r, a);
- }
- }
- };
- ie(e).sseListener = a;
- i.addEventListener(r, a);
- } else {
- ue(e, "htmx:noSSESourceError");
- }
- }
- function xt(e) {
- if (!oe(e)) {
- ie(e).sseEventSource.close();
- return true;
- }
- }
- function yt(e) {
- return ie(e).sseEventSource != null;
- }
- function bt(e, t, r, n) {
- var i = function () {
- if (!r.loaded) {
- r.loaded = true;
- t(e);
- }
- };
- if (n) {
- setTimeout(i, n);
- } else {
- i();
- }
- }
- function wt(t, i, e) {
- var a = false;
- ae(b, function (r) {
- if (o(t, "hx-" + r)) {
- var n = ee(t, "hx-" + r);
- a = true;
- i.path = n;
- i.verb = r;
- e.forEach(function (e) {
- St(t, e, i, function (e, t) {
- if (v(e, Y.config.disableSelector)) {
- m(e);
- return;
- }
- ce(r, n, e, t);
- });
- });
- }
- });
- return a;
- }
- function St(n, e, t, r) {
- if (e.sseEvent) {
- pt(n, r, e.sseEvent);
- } else if (e.trigger === "revealed") {
- st();
- it(n, r, t, e);
- lt(n);
- } else if (e.trigger === "intersect") {
- var i = {};
- if (e.root) {
- i.root = le(n, e.root);
- }
- if (e.threshold) {
- i.threshold = parseFloat(e.threshold);
- }
- var a = new IntersectionObserver(function (e) {
- for (var t = 0; t < e.length; t++) {
- var r = e[t];
- if (r.isIntersecting) {
- fe(n, "intersect");
- break;
- }
- }
- }, i);
- a.observe(n);
- it(n, r, t, e);
- } else if (e.trigger === "load") {
- if (!nt(e, n, Mt("load", { elt: n }))) {
- bt(n, r, t, e.delay);
- }
- } else if (e.pollInterval) {
- t.polling = true;
- Ye(n, r, e);
- } else {
- it(n, r, t, e);
- }
- }
- function Et(e) {
- if (Y.config.allowScriptTags && (e.type === "text/javascript" || e.type === "module" || e.type === "")) {
- var t = te().createElement("script");
- ae(e.attributes, function (e) {
- t.setAttribute(e.name, e.value);
- });
- t.textContent = e.textContent;
- t.async = false;
- if (Y.config.inlineScriptNonce) {
- t.nonce = Y.config.inlineScriptNonce;
- }
- var r = e.parentElement;
- try {
- r.insertBefore(t, e);
- } catch (e) {
- y(e);
- } finally {
- if (e.parentElement) {
- e.parentElement.removeChild(e);
- }
- }
- }
- }
- function Ct(e) {
- if (h(e, "script")) {
- Et(e);
- }
- ae(f(e, "script"), function (e) {
- Et(e);
- });
- }
- function Tt() {
- return document.querySelector("[hx-boost], [data-hx-boost]");
- }
- function Rt(e) {
- var t = null;
- var r = [];
- if (document.evaluate) {
- var n = document.evaluate(
- '//*[@*[ starts-with(name(), "hx-on:") or starts-with(name(), "data-hx-on:") ]]',
- e,
- );
- while ((t = n.iterateNext())) r.push(t);
- } else {
- var i = document.getElementsByTagName("*");
- for (var a = 0; a < i.length; a++) {
- var o = i[a].attributes;
- for (var s = 0; s < o.length; s++) {
- var l = o[s].name;
- if (g(l, "hx-on:") || g(l, "data-hx-on:")) {
- r.push(i[a]);
- }
- }
- }
- }
- return r;
- }
- function Ot(e) {
- if (e.querySelectorAll) {
- var t = Tt() ? ", a" : "";
- var r = e.querySelectorAll(
- w +
- t +
- ", form, [type='submit'], [hx-sse], [data-hx-sse], [hx-ws]," +
- " [data-hx-ws], [hx-ext], [data-hx-ext], [hx-trigger], [data-hx-trigger], [hx-on], [data-hx-on]",
- );
- return r;
- } else {
- return [];
- }
- }
- function qt(e) {
- var n = s("#" + Q(e, "form")) || v(e, "form");
- if (!n) {
- return;
- }
- var t = function (e) {
- var t = v(e.target, "button, input[type='submit']");
- if (t !== null) {
- var r = ie(n);
- r.lastButtonClicked = t;
- }
- };
- e.addEventListener("click", t);
- e.addEventListener("focusin", t);
- e.addEventListener("focusout", function (e) {
- var t = ie(n);
- t.lastButtonClicked = null;
- });
- }
- function Ht(e) {
- var t = We(e);
- var r = 0;
- for (let e = 0; e < t.length; e++) {
- const n = t[e];
- if (n === "{") {
- r++;
- } else if (n === "}") {
- r--;
- }
- }
- return r;
- }
- function Lt(t, e, r) {
- var n = ie(t);
- n.onHandlers = [];
- var i;
- var a = function (e) {
- return gr(t, function () {
- if (!i) {
- i = new Function("event", r);
- }
- i.call(t, e);
- });
- };
- t.addEventListener(e, a);
- n.onHandlers.push({ event: e, listener: a });
- }
- function At(e) {
- var t = ee(e, "hx-on");
- if (t) {
- var r = {};
- var n = t.split("\n");
- var i = null;
- var a = 0;
- while (n.length > 0) {
- var o = n.shift();
- var s = o.match(/^\s*([a-zA-Z:\-\.]+:)(.*)/);
- if (a === 0 && s) {
- o.split(":");
- i = s[1].slice(0, -1);
- r[i] = s[2];
- } else {
- r[i] += o;
- }
- a += Ht(o);
- }
- for (var l in r) {
- Lt(e, l, r[l]);
- }
- }
- }
- function Nt(t) {
- Oe(t);
- for (var e = 0; e < t.attributes.length; e++) {
- var r = t.attributes[e].name;
- var n = t.attributes[e].value;
- if (g(r, "hx-on:") || g(r, "data-hx-on:")) {
- let e = r.slice(r.indexOf(":") + 1);
- if (g(e, ":")) e = "htmx" + e;
- Lt(t, e, n);
- }
- }
- }
- function It(t) {
- if (v(t, Y.config.disableSelector)) {
- m(t);
- return;
- }
- var r = ie(t);
- if (r.initHash !== Re(t)) {
- qe(t);
- r.initHash = Re(t);
- At(t);
- fe(t, "htmx:beforeProcessNode");
- if (t.value) {
- r.lastValue = t.value;
- }
- var e = Ze(t);
- var n = wt(t, r, e);
- if (!n) {
- if (re(t, "hx-boost") === "true") {
- et(t, r, e);
- } else if (o(t, "hx-trigger")) {
- e.forEach(function (e) {
- St(t, e, r, function () {});
- });
- }
- }
- if (t.tagName === "FORM" || (Q(t, "type") === "submit" && o(t, "form"))) {
- qt(t);
- }
- var i = ee(t, "hx-sse");
- if (i) {
- dt(t, r, i);
- }
- var a = ee(t, "hx-ws");
- if (a) {
- ut(t, r, a);
- }
- fe(t, "htmx:afterProcessNode");
- }
- }
- function Pt(e) {
- e = s(e);
- if (v(e, Y.config.disableSelector)) {
- m(e);
- return;
- }
- It(e);
- ae(Ot(e), function (e) {
- It(e);
- });
- ae(Rt(e), Nt);
- }
- function kt(e) {
- return e.replace(/([a-z0-9])([A-Z])/g, "$1-$2").toLowerCase();
- }
- function Mt(e, t) {
- var r;
- if (window.CustomEvent && typeof window.CustomEvent === "function") {
- r = new CustomEvent(e, { bubbles: true, cancelable: true, detail: t });
- } else {
- r = te().createEvent("CustomEvent");
- r.initCustomEvent(e, true, true, t);
- }
- return r;
- }
- function ue(e, t, r) {
- fe(e, t, se({ error: t }, r));
- }
- function Dt(e) {
- return e === "htmx:afterProcessNode";
- }
- function C(e, t) {
- ae(Lr(e), function (e) {
- try {
- t(e);
- } catch (e) {
- y(e);
- }
- });
- }
- function y(e) {
- if (console.error) {
- console.error(e);
- } else if (console.log) {
- console.log("ERROR: ", e);
- }
- }
- function fe(e, t, r) {
- e = s(e);
- if (r == null) {
- r = {};
- }
- r["elt"] = e;
- var n = Mt(t, r);
- if (Y.logger && !Dt(t)) {
- Y.logger(e, t, r);
- }
- if (r.error) {
- y(r.error);
- fe(e, "htmx:error", { errorInfo: r });
- }
- var i = e.dispatchEvent(n);
- var a = kt(t);
- if (i && a !== t) {
- var o = Mt(a, n.detail);
- i = i && e.dispatchEvent(o);
- }
- C(e, function (e) {
- i = i && e.onEvent(t, n) !== false && !n.defaultPrevented;
- });
- return i;
- }
- var Xt = location.pathname + location.search;
- function Ft() {
- var e = te().querySelector("[hx-history-elt],[data-hx-history-elt]");
- return e || te().body;
- }
- function Ut(e, t, r, n) {
- if (!M()) {
- return;
- }
- e = D(e);
- var i = S(localStorage.getItem("htmx-history-cache")) || [];
- for (var a = 0; a < i.length; a++) {
- if (i[a].url === e) {
- i.splice(a, 1);
- break;
- }
- }
- var o = { url: e, content: t, title: r, scroll: n };
- fe(te().body, "htmx:historyItemCreated", { item: o, cache: i });
- i.push(o);
- while (i.length > Y.config.historyCacheSize) {
- i.shift();
- }
- while (i.length > 0) {
- try {
- localStorage.setItem("htmx-history-cache", JSON.stringify(i));
- break;
- } catch (e) {
- ue(te().body, "htmx:historyCacheError", { cause: e, cache: i });
- i.shift();
- }
- }
- }
- function Bt(e) {
- if (!M()) {
- return null;
- }
- e = D(e);
- var t = S(localStorage.getItem("htmx-history-cache")) || [];
- for (var r = 0; r < t.length; r++) {
- if (t[r].url === e) {
- return t[r];
- }
- }
- return null;
- }
- function Vt(e) {
- var t = Y.config.requestClass;
- var r = e.cloneNode(true);
- ae(f(r, "." + t), function (e) {
- n(e, t);
- });
- return r.innerHTML;
- }
- function jt() {
- var e = Ft();
- var t = Xt || location.pathname + location.search;
- var r;
- try {
- r = te().querySelector('[hx-history="false" i],[data-hx-history="false" i]');
- } catch (e) {
- r = te().querySelector('[hx-history="false"],[data-hx-history="false"]');
- }
- if (!r) {
- fe(te().body, "htmx:beforeHistorySave", { path: t, historyElt: e });
- Ut(t, Vt(e), te().title, window.scrollY);
- }
- if (Y.config.historyEnabled) history.replaceState({ htmx: true }, te().title, window.location.href);
- }
- function _t(e) {
- if (Y.config.getCacheBusterParam) {
- e = e.replace(/org\.htmx\.cache-buster=[^&]*&?/, "");
- if (_(e, "&") || _(e, "?")) {
- e = e.slice(0, -1);
- }
- }
- if (Y.config.historyEnabled) {
- history.pushState({ htmx: true }, "", e);
- }
- Xt = e;
- }
- function zt(e) {
- if (Y.config.historyEnabled) history.replaceState({ htmx: true }, "", e);
- Xt = e;
- }
- function Wt(e) {
- ae(e, function (e) {
- e.call();
- });
- }
- function $t(a) {
- var e = new XMLHttpRequest();
- var o = { path: a, xhr: e };
- fe(te().body, "htmx:historyCacheMiss", o);
- e.open("GET", a, true);
- e.setRequestHeader("HX-History-Restore-Request", "true");
- e.onload = function () {
- if (this.status >= 200 && this.status < 400) {
- fe(te().body, "htmx:historyCacheMissLoad", o);
- var e = l(this.response);
- e = e.querySelector("[hx-history-elt],[data-hx-history-elt]") || e;
- var t = Ft();
- var r = T(t);
- var n = Xe(this.response);
- if (n) {
- var i = E("title");
- if (i) {
- i.innerHTML = n;
- } else {
- window.document.title = n;
- }
- }
- ke(t, e, r);
- Wt(r.tasks);
- Xt = a;
- fe(te().body, "htmx:historyRestore", { path: a, cacheMiss: true, serverResponse: this.response });
- } else {
- ue(te().body, "htmx:historyCacheMissLoadError", o);
- }
- };
- e.send();
- }
- function Gt(e) {
- jt();
- e = e || location.pathname + location.search;
- var t = Bt(e);
- if (t) {
- var r = l(t.content);
- var n = Ft();
- var i = T(n);
- ke(n, r, i);
- Wt(i.tasks);
- document.title = t.title;
- setTimeout(function () {
- window.scrollTo(0, t.scroll);
- }, 0);
- Xt = e;
- fe(te().body, "htmx:historyRestore", { path: e, item: t });
- } else {
- if (Y.config.refreshOnHistoryMiss) {
- window.location.reload(true);
- } else {
- $t(e);
- }
- }
- }
- function Jt(e) {
- var t = ve(e, "hx-indicator");
- if (t == null) {
- t = [e];
- }
- ae(t, function (e) {
- var t = ie(e);
- t.requestCount = (t.requestCount || 0) + 1;
- e.classList["add"].call(e.classList, Y.config.requestClass);
- });
- return t;
- }
- function Zt(e) {
- var t = ve(e, "hx-disabled-elt");
- if (t == null) {
- t = [];
- }
- ae(t, function (e) {
- var t = ie(e);
- t.requestCount = (t.requestCount || 0) + 1;
- e.setAttribute("disabled", "");
- });
- return t;
- }
- function Kt(e, t) {
- ae(e, function (e) {
- var t = ie(e);
- t.requestCount = (t.requestCount || 0) - 1;
- if (t.requestCount === 0) {
- e.classList["remove"].call(e.classList, Y.config.requestClass);
- }
- });
- ae(t, function (e) {
- var t = ie(e);
- t.requestCount = (t.requestCount || 0) - 1;
- if (t.requestCount === 0) {
- e.removeAttribute("disabled");
- }
- });
- }
- function Yt(e, t) {
- for (var r = 0; r < e.length; r++) {
- var n = e[r];
- if (n.isSameNode(t)) {
- return true;
- }
- }
- return false;
- }
- function Qt(e) {
- if (e.name === "" || e.name == null || e.disabled) {
- return false;
- }
- if (
- e.type === "button" ||
- e.type === "submit" ||
- e.tagName === "image" ||
- e.tagName === "reset" ||
- e.tagName === "file"
- ) {
- return false;
- }
- if (e.type === "checkbox" || e.type === "radio") {
- return e.checked;
- }
- return true;
- }
- function er(e, t, r) {
- if (e != null && t != null) {
- var n = r[e];
- if (n === undefined) {
- r[e] = t;
- } else if (Array.isArray(n)) {
- if (Array.isArray(t)) {
- r[e] = n.concat(t);
- } else {
- n.push(t);
- }
- } else {
- if (Array.isArray(t)) {
- r[e] = [n].concat(t);
- } else {
- r[e] = [n, t];
- }
- }
- }
- }
- function tr(t, r, n, e, i) {
- if (e == null || Yt(t, e)) {
- return;
- } else {
- t.push(e);
- }
- if (Qt(e)) {
- var a = Q(e, "name");
- var o = e.value;
- if (e.multiple) {
- o = I(e.querySelectorAll("option:checked")).map(function (e) {
- return e.value;
- });
- }
- if (e.files) {
- o = I(e.files);
- }
- er(a, o, r);
- if (i) {
- rr(e, n);
- }
- }
- if (h(e, "form")) {
- var s = e.elements;
- ae(s, function (e) {
- tr(t, r, n, e, i);
- });
- }
- }
- function rr(e, t) {
- if (e.willValidate) {
- fe(e, "htmx:validation:validate");
- if (!e.checkValidity()) {
- t.push({ elt: e, message: e.validationMessage, validity: e.validity });
- fe(e, "htmx:validation:failed", { message: e.validationMessage, validity: e.validity });
- }
- }
- }
- function nr(e, t) {
- var r = [];
- var n = {};
- var i = {};
- var a = [];
- var o = ie(e);
- var s = (h(e, "form") && e.noValidate !== true) || ee(e, "hx-validate") === "true";
- if (o.lastButtonClicked) {
- s = s && o.lastButtonClicked.formNoValidate !== true;
- }
- if (t !== "get") {
- tr(r, i, a, v(e, "form"), s);
- }
- tr(r, n, a, e, s);
- if (o.lastButtonClicked || e.tagName === "BUTTON" || (e.tagName === "INPUT" && Q(e, "type") === "submit")) {
- var l = o.lastButtonClicked || e;
- var u = Q(l, "name");
- er(u, l.value, i);
- }
- var f = ve(e, "hx-include");
- ae(f, function (e) {
- tr(r, n, a, e, s);
- if (!h(e, "form")) {
- ae(e.querySelectorAll(Je), function (e) {
- tr(r, n, a, e, s);
- });
- }
- });
- n = se(n, i);
- return { errors: a, values: n };
- }
- function ir(e, t, r) {
- if (e !== "") {
- e += "&";
- }
- if (String(r) === "[object Object]") {
- r = JSON.stringify(r);
- }
- var n = encodeURIComponent(r);
- e += encodeURIComponent(t) + "=" + n;
- return e;
- }
- function ar(e) {
- var t = "";
- for (var r in e) {
- if (e.hasOwnProperty(r)) {
- var n = e[r];
- if (Array.isArray(n)) {
- ae(n, function (e) {
- t = ir(t, r, e);
- });
- } else {
- t = ir(t, r, n);
- }
- }
- }
- return t;
- }
- function or(e) {
- var t = new FormData();
- for (var r in e) {
- if (e.hasOwnProperty(r)) {
- var n = e[r];
- if (Array.isArray(n)) {
- ae(n, function (e) {
- t.append(r, e);
- });
- } else {
- t.append(r, n);
- }
- }
- }
- return t;
- }
- function sr(e, t, r) {
- var n = {
- "HX-Request": "true",
- "HX-Trigger": Q(e, "id"),
- "HX-Trigger-Name": Q(e, "name"),
- "HX-Target": ee(t, "id"),
- "HX-Current-URL": te().location.href,
- };
- dr(e, "hx-headers", false, n);
- if (r !== undefined) {
- n["HX-Prompt"] = r;
- }
- if (ie(e).boosted) {
- n["HX-Boosted"] = "true";
- }
- return n;
- }
- function lr(t, e) {
- var r = re(e, "hx-params");
- if (r) {
- if (r === "none") {
- return {};
- } else if (r === "*") {
- return t;
- } else if (r.indexOf("not ") === 0) {
- ae(r.substr(4).split(","), function (e) {
- e = e.trim();
- delete t[e];
- });
- return t;
- } else {
- var n = {};
- ae(r.split(","), function (e) {
- e = e.trim();
- n[e] = t[e];
- });
- return n;
- }
- } else {
- return t;
- }
- }
- function ur(e) {
- return Q(e, "href") && Q(e, "href").indexOf("#") >= 0;
- }
- function fr(e, t) {
- var r = t ? t : re(e, "hx-swap");
- var n = {
- swapStyle: ie(e).boosted ? "innerHTML" : Y.config.defaultSwapStyle,
- swapDelay: Y.config.defaultSwapDelay,
- settleDelay: Y.config.defaultSettleDelay,
- };
- if (ie(e).boosted && !ur(e)) {
- n["show"] = "top";
- }
- if (r) {
- var i = k(r);
- if (i.length > 0) {
- for (var a = 0; a < i.length; a++) {
- var o = i[a];
- if (o.indexOf("swap:") === 0) {
- n["swapDelay"] = d(o.substr(5));
- } else if (o.indexOf("settle:") === 0) {
- n["settleDelay"] = d(o.substr(7));
- } else if (o.indexOf("transition:") === 0) {
- n["transition"] = o.substr(11) === "true";
- } else if (o.indexOf("ignoreTitle:") === 0) {
- n["ignoreTitle"] = o.substr(12) === "true";
- } else if (o.indexOf("scroll:") === 0) {
- var s = o.substr(7);
- var l = s.split(":");
- var u = l.pop();
- var f = l.length > 0 ? l.join(":") : null;
- n["scroll"] = u;
- n["scrollTarget"] = f;
- } else if (o.indexOf("show:") === 0) {
- var c = o.substr(5);
- var l = c.split(":");
- var h = l.pop();
- var f = l.length > 0 ? l.join(":") : null;
- n["show"] = h;
- n["showTarget"] = f;
- } else if (o.indexOf("focus-scroll:") === 0) {
- var v = o.substr("focus-scroll:".length);
- n["focusScroll"] = v == "true";
- } else if (a == 0) {
- n["swapStyle"] = o;
- } else {
- y("Unknown modifier in hx-swap: " + o);
- }
- }
- }
- }
- return n;
- }
- function cr(e) {
- return (
- re(e, "hx-encoding") === "multipart/form-data" ||
- (h(e, "form") && Q(e, "enctype") === "multipart/form-data")
- );
- }
- function hr(t, r, n) {
- var i = null;
- C(r, function (e) {
- if (i == null) {
- i = e.encodeParameters(t, n, r);
- }
- });
- if (i != null) {
- return i;
- } else {
- if (cr(r)) {
- return or(n);
- } else {
- return ar(n);
- }
- }
- }
- function T(e) {
- return { tasks: [], elts: [e] };
- }
- function vr(e, t) {
- var r = e[0];
- var n = e[e.length - 1];
- if (t.scroll) {
- var i = null;
- if (t.scrollTarget) {
- i = le(r, t.scrollTarget);
- }
- if (t.scroll === "top" && (r || i)) {
- i = i || r;
- i.scrollTop = 0;
- }
- if (t.scroll === "bottom" && (n || i)) {
- i = i || n;
- i.scrollTop = i.scrollHeight;
- }
- }
- if (t.show) {
- var i = null;
- if (t.showTarget) {
- var a = t.showTarget;
- if (t.showTarget === "window") {
- a = "body";
- }
- i = le(r, a);
- }
- if (t.show === "top" && (r || i)) {
- i = i || r;
- i.scrollIntoView({ block: "start", behavior: Y.config.scrollBehavior });
- }
- if (t.show === "bottom" && (n || i)) {
- i = i || n;
- i.scrollIntoView({ block: "end", behavior: Y.config.scrollBehavior });
- }
- }
- }
- function dr(e, t, r, n) {
- if (n == null) {
- n = {};
- }
- if (e == null) {
- return n;
- }
- var i = ee(e, t);
- if (i) {
- var a = i.trim();
- var o = r;
- if (a === "unset") {
- return null;
- }
- if (a.indexOf("javascript:") === 0) {
- a = a.substr(11);
- o = true;
- } else if (a.indexOf("js:") === 0) {
- a = a.substr(3);
- o = true;
- }
- if (a.indexOf("{") !== 0) {
- a = "{" + a + "}";
- }
- var s;
- if (o) {
- s = gr(
- e,
- function () {
- return Function("return (" + a + ")")();
- },
- {},
- );
- } else {
- s = S(a);
- }
- for (var l in s) {
- if (s.hasOwnProperty(l)) {
- if (n[l] == null) {
- n[l] = s[l];
- }
- }
- }
- }
- return dr(u(e), t, r, n);
- }
- function gr(e, t, r) {
- if (Y.config.allowEval) {
- return t();
- } else {
- ue(e, "htmx:evalDisallowedError");
- return r;
- }
- }
- function mr(e, t) {
- return dr(e, "hx-vars", true, t);
- }
- function pr(e, t) {
- return dr(e, "hx-vals", false, t);
- }
- function xr(e) {
- return se(mr(e), pr(e));
- }
- function yr(t, r, n) {
- if (n !== null) {
- try {
- t.setRequestHeader(r, n);
- } catch (e) {
- t.setRequestHeader(r, encodeURIComponent(n));
- t.setRequestHeader(r + "-URI-AutoEncoded", "true");
- }
- }
- }
- function br(t) {
- if (t.responseURL && typeof URL !== "undefined") {
- try {
- var e = new URL(t.responseURL);
- return e.pathname + e.search;
- } catch (e) {
- ue(te().body, "htmx:badResponseUrl", { url: t.responseURL });
- }
- }
- }
- function R(e, t) {
- return e.getAllResponseHeaders().match(t);
- }
- function wr(e, t, r) {
- e = e.toLowerCase();
- if (r) {
- if (r instanceof Element || L(r, "String")) {
- return ce(e, t, null, null, { targetOverride: s(r), returnPromise: true });
- } else {
- return ce(e, t, s(r.source), r.event, {
- handler: r.handler,
- headers: r.headers,
- values: r.values,
- targetOverride: s(r.target),
- swapOverride: r.swap,
- returnPromise: true,
- });
- }
- } else {
- return ce(e, t, null, null, { returnPromise: true });
- }
- }
- function Sr(e) {
- var t = [];
- while (e) {
- t.push(e);
- e = e.parentElement;
- }
- return t;
- }
- function Er(e, t, r) {
- var n;
- var i;
- if (typeof URL === "function") {
- i = new URL(t, document.location.href);
- var a = document.location.origin;
- n = a === i.origin;
- } else {
- i = t;
- n = g(t, document.location.origin);
- }
- if (Y.config.selfRequestsOnly) {
- if (!n) {
- return false;
- }
- }
- return fe(e, "htmx:validateUrl", se({ url: i, sameHost: n }, r));
- }
- function ce(e, t, n, r, i, M) {
- var a = null;
- var o = null;
- i = i != null ? i : {};
- if (i.returnPromise && typeof Promise !== "undefined") {
- var s = new Promise(function (e, t) {
- a = e;
- o = t;
- });
- }
- if (n == null) {
- n = te().body;
- }
- var D = i.handler || Tr;
- if (!oe(n)) {
- ne(a);
- return s;
- }
- var l = i.targetOverride || ge(n);
- if (l == null || l == he) {
- ue(n, "htmx:targetError", { target: ee(n, "hx-target") });
- ne(o);
- return s;
- }
- var u = ie(n);
- var f = u.lastButtonClicked;
- if (f) {
- var c = Q(f, "formaction");
- if (c != null) {
- t = c;
- }
- var h = Q(f, "formmethod");
- if (h != null) {
- e = h;
- }
- }
- if (!M) {
- var X = function () {
- return ce(e, t, n, r, i, true);
- };
- var F = { target: l, elt: n, path: t, verb: e, triggeringEvent: r, etc: i, issueRequest: X };
- if (fe(n, "htmx:confirm", F) === false) {
- ne(a);
- return s;
- }
- }
- var v = n;
- var d = re(n, "hx-sync");
- var g = null;
- var m = false;
- if (d) {
- var p = d.split(":");
- var x = p[0].trim();
- if (x === "this") {
- v = de(n, "hx-sync");
- } else {
- v = le(n, x);
- }
- d = (p[1] || "drop").trim();
- u = ie(v);
- if (d === "drop" && u.xhr && u.abortable !== true) {
- ne(a);
- return s;
- } else if (d === "abort") {
- if (u.xhr) {
- ne(a);
- return s;
- } else {
- m = true;
- }
- } else if (d === "replace") {
- fe(v, "htmx:abort");
- } else if (d.indexOf("queue") === 0) {
- var U = d.split(" ");
- g = (U[1] || "last").trim();
- }
- }
- if (u.xhr) {
- if (u.abortable) {
- fe(v, "htmx:abort");
- } else {
- if (g == null) {
- if (r) {
- var y = ie(r);
- if (y && y.triggerSpec && y.triggerSpec.queue) {
- g = y.triggerSpec.queue;
- }
- }
- if (g == null) {
- g = "last";
- }
- }
- if (u.queuedRequests == null) {
- u.queuedRequests = [];
- }
- if (g === "first" && u.queuedRequests.length === 0) {
- u.queuedRequests.push(function () {
- ce(e, t, n, r, i);
- });
- } else if (g === "all") {
- u.queuedRequests.push(function () {
- ce(e, t, n, r, i);
- });
- } else if (g === "last") {
- u.queuedRequests = [];
- u.queuedRequests.push(function () {
- ce(e, t, n, r, i);
- });
- }
- ne(a);
- return s;
- }
- }
- var b = new XMLHttpRequest();
- u.xhr = b;
- u.abortable = m;
- var w = function () {
- u.xhr = null;
- u.abortable = false;
- if (u.queuedRequests != null && u.queuedRequests.length > 0) {
- var e = u.queuedRequests.shift();
- e();
- }
- };
- var B = re(n, "hx-prompt");
- if (B) {
- var S = prompt(B);
- if (S === null || !fe(n, "htmx:prompt", { prompt: S, target: l })) {
- ne(a);
- w();
- return s;
- }
- }
- var V = re(n, "hx-confirm");
- if (V) {
- if (!confirm(V)) {
- ne(a);
- w();
- return s;
- }
- }
- var E = sr(n, l, S);
- if (i.headers) {
- E = se(E, i.headers);
- }
- var j = nr(n, e);
- var C = j.errors;
- var T = j.values;
- if (i.values) {
- T = se(T, i.values);
- }
- var _ = xr(n);
- var z = se(T, _);
- var R = lr(z, n);
- if (e !== "get" && !cr(n)) {
- E["Content-Type"] = "application/x-www-form-urlencoded";
- }
- if (Y.config.getCacheBusterParam && e === "get") {
- R["org.htmx.cache-buster"] = Q(l, "id") || "true";
- }
- if (t == null || t === "") {
- t = te().location.href;
- }
- var O = dr(n, "hx-request");
- var W = ie(n).boosted;
- var q = Y.config.methodsThatUseUrlParams.indexOf(e) >= 0;
- var H = {
- boosted: W,
- useUrlParams: q,
- parameters: R,
- unfilteredParameters: z,
- headers: E,
- target: l,
- verb: e,
- errors: C,
- withCredentials: i.credentials || O.credentials || Y.config.withCredentials,
- timeout: i.timeout || O.timeout || Y.config.timeout,
- path: t,
- triggeringEvent: r,
- };
- if (!fe(n, "htmx:configRequest", H)) {
- ne(a);
- w();
- return s;
- }
- t = H.path;
- e = H.verb;
- E = H.headers;
- R = H.parameters;
- C = H.errors;
- q = H.useUrlParams;
- if (C && C.length > 0) {
- fe(n, "htmx:validation:halted", H);
- ne(a);
- w();
- return s;
- }
- var $ = t.split("#");
- var G = $[0];
- var L = $[1];
- var A = t;
- if (q) {
- A = G;
- var J = Object.keys(R).length !== 0;
- if (J) {
- if (A.indexOf("?") < 0) {
- A += "?";
- } else {
- A += "&";
- }
- A += ar(R);
- if (L) {
- A += "#" + L;
- }
- }
- }
- if (!Er(n, A, H)) {
- ue(n, "htmx:invalidPath", H);
- ne(o);
- return s;
- }
- b.open(e.toUpperCase(), A, true);
- b.overrideMimeType("text/html");
- b.withCredentials = H.withCredentials;
- b.timeout = H.timeout;
- if (O.noHeaders) {
- } else {
- for (var N in E) {
- if (E.hasOwnProperty(N)) {
- var Z = E[N];
- yr(b, N, Z);
- }
- }
- }
- var I = {
- xhr: b,
- target: l,
- requestConfig: H,
- etc: i,
- boosted: W,
- pathInfo: { requestPath: t, finalRequestPath: A, anchor: L },
- };
- b.onload = function () {
- try {
- var e = Sr(n);
- I.pathInfo.responsePath = br(b);
- D(n, I);
- Kt(P, k);
- fe(n, "htmx:afterRequest", I);
- fe(n, "htmx:afterOnLoad", I);
- if (!oe(n)) {
- var t = null;
- while (e.length > 0 && t == null) {
- var r = e.shift();
- if (oe(r)) {
- t = r;
- }
- }
- if (t) {
- fe(t, "htmx:afterRequest", I);
- fe(t, "htmx:afterOnLoad", I);
- }
- }
- ne(a);
- w();
- } catch (e) {
- ue(n, "htmx:onLoadError", se({ error: e }, I));
- throw e;
- }
- };
- b.onerror = function () {
- Kt(P, k);
- ue(n, "htmx:afterRequest", I);
- ue(n, "htmx:sendError", I);
- ne(o);
- w();
- };
- b.onabort = function () {
- Kt(P, k);
- ue(n, "htmx:afterRequest", I);
- ue(n, "htmx:sendAbort", I);
- ne(o);
- w();
- };
- b.ontimeout = function () {
- Kt(P, k);
- ue(n, "htmx:afterRequest", I);
- ue(n, "htmx:timeout", I);
- ne(o);
- w();
- };
- if (!fe(n, "htmx:beforeRequest", I)) {
- ne(a);
- w();
- return s;
- }
- var P = Jt(n);
- var k = Zt(n);
- ae(["loadstart", "loadend", "progress", "abort"], function (t) {
- ae([b, b.upload], function (e) {
- e.addEventListener(t, function (e) {
- fe(n, "htmx:xhr:" + t, {
- lengthComputable: e.lengthComputable,
- loaded: e.loaded,
- total: e.total,
- });
- });
- });
- });
- fe(n, "htmx:beforeSend", I);
- var K = q ? null : hr(b, n, R);
- b.send(K);
- return s;
- }
- function Cr(e, t) {
- var r = t.xhr;
- var n = null;
- var i = null;
- if (R(r, /HX-Push:/i)) {
- n = r.getResponseHeader("HX-Push");
- i = "push";
- } else if (R(r, /HX-Push-Url:/i)) {
- n = r.getResponseHeader("HX-Push-Url");
- i = "push";
- } else if (R(r, /HX-Replace-Url:/i)) {
- n = r.getResponseHeader("HX-Replace-Url");
- i = "replace";
- }
- if (n) {
- if (n === "false") {
- return {};
- } else {
- return { type: i, path: n };
- }
- }
- var a = t.pathInfo.finalRequestPath;
- var o = t.pathInfo.responsePath;
- var s = re(e, "hx-push-url");
- var l = re(e, "hx-replace-url");
- var u = ie(e).boosted;
- var f = null;
- var c = null;
- if (s) {
- f = "push";
- c = s;
- } else if (l) {
- f = "replace";
- c = l;
- } else if (u) {
- f = "push";
- c = o || a;
- }
- if (c) {
- if (c === "false") {
- return {};
- }
- if (c === "true") {
- c = o || a;
- }
- if (t.pathInfo.anchor && c.indexOf("#") === -1) {
- c = c + "#" + t.pathInfo.anchor;
- }
- return { type: f, path: c };
- } else {
- return {};
- }
- }
- function Tr(l, u) {
- var f = u.xhr;
- var c = u.target;
- var e = u.etc;
- var t = u.requestConfig;
- if (!fe(l, "htmx:beforeOnLoad", u)) return;
- if (R(f, /HX-Trigger:/i)) {
- Ue(f, "HX-Trigger", l);
- }
- if (R(f, /HX-Location:/i)) {
- jt();
- var r = f.getResponseHeader("HX-Location");
- var h;
- if (r.indexOf("{") === 0) {
- h = S(r);
- r = h["path"];
- delete h["path"];
- }
- wr("GET", r, h).then(function () {
- _t(r);
- });
- return;
- }
- var n = R(f, /HX-Refresh:/i) && "true" === f.getResponseHeader("HX-Refresh");
- if (R(f, /HX-Redirect:/i)) {
- location.href = f.getResponseHeader("HX-Redirect");
- n && location.reload();
- return;
- }
- if (n) {
- location.reload();
- return;
- }
- if (R(f, /HX-Retarget:/i)) {
- u.target = te().querySelector(f.getResponseHeader("HX-Retarget"));
- }
- var v = Cr(l, u);
- var i = f.status >= 200 && f.status < 400 && f.status !== 204;
- var d = f.response;
- var a = f.status >= 400;
- var g = Y.config.ignoreTitle;
- var o = se({ shouldSwap: i, serverResponse: d, isError: a, ignoreTitle: g }, u);
- if (!fe(c, "htmx:beforeSwap", o)) return;
- c = o.target;
- d = o.serverResponse;
- a = o.isError;
- g = o.ignoreTitle;
- u.target = c;
- u.failed = a;
- u.successful = !a;
- if (o.shouldSwap) {
- if (f.status === 286) {
- Ke(l);
- }
- C(l, function (e) {
- d = e.transformResponse(d, f, l);
- });
- if (v.type) {
- jt();
- }
- var s = e.swapOverride;
- if (R(f, /HX-Reswap:/i)) {
- s = f.getResponseHeader("HX-Reswap");
- }
- var h = fr(l, s);
- if (h.hasOwnProperty("ignoreTitle")) {
- g = h.ignoreTitle;
- }
- c.classList.add(Y.config.swappingClass);
- var m = null;
- var p = null;
- var x = function () {
- try {
- var e = document.activeElement;
- var t = {};
- try {
- t = { elt: e, start: e ? e.selectionStart : null, end: e ? e.selectionEnd : null };
- } catch (e) {}
- var r;
- if (R(f, /HX-Reselect:/i)) {
- r = f.getResponseHeader("HX-Reselect");
- }
- var n = T(c);
- Fe(h.swapStyle, c, l, d, n, r);
- if (t.elt && !oe(t.elt) && Q(t.elt, "id")) {
- var i = document.getElementById(Q(t.elt, "id"));
- var a = {
- preventScroll:
- h.focusScroll !== undefined ? !h.focusScroll : !Y.config.defaultFocusScroll,
- };
- if (i) {
- if (t.start && i.setSelectionRange) {
- try {
- i.setSelectionRange(t.start, t.end);
- } catch (e) {}
- }
- i.focus(a);
- }
- }
- c.classList.remove(Y.config.swappingClass);
- ae(n.elts, function (e) {
- if (e.classList) {
- e.classList.add(Y.config.settlingClass);
- }
- fe(e, "htmx:afterSwap", u);
- });
- if (R(f, /HX-Trigger-After-Swap:/i)) {
- var o = l;
- if (!oe(l)) {
- o = te().body;
- }
- Ue(f, "HX-Trigger-After-Swap", o);
- }
- var s = function () {
- ae(n.tasks, function (e) {
- e.call();
- });
- ae(n.elts, function (e) {
- if (e.classList) {
- e.classList.remove(Y.config.settlingClass);
- }
- fe(e, "htmx:afterSettle", u);
- });
- if (v.type) {
- if (v.type === "push") {
- _t(v.path);
- fe(te().body, "htmx:pushedIntoHistory", { path: v.path });
- } else {
- zt(v.path);
- fe(te().body, "htmx:replacedInHistory", { path: v.path });
- }
- }
- if (u.pathInfo.anchor) {
- var e = E("#" + u.pathInfo.anchor);
- if (e) {
- e.scrollIntoView({ block: "start", behavior: "auto" });
- }
- }
- if (n.title && !g) {
- var t = E("title");
- if (t) {
- t.innerHTML = n.title;
- } else {
- window.document.title = n.title;
- }
- }
- vr(n.elts, h);
- if (R(f, /HX-Trigger-After-Settle:/i)) {
- var r = l;
- if (!oe(l)) {
- r = te().body;
- }
- Ue(f, "HX-Trigger-After-Settle", r);
- }
- ne(m);
- };
- if (h.settleDelay > 0) {
- setTimeout(s, h.settleDelay);
- } else {
- s();
- }
- } catch (e) {
- ue(l, "htmx:swapError", u);
- ne(p);
- throw e;
- }
- };
- var y = Y.config.globalViewTransitions;
- if (h.hasOwnProperty("transition")) {
- y = h.transition;
- }
- if (
- y &&
- fe(l, "htmx:beforeTransition", u) &&
- typeof Promise !== "undefined" &&
- document.startViewTransition
- ) {
- var b = new Promise(function (e, t) {
- m = e;
- p = t;
- });
- var w = x;
- x = function () {
- document.startViewTransition(function () {
- w();
- return b;
- });
- };
- }
- if (h.swapDelay > 0) {
- setTimeout(x, h.swapDelay);
- } else {
- x();
- }
- }
- if (a) {
- ue(
- l,
- "htmx:responseError",
- se({ error: "Response Status Error Code " + f.status + " from " + u.pathInfo.requestPath }, u),
- );
- }
- }
- var Rr = {};
- function Or() {
- return {
- init: function (e) {
- return null;
- },
- onEvent: function (e, t) {
- return true;
- },
- transformResponse: function (e, t, r) {
- return e;
- },
- isInlineSwap: function (e) {
- return false;
- },
- handleSwap: function (e, t, r, n) {
- return false;
- },
- encodeParameters: function (e, t, r) {
- return null;
- },
- };
- }
- function qr(e, t) {
- if (t.init) {
- t.init(r);
- }
- Rr[e] = se(Or(), t);
- }
- function Hr(e) {
- delete Rr[e];
- }
- function Lr(e, r, n) {
- if (e == undefined) {
- return r;
- }
- if (r == undefined) {
- r = [];
- }
- if (n == undefined) {
- n = [];
- }
- var t = ee(e, "hx-ext");
- if (t) {
- ae(t.split(","), function (e) {
- e = e.replace(/ /g, "");
- if (e.slice(0, 7) == "ignore:") {
- n.push(e.slice(7));
- return;
- }
- if (n.indexOf(e) < 0) {
- var t = Rr[e];
- if (t && r.indexOf(t) < 0) {
- r.push(t);
- }
- }
- });
- }
- return Lr(u(e), r, n);
- }
- var Ar = false;
- te().addEventListener("DOMContentLoaded", function () {
- Ar = true;
- });
- function Nr(e) {
- if (Ar || te().readyState === "complete") {
- e();
- } else {
- te().addEventListener("DOMContentLoaded", e);
- }
- }
- function Ir() {
- if (Y.config.includeIndicatorStyles !== false) {
- te().head.insertAdjacentHTML(
- "beforeend",
- "",
- );
- }
- }
- function Pr() {
- var e = te().querySelector('meta[name="htmx-config"]');
- if (e) {
- return S(e.content);
- } else {
- return null;
- }
- }
- function kr() {
- var e = Pr();
- if (e) {
- Y.config = se(Y.config, e);
- }
- }
- Nr(function () {
- kr();
- Ir();
- var e = te().body;
- Pt(e);
- var t = te().querySelectorAll("[hx-trigger='restored'],[data-hx-trigger='restored']");
- e.addEventListener("htmx:abort", function (e) {
- var t = e.target;
- var r = ie(t);
- if (r && r.xhr) {
- r.xhr.abort();
- }
- });
- var r = window.onpopstate;
- window.onpopstate = function (e) {
- if (e.state && e.state.htmx) {
- Gt();
- ae(t, function (e) {
- fe(e, "htmx:restored", { document: te(), triggerEvent: fe });
- });
- } else {
- if (r) {
- r(e);
- }
- }
- };
- setTimeout(function () {
- fe(e, "htmx:load", {});
- e = null;
- }, 0);
- });
- return Y;
- })();
-});
+var htmx=function(){"use strict";const Q={onLoad:null,process:null,on:null,off:null,trigger:null,ajax:null,find:null,findAll:null,closest:null,values:function(e,t){const n=cn(e,t||"post");return n.values},remove:null,addClass:null,removeClass:null,toggleClass:null,takeClass:null,swap:null,defineExtension:null,removeExtension:null,logAll:null,logNone:null,logger:null,config:{historyEnabled:true,historyCacheSize:10,refreshOnHistoryMiss:false,defaultSwapStyle:"innerHTML",defaultSwapDelay:0,defaultSettleDelay:20,includeIndicatorStyles:true,indicatorClass:"htmx-indicator",requestClass:"htmx-request",addedClass:"htmx-added",settlingClass:"htmx-settling",swappingClass:"htmx-swapping",allowEval:true,allowScriptTags:true,inlineScriptNonce:"",inlineStyleNonce:"",attributesToSettle:["class","style","width","height"],withCredentials:false,timeout:0,wsReconnectDelay:"full-jitter",wsBinaryType:"blob",disableSelector:"[hx-disable], [data-hx-disable]",scrollBehavior:"instant",defaultFocusScroll:false,getCacheBusterParam:false,globalViewTransitions:false,methodsThatUseUrlParams:["get","delete"],selfRequestsOnly:true,ignoreTitle:false,scrollIntoViewOnBoost:true,triggerSpecsCache:null,disableInheritance:false,responseHandling:[{code:"204",swap:false},{code:"[23]..",swap:true},{code:"[45]..",swap:false,error:true}],allowNestedOobSwaps:true},parseInterval:null,_:null,version:"2.0.1"};Q.onLoad=$;Q.process=kt;Q.on=be;Q.off=we;Q.trigger=he;Q.ajax=Hn;Q.find=r;Q.findAll=p;Q.closest=g;Q.remove=K;Q.addClass=Y;Q.removeClass=o;Q.toggleClass=W;Q.takeClass=ge;Q.swap=ze;Q.defineExtension=Un;Q.removeExtension=Bn;Q.logAll=z;Q.logNone=J;Q.parseInterval=d;Q._=_;const n={addTriggerHandler:Et,bodyContains:le,canAccessLocalStorage:j,findThisElement:Ee,filterValues:dn,swap:ze,hasAttribute:s,getAttributeValue:te,getClosestAttributeValue:re,getClosestMatch:T,getExpressionVars:Cn,getHeaders:hn,getInputValues:cn,getInternalData:ie,getSwapSpecification:pn,getTriggerSpecs:lt,getTarget:Ce,makeFragment:k,mergeObjects:ue,makeSettleInfo:xn,oobSwap:Te,querySelectorExt:fe,settleImmediately:Gt,shouldCancel:dt,triggerEvent:he,triggerErrorEvent:ae,withExtensions:Ut};const v=["get","post","put","delete","patch"];const R=v.map(function(e){return"[hx-"+e+"], [data-hx-"+e+"]"}).join(", ");const O=e("head");function e(e,t=false){return new RegExp(`<${e}(\\s[^>]*>|>)([\\s\\S]*?)<\\/${e}>`,t?"gim":"im")}function d(e){if(e==undefined){return undefined}let t=NaN;if(e.slice(-2)=="ms"){t=parseFloat(e.slice(0,-2))}else if(e.slice(-1)=="s"){t=parseFloat(e.slice(0,-1))*1e3}else if(e.slice(-1)=="m"){t=parseFloat(e.slice(0,-1))*1e3*60}else{t=parseFloat(e)}return isNaN(t)?undefined:t}function ee(e,t){return e instanceof Element&&e.getAttribute(t)}function s(e,t){return!!e.hasAttribute&&(e.hasAttribute(t)||e.hasAttribute("data-"+t))}function te(e,t){return ee(e,t)||ee(e,"data-"+t)}function u(e){const t=e.parentElement;if(!t&&e.parentNode instanceof ShadowRoot)return e.parentNode;return t}function ne(){return document}function H(e,t){return e.getRootNode?e.getRootNode({composed:t}):ne()}function T(e,t){while(e&&!t(e)){e=u(e)}return e||null}function q(e,t,n){const r=te(t,n);const o=te(t,"hx-disinherit");var i=te(t,"hx-inherit");if(e!==t){if(Q.config.disableInheritance){if(i&&(i==="*"||i.split(" ").indexOf(n)>=0)){return r}else{return null}}if(o&&(o==="*"||o.split(" ").indexOf(n)>=0)){return"unset"}}return r}function re(t,n){let r=null;T(t,function(e){return!!(r=q(t,ce(e),n))});if(r!=="unset"){return r}}function a(e,t){const n=e instanceof Element&&(e.matches||e.matchesSelector||e.msMatchesSelector||e.mozMatchesSelector||e.webkitMatchesSelector||e.oMatchesSelector);return!!n&&n.call(e,t)}function L(e){const t=/<([a-z][^\/\0>\x20\t\r\n\f]*)/i;const n=t.exec(e);if(n){return n[1].toLowerCase()}else{return""}}function N(e){const t=new DOMParser;return t.parseFromString(e,"text/html")}function A(e,t){while(t.childNodes.length>0){e.append(t.childNodes[0])}}function I(e){const t=ne().createElement("script");se(e.attributes,function(e){t.setAttribute(e.name,e.value)});t.textContent=e.textContent;t.async=false;if(Q.config.inlineScriptNonce){t.nonce=Q.config.inlineScriptNonce}return t}function P(e){return e.matches("script")&&(e.type==="text/javascript"||e.type==="module"||e.type==="")}function D(e){Array.from(e.querySelectorAll("script")).forEach(e=>{if(P(e)){const t=I(e);const n=e.parentNode;try{n.insertBefore(t,e)}catch(e){w(e)}finally{e.remove()}}})}function k(e){const t=e.replace(O,"");const n=L(t);let r;if(n==="html"){r=new DocumentFragment;const i=N(e);A(r,i.body);r.title=i.title}else if(n==="body"){r=new DocumentFragment;const i=N(t);A(r,i.body);r.title=i.title}else{const i=N(' '+t+" ");r=i.querySelector("template").content;r.title=i.title;var o=r.querySelector("title");if(o&&o.parentNode===r){o.remove();r.title=o.innerText}}if(r){if(Q.config.allowScriptTags){D(r)}else{r.querySelectorAll("script").forEach(e=>e.remove())}}return r}function oe(e){if(e){e()}}function t(e,t){return Object.prototype.toString.call(e)==="[object "+t+"]"}function M(e){return typeof e==="function"}function X(e){return t(e,"Object")}function ie(e){const t="htmx-internal-data";let n=e[t];if(!n){n=e[t]={}}return n}function F(t){const n=[];if(t){for(let e=0;e=0}function le(e){const t=e.getRootNode&&e.getRootNode();if(t&&t instanceof window.ShadowRoot){return ne().body.contains(t.host)}else{return ne().body.contains(e)}}function B(e){return e.trim().split(/\s+/)}function ue(e,t){for(const n in t){if(t.hasOwnProperty(n)){e[n]=t[n]}}return e}function S(e){try{return JSON.parse(e)}catch(e){w(e);return null}}function j(){const e="htmx:localStorageTest";try{localStorage.setItem(e,e);localStorage.removeItem(e);return true}catch(e){return false}}function V(t){try{const e=new URL(t);if(e){t=e.pathname+e.search}if(!/^\/$/.test(t)){t=t.replace(/\/+$/,"")}return t}catch(e){return t}}function _(e){return vn(ne().body,function(){return eval(e)})}function $(t){const e=Q.on("htmx:load",function(e){t(e.detail.elt)});return e}function z(){Q.logger=function(e,t,n){if(console){console.log(t,e,n)}}}function J(){Q.logger=null}function r(e,t){if(typeof e!=="string"){return e.querySelector(t)}else{return r(ne(),e)}}function p(e,t){if(typeof e!=="string"){return e.querySelectorAll(t)}else{return p(ne(),e)}}function E(){return window}function K(e,t){e=y(e);if(t){E().setTimeout(function(){K(e);e=null},t)}else{u(e).removeChild(e)}}function ce(e){return e instanceof Element?e:null}function G(e){return e instanceof HTMLElement?e:null}function Z(e){return typeof e==="string"?e:null}function h(e){return e instanceof Element||e instanceof Document||e instanceof DocumentFragment?e:null}function Y(e,t,n){e=ce(y(e));if(!e){return}if(n){E().setTimeout(function(){Y(e,t);e=null},n)}else{e.classList&&e.classList.add(t)}}function o(e,t,n){let r=ce(y(e));if(!r){return}if(n){E().setTimeout(function(){o(r,t);r=null},n)}else{if(r.classList){r.classList.remove(t);if(r.classList.length===0){r.removeAttribute("class")}}}}function W(e,t){e=y(e);e.classList.toggle(t)}function ge(e,t){e=y(e);se(e.parentElement.children,function(e){o(e,t)});Y(ce(e),t)}function g(e,t){e=ce(y(e));if(e&&e.closest){return e.closest(t)}else{do{if(e==null||a(e,t)){return e}}while(e=e&&ce(u(e)));return null}}function l(e,t){return e.substring(0,t.length)===t}function pe(e,t){return e.substring(e.length-t.length)===t}function i(e){const t=e.trim();if(l(t,"<")&&pe(t,"/>")){return t.substring(1,t.length-2)}else{return t}}function m(e,t,n){e=y(e);if(t.indexOf("closest ")===0){return[g(ce(e),i(t.substr(8)))]}else if(t.indexOf("find ")===0){return[r(h(e),i(t.substr(5)))]}else if(t==="next"){return[ce(e).nextElementSibling]}else if(t.indexOf("next ")===0){return[me(e,i(t.substr(5)),!!n)]}else if(t==="previous"){return[ce(e).previousElementSibling]}else if(t.indexOf("previous ")===0){return[ye(e,i(t.substr(9)),!!n)]}else if(t==="document"){return[document]}else if(t==="window"){return[window]}else if(t==="body"){return[document.body]}else if(t==="root"){return[H(e,!!n)]}else if(t.indexOf("global ")===0){return m(e,t.slice(7),true)}else{return F(h(H(e,!!n)).querySelectorAll(i(t)))}}var me=function(t,e,n){const r=h(H(t,n)).querySelectorAll(e);for(let e=0;e=0;e--){const o=r[e];if(o.compareDocumentPosition(t)===Node.DOCUMENT_POSITION_FOLLOWING){return o}}};function fe(e,t){if(typeof e!=="string"){return m(e,t)[0]}else{return m(ne().body,e)[0]}}function y(e,t){if(typeof e==="string"){return r(h(t)||document,e)}else{return e}}function xe(e,t,n){if(M(t)){return{target:ne().body,event:Z(e),listener:t}}else{return{target:y(e),event:Z(t),listener:n}}}function be(t,n,r){_n(function(){const e=xe(t,n,r);e.target.addEventListener(e.event,e.listener)});const e=M(n);return e?n:r}function we(t,n,r){_n(function(){const e=xe(t,n,r);e.target.removeEventListener(e.event,e.listener)});return M(n)?n:r}const ve=ne().createElement("output");function Se(e,t){const n=re(e,t);if(n){if(n==="this"){return[Ee(e,t)]}else{const r=m(e,n);if(r.length===0){w('The selector "'+n+'" on '+t+" returned no matches!");return[ve]}else{return r}}}}function Ee(e,t){return ce(T(e,function(e){return te(ce(e),t)!=null}))}function Ce(e){const t=re(e,"hx-target");if(t){if(t==="this"){return Ee(e,"hx-target")}else{return fe(e,t)}}else{const n=ie(e);if(n.boosted){return ne().body}else{return e}}}function Re(t){const n=Q.config.attributesToSettle;for(let e=0;e0){s=e.substr(0,e.indexOf(":"));t=e.substr(e.indexOf(":")+1,e.length)}else{s=e}const n=ne().querySelectorAll(t);if(n){se(n,function(e){let t;const n=o.cloneNode(true);t=ne().createDocumentFragment();t.appendChild(n);if(!He(s,e)){t=h(n)}const r={shouldSwap:true,target:e,fragment:t};if(!he(e,"htmx:oobBeforeSwap",r))return;e=r.target;if(r.shouldSwap){_e(s,e,e,t,i)}se(i.elts,function(e){he(e,"htmx:oobAfterSwap",r)})});o.parentNode.removeChild(o)}else{o.parentNode.removeChild(o);ae(ne().body,"htmx:oobErrorNoTarget",{content:o})}return e}function qe(e){se(p(e,"[hx-preserve], [data-hx-preserve]"),function(e){const t=te(e,"id");const n=ne().getElementById(t);if(n!=null){e.parentNode.replaceChild(n,e)}})}function Le(l,e,u){se(e.querySelectorAll("[id]"),function(t){const n=ee(t,"id");if(n&&n.length>0){const r=n.replace("'","\\'");const o=t.tagName.replace(":","\\:");const e=h(l);const i=e&&e.querySelector(o+"[id='"+r+"']");if(i&&i!==e){const s=t.cloneNode();Oe(t,i);u.tasks.push(function(){Oe(t,s)})}}})}function Ne(e){return function(){o(e,Q.config.addedClass);kt(ce(e));Ae(h(e));he(e,"htmx:load")}}function Ae(e){const t="[autofocus]";const n=G(a(e,t)?e:e.querySelector(t));if(n!=null){n.focus()}}function c(e,t,n,r){Le(e,n,r);while(n.childNodes.length>0){const o=n.firstChild;Y(ce(o),Q.config.addedClass);e.insertBefore(o,t);if(o.nodeType!==Node.TEXT_NODE&&o.nodeType!==Node.COMMENT_NODE){r.tasks.push(Ne(o))}}}function Ie(e,t){let n=0;while(n0){E().setTimeout(l,r.settleDelay)}else{l()}}function Je(e,t,n){const r=e.getResponseHeader(t);if(r.indexOf("{")===0){const o=S(r);for(const i in o){if(o.hasOwnProperty(i)){let e=o[i];if(!X(e)){e={value:e}}he(n,i,e)}}}else{const s=r.split(",");for(let e=0;e0){const s=o[0];if(s==="]"){e--;if(e===0){if(n===null){t=t+"true"}o.shift();t+=")})";try{const l=vn(r,function(){return Function(t)()},function(){return true});l.source=t;return l}catch(e){ae(ne().body,"htmx:syntax:error",{error:e,source:t});return null}}}else if(s==="["){e++}if(nt(s,n,i)){t+="(("+i+"."+s+") ? ("+i+"."+s+") : (window."+s+"))"}else{t=t+s}n=o.shift()}}}function b(e,t){let n="";while(e.length>0&&!t.test(e[0])){n+=e.shift()}return n}function ot(e){let t;if(e.length>0&&Qe.test(e[0])){e.shift();t=b(e,et).trim();e.shift()}else{t=b(e,x)}return t}const it="input, textarea, select";function st(e,t,n){const r=[];const o=tt(t);do{b(o,We);const l=o.length;const u=b(o,/[,\[\s]/);if(u!==""){if(u==="every"){const c={trigger:"every"};b(o,We);c.pollInterval=d(b(o,/[,\[\s]/));b(o,We);var i=rt(e,o,"event");if(i){c.eventFilter=i}r.push(c)}else{const f={trigger:u};var i=rt(e,o,"event");if(i){f.eventFilter=i}while(o.length>0&&o[0]!==","){b(o,We);const a=o.shift();if(a==="changed"){f.changed=true}else if(a==="once"){f.once=true}else if(a==="consume"){f.consume=true}else if(a==="delay"&&o[0]===":"){o.shift();f.delay=d(b(o,x))}else if(a==="from"&&o[0]===":"){o.shift();if(Qe.test(o[0])){var s=ot(o)}else{var s=b(o,x);if(s==="closest"||s==="find"||s==="next"||s==="previous"){o.shift();const h=ot(o);if(h.length>0){s+=" "+h}}}f.from=s}else if(a==="target"&&o[0]===":"){o.shift();f.target=ot(o)}else if(a==="throttle"&&o[0]===":"){o.shift();f.throttle=d(b(o,x))}else if(a==="queue"&&o[0]===":"){o.shift();f.queue=b(o,x)}else if(a==="root"&&o[0]===":"){o.shift();f[a]=ot(o)}else if(a==="threshold"&&o[0]===":"){o.shift();f[a]=b(o,x)}else{ae(e,"htmx:syntax:error",{token:o.shift()})}}r.push(f)}}if(o.length===l){ae(e,"htmx:syntax:error",{token:o.shift()})}b(o,We)}while(o[0]===","&&o.shift());if(n){n[t]=r}return r}function lt(e){const t=te(e,"hx-trigger");let n=[];if(t){const r=Q.config.triggerSpecsCache;n=r&&r[t]||st(e,t,r)}if(n.length>0){return n}else if(a(e,"form")){return[{trigger:"submit"}]}else if(a(e,'input[type="button"], input[type="submit"]')){return[{trigger:"click"}]}else if(a(e,it)){return[{trigger:"change"}]}else{return[{trigger:"click"}]}}function ut(e){ie(e).cancelled=true}function ct(e,t,n){const r=ie(e);r.timeout=E().setTimeout(function(){if(le(e)&&r.cancelled!==true){if(!pt(n,e,Xt("hx:poll:trigger",{triggerSpec:n,target:e}))){t(e)}ct(e,t,n)}},n.pollInterval)}function ft(e){return location.hostname===e.hostname&&ee(e,"href")&&ee(e,"href").indexOf("#")!==0}function at(e){return g(e,Q.config.disableSelector)}function ht(t,n,e){if(t instanceof HTMLAnchorElement&&ft(t)&&(t.target===""||t.target==="_self")||t.tagName==="FORM"){n.boosted=true;let r,o;if(t.tagName==="A"){r="get";o=ee(t,"href")}else{const i=ee(t,"method");r=i?i.toLowerCase():"get";if(r==="get"){}o=ee(t,"action")}e.forEach(function(e){mt(t,function(e,t){const n=ce(e);if(at(n)){f(n);return}de(r,o,n,t)},n,e,true)})}}function dt(e,t){const n=ce(t);if(!n){return false}if(e.type==="submit"||e.type==="click"){if(n.tagName==="FORM"){return true}if(a(n,'input[type="submit"], button')&&g(n,"form")!==null){return true}if(n instanceof HTMLAnchorElement&&n.href&&(n.getAttribute("href")==="#"||n.getAttribute("href").indexOf("#")!==0)){return true}}return false}function gt(e,t){return ie(e).boosted&&e instanceof HTMLAnchorElement&&t.type==="click"&&(t.ctrlKey||t.metaKey)}function pt(e,t,n){const r=e.eventFilter;if(r){try{return r.call(t,n)!==true}catch(e){const o=r.source;ae(ne().body,"htmx:eventFilter:error",{error:e,source:o});return true}}return false}function mt(s,l,e,u,c){const f=ie(s);let t;if(u.from){t=m(s,u.from)}else{t=[s]}if(u.changed){t.forEach(function(e){const t=ie(e);t.lastValue=e.value})}se(t,function(o){const i=function(e){if(!le(s)){o.removeEventListener(u.trigger,i);return}if(gt(s,e)){return}if(c||dt(e,s)){e.preventDefault()}if(pt(u,s,e)){return}const t=ie(e);t.triggerSpec=u;if(t.handledFor==null){t.handledFor=[]}if(t.handledFor.indexOf(s)<0){t.handledFor.push(s);if(u.consume){e.stopPropagation()}if(u.target&&e.target){if(!a(ce(e.target),u.target)){return}}if(u.once){if(f.triggeredOnce){return}else{f.triggeredOnce=true}}if(u.changed){const n=ie(o);const r=o.value;if(n.lastValue===r){return}n.lastValue=r}if(f.delayed){clearTimeout(f.delayed)}if(f.throttle){return}if(u.throttle>0){if(!f.throttle){l(s,e);f.throttle=E().setTimeout(function(){f.throttle=null},u.throttle)}}else if(u.delay>0){f.delayed=E().setTimeout(function(){l(s,e)},u.delay)}else{he(s,"htmx:trigger");l(s,e)}}};if(e.listenerInfos==null){e.listenerInfos=[]}e.listenerInfos.push({trigger:u.trigger,listener:i,on:o});o.addEventListener(u.trigger,i)})}let yt=false;let xt=null;function bt(){if(!xt){xt=function(){yt=true};window.addEventListener("scroll",xt);setInterval(function(){if(yt){yt=false;se(ne().querySelectorAll("[hx-trigger*='revealed'],[data-hx-trigger*='revealed']"),function(e){wt(e)})}},200)}}function wt(e){if(!s(e,"data-hx-revealed")&&U(e)){e.setAttribute("data-hx-revealed","true");const t=ie(e);if(t.initHash){he(e,"revealed")}else{e.addEventListener("htmx:afterProcessNode",function(){he(e,"revealed")},{once:true})}}}function vt(e,t,n,r){const o=function(){if(!n.loaded){n.loaded=true;t(e)}};if(r>0){E().setTimeout(o,r)}else{o()}}function St(t,n,e){let i=false;se(v,function(r){if(s(t,"hx-"+r)){const o=te(t,"hx-"+r);i=true;n.path=o;n.verb=r;e.forEach(function(e){Et(t,e,n,function(e,t){const n=ce(e);if(g(n,Q.config.disableSelector)){f(n);return}de(r,o,n,t)})})}});return i}function Et(r,e,t,n){if(e.trigger==="revealed"){bt();mt(r,n,t,e);wt(ce(r))}else if(e.trigger==="intersect"){const o={};if(e.root){o.root=fe(r,e.root)}if(e.threshold){o.threshold=parseFloat(e.threshold)}const i=new IntersectionObserver(function(t){for(let e=0;e0){t.polling=true;ct(ce(r),n,e)}else{mt(r,n,t,e)}}function Ct(e){const t=ce(e);if(!t){return false}const n=t.attributes;for(let e=0;e", "+e).join(""));return o}else{return[]}}function qt(e){const t=g(ce(e.target),"button, input[type='submit']");const n=Nt(e);if(n){n.lastButtonClicked=t}}function Lt(e){const t=Nt(e);if(t){t.lastButtonClicked=null}}function Nt(e){const t=g(ce(e.target),"button, input[type='submit']");if(!t){return}const n=y("#"+ee(t,"form"),t.getRootNode())||g(t,"form");if(!n){return}return ie(n)}function At(e){e.addEventListener("click",qt);e.addEventListener("focusin",qt);e.addEventListener("focusout",Lt)}function It(t,e,n){const r=ie(t);if(!Array.isArray(r.onHandlers)){r.onHandlers=[]}let o;const i=function(e){vn(t,function(){if(at(t)){return}if(!o){o=new Function("event",n)}o.call(t,e)})};t.addEventListener(e,i);r.onHandlers.push({event:e,listener:i})}function Pt(t){De(t);for(let e=0;eQ.config.historyCacheSize){i.shift()}while(i.length>0){try{localStorage.setItem("htmx-history-cache",JSON.stringify(i));break}catch(e){ae(ne().body,"htmx:historyCacheError",{cause:e,cache:i});i.shift()}}}function _t(t){if(!j()){return null}t=V(t);const n=S(localStorage.getItem("htmx-history-cache"))||[];for(let e=0;e=200&&this.status<400){he(ne().body,"htmx:historyCacheMissLoad",i);const e=k(this.response);const t=e.querySelector("[hx-history-elt],[data-hx-history-elt]")||e;const n=jt();const r=xn(n);kn(e.title);Ve(n,t,r);Gt(r.tasks);Bt=o;he(ne().body,"htmx:historyRestore",{path:o,cacheMiss:true,serverResponse:this.response})}else{ae(ne().body,"htmx:historyCacheMissLoadError",i)}};e.send()}function Yt(e){zt();e=e||location.pathname+location.search;const t=_t(e);if(t){const n=k(t.content);const r=jt();const o=xn(r);kn(n.title);Ve(r,n,o);Gt(o.tasks);E().setTimeout(function(){window.scrollTo(0,t.scroll)},0);Bt=e;he(ne().body,"htmx:historyRestore",{path:e,item:t})}else{if(Q.config.refreshOnHistoryMiss){window.location.reload(true)}else{Zt(e)}}}function Wt(e){let t=Se(e,"hx-indicator");if(t==null){t=[e]}se(t,function(e){const t=ie(e);t.requestCount=(t.requestCount||0)+1;e.classList.add.call(e.classList,Q.config.requestClass)});return t}function Qt(e){let t=Se(e,"hx-disabled-elt");if(t==null){t=[]}se(t,function(e){const t=ie(e);t.requestCount=(t.requestCount||0)+1;e.setAttribute("disabled","")});return t}function en(e,t){se(e,function(e){const t=ie(e);t.requestCount=(t.requestCount||0)-1;if(t.requestCount===0){e.classList.remove.call(e.classList,Q.config.requestClass)}});se(t,function(e){const t=ie(e);t.requestCount=(t.requestCount||0)-1;if(t.requestCount===0){e.removeAttribute("disabled")}})}function tn(t,n){for(let e=0;en.indexOf(e)<0)}else{e=e.filter(e=>e!==n)}r.delete(t);se(e,e=>r.append(t,e))}}function sn(t,n,r,o,i){if(o==null||tn(t,o)){return}else{t.push(o)}if(nn(o)){const s=ee(o,"name");let e=o.value;if(o instanceof HTMLSelectElement&&o.multiple){e=F(o.querySelectorAll("option:checked")).map(function(e){return e.value})}if(o instanceof HTMLInputElement&&o.files){e=F(o.files)}rn(s,e,n);if(i){ln(o,r)}}if(o instanceof HTMLFormElement){se(o.elements,function(e){if(t.indexOf(e)>=0){on(e.name,e.value,n)}else{t.push(e)}if(i){ln(e,r)}});new FormData(o).forEach(function(e,t){if(e instanceof File&&e.name===""){return}rn(t,e,n)})}}function ln(e,t){const n=e;if(n.willValidate){he(n,"htmx:validation:validate");if(!n.checkValidity()){t.push({elt:n,message:n.validationMessage,validity:n.validity});he(n,"htmx:validation:failed",{message:n.validationMessage,validity:n.validity})}}}function un(t,e){for(const n of e.keys()){t.delete(n);e.getAll(n).forEach(function(e){t.append(n,e)})}return t}function cn(e,t){const n=[];const r=new FormData;const o=new FormData;const i=[];const s=ie(e);if(s.lastButtonClicked&&!le(s.lastButtonClicked)){s.lastButtonClicked=null}let l=e instanceof HTMLFormElement&&e.noValidate!==true||te(e,"hx-validate")==="true";if(s.lastButtonClicked){l=l&&s.lastButtonClicked.formNoValidate!==true}if(t!=="get"){sn(n,o,i,g(e,"form"),l)}sn(n,r,i,e,l);if(s.lastButtonClicked||e.tagName==="BUTTON"||e.tagName==="INPUT"&&ee(e,"type")==="submit"){const c=s.lastButtonClicked||e;const f=ee(c,"name");rn(f,c.value,o)}const u=Se(e,"hx-include");se(u,function(e){sn(n,r,i,ce(e),l);if(!a(e,"form")){se(h(e).querySelectorAll(it),function(e){sn(n,r,i,e,l)})}});un(r,o);return{errors:i,formData:r,values:An(r)}}function fn(e,t,n){if(e!==""){e+="&"}if(String(n)==="[object Object]"){n=JSON.stringify(n)}const r=encodeURIComponent(n);e+=encodeURIComponent(t)+"="+r;return e}function an(e){e=Ln(e);let n="";e.forEach(function(e,t){n=fn(n,t,e)});return n}function hn(e,t,n){const r={"HX-Request":"true","HX-Trigger":ee(e,"id"),"HX-Trigger-Name":ee(e,"name"),"HX-Target":te(t,"id"),"HX-Current-URL":ne().location.href};wn(e,"hx-headers",false,r);if(n!==undefined){r["HX-Prompt"]=n}if(ie(e).boosted){r["HX-Boosted"]="true"}return r}function dn(n,e){const t=re(e,"hx-params");if(t){if(t==="none"){return new FormData}else if(t==="*"){return n}else if(t.indexOf("not ")===0){se(t.substr(4).split(","),function(e){e=e.trim();n.delete(e)});return n}else{const r=new FormData;se(t.split(","),function(t){t=t.trim();if(n.has(t)){n.getAll(t).forEach(function(e){r.append(t,e)})}});return r}}else{return n}}function gn(e){return!!ee(e,"href")&&ee(e,"href").indexOf("#")>=0}function pn(e,t){const n=t||re(e,"hx-swap");const r={swapStyle:ie(e).boosted?"innerHTML":Q.config.defaultSwapStyle,swapDelay:Q.config.defaultSwapDelay,settleDelay:Q.config.defaultSettleDelay};if(Q.config.scrollIntoViewOnBoost&&ie(e).boosted&&!gn(e)){r.show="top"}if(n){const s=B(n);if(s.length>0){for(let e=0;e0?o.join(":"):null;r.scroll=c;r.scrollTarget=i}else if(l.indexOf("show:")===0){const f=l.substr(5);var o=f.split(":");const a=o.pop();var i=o.length>0?o.join(":"):null;r.show=a;r.showTarget=i}else if(l.indexOf("focus-scroll:")===0){const h=l.substr("focus-scroll:".length);r.focusScroll=h=="true"}else if(e==0){r.swapStyle=l}else{w("Unknown modifier in hx-swap: "+l)}}}}return r}function mn(e){return re(e,"hx-encoding")==="multipart/form-data"||a(e,"form")&&ee(e,"enctype")==="multipart/form-data"}function yn(t,n,r){let o=null;Ut(n,function(e){if(o==null){o=e.encodeParameters(t,r,n)}});if(o!=null){return o}else{if(mn(n)){return un(new FormData,Ln(r))}else{return an(r)}}}function xn(e){return{tasks:[],elts:[e]}}function bn(e,t){const n=e[0];const r=e[e.length-1];if(t.scroll){var o=null;if(t.scrollTarget){o=ce(fe(n,t.scrollTarget))}if(t.scroll==="top"&&(n||o)){o=o||n;o.scrollTop=0}if(t.scroll==="bottom"&&(r||o)){o=o||r;o.scrollTop=o.scrollHeight}}if(t.show){var o=null;if(t.showTarget){let e=t.showTarget;if(t.showTarget==="window"){e="body"}o=ce(fe(n,e))}if(t.show==="top"&&(n||o)){o=o||n;o.scrollIntoView({block:"start",behavior:Q.config.scrollBehavior})}if(t.show==="bottom"&&(r||o)){o=o||r;o.scrollIntoView({block:"end",behavior:Q.config.scrollBehavior})}}}function wn(r,e,o,i){if(i==null){i={}}if(r==null){return i}const s=te(r,e);if(s){let e=s.trim();let t=o;if(e==="unset"){return null}if(e.indexOf("javascript:")===0){e=e.substr(11);t=true}else if(e.indexOf("js:")===0){e=e.substr(3);t=true}if(e.indexOf("{")!==0){e="{"+e+"}"}let n;if(t){n=vn(r,function(){return Function("return ("+e+")")()},{})}else{n=S(e)}for(const l in n){if(n.hasOwnProperty(l)){if(i[l]==null){i[l]=n[l]}}}}return wn(ce(u(r)),e,o,i)}function vn(e,t,n){if(Q.config.allowEval){return t()}else{ae(e,"htmx:evalDisallowedError");return n}}function Sn(e,t){return wn(e,"hx-vars",true,t)}function En(e,t){return wn(e,"hx-vals",false,t)}function Cn(e){return ue(Sn(e),En(e))}function Rn(t,n,r){if(r!==null){try{t.setRequestHeader(n,r)}catch(e){t.setRequestHeader(n,encodeURIComponent(r));t.setRequestHeader(n+"-URI-AutoEncoded","true")}}}function On(t){if(t.responseURL&&typeof URL!=="undefined"){try{const e=new URL(t.responseURL);return e.pathname+e.search}catch(e){ae(ne().body,"htmx:badResponseUrl",{url:t.responseURL})}}}function C(e,t){return t.test(e.getAllResponseHeaders())}function Hn(e,t,n){e=e.toLowerCase();if(n){if(n instanceof Element||typeof n==="string"){return de(e,t,null,null,{targetOverride:y(n),returnPromise:true})}else{return de(e,t,y(n.source),n.event,{handler:n.handler,headers:n.headers,values:n.values,targetOverride:y(n.target),swapOverride:n.swap,select:n.select,returnPromise:true})}}else{return de(e,t,null,null,{returnPromise:true})}}function Tn(e){const t=[];while(e){t.push(e);e=e.parentElement}return t}function qn(e,t,n){let r;let o;if(typeof URL==="function"){o=new URL(t,document.location.href);const i=document.location.origin;r=i===o.origin}else{o=t;r=l(t,document.location.origin)}if(Q.config.selfRequestsOnly){if(!r){return false}}return he(e,"htmx:validateUrl",ue({url:o,sameHost:r},n))}function Ln(e){if(e instanceof FormData)return e;const t=new FormData;for(const n in e){if(e.hasOwnProperty(n)){if(typeof e[n].forEach==="function"){e[n].forEach(function(e){t.append(n,e)})}else if(typeof e[n]==="object"){t.append(n,JSON.stringify(e[n]))}else{t.append(n,e[n])}}}return t}function Nn(r,o,e){return new Proxy(e,{get:function(t,e){if(typeof e==="number")return t[e];if(e==="length")return t.length;if(e==="push"){return function(e){t.push(e);r.append(o,e)}}if(typeof t[e]==="function"){return function(){t[e].apply(t,arguments);r.delete(o);t.forEach(function(e){r.append(o,e)})}}if(t[e]&&t[e].length===1){return t[e][0]}else{return t[e]}},set:function(e,t,n){e[t]=n;r.delete(o);e.forEach(function(e){r.append(o,e)});return true}})}function An(r){return new Proxy(r,{get:function(e,t){if(typeof t==="symbol"){return Reflect.get(e,t)}if(t==="toJSON"){return()=>Object.fromEntries(r)}if(t in e){if(typeof e[t]==="function"){return function(){return r[t].apply(r,arguments)}}else{return e[t]}}const n=r.getAll(t);if(n.length===0){return undefined}else if(n.length===1){return n[0]}else{return Nn(e,t,n)}},set:function(t,n,e){if(typeof n!=="string"){return false}t.delete(n);if(typeof e.forEach==="function"){e.forEach(function(e){t.append(n,e)})}else{t.append(n,e)}return true},deleteProperty:function(e,t){if(typeof t==="string"){e.delete(t)}return true},ownKeys:function(e){return Reflect.ownKeys(Object.fromEntries(e))},getOwnPropertyDescriptor:function(e,t){return Reflect.getOwnPropertyDescriptor(Object.fromEntries(e),t)}})}function de(t,n,r,o,i,k){let s=null;let l=null;i=i!=null?i:{};if(i.returnPromise&&typeof Promise!=="undefined"){var e=new Promise(function(e,t){s=e;l=t})}if(r==null){r=ne().body}const M=i.handler||Mn;const X=i.select||null;if(!le(r)){oe(s);return e}const u=i.targetOverride||ce(Ce(r));if(u==null||u==ve){ae(r,"htmx:targetError",{target:te(r,"hx-target")});oe(l);return e}let c=ie(r);const f=c.lastButtonClicked;if(f){const L=ee(f,"formaction");if(L!=null){n=L}const N=ee(f,"formmethod");if(N!=null){if(N.toLowerCase()!=="dialog"){t=N}}}const a=re(r,"hx-confirm");if(k===undefined){const K=function(e){return de(t,n,r,o,i,!!e)};const G={target:u,elt:r,path:n,verb:t,triggeringEvent:o,etc:i,issueRequest:K,question:a};if(he(r,"htmx:confirm",G)===false){oe(s);return e}}let h=r;let d=re(r,"hx-sync");let g=null;let F=false;if(d){const A=d.split(":");const I=A[0].trim();if(I==="this"){h=Ee(r,"hx-sync")}else{h=ce(fe(r,I))}d=(A[1]||"drop").trim();c=ie(h);if(d==="drop"&&c.xhr&&c.abortable!==true){oe(s);return e}else if(d==="abort"){if(c.xhr){oe(s);return e}else{F=true}}else if(d==="replace"){he(h,"htmx:abort")}else if(d.indexOf("queue")===0){const Z=d.split(" ");g=(Z[1]||"last").trim()}}if(c.xhr){if(c.abortable){he(h,"htmx:abort")}else{if(g==null){if(o){const P=ie(o);if(P&&P.triggerSpec&&P.triggerSpec.queue){g=P.triggerSpec.queue}}if(g==null){g="last"}}if(c.queuedRequests==null){c.queuedRequests=[]}if(g==="first"&&c.queuedRequests.length===0){c.queuedRequests.push(function(){de(t,n,r,o,i)})}else if(g==="all"){c.queuedRequests.push(function(){de(t,n,r,o,i)})}else if(g==="last"){c.queuedRequests=[];c.queuedRequests.push(function(){de(t,n,r,o,i)})}oe(s);return e}}const p=new XMLHttpRequest;c.xhr=p;c.abortable=F;const m=function(){c.xhr=null;c.abortable=false;if(c.queuedRequests!=null&&c.queuedRequests.length>0){const e=c.queuedRequests.shift();e()}};const U=re(r,"hx-prompt");if(U){var y=prompt(U);if(y===null||!he(r,"htmx:prompt",{prompt:y,target:u})){oe(s);m();return e}}if(a&&!k){if(!confirm(a)){oe(s);m();return e}}let x=hn(r,u,y);if(t!=="get"&&!mn(r)){x["Content-Type"]="application/x-www-form-urlencoded"}if(i.headers){x=ue(x,i.headers)}const B=cn(r,t);let b=B.errors;const j=B.formData;if(i.values){un(j,Ln(i.values))}const V=Ln(Cn(r));const w=un(j,V);let v=dn(w,r);if(Q.config.getCacheBusterParam&&t==="get"){v.set("org.htmx.cache-buster",ee(u,"id")||"true")}if(n==null||n===""){n=ne().location.href}const S=wn(r,"hx-request");const _=ie(r).boosted;let E=Q.config.methodsThatUseUrlParams.indexOf(t)>=0;const C={boosted:_,useUrlParams:E,formData:v,parameters:An(v),unfilteredFormData:w,unfilteredParameters:An(w),headers:x,target:u,verb:t,errors:b,withCredentials:i.credentials||S.credentials||Q.config.withCredentials,timeout:i.timeout||S.timeout||Q.config.timeout,path:n,triggeringEvent:o};if(!he(r,"htmx:configRequest",C)){oe(s);m();return e}n=C.path;t=C.verb;x=C.headers;v=Ln(C.parameters);b=C.errors;E=C.useUrlParams;if(b&&b.length>0){he(r,"htmx:validation:halted",C);oe(s);m();return e}const $=n.split("#");const z=$[0];const R=$[1];let O=n;if(E){O=z;const Y=!v.keys().next().done;if(Y){if(O.indexOf("?")<0){O+="?"}else{O+="&"}O+=an(v);if(R){O+="#"+R}}}if(!qn(r,O,C)){ae(r,"htmx:invalidPath",C);oe(l);return e}p.open(t.toUpperCase(),O,true);p.overrideMimeType("text/html");p.withCredentials=C.withCredentials;p.timeout=C.timeout;if(S.noHeaders){}else{for(const D in x){if(x.hasOwnProperty(D)){const W=x[D];Rn(p,D,W)}}}const H={xhr:p,target:u,requestConfig:C,etc:i,boosted:_,select:X,pathInfo:{requestPath:n,finalRequestPath:O,responsePath:null,anchor:R}};p.onload=function(){try{const t=Tn(r);H.pathInfo.responsePath=On(p);M(r,H);en(T,q);he(r,"htmx:afterRequest",H);he(r,"htmx:afterOnLoad",H);if(!le(r)){let e=null;while(t.length>0&&e==null){const n=t.shift();if(le(n)){e=n}}if(e){he(e,"htmx:afterRequest",H);he(e,"htmx:afterOnLoad",H)}}oe(s);m()}catch(e){ae(r,"htmx:onLoadError",ue({error:e},H));throw e}};p.onerror=function(){en(T,q);ae(r,"htmx:afterRequest",H);ae(r,"htmx:sendError",H);oe(l);m()};p.onabort=function(){en(T,q);ae(r,"htmx:afterRequest",H);ae(r,"htmx:sendAbort",H);oe(l);m()};p.ontimeout=function(){en(T,q);ae(r,"htmx:afterRequest",H);ae(r,"htmx:timeout",H);oe(l);m()};if(!he(r,"htmx:beforeRequest",H)){oe(s);m();return e}var T=Wt(r);var q=Qt(r);se(["loadstart","loadend","progress","abort"],function(t){se([p,p.upload],function(e){e.addEventListener(t,function(e){he(r,"htmx:xhr:"+t,{lengthComputable:e.lengthComputable,loaded:e.loaded,total:e.total})})})});he(r,"htmx:beforeSend",H);const J=E?null:yn(p,r,v);p.send(J);return e}function In(e,t){const n=t.xhr;let r=null;let o=null;if(C(n,/HX-Push:/i)){r=n.getResponseHeader("HX-Push");o="push"}else if(C(n,/HX-Push-Url:/i)){r=n.getResponseHeader("HX-Push-Url");o="push"}else if(C(n,/HX-Replace-Url:/i)){r=n.getResponseHeader("HX-Replace-Url");o="replace"}if(r){if(r==="false"){return{}}else{return{type:o,path:r}}}const i=t.pathInfo.finalRequestPath;const s=t.pathInfo.responsePath;const l=re(e,"hx-push-url");const u=re(e,"hx-replace-url");const c=ie(e).boosted;let f=null;let a=null;if(l){f="push";a=l}else if(u){f="replace";a=u}else if(c){f="push";a=s||i}if(a){if(a==="false"){return{}}if(a==="true"){a=s||i}if(t.pathInfo.anchor&&a.indexOf("#")===-1){a=a+"#"+t.pathInfo.anchor}return{type:f,path:a}}else{return{}}}function Pn(e,t){var n=new RegExp(e.code);return n.test(t.toString(10))}function Dn(e){for(var t=0;t0){E().setTimeout(e,y.swapDelay)}else{e()}}if(a){ae(o,"htmx:responseError",ue({error:"Response Status Error Code "+s.status+" from "+i.pathInfo.requestPath},i))}}const Xn={};function Fn(){return{init:function(e){return null},getSelectors:function(){return null},onEvent:function(e,t){return true},transformResponse:function(e,t,n){return e},isInlineSwap:function(e){return false},handleSwap:function(e,t,n,r){return false},encodeParameters:function(e,t,n){return null}}}function Un(e,t){if(t.init){t.init(n)}Xn[e]=ue(Fn(),t)}function Bn(e){delete Xn[e]}function jn(e,n,r){if(n==undefined){n=[]}if(e==undefined){return n}if(r==undefined){r=[]}const t=te(e,"hx-ext");if(t){se(t.split(","),function(e){e=e.replace(/ /g,"");if(e.slice(0,7)=="ignore:"){r.push(e.slice(7));return}if(r.indexOf(e)<0){const t=Xn[e];if(t&&n.indexOf(t)<0){n.push(t)}}})}return jn(ce(u(e)),n,r)}var Vn=false;ne().addEventListener("DOMContentLoaded",function(){Vn=true});function _n(e){if(Vn||ne().readyState==="complete"){e()}else{ne().addEventListener("DOMContentLoaded",e)}}function $n(){if(Q.config.includeIndicatorStyles!==false){const e=Q.config.inlineStyleNonce?` nonce="${Q.config.inlineStyleNonce}"`:"";ne().head.insertAdjacentHTML("beforeend","")}}function zn(){const e=ne().querySelector('meta[name="htmx-config"]');if(e){return S(e.content)}else{return null}}function Jn(){const e=zn();if(e){Q.config=ue(Q.config,e)}}_n(function(){Jn();$n();let e=ne().body;kt(e);const t=ne().querySelectorAll("[hx-trigger='restored'],[data-hx-trigger='restored']");e.addEventListener("htmx:abort",function(e){const t=e.target;const n=ie(t);if(n&&n.xhr){n.xhr.abort()}});const n=window.onpopstate?window.onpopstate.bind(window):null;window.onpopstate=function(e){if(e.state&&e.state.htmx){Yt();se(t,function(e){he(e,"htmx:restored",{document:ne(),triggerEvent:he})})}else{if(n){n(e)}}};E().setTimeout(function(){he(e,"htmx:load",{});e=null},0)});return Q}();
\ No newline at end of file
diff --git a/frontend/public/tailwind.css b/frontend/public/tailwind.css
index c52b8bb3..e7d2a3db 100644
--- a/frontend/public/tailwind.css
+++ b/frontend/public/tailwind.css
@@ -556,6 +556,40 @@ video {
--tw-backdrop-sepia: ;
}
+.container {
+ width: 100%;
+}
+
+@media (min-width: 640px) {
+ .container {
+ max-width: 640px;
+ }
+}
+
+@media (min-width: 768px) {
+ .container {
+ max-width: 768px;
+ }
+}
+
+@media (min-width: 1024px) {
+ .container {
+ max-width: 1024px;
+ }
+}
+
+@media (min-width: 1280px) {
+ .container {
+ max-width: 1280px;
+ }
+}
+
+@media (min-width: 1536px) {
+ .container {
+ max-width: 1536px;
+ }
+}
+
.invisible {
visibility: hidden;
}
@@ -663,6 +697,11 @@ video {
display: none;
}
+.size-6 {
+ width: 1.5rem;
+ height: 1.5rem;
+}
+
.h-5 {
height: 1.25rem;
}
@@ -699,18 +738,22 @@ video {
width: 2.5rem;
}
-.w-12 {
- width: 3rem;
-}
-
.w-14 {
width: 3.5rem;
}
+.w-24 {
+ width: 6rem;
+}
+
.w-36 {
width: 9rem;
}
+.w-44 {
+ width: 11rem;
+}
+
.w-5 {
width: 1.25rem;
}
@@ -735,22 +778,6 @@ video {
width: 100%;
}
-.w-32 {
- width: 8rem;
-}
-
-.w-20 {
- width: 5rem;
-}
-
-.w-28 {
- width: 7rem;
-}
-
-.w-24 {
- width: 6rem;
-}
-
.flex-grow {
flex-grow: 1;
}
@@ -784,6 +811,11 @@ video {
transform: translate(var(--tw-translate-x), var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y));
}
+.-translate-y-\[22px\] {
+ --tw-translate-y: -22px;
+ transform: translate(var(--tw-translate-x), var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y));
+}
+
.-translate-y-\[9px\] {
--tw-translate-y: -9px;
transform: translate(var(--tw-translate-x), var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y));
@@ -892,10 +924,18 @@ video {
overflow: auto;
}
+.overflow-scroll {
+ overflow: scroll;
+}
+
.overflow-x-auto {
overflow-x: auto;
}
+.overflow-x-scroll {
+ overflow-x: scroll;
+}
+
.whitespace-nowrap {
white-space: nowrap;
}
@@ -978,6 +1018,11 @@ video {
background-color: rgb(59 130 246 / var(--tw-bg-opacity));
}
+.bg-cyan-50 {
+ --tw-bg-opacity: 1;
+ background-color: rgb(236 254 255 / var(--tw-bg-opacity));
+}
+
.bg-gray-100 {
--tw-bg-opacity: 1;
background-color: rgb(243 244 246 / var(--tw-bg-opacity));
@@ -1105,6 +1150,10 @@ video {
padding-bottom: 0.75rem;
}
+.pb-6 {
+ padding-bottom: 1.5rem;
+}
+
.pt-0 {
padding-top: 0px;
}
@@ -1176,6 +1225,11 @@ video {
color: rgb(29 78 216 / var(--tw-text-opacity));
}
+.text-cyan-700 {
+ --tw-text-opacity: 1;
+ color: rgb(14 116 144 / var(--tw-text-opacity));
+}
+
.text-gray-500 {
--tw-text-opacity: 1;
color: rgb(107 114 128 / var(--tw-text-opacity));
@@ -1221,6 +1275,10 @@ video {
color: rgb(234 179 8 / var(--tw-text-opacity));
}
+.accent-yellow-400 {
+ accent-color: #facc15;
+}
+
.opacity-0 {
opacity: 0;
}
@@ -1263,6 +1321,12 @@ video {
box-shadow: var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow, 0 0 #0000);
}
+.ring-2 {
+ --tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);
+ --tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);
+ box-shadow: var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow, 0 0 #0000);
+}
+
.ring-inset {
--tw-ring-inset: inset;
}
@@ -1271,6 +1335,10 @@ video {
--tw-ring-color: rgb(29 78 216 / 0.1);
}
+.ring-cyan-700\/10 {
+ --tw-ring-color: rgb(14 116 144 / 0.1);
+}
+
.ring-gray-500\/10 {
--tw-ring-color: rgb(107 114 128 / 0.1);
}
@@ -1279,6 +1347,11 @@ video {
--tw-ring-color: rgb(21 128 61 / 0.1);
}
+.ring-neutral-500 {
+ --tw-ring-opacity: 1;
+ --tw-ring-color: rgb(115 115 115 / var(--tw-ring-opacity));
+}
+
.ring-red-700\/10 {
--tw-ring-color: rgb(185 28 28 / 0.1);
}
@@ -1375,6 +1448,10 @@ samp {
border: 2px solid #ffffff;
}
+.\*\:border > * {
+ border-width: 1px;
+}
+
.\*\:border-b > * {
border-bottom-width: 1px;
}
@@ -1384,6 +1461,11 @@ samp {
border-color: rgb(229 231 235 / var(--tw-border-opacity));
}
+.\*\:border-neutral-300 > * {
+ --tw-border-opacity: 1;
+ border-color: rgb(212 212 212 / var(--tw-border-opacity));
+}
+
.\*\:p-1 > * {
padding: 0.25rem;
}
@@ -1392,6 +1474,16 @@ samp {
padding: 0.75rem;
}
+.\*\:px-2 > * {
+ padding-left: 0.5rem;
+ padding-right: 0.5rem;
+}
+
+.\*\:py-1 > * {
+ padding-top: 0.25rem;
+ padding-bottom: 0.25rem;
+}
+
.\*\:text-sm > * {
font-size: 0.875rem;
line-height: 1.25rem;
@@ -1470,6 +1562,17 @@ samp {
color: rgb(239 68 68 / var(--tw-text-opacity));
}
+.focus\:ring-2:focus {
+ --tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);
+ --tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);
+ box-shadow: var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow, 0 0 #0000);
+}
+
+.focus\:ring-yellow-400:focus {
+ --tw-ring-opacity: 1;
+ --tw-ring-color: rgb(250 204 21 / var(--tw-ring-opacity));
+}
+
.active\:scale-90:active {
--tw-scale-x: .9;
--tw-scale-y: .9;
@@ -1532,6 +1635,11 @@ samp {
.sm\:grid-cols-2 {
grid-template-columns: repeat(2, minmax(0, 1fr));
}
+
+ .sm\:text-base {
+ font-size: 1rem;
+ line-height: 1.5rem;
+ }
}
@media (min-width: 768px) {
diff --git a/frontend/server.go b/frontend/server.go
index 8b5bf870..2d59bfa4 100644
--- a/frontend/server.go
+++ b/frontend/server.go
@@ -133,6 +133,11 @@ func (s *Server) Serve() error {
}
return Items
},
+ "StrAdd": func(x, y string) int {
+ a, _ := strconv.Atoi(x)
+ b, _ := strconv.Atoi(y)
+ return a + b
+ },
}
views, err := fs.Sub(staticFS, "views")
diff --git a/frontend/views/base.gohtml b/frontend/views/base.gohtml
index 4f434427..fa0024c7 100644
--- a/frontend/views/base.gohtml
+++ b/frontend/views/base.gohtml
@@ -41,6 +41,12 @@
hx-boost="true">
Floorplan
+
+ Nodes
+
{{ if eq .Auth.Role "admin" }}
{{ if eq .Auth.User nil }}
-
Login
+
Login
{{ else }}
diff --git a/frontend/views/fragments/rack/actions.gohtml b/frontend/views/fragments/actions.gohtml
similarity index 100%
rename from frontend/views/fragments/rack/actions.gohtml
rename to frontend/views/fragments/actions.gohtml
diff --git a/frontend/views/fragments/bonds.gohtml b/frontend/views/fragments/bonds.gohtml
new file mode 100644
index 00000000..5ad213ed
--- /dev/null
+++ b/frontend/views/fragments/bonds.gohtml
@@ -0,0 +1,80 @@
+
+
+ Bond:
+
+
+ FQDN:
+
+
+
+
Peers:
+
+
+
+ {{ template "icon-chevron-up" }}
+
+
+ {{ template "icon-chevron-down" }}
+
+
+ {{ template "icon-x-mark" }}
+
+
+
+
+ Mac:
+
+
+
+ Ip:
+
+
+
+ Ifname:
+
+
+
+ BMC:
+
+ true
+ false
+
+
+
+ Vlan:
+
+
+
+ Mtu:
+
+
+
diff --git a/frontend/views/fragments/host/form.gohtml b/frontend/views/fragments/host/form.gohtml
index d87d8960..089d6619 100644
--- a/frontend/views/fragments/host/form.gohtml
+++ b/frontend/views/fragments/host/form.gohtml
@@ -1,4 +1,8 @@
-