diff --git a/.gitignore b/.gitignore
index 1de340b..276ae74 100644
--- a/.gitignore
+++ b/.gitignore
@@ -366,3 +366,5 @@ nul
tmpclaude*
PlanTempus.Application/tmpclaude*
+
+PlanTempus.Application/wwwroot/js/app.js
diff --git a/PlanTempus.Application/wwwroot/css/COMPONENT-CATALOG.md b/PlanTempus.Application/wwwroot/css/COMPONENT-CATALOG.md
index 66dea9e..aca7a8c 100644
--- a/PlanTempus.Application/wwwroot/css/COMPONENT-CATALOG.md
+++ b/PlanTempus.Application/wwwroot/css/COMPONENT-CATALOG.md
@@ -8,45 +8,40 @@ Reference for alle genbrugelige komponenter. **LAV ALDRIG EN NY KOMPONENT HVIS D
**VIGTIGT:** Disse base patterns skal ALTID bruges som foundation for nye features.
-### Grid + Subgrid Table Pattern
+### Data Table Pattern (ANBEFALET)
-Alle tabeller skal bruge dette pattern:
+Alle nye tabeller skal bruge `swp-data-table` fra components.css:
```html
-
-
- Kolonne 1
-
-
-
- Data
-
-
-
+
+
+
+ Kolonne 1
+ Kolonne 2
+
+
+ Data 1
+ Data 2
+
+
+
```
**CSS Pattern:**
```css
-swp-feature-table {
- display: grid;
- grid-template-columns: /* feature-specific */;
+/* I feature CSS - definer kun kolonner via context class */
+swp-card.feature-context swp-data-table {
+ grid-template-columns: 1fr 120px 80px;
}
-swp-feature-table-header,
-swp-feature-table-body {
- display: grid;
- grid-column: 1 / -1;
- grid-template-columns: subgrid;
-}
-
-swp-feature-row {
- display: grid;
- grid-column: 1 / -1;
- grid-template-columns: subgrid;
- align-items: center;
+/* Kolonne-specifik styling med nth-child */
+swp-card.feature-context swp-data-table-cell:nth-child(2) {
+ font-family: var(--font-mono);
}
```
+**VIGTIGT:** Context class skal være på `swp-card`, IKKE en wrapper div!
+
### List Item Pattern
Alle lister (notifikationer, bookinger, attentions) bruger:
@@ -211,57 +206,56 @@ Automatisk dot via `::before` pseudo-element.
---
-## Tables - Grid + Subgrid Pattern
+## Tables - swp-data-table Pattern
-### Struktur (ALTID følg dette mønster)
+**BRUG ALTID `swp-data-table`** for nye tabeller. Den generiske komponent er defineret i components.css.
+
+### Struktur
```html
-
-
- Kolonne 1
- Kolonne 2
-
-
-
- Data 1
- Data 2
-
-
-
+
+
+
+ Kolonne 1
+ Kolonne 2
+
+
+ Data 1
+ Data 2
+
+
+
```
### CSS Pattern
```css
-swp-[feature]-table {
- display: grid;
- grid-template-columns: /* definer kolonner her */;
+/* Definer kolonner via context class på swp-card */
+swp-card.context-class swp-data-table {
+ grid-template-columns: 1fr 120px 80px;
}
-swp-[feature]-table-header,
-swp-[feature]-table-body {
- display: grid;
- grid-column: 1 / -1;
- grid-template-columns: subgrid;
-}
-
-swp-[feature]-row {
- display: grid;
- grid-column: 1 / -1;
- grid-template-columns: subgrid;
- align-items: center;
+/* Kolonne-specifik styling med nth-child */
+swp-card.context-class swp-data-table-cell:nth-child(2) {
+ font-family: var(--font-mono);
}
```
### Eksisterende tabeller
-| Feature | Container | Row | CSS fil |
-|---------|-----------|-----|---------|
-| Cash | `swp-cash-table` | `swp-cash-table-row` | cash.css |
-| Employees | `swp-employee-table` | `swp-employee-row` | employees.css |
-| Salary | `swp-salary-table` | `swp-salary-table-row` | employees.css |
-| **Data (generisk)** | `swp-data-table` | `swp-data-table-row` | components.css |
-| Bookings (dashboard) | `swp-booking-list` | `swp-booking-item` | bookings.css |
+| Feature | Card Class | CSS fil |
+|---------|------------|---------|
+| Employees list | `swp-card.employees-list` | employees.css |
+| Salary history | `swp-card.salary-history` | employees.css |
+| Invoice history | `swp-card.invoice-history` | account.css |
+| Stats bookings | `swp-card.stats-bookings` | employees.css |
+| Cash | `swp-cash-table` (kompleks) | cash.css |
+
+**Lister (ikke tabeller):**
+
+| Feature | Container | Item | CSS fil |
+|---------|-----------|------|---------|
+| Bookings | `swp-booking-list` | `swp-booking-item` | bookings.css |
| Notifications | `swp-notification-list` | `swp-notification-item` | notifications.css |
| Attentions | `swp-attention-list` | `swp-attention-item` | attentions.css |
@@ -269,22 +263,28 @@ swp-[feature]-row {
## Table Cells - Standard Styling
+Base styling er i components.css. Tilpas kun via context class:
+
```css
-/* Header cells */
-swp-[feature]-table-header swp-[feature]-cell {
+/* Header cells (automatisk fra components.css) */
+swp-data-table-header swp-data-table-cell {
font-size: var(--font-size-xs);
font-weight: var(--font-weight-semibold);
text-transform: uppercase;
- letter-spacing: 0.5px;
color: var(--color-text-secondary);
}
-/* Body cells */
-swp-[feature]-cell {
- padding: var(--spacing-5);
- font-size: var(--font-size-base); /* ALTID base, ikke sm */
+/* Body cells (automatisk fra components.css) */
+swp-data-table-cell {
+ padding: var(--spacing-4);
+ font-size: var(--font-size-base);
color: var(--color-text);
}
+
+/* Feature-specifik tilpasning via context */
+swp-card.my-feature swp-data-table-cell:nth-child(3) {
+ text-align: right;
+}
```
---
@@ -428,73 +428,29 @@ Simpel liste med tekst + badge (f.eks. planlagt fravær).
---
-## Salary Table (employees.css)
+## Salary History Table (employees.css)
-Bruger Grid + Subgrid mønsteret.
+Bruger `swp-data-table` med `.salary-history` context class.
```html
-
-
- Periode
- Bruttoløn
-
-
-
-
- Januar 2026
- 34.063,50 kr
-
-
-
-
-```
-
-Rækker har hover-effekt og chevron bliver teal ved hover.
-
----
-
-## Data Table - Generisk (components.css)
-
-Generisk tabel med Grid + Subgrid.
-
-**Struktur:**
-- `swp-data-table` = grid (kolonner i feature CSS)
-- `swp-data-table-header` = subgrid (celler direkte i)
-- `swp-data-table-row` = subgrid
-- `swp-data-table-cell` = celler
-
-**Brug:** Wrap i container med klasse der definerer kolonner.
-
-```css
-/* I feature CSS (f.eks. employees.css) */
-.stats-bookings swp-data-table {
- grid-template-columns: 90px 60px 1fr 1fr 80px 100px 100px;
-}
-
-/* Kolonne-specifik styling med nth-child */
-.stats-bookings swp-data-table-row swp-data-table-cell:nth-child(1),
-.stats-bookings swp-data-table-row swp-data-table-cell:nth-child(2) {
- font-family: var(--font-mono);
- color: var(--color-text-secondary);
-}
-```
-
-```html
-
+
+ Lønhistorik
- Kolonne 1
- Kolonne 2
+ Periode
+ Bruttoløn
+
- Data 1
- Data 2
+ Januar 2026
+ 34.063,50 kr
+
```
-**Kolonne-styling:** Brug `nth-child()` i feature CSS frem for klasser på celler.
+Rækker har hover-effekt og chevron bliver teal ved hover.
---
@@ -579,11 +535,13 @@ Dashed border knap til tilføjelse af elementer.
|-----|---------|
| `design-tokens.css` | Farver, spacing, fonts, shadows |
| `design-system.css` | Base resets, typography |
-| `page.css` | Page structure |
-| `components.css` | Buttons, badges, cards, section-label, add-button, avatars, icon-btn, data-table |
+| `page.css` | Page structure, sticky-header |
+| `components.css` | Buttons, badges, cards, section-label, add-button, avatars, icon-btn, **swp-data-table** |
| `stats.css` | Stat cards, stat rows |
| `tabs.css` | Tab bar, tab content |
-| `employees.css` | Employee table, user info, edit forms, document lists, salary table |
+| `employees.css` | User info, edit forms, document lists, context styles (.employees-list, .salary-history, .stats-bookings) |
+| `account.css` | Account/billing styles, context styles (.invoice-history) |
| `bookings.css` | Booking list items |
| `notifications.css` | Notification items |
| `attentions.css` | Attention items |
+| `cash.css` | Cash register (swp-cash-table - kompleks, ikke migreret) |
diff --git a/PlanTempus.Application/wwwroot/css/employees.css b/PlanTempus.Application/wwwroot/css/employees.css
index 205dd75..f724456 100644
--- a/PlanTempus.Application/wwwroot/css/employees.css
+++ b/PlanTempus.Application/wwwroot/css/employees.css
@@ -814,6 +814,23 @@ swp-simple-item {
}
}
+/* ===========================================
+ FOCUS HIGHLIGHT (double-click to edit)
+ =========================================== */
+@keyframes focus-blink {
+ 0%, 100% { background: transparent; }
+ 25%, 75% { background: var(--bg-teal-medium); }
+}
+
+swp-data-row.focus-highlight {
+ animation: focus-blink 1s ease-out;
+
+ & input {
+ outline: 2px solid var(--color-teal);
+ outline-offset: 2px;
+ }
+}
+
/* ===========================================
STATS BOOKINGS TABLE
=========================================== */
diff --git a/PlanTempus.Application/wwwroot/js/app.js b/PlanTempus.Application/wwwroot/js/app.js
deleted file mode 100644
index d554ee1..0000000
--- a/PlanTempus.Application/wwwroot/js/app.js
+++ /dev/null
@@ -1,1176 +0,0 @@
-var __defProp = Object.defineProperty;
-var __name = (target, value) => __defProp(target, "name", { value, configurable: true });
-
-// wwwroot/ts/modules/sidebar.ts
-var _SidebarController = class _SidebarController {
- constructor() {
- this.menuToggle = null;
- this.appLayout = null;
- this.menuTooltip = null;
- this.menuToggle = document.getElementById("menuToggle");
- this.appLayout = document.querySelector("swp-app-layout");
- this.menuTooltip = document.getElementById("menuTooltip");
- this.setupListeners();
- this.setupTooltips();
- this.restoreState();
- }
- /**
- * Check if sidebar is collapsed
- */
- get isCollapsed() {
- return this.appLayout?.classList.contains("menu-collapsed") ?? false;
- }
- /**
- * Toggle sidebar collapsed state
- */
- toggle() {
- if (!this.appLayout) return;
- this.appLayout.classList.toggle("menu-collapsed");
- localStorage.setItem("sidebar-collapsed", String(this.isCollapsed));
- }
- /**
- * Collapse the sidebar
- */
- collapse() {
- this.appLayout?.classList.add("menu-collapsed");
- localStorage.setItem("sidebar-collapsed", "true");
- }
- /**
- * Expand the sidebar
- */
- expand() {
- this.appLayout?.classList.remove("menu-collapsed");
- localStorage.setItem("sidebar-collapsed", "false");
- }
- setupListeners() {
- this.menuToggle?.addEventListener("click", () => this.toggle());
- }
- setupTooltips() {
- if (!this.menuTooltip) return;
- const menuItems = document.querySelectorAll("swp-side-menu-item[data-tooltip]");
- menuItems.forEach((item) => {
- item.addEventListener("mouseenter", () => this.showTooltip(item));
- item.addEventListener("mouseleave", () => this.hideTooltip());
- });
- }
- showTooltip(item) {
- if (!this.isCollapsed || !this.menuTooltip) return;
- const rect = item.getBoundingClientRect();
- const tooltipText = item.dataset.tooltip;
- if (!tooltipText) return;
- this.menuTooltip.textContent = tooltipText;
- this.menuTooltip.style.left = `${rect.right + 8}px`;
- this.menuTooltip.style.top = `${rect.top + rect.height / 2}px`;
- this.menuTooltip.style.transform = "translateY(-50%)";
- this.menuTooltip.showPopover();
- }
- hideTooltip() {
- this.menuTooltip?.hidePopover();
- }
- restoreState() {
- if (!this.appLayout) return;
- if (localStorage.getItem("sidebar-collapsed") === "true") {
- this.appLayout.classList.add("menu-collapsed");
- }
- }
-};
-__name(_SidebarController, "SidebarController");
-var SidebarController = _SidebarController;
-
-// wwwroot/ts/modules/drawers.ts
-var _DrawerController = class _DrawerController {
- constructor() {
- this.profileDrawer = null;
- this.notificationDrawer = null;
- this.todoDrawer = null;
- this.newTodoDrawer = null;
- this.overlay = null;
- this.activeDrawer = null;
- this.activeGenericDrawer = null;
- this.profileDrawer = document.getElementById("profileDrawer");
- this.notificationDrawer = document.getElementById("notificationDrawer");
- this.todoDrawer = document.getElementById("todoDrawer");
- this.newTodoDrawer = document.getElementById("newTodoDrawer");
- this.overlay = document.getElementById("drawerOverlay");
- this.setupListeners();
- this.setupGenericDrawers();
- }
- /**
- * Get currently active drawer name
- */
- get active() {
- return this.activeDrawer;
- }
- /**
- * Open a drawer by name
- */
- open(name) {
- this.closeAll();
- const drawer = this.getDrawer(name);
- if (drawer && this.overlay) {
- drawer.classList.add("active");
- this.overlay.classList.add("active");
- document.body.style.overflow = "hidden";
- this.activeDrawer = name;
- }
- }
- /**
- * Close a specific drawer
- */
- close(name) {
- const drawer = this.getDrawer(name);
- drawer?.classList.remove("active");
- if (this.overlay && !document.querySelector('.active[class*="drawer"]')) {
- this.overlay.classList.remove("active");
- document.body.style.overflow = "";
- }
- if (this.activeDrawer === name) {
- this.activeDrawer = null;
- }
- }
- /**
- * Close all drawers
- */
- closeAll() {
- [this.profileDrawer, this.notificationDrawer, this.todoDrawer, this.newTodoDrawer].forEach((drawer) => drawer?.classList.remove("active"));
- this.closeGenericDrawer();
- this.overlay?.classList.remove("active");
- document.body.style.overflow = "";
- this.activeDrawer = null;
- }
- /**
- * Open a generic drawer by ID
- */
- openGenericDrawer(drawerId) {
- this.closeAll();
- const drawer = document.getElementById(drawerId);
- if (drawer && this.overlay) {
- drawer.classList.add("open");
- this.overlay.classList.add("active");
- document.body.style.overflow = "hidden";
- this.activeGenericDrawer = drawer;
- }
- }
- /**
- * Close the currently open generic drawer
- */
- closeGenericDrawer() {
- this.activeGenericDrawer?.classList.remove("open");
- this.activeGenericDrawer = null;
- }
- /**
- * Open profile drawer
- */
- openProfile() {
- this.open("profile");
- }
- /**
- * Open notification drawer
- */
- openNotification() {
- this.open("notification");
- }
- /**
- * Open todo drawer (slides on top of profile)
- */
- openTodo() {
- this.todoDrawer?.classList.add("active");
- }
- /**
- * Close todo drawer
- */
- closeTodo() {
- this.todoDrawer?.classList.remove("active");
- this.closeNewTodo();
- }
- /**
- * Open new todo drawer
- */
- openNewTodo() {
- this.newTodoDrawer?.classList.add("active");
- }
- /**
- * Close new todo drawer
- */
- closeNewTodo() {
- this.newTodoDrawer?.classList.remove("active");
- }
- /**
- * Mark all notifications as read
- */
- markAllNotificationsRead() {
- if (!this.notificationDrawer) return;
- const unreadItems = this.notificationDrawer.querySelectorAll(
- 'swp-notification-item[data-unread="true"]'
- );
- unreadItems.forEach((item) => item.removeAttribute("data-unread"));
- const badge = document.querySelector("swp-notification-badge");
- if (badge) {
- badge.style.display = "none";
- }
- }
- getDrawer(name) {
- switch (name) {
- case "profile":
- return this.profileDrawer;
- case "notification":
- return this.notificationDrawer;
- case "todo":
- return this.todoDrawer;
- case "newTodo":
- return this.newTodoDrawer;
- }
- }
- setupListeners() {
- document.getElementById("profileTrigger")?.addEventListener("click", () => this.openProfile());
- document.getElementById("drawerClose")?.addEventListener("click", () => this.close("profile"));
- document.getElementById("notificationsBtn")?.addEventListener("click", () => this.openNotification());
- document.getElementById("notificationDrawerClose")?.addEventListener("click", () => this.close("notification"));
- document.getElementById("markAllRead")?.addEventListener("click", () => this.markAllNotificationsRead());
- document.getElementById("openTodoDrawer")?.addEventListener("click", () => this.openTodo());
- document.getElementById("todoDrawerBack")?.addEventListener("click", () => this.closeTodo());
- document.getElementById("addTodoBtn")?.addEventListener("click", () => this.openNewTodo());
- document.getElementById("newTodoDrawerBack")?.addEventListener("click", () => this.closeNewTodo());
- document.getElementById("cancelNewTodo")?.addEventListener("click", () => this.closeNewTodo());
- document.getElementById("saveNewTodo")?.addEventListener("click", () => this.closeNewTodo());
- this.overlay?.addEventListener("click", () => this.closeAll());
- document.addEventListener("keydown", (e) => {
- if (e.key === "Escape") this.closeAll();
- });
- this.todoDrawer?.addEventListener("click", (e) => this.handleTodoClick(e));
- document.addEventListener("click", (e) => this.handleVisibilityClick(e));
- }
- handleTodoClick(e) {
- const target = e.target;
- const todoItem = target.closest("swp-todo-item");
- const checkbox = target.closest("swp-todo-checkbox");
- if (checkbox && todoItem) {
- const isCompleted = todoItem.dataset.completed === "true";
- if (isCompleted) {
- todoItem.removeAttribute("data-completed");
- } else {
- todoItem.dataset.completed = "true";
- }
- }
- const sectionHeader = target.closest("swp-todo-section-header");
- if (sectionHeader) {
- const section = sectionHeader.closest("swp-todo-section");
- section?.classList.toggle("collapsed");
- }
- }
- handleVisibilityClick(e) {
- const target = e.target;
- const option = target.closest("swp-visibility-option");
- if (option) {
- document.querySelectorAll("swp-visibility-option").forEach((o) => o.classList.remove("active"));
- option.classList.add("active");
- }
- }
- /**
- * Setup generic drawer triggers and close buttons
- * Uses data-drawer-trigger="drawer-id" and data-drawer-close attributes
- */
- setupGenericDrawers() {
- document.addEventListener("click", (e) => {
- const target = e.target;
- const trigger = target.closest("[data-drawer-trigger]");
- if (trigger) {
- const drawerId = trigger.dataset.drawerTrigger;
- if (drawerId) {
- this.openGenericDrawer(drawerId);
- }
- }
- });
- document.addEventListener("click", (e) => {
- const target = e.target;
- const closeBtn = target.closest("[data-drawer-close]");
- if (closeBtn) {
- this.closeGenericDrawer();
- this.overlay?.classList.remove("active");
- document.body.style.overflow = "";
- }
- });
- }
-};
-__name(_DrawerController, "DrawerController");
-var DrawerController = _DrawerController;
-
-// wwwroot/ts/modules/theme.ts
-var _ThemeController = class _ThemeController {
- constructor() {
- this.root = document.documentElement;
- this.themeOptions = document.querySelectorAll("swp-theme-option");
- this.applyTheme(this.current);
- this.updateUI();
- this.setupListeners();
- }
- /**
- * Get the current theme setting
- */
- get current() {
- const stored = localStorage.getItem(_ThemeController.STORAGE_KEY);
- if (stored === "dark" || stored === "light" || stored === "system") {
- return stored;
- }
- return "system";
- }
- /**
- * Check if dark mode is currently active
- */
- get isDark() {
- return this.root.classList.contains(_ThemeController.DARK_CLASS) || this.systemPrefersDark && !this.root.classList.contains(_ThemeController.LIGHT_CLASS);
- }
- /**
- * Check if system prefers dark mode
- */
- get systemPrefersDark() {
- return window.matchMedia("(prefers-color-scheme: dark)").matches;
- }
- /**
- * Set theme and persist preference
- */
- set(theme) {
- localStorage.setItem(_ThemeController.STORAGE_KEY, theme);
- this.applyTheme(theme);
- this.updateUI();
- }
- /**
- * Toggle between light and dark themes
- */
- toggle() {
- this.set(this.isDark ? "light" : "dark");
- }
- applyTheme(theme) {
- this.root.classList.remove(_ThemeController.DARK_CLASS, _ThemeController.LIGHT_CLASS);
- if (theme === "dark") {
- this.root.classList.add(_ThemeController.DARK_CLASS);
- } else if (theme === "light") {
- this.root.classList.add(_ThemeController.LIGHT_CLASS);
- }
- }
- updateUI() {
- if (!this.themeOptions) return;
- const darkActive = this.isDark;
- this.themeOptions.forEach((option) => {
- const theme = option.dataset.theme;
- const isActive = theme === "dark" && darkActive || theme === "light" && !darkActive;
- option.classList.toggle("active", isActive);
- });
- }
- setupListeners() {
- this.themeOptions.forEach((option) => {
- option.addEventListener("click", (e) => this.handleOptionClick(e));
- });
- window.matchMedia("(prefers-color-scheme: dark)").addEventListener("change", () => this.handleSystemChange());
- }
- handleOptionClick(e) {
- const target = e.target;
- const option = target.closest("swp-theme-option");
- if (option) {
- const theme = option.dataset.theme;
- if (theme) {
- this.set(theme);
- }
- }
- }
- handleSystemChange() {
- if (this.current === "system") {
- this.updateUI();
- }
- }
-};
-__name(_ThemeController, "ThemeController");
-_ThemeController.STORAGE_KEY = "theme-preference";
-_ThemeController.DARK_CLASS = "dark-mode";
-_ThemeController.LIGHT_CLASS = "light-mode";
-var ThemeController = _ThemeController;
-
-// wwwroot/ts/modules/search.ts
-var _SearchController = class _SearchController {
- constructor() {
- this.input = null;
- this.container = null;
- this.input = document.getElementById("globalSearch");
- this.container = document.querySelector("swp-topbar-search");
- this.setupListeners();
- }
- /**
- * Get current search value
- */
- get value() {
- return this.input?.value ?? "";
- }
- /**
- * Set search value
- */
- set value(val) {
- if (this.input) {
- this.input.value = val;
- }
- }
- /**
- * Focus the search input
- */
- focus() {
- this.input?.focus();
- }
- /**
- * Blur the search input
- */
- blur() {
- this.input?.blur();
- }
- /**
- * Clear the search input
- */
- clear() {
- this.value = "";
- }
- setupListeners() {
- document.addEventListener("keydown", (e) => this.handleKeyboard(e));
- if (this.input) {
- this.input.addEventListener("input", (e) => this.handleInput(e));
- const form = this.input.closest("form");
- form?.addEventListener("submit", (e) => this.handleSubmit(e));
- }
- }
- handleKeyboard(e) {
- if ((e.metaKey || e.ctrlKey) && e.key === "k") {
- e.preventDefault();
- this.focus();
- return;
- }
- if (e.key === "Escape" && document.activeElement === this.input) {
- this.blur();
- }
- }
- handleInput(e) {
- const target = e.target;
- const query = target.value.trim();
- document.dispatchEvent(new CustomEvent("app:search", {
- detail: { query },
- bubbles: true
- }));
- }
- handleSubmit(e) {
- e.preventDefault();
- const query = this.value.trim();
- if (!query) return;
- document.dispatchEvent(new CustomEvent("app:search-submit", {
- detail: { query },
- bubbles: true
- }));
- }
-};
-__name(_SearchController, "SearchController");
-var SearchController = _SearchController;
-
-// wwwroot/ts/modules/lockscreen.ts
-var _LockScreenController = class _LockScreenController {
- constructor(drawers) {
- // Demo PIN
- this.lockScreen = null;
- this.pinInput = null;
- this.pinKeypad = null;
- this.lockTimeEl = null;
- this.pinDigits = null;
- this.currentPin = "";
- this.drawers = null;
- this.drawers = drawers ?? null;
- this.lockScreen = document.getElementById("lockScreen");
- this.pinInput = document.getElementById("pinInput");
- this.pinKeypad = document.getElementById("pinKeypad");
- this.lockTimeEl = document.getElementById("lockTime");
- this.pinDigits = this.pinInput?.querySelectorAll("swp-pin-digit") ?? null;
- this.setupListeners();
- }
- /**
- * Check if lock screen is active
- */
- get isActive() {
- return this.lockScreen?.classList.contains("active") ?? false;
- }
- /**
- * Show the lock screen
- */
- show() {
- this.drawers?.closeAll();
- if (this.lockScreen) {
- this.lockScreen.classList.add("active");
- document.body.style.overflow = "hidden";
- }
- this.currentPin = "";
- this.updateDisplay();
- if (this.lockTimeEl) {
- this.lockTimeEl.textContent = `L\xE5st kl. ${this.formatTime()}`;
- }
- }
- /**
- * Hide the lock screen
- */
- hide() {
- if (this.lockScreen) {
- this.lockScreen.classList.remove("active");
- document.body.style.overflow = "";
- }
- this.currentPin = "";
- this.updateDisplay();
- }
- formatTime() {
- const now = /* @__PURE__ */ new Date();
- const hours = now.getHours().toString().padStart(2, "0");
- const minutes = now.getMinutes().toString().padStart(2, "0");
- return `${hours}:${minutes}`;
- }
- updateDisplay() {
- if (!this.pinDigits) return;
- this.pinDigits.forEach((digit, index) => {
- digit.classList.remove("filled", "error");
- if (index < this.currentPin.length) {
- digit.textContent = "\u2022";
- digit.classList.add("filled");
- } else {
- digit.textContent = "";
- }
- });
- }
- showError() {
- if (!this.pinDigits) return;
- this.pinDigits.forEach((digit) => digit.classList.add("error"));
- this.pinInput?.classList.add("shake");
- setTimeout(() => {
- this.currentPin = "";
- this.updateDisplay();
- this.pinInput?.classList.remove("shake");
- }, 500);
- }
- verify() {
- if (this.currentPin === _LockScreenController.CORRECT_PIN) {
- this.hide();
- } else {
- this.showError();
- }
- }
- addDigit(digit) {
- if (this.currentPin.length >= 4) return;
- this.currentPin += digit;
- this.updateDisplay();
- if (this.currentPin.length === 4) {
- setTimeout(() => this.verify(), 200);
- }
- }
- removeDigit() {
- if (this.currentPin.length === 0) return;
- this.currentPin = this.currentPin.slice(0, -1);
- this.updateDisplay();
- }
- clearPin() {
- this.currentPin = "";
- this.updateDisplay();
- }
- setupListeners() {
- this.pinKeypad?.addEventListener("click", (e) => this.handleKeypadClick(e));
- document.addEventListener("keydown", (e) => this.handleKeyboard(e));
- document.querySelector("swp-side-menu-action.lock")?.addEventListener("click", () => this.show());
- }
- handleKeypadClick(e) {
- const target = e.target;
- const key = target.closest("swp-pin-key");
- if (!key) return;
- const digit = key.dataset.digit;
- const action = key.dataset.action;
- if (digit) {
- this.addDigit(digit);
- } else if (action === "backspace") {
- this.removeDigit();
- } else if (action === "clear") {
- this.clearPin();
- }
- }
- handleKeyboard(e) {
- if (!this.isActive) return;
- e.preventDefault();
- if (e.key >= "0" && e.key <= "9") {
- this.addDigit(e.key);
- } else if (e.key === "Backspace") {
- this.removeDigit();
- } else if (e.key === "Escape") {
- this.clearPin();
- }
- }
-};
-__name(_LockScreenController, "LockScreenController");
-_LockScreenController.CORRECT_PIN = "1234";
-var LockScreenController = _LockScreenController;
-
-// wwwroot/ts/modules/cash.ts
-var _CashController = class _CashController {
- constructor() {
- // Base values (from system - would come from server in real app)
- this.startBalance = 2e3;
- this.cashSales = 3540;
- this.setupTabs();
- this.setupCashCalculation();
- this.setupCheckboxSelection();
- this.setupApprovalCheckbox();
- this.setupDateFilters();
- this.setupRowToggle();
- this.setupDraftRowClick();
- }
- /**
- * Setup tab switching functionality
- */
- setupTabs() {
- const tabs = document.querySelectorAll("swp-tab[data-tab]");
- tabs.forEach((tab) => {
- tab.addEventListener("click", () => {
- const targetTab = tab.dataset.tab;
- if (targetTab) {
- this.switchToTab(targetTab);
- }
- });
- });
- }
- /**
- * Switch to a specific tab by name
- */
- switchToTab(targetTab) {
- const tabs = document.querySelectorAll("swp-tab[data-tab]");
- const contents = document.querySelectorAll("swp-tab-content[data-tab]");
- const statsBars = document.querySelectorAll("swp-cash-stats[data-for-tab]");
- tabs.forEach((t) => {
- if (t.dataset.tab === targetTab) {
- t.classList.add("active");
- } else {
- t.classList.remove("active");
- }
- });
- contents.forEach((content) => {
- if (content.dataset.tab === targetTab) {
- content.classList.add("active");
- } else {
- content.classList.remove("active");
- }
- });
- statsBars.forEach((stats) => {
- if (stats.dataset.forTab === targetTab) {
- stats.classList.add("active");
- } else {
- stats.classList.remove("active");
- }
- });
- }
- /**
- * Setup cash calculation with real-time updates
- */
- setupCashCalculation() {
- const payoutsInput = document.getElementById("payouts");
- const toBankInput = document.getElementById("toBank");
- const actualCashInput = document.getElementById("actualCash");
- if (!payoutsInput || !toBankInput || !actualCashInput) return;
- const calculate = /* @__PURE__ */ __name(() => this.calculateCash(payoutsInput, toBankInput, actualCashInput), "calculate");
- payoutsInput.addEventListener("input", calculate);
- toBankInput.addEventListener("input", calculate);
- actualCashInput.addEventListener("input", calculate);
- calculate();
- }
- /**
- * Calculate expected cash and difference
- */
- calculateCash(payoutsInput, toBankInput, actualCashInput) {
- const payouts = this.parseNumber(payoutsInput.value);
- const toBank = this.parseNumber(toBankInput.value);
- const actual = this.parseNumber(actualCashInput.value);
- const expectedCash = this.startBalance + this.cashSales - payouts - toBank;
- const expectedElement = document.getElementById("expectedCash");
- if (expectedElement) {
- expectedElement.textContent = this.formatNumber(expectedCash);
- }
- this.updateDifference(actual, expectedCash, actualCashInput.value);
- }
- /**
- * Update difference box with color coding
- */
- updateDifference(actual, expected, rawValue) {
- const box = document.getElementById("differenceBox");
- const value = document.getElementById("differenceValue");
- if (!box || !value) return;
- const diff = actual - expected;
- box.classList.remove("positive", "negative", "neutral");
- if (actual === 0 && rawValue === "") {
- value.textContent = "\u2013 kr";
- box.classList.add("neutral");
- } else if (diff > 0) {
- value.textContent = "+" + this.formatNumber(diff) + " kr";
- box.classList.add("positive");
- } else if (diff < 0) {
- value.textContent = this.formatNumber(diff) + " kr";
- box.classList.add("negative");
- } else {
- value.textContent = "0,00 kr";
- box.classList.add("neutral");
- }
- }
- /**
- * Setup checkbox selection for table rows
- */
- setupCheckboxSelection() {
- const selectAll = document.getElementById("selectAll");
- const rowCheckboxes = document.querySelectorAll(".row-select");
- const exportBtn = document.getElementById("exportBtn");
- const selectionCount = document.getElementById("selectionCount");
- if (!selectAll || !exportBtn || !selectionCount) return;
- const updateSelection = /* @__PURE__ */ __name(() => {
- const checked = document.querySelectorAll(".row-select:checked");
- const count = checked.length;
- selectionCount.textContent = count === 0 ? "0 valgt" : `${count} valgt`;
- exportBtn.disabled = count === 0;
- selectAll.checked = count === rowCheckboxes.length && count > 0;
- selectAll.indeterminate = count > 0 && count < rowCheckboxes.length;
- }, "updateSelection");
- selectAll.addEventListener("change", () => {
- rowCheckboxes.forEach((cb) => cb.checked = selectAll.checked);
- updateSelection();
- });
- rowCheckboxes.forEach((cb) => {
- cb.addEventListener("change", updateSelection);
- cb.addEventListener("click", (e) => e.stopPropagation());
- });
- }
- /**
- * Setup approval checkbox to enable/disable approve button
- */
- setupApprovalCheckbox() {
- const checkbox = document.getElementById("confirmCheckbox");
- const approveBtn = document.getElementById("approveBtn");
- if (!checkbox || !approveBtn) return;
- checkbox.addEventListener("change", () => {
- approveBtn.disabled = !checkbox.checked;
- });
- }
- /**
- * Setup date filter defaults (last 30 days)
- */
- setupDateFilters() {
- const dateFrom = document.getElementById("dateFrom");
- const dateTo = document.getElementById("dateTo");
- if (!dateFrom || !dateTo) return;
- const today = /* @__PURE__ */ new Date();
- const thirtyDaysAgo = new Date(today);
- thirtyDaysAgo.setDate(today.getDate() - 30);
- dateTo.value = this.formatDateISO(today);
- dateFrom.value = this.formatDateISO(thirtyDaysAgo);
- }
- /**
- * Format number as Danish currency
- */
- formatNumber(num) {
- return num.toLocaleString("da-DK", {
- minimumFractionDigits: 2,
- maximumFractionDigits: 2
- });
- }
- /**
- * Parse Danish number format
- */
- parseNumber(str) {
- if (!str) return 0;
- return parseFloat(str.replace(/\./g, "").replace(",", ".")) || 0;
- }
- /**
- * Format date as ISO string (YYYY-MM-DD)
- */
- formatDateISO(date) {
- return date.toISOString().split("T")[0];
- }
- /**
- * Setup row toggle for expandable details
- */
- setupRowToggle() {
- const rows = document.querySelectorAll("swp-cash-table-row[data-id]:not(.draft-row)");
- rows.forEach((row) => {
- const rowId = row.getAttribute("data-id");
- if (!rowId) return;
- const detail = document.querySelector(`swp-cash-row-detail[data-for="${rowId}"]`);
- if (!detail) return;
- row.addEventListener("click", (e) => {
- if (e.target.closest('input[type="checkbox"]')) return;
- const icon = row.querySelector("swp-row-toggle i");
- const isExpanded = row.classList.contains("expanded");
- document.querySelectorAll("swp-cash-table-row.expanded").forEach((r) => {
- if (r !== row) {
- const otherId = r.getAttribute("data-id");
- if (otherId) {
- const otherDetail = document.querySelector(`swp-cash-row-detail[data-for="${otherId}"]`);
- const otherIcon = r.querySelector("swp-row-toggle i");
- if (otherDetail && otherIcon) {
- this.collapseRow(r, otherDetail, otherIcon);
- }
- }
- }
- });
- if (isExpanded) {
- this.collapseRow(row, detail, icon);
- } else {
- this.expandRow(row, detail, icon);
- }
- });
- });
- }
- /**
- * Expand a row with animation
- */
- expandRow(row, detail, icon) {
- row.classList.add("expanded");
- detail.classList.add("expanded");
- icon?.animate([
- { transform: "rotate(0deg)" },
- { transform: "rotate(90deg)" }
- ], {
- duration: 200,
- easing: "ease-out",
- fill: "forwards"
- });
- const content = detail.querySelector("swp-row-detail-content");
- if (content) {
- const height = content.offsetHeight;
- detail.animate([
- { height: "0px", opacity: 0 },
- { height: `${height}px`, opacity: 1 }
- ], {
- duration: 250,
- easing: "ease-out",
- fill: "forwards"
- });
- }
- }
- /**
- * Collapse a row with animation
- */
- collapseRow(row, detail, icon) {
- icon?.animate([
- { transform: "rotate(90deg)" },
- { transform: "rotate(0deg)" }
- ], {
- duration: 200,
- easing: "ease-out",
- fill: "forwards"
- });
- const content = detail.querySelector("swp-row-detail-content");
- if (content) {
- const height = content.offsetHeight;
- const animation = detail.animate([
- { height: `${height}px`, opacity: 1 },
- { height: "0px", opacity: 0 }
- ], {
- duration: 200,
- easing: "ease-out",
- fill: "forwards"
- });
- animation.onfinish = () => {
- row.classList.remove("expanded");
- detail.classList.remove("expanded");
- };
- } else {
- row.classList.remove("expanded");
- detail.classList.remove("expanded");
- }
- }
- /**
- * Setup draft row click to navigate to reconciliation tab
- */
- setupDraftRowClick() {
- const draftRow = document.querySelector("swp-cash-table-row.draft-row");
- if (!draftRow) return;
- draftRow.style.cursor = "pointer";
- draftRow.addEventListener("click", (e) => {
- if (e.target.closest('input[type="checkbox"]')) return;
- this.switchToTab("afstemning");
- });
- }
-};
-__name(_CashController, "CashController");
-var CashController = _CashController;
-
-// wwwroot/ts/modules/employees.ts
-var _EmployeesController = class _EmployeesController {
- constructor() {
- this.ratesSync = null;
- this.listView = null;
- this.detailView = null;
- this.listView = document.getElementById("employees-list-view");
- this.detailView = document.getElementById("employee-detail-view");
- if (!this.listView) return;
- this.setupListTabs();
- this.setupDetailTabs();
- this.setupChevronNavigation();
- this.setupBackNavigation();
- this.setupHistoryNavigation();
- this.restoreStateFromUrl();
- this.ratesSync = new RatesSyncController();
- }
- /**
- * Setup popstate listener for browser back/forward
- */
- setupHistoryNavigation() {
- window.addEventListener("popstate", (e) => {
- if (e.state?.employeeKey) {
- this.showDetailViewInternal(e.state.employeeKey);
- } else {
- this.showListViewInternal();
- }
- });
- }
- /**
- * Restore view state from URL on page load
- */
- restoreStateFromUrl() {
- const hash = window.location.hash;
- if (hash.startsWith("#employee-")) {
- const employeeKey = hash.substring(1);
- this.showDetailViewInternal(employeeKey);
- }
- }
- /**
- * Setup tab switching for the list view
- */
- setupListTabs() {
- if (!this.listView) return;
- const tabs = this.listView.querySelectorAll("swp-tab-bar > swp-tab[data-tab]");
- tabs.forEach((tab) => {
- tab.addEventListener("click", () => {
- const targetTab = tab.dataset.tab;
- if (targetTab) {
- this.switchTab(this.listView, targetTab);
- }
- });
- });
- }
- /**
- * Setup tab switching for the detail view
- */
- setupDetailTabs() {
- if (!this.detailView) return;
- const tabs = this.detailView.querySelectorAll("swp-tab-bar > swp-tab[data-tab]");
- tabs.forEach((tab) => {
- tab.addEventListener("click", () => {
- const targetTab = tab.dataset.tab;
- if (targetTab) {
- this.switchTab(this.detailView, targetTab);
- }
- });
- });
- }
- /**
- * Switch to a specific tab within a container
- */
- switchTab(container, targetTab) {
- const tabs = container.querySelectorAll("swp-tab-bar > swp-tab[data-tab]");
- const contents = container.querySelectorAll("swp-tab-content[data-tab]");
- tabs.forEach((t) => {
- t.classList.toggle("active", t.dataset.tab === targetTab);
- });
- contents.forEach((content) => {
- content.classList.toggle("active", content.dataset.tab === targetTab);
- });
- }
- /**
- * Setup row click to show detail view
- * Ignores clicks on action buttons
- */
- setupChevronNavigation() {
- document.addEventListener("click", (e) => {
- const target = e.target;
- if (target.closest("swp-icon-btn") || target.closest("swp-table-actions")) {
- return;
- }
- const row = target.closest("swp-data-table-row[data-employee-detail]");
- if (row) {
- const employeeKey = row.dataset.employeeDetail;
- if (employeeKey) {
- this.showDetailView(employeeKey);
- }
- }
- });
- }
- /**
- * Setup back button to return to list view
- */
- setupBackNavigation() {
- document.addEventListener("click", (e) => {
- const target = e.target;
- const backLink = target.closest("[data-employee-back]");
- if (backLink) {
- this.showListView();
- }
- });
- }
- /**
- * Show the detail view and hide list view (with history push)
- */
- showDetailView(employeeKey) {
- history.pushState(
- { employeeKey },
- "",
- `#${employeeKey}`
- );
- this.showDetailViewInternal(employeeKey);
- }
- /**
- * Show detail view without modifying history (for popstate)
- */
- showDetailViewInternal(employeeKey) {
- if (this.listView && this.detailView) {
- this.listView.style.display = "none";
- this.detailView.style.display = "block";
- this.detailView.dataset.employee = employeeKey;
- this.switchTab(this.detailView, "general");
- }
- }
- /**
- * Show the list view and hide detail view (with history push)
- */
- showListView() {
- history.pushState(
- {},
- "",
- window.location.pathname
- );
- this.showListViewInternal();
- }
- /**
- * Show list view without modifying history (for popstate)
- */
- showListViewInternal() {
- if (this.listView && this.detailView) {
- this.detailView.style.display = "none";
- this.listView.style.display = "block";
- }
- }
-};
-__name(_EmployeesController, "EmployeesController");
-var EmployeesController = _EmployeesController;
-var _RatesSyncController = class _RatesSyncController {
- constructor() {
- this.drawer = null;
- this.drawer = document.getElementById("rates-drawer");
- if (!this.drawer) return;
- this.setupCheckboxListeners();
- this.setupInputListeners();
- }
- /**
- * Extract rate key from checkbox ID (e.g., "rate-normal-enabled" → "normal")
- */
- extractRateKey(checkboxId) {
- const match = checkboxId.match(/^rate-(.+)-enabled$/);
- return match ? match[1] : null;
- }
- /**
- * Setup checkbox change listeners in drawer
- */
- setupCheckboxListeners() {
- if (!this.drawer) return;
- this.drawer.addEventListener("change", (e) => {
- const target = e.target;
- if (target.type !== "checkbox" || !target.id) return;
- const rateKey = this.extractRateKey(target.id);
- if (!rateKey) return;
- const isChecked = target.checked;
- const row = target.closest("swp-data-row");
- if (!row) return;
- const label = row.querySelector("swp-data-label");
- const input = row.querySelector("swp-data-input");
- if (label) label.classList.toggle("disabled", !isChecked);
- if (input) input.classList.toggle("disabled", !isChecked);
- this.toggleCardRow(rateKey, isChecked);
- if (isChecked) {
- const textInput = document.getElementById(`rate-${rateKey}`);
- if (textInput) {
- this.syncValueToCard(rateKey, textInput.value);
- }
- }
- });
- }
- /**
- * Setup input change listeners in drawer
- */
- setupInputListeners() {
- if (!this.drawer) return;
- this.drawer.addEventListener("input", (e) => {
- const target = e.target;
- if (target.type !== "text" || !target.id) return;
- const match = target.id.match(/^rate-(.+)$/);
- if (!match) return;
- const rateKey = match[1];
- if (rateKey.endsWith("-enabled")) return;
- this.syncValueToCard(rateKey, target.value);
- });
- }
- /**
- * Toggle card row visibility by ID
- */
- toggleCardRow(rateKey, visible) {
- const cardRow = document.getElementById(`card-${rateKey}`);
- if (cardRow) {
- cardRow.style.display = visible ? "" : "none";
- }
- }
- /**
- * Format number with 2 decimals using Danish locale (comma as decimal separator)
- */
- formatNumber(value) {
- const normalized = value.replace(",", ".");
- const num = parseFloat(normalized);
- if (isNaN(num)) return value;
- return num.toFixed(2).replace(".", ",");
- }
- /**
- * Sync value from drawer to card by ID
- */
- syncValueToCard(rateKey, value) {
- const cardInput = document.getElementById(`value-${rateKey}`);
- if (!cardInput) return;
- const textInput = document.getElementById(`rate-${rateKey}`);
- const inputContainer = textInput?.closest("swp-data-input");
- const unit = inputContainer?.textContent?.trim().replace(value, "").trim() || "kr";
- const formattedValue = this.formatNumber(value);
- cardInput.value = `${formattedValue} ${unit}`;
- }
-};
-__name(_RatesSyncController, "RatesSyncController");
-var RatesSyncController = _RatesSyncController;
-
-// wwwroot/ts/app.ts
-var _App = class _App {
- constructor() {
- this.sidebar = new SidebarController();
- this.drawers = new DrawerController();
- this.theme = new ThemeController();
- this.search = new SearchController();
- this.lockScreen = new LockScreenController(this.drawers);
- this.cash = new CashController();
- this.employees = new EmployeesController();
- }
-};
-__name(_App, "App");
-var App = _App;
-var app;
-function init() {
- app = new App();
- if (typeof window !== "undefined") {
- window.app = app;
- }
-}
-__name(init, "init");
-if (document.readyState === "loading") {
- document.addEventListener("DOMContentLoaded", init);
-} else {
- init();
-}
-var app_default = App;
-export {
- App,
- app,
- app_default as default
-};
-//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../ts/modules/sidebar.ts", "../ts/modules/drawers.ts", "../ts/modules/theme.ts", "../ts/modules/search.ts", "../ts/modules/lockscreen.ts", "../ts/modules/cash.ts", "../ts/modules/employees.ts", "../ts/app.ts"],
  "sourcesContent": ["/**\n * Sidebar Controller\n *\n * Handles sidebar collapse/expand and tooltip functionality\n */\n\nexport class SidebarController {\n  private menuToggle: HTMLElement | null = null;\n  private appLayout: HTMLElement | null = null;\n  private menuTooltip: HTMLElement | null = null;\n\n  constructor() {\n    this.menuToggle = document.getElementById('menuToggle');\n    this.appLayout = document.querySelector('swp-app-layout');\n    this.menuTooltip = document.getElementById('menuTooltip');\n\n    this.setupListeners();\n    this.setupTooltips();\n    this.restoreState();\n  }\n\n  /**\n   * Check if sidebar is collapsed\n   */\n  get isCollapsed(): boolean {\n    return this.appLayout?.classList.contains('menu-collapsed') ?? false;\n  }\n\n  /**\n   * Toggle sidebar collapsed state\n   */\n  toggle(): void {\n    if (!this.appLayout) return;\n\n    this.appLayout.classList.toggle('menu-collapsed');\n    localStorage.setItem('sidebar-collapsed', String(this.isCollapsed));\n  }\n\n  /**\n   * Collapse the sidebar\n   */\n  collapse(): void {\n    this.appLayout?.classList.add('menu-collapsed');\n    localStorage.setItem('sidebar-collapsed', 'true');\n  }\n\n  /**\n   * Expand the sidebar\n   */\n  expand(): void {\n    this.appLayout?.classList.remove('menu-collapsed');\n    localStorage.setItem('sidebar-collapsed', 'false');\n  }\n\n  private setupListeners(): void {\n    this.menuToggle?.addEventListener('click', () => this.toggle());\n  }\n\n  private setupTooltips(): void {\n    if (!this.menuTooltip) return;\n\n    const menuItems = document.querySelectorAll<HTMLElement>('swp-side-menu-item[data-tooltip]');\n\n    menuItems.forEach(item => {\n      item.addEventListener('mouseenter', () => this.showTooltip(item));\n      item.addEventListener('mouseleave', () => this.hideTooltip());\n    });\n  }\n\n  private showTooltip(item: HTMLElement): void {\n    if (!this.isCollapsed || !this.menuTooltip) return;\n\n    const rect = item.getBoundingClientRect();\n    const tooltipText = item.dataset.tooltip;\n\n    if (!tooltipText) return;\n\n    this.menuTooltip.textContent = tooltipText;\n    this.menuTooltip.style.left = `${rect.right + 8}px`;\n    this.menuTooltip.style.top = `${rect.top + rect.height / 2}px`;\n    this.menuTooltip.style.transform = 'translateY(-50%)';\n    this.menuTooltip.showPopover();\n  }\n\n  private hideTooltip(): void {\n    this.menuTooltip?.hidePopover();\n  }\n\n  private restoreState(): void {\n    if (!this.appLayout) return;\n\n    if (localStorage.getItem('sidebar-collapsed') === 'true') {\n      this.appLayout.classList.add('menu-collapsed');\n    }\n  }\n}\n", "/**\n * Drawer Controller\n *\n * Handles all drawer functionality including profile, notifications, and todo drawers\n */\n\nexport type DrawerName = 'profile' | 'notification' | 'todo' | 'newTodo';\n\nexport class DrawerController {\n  private profileDrawer: HTMLElement | null = null;\n  private notificationDrawer: HTMLElement | null = null;\n  private todoDrawer: HTMLElement | null = null;\n  private newTodoDrawer: HTMLElement | null = null;\n  private overlay: HTMLElement | null = null;\n  private activeDrawer: DrawerName | null = null;\n  private activeGenericDrawer: HTMLElement | null = null;\n\n  constructor() {\n    this.profileDrawer = document.getElementById('profileDrawer');\n    this.notificationDrawer = document.getElementById('notificationDrawer');\n    this.todoDrawer = document.getElementById('todoDrawer');\n    this.newTodoDrawer = document.getElementById('newTodoDrawer');\n    this.overlay = document.getElementById('drawerOverlay');\n\n    this.setupListeners();\n    this.setupGenericDrawers();\n  }\n\n  /**\n   * Get currently active drawer name\n   */\n  get active(): DrawerName | null {\n    return this.activeDrawer;\n  }\n\n  /**\n   * Open a drawer by name\n   */\n  open(name: DrawerName): void {\n    this.closeAll();\n\n    const drawer = this.getDrawer(name);\n    if (drawer && this.overlay) {\n      drawer.classList.add('active');\n      this.overlay.classList.add('active');\n      document.body.style.overflow = 'hidden';\n      this.activeDrawer = name;\n    }\n  }\n\n  /**\n   * Close a specific drawer\n   */\n  close(name: DrawerName): void {\n    const drawer = this.getDrawer(name);\n    drawer?.classList.remove('active');\n\n    // Only hide overlay if no drawers are active\n    if (this.overlay && !document.querySelector('.active[class*=\"drawer\"]')) {\n      this.overlay.classList.remove('active');\n      document.body.style.overflow = '';\n    }\n\n    if (this.activeDrawer === name) {\n      this.activeDrawer = null;\n    }\n  }\n\n  /**\n   * Close all drawers\n   */\n  closeAll(): void {\n    [this.profileDrawer, this.notificationDrawer, this.todoDrawer, this.newTodoDrawer]\n      .forEach(drawer => drawer?.classList.remove('active'));\n\n    // Close any generic drawers\n    this.closeGenericDrawer();\n\n    this.overlay?.classList.remove('active');\n    document.body.style.overflow = '';\n    this.activeDrawer = null;\n  }\n\n  /**\n   * Open a generic drawer by ID\n   */\n  openGenericDrawer(drawerId: string): void {\n    this.closeAll();\n\n    const drawer = document.getElementById(drawerId);\n    if (drawer && this.overlay) {\n      drawer.classList.add('open');\n      this.overlay.classList.add('active');\n      document.body.style.overflow = 'hidden';\n      this.activeGenericDrawer = drawer;\n    }\n  }\n\n  /**\n   * Close the currently open generic drawer\n   */\n  closeGenericDrawer(): void {\n    this.activeGenericDrawer?.classList.remove('open');\n    this.activeGenericDrawer = null;\n  }\n\n  /**\n   * Open profile drawer\n   */\n  openProfile(): void {\n    this.open('profile');\n  }\n\n  /**\n   * Open notification drawer\n   */\n  openNotification(): void {\n    this.open('notification');\n  }\n\n  /**\n   * Open todo drawer (slides on top of profile)\n   */\n  openTodo(): void {\n    this.todoDrawer?.classList.add('active');\n  }\n\n  /**\n   * Close todo drawer\n   */\n  closeTodo(): void {\n    this.todoDrawer?.classList.remove('active');\n    this.closeNewTodo();\n  }\n\n  /**\n   * Open new todo drawer\n   */\n  openNewTodo(): void {\n    this.newTodoDrawer?.classList.add('active');\n  }\n\n  /**\n   * Close new todo drawer\n   */\n  closeNewTodo(): void {\n    this.newTodoDrawer?.classList.remove('active');\n  }\n\n  /**\n   * Mark all notifications as read\n   */\n  markAllNotificationsRead(): void {\n    if (!this.notificationDrawer) return;\n\n    const unreadItems = this.notificationDrawer.querySelectorAll<HTMLElement>(\n      'swp-notification-item[data-unread=\"true\"]'\n    );\n    unreadItems.forEach(item => item.removeAttribute('data-unread'));\n\n    const badge = document.querySelector<HTMLElement>('swp-notification-badge');\n    if (badge) {\n      badge.style.display = 'none';\n    }\n  }\n\n  private getDrawer(name: DrawerName): HTMLElement | null {\n    switch (name) {\n      case 'profile': return this.profileDrawer;\n      case 'notification': return this.notificationDrawer;\n      case 'todo': return this.todoDrawer;\n      case 'newTodo': return this.newTodoDrawer;\n    }\n  }\n\n  private setupListeners(): void {\n    // Profile drawer triggers\n    document.getElementById('profileTrigger')\n      ?.addEventListener('click', () => this.openProfile());\n    document.getElementById('drawerClose')\n      ?.addEventListener('click', () => this.close('profile'));\n\n    // Notification drawer triggers\n    document.getElementById('notificationsBtn')\n      ?.addEventListener('click', () => this.openNotification());\n    document.getElementById('notificationDrawerClose')\n      ?.addEventListener('click', () => this.close('notification'));\n    document.getElementById('markAllRead')\n      ?.addEventListener('click', () => this.markAllNotificationsRead());\n\n    // Todo drawer triggers\n    document.getElementById('openTodoDrawer')\n      ?.addEventListener('click', () => this.openTodo());\n    document.getElementById('todoDrawerBack')\n      ?.addEventListener('click', () => this.closeTodo());\n\n    // New todo drawer triggers\n    document.getElementById('addTodoBtn')\n      ?.addEventListener('click', () => this.openNewTodo());\n    document.getElementById('newTodoDrawerBack')\n      ?.addEventListener('click', () => this.closeNewTodo());\n    document.getElementById('cancelNewTodo')\n      ?.addEventListener('click', () => this.closeNewTodo());\n    document.getElementById('saveNewTodo')\n      ?.addEventListener('click', () => this.closeNewTodo());\n\n    // Overlay click closes all\n    this.overlay?.addEventListener('click', () => this.closeAll());\n\n    // Escape key closes all\n    document.addEventListener('keydown', (e: KeyboardEvent) => {\n      if (e.key === 'Escape') this.closeAll();\n    });\n\n    // Todo interactions\n    this.todoDrawer?.addEventListener('click', (e) => this.handleTodoClick(e));\n\n    // Visibility options\n    document.addEventListener('click', (e) => this.handleVisibilityClick(e));\n  }\n\n  private handleTodoClick(e: Event): void {\n    const target = e.target as HTMLElement;\n    const todoItem = target.closest<HTMLElement>('swp-todo-item');\n    const checkbox = target.closest<HTMLElement>('swp-todo-checkbox');\n\n    if (checkbox && todoItem) {\n      const isCompleted = todoItem.dataset.completed === 'true';\n      if (isCompleted) {\n        todoItem.removeAttribute('data-completed');\n      } else {\n        todoItem.dataset.completed = 'true';\n      }\n    }\n\n    // Toggle section collapse\n    const sectionHeader = target.closest<HTMLElement>('swp-todo-section-header');\n    if (sectionHeader) {\n      const section = sectionHeader.closest<HTMLElement>('swp-todo-section');\n      section?.classList.toggle('collapsed');\n    }\n  }\n\n  private handleVisibilityClick(e: Event): void {\n    const target = e.target as HTMLElement;\n    const option = target.closest<HTMLElement>('swp-visibility-option');\n\n    if (option) {\n      document.querySelectorAll<HTMLElement>('swp-visibility-option')\n        .forEach(o => o.classList.remove('active'));\n      option.classList.add('active');\n    }\n  }\n\n  /**\n   * Setup generic drawer triggers and close buttons\n   * Uses data-drawer-trigger=\"drawer-id\" and data-drawer-close attributes\n   */\n  private setupGenericDrawers(): void {\n    // Handle drawer triggers\n    document.addEventListener('click', (e: Event) => {\n      const target = e.target as HTMLElement;\n      const trigger = target.closest<HTMLElement>('[data-drawer-trigger]');\n\n      if (trigger) {\n        const drawerId = trigger.dataset.drawerTrigger;\n        if (drawerId) {\n          this.openGenericDrawer(drawerId);\n        }\n      }\n    });\n\n    // Handle drawer close buttons\n    document.addEventListener('click', (e: Event) => {\n      const target = e.target as HTMLElement;\n      const closeBtn = target.closest<HTMLElement>('[data-drawer-close]');\n\n      if (closeBtn) {\n        this.closeGenericDrawer();\n        this.overlay?.classList.remove('active');\n        document.body.style.overflow = '';\n      }\n    });\n  }\n}\n", "/**\n * Theme Controller\n *\n * Handles dark/light mode switching and system preference detection\n */\n\nexport type Theme = 'light' | 'dark' | 'system';\n\nexport class ThemeController {\n  private static readonly STORAGE_KEY = 'theme-preference';\n  private static readonly DARK_CLASS = 'dark-mode';\n  private static readonly LIGHT_CLASS = 'light-mode';\n\n  private root: HTMLElement;\n  private themeOptions: NodeListOf<HTMLElement>;\n\n  constructor() {\n    this.root = document.documentElement;\n    this.themeOptions = document.querySelectorAll<HTMLElement>('swp-theme-option');\n\n    this.applyTheme(this.current);\n    this.updateUI();\n    this.setupListeners();\n  }\n\n  /**\n   * Get the current theme setting\n   */\n  get current(): Theme {\n    const stored = localStorage.getItem(ThemeController.STORAGE_KEY) as Theme | null;\n    if (stored === 'dark' || stored === 'light' || stored === 'system') {\n      return stored;\n    }\n    return 'system';\n  }\n\n  /**\n   * Check if dark mode is currently active\n   */\n  get isDark(): boolean {\n    return this.root.classList.contains(ThemeController.DARK_CLASS) ||\n      (this.systemPrefersDark && !this.root.classList.contains(ThemeController.LIGHT_CLASS));\n  }\n\n  /**\n   * Check if system prefers dark mode\n   */\n  get systemPrefersDark(): boolean {\n    return window.matchMedia('(prefers-color-scheme: dark)').matches;\n  }\n\n  /**\n   * Set theme and persist preference\n   */\n  set(theme: Theme): void {\n    localStorage.setItem(ThemeController.STORAGE_KEY, theme);\n    this.applyTheme(theme);\n    this.updateUI();\n  }\n\n  /**\n   * Toggle between light and dark themes\n   */\n  toggle(): void {\n    this.set(this.isDark ? 'light' : 'dark');\n  }\n\n  private applyTheme(theme: Theme): void {\n    this.root.classList.remove(ThemeController.DARK_CLASS, ThemeController.LIGHT_CLASS);\n\n    if (theme === 'dark') {\n      this.root.classList.add(ThemeController.DARK_CLASS);\n    } else if (theme === 'light') {\n      this.root.classList.add(ThemeController.LIGHT_CLASS);\n    }\n    // 'system' leaves both classes off, letting CSS media query handle it\n  }\n\n  private updateUI(): void {\n    if (!this.themeOptions) return;\n\n    const darkActive = this.isDark;\n\n    this.themeOptions.forEach(option => {\n      const theme = option.dataset.theme as Theme;\n      const isActive = (theme === 'dark' && darkActive) || (theme === 'light' && !darkActive);\n      option.classList.toggle('active', isActive);\n    });\n  }\n\n  private setupListeners(): void {\n    // Theme option clicks\n    this.themeOptions.forEach(option => {\n      option.addEventListener('click', (e) => this.handleOptionClick(e));\n    });\n\n    // System theme changes\n    window.matchMedia('(prefers-color-scheme: dark)')\n      .addEventListener('change', () => this.handleSystemChange());\n  }\n\n  private handleOptionClick(e: Event): void {\n    const target = e.target as HTMLElement;\n    const option = target.closest<HTMLElement>('swp-theme-option');\n\n    if (option) {\n      const theme = option.dataset.theme as Theme;\n      if (theme) {\n        this.set(theme);\n      }\n    }\n  }\n\n  private handleSystemChange(): void {\n    // Only react to system changes if we're using system preference\n    if (this.current === 'system') {\n      this.updateUI();\n    }\n  }\n}\n", "/**\n * Search Controller\n *\n * Handles global search functionality and keyboard shortcuts\n */\n\nexport class SearchController {\n  private input: HTMLInputElement | null = null;\n  private container: HTMLElement | null = null;\n\n  constructor() {\n    this.input = document.getElementById('globalSearch') as HTMLInputElement | null;\n    this.container = document.querySelector<HTMLElement>('swp-topbar-search');\n\n    this.setupListeners();\n  }\n\n  /**\n   * Get current search value\n   */\n  get value(): string {\n    return this.input?.value ?? '';\n  }\n\n  /**\n   * Set search value\n   */\n  set value(val: string) {\n    if (this.input) {\n      this.input.value = val;\n    }\n  }\n\n  /**\n   * Focus the search input\n   */\n  focus(): void {\n    this.input?.focus();\n  }\n\n  /**\n   * Blur the search input\n   */\n  blur(): void {\n    this.input?.blur();\n  }\n\n  /**\n   * Clear the search input\n   */\n  clear(): void {\n    this.value = '';\n  }\n\n  private setupListeners(): void {\n    // Keyboard shortcuts\n    document.addEventListener('keydown', (e) => this.handleKeyboard(e));\n\n    // Input handlers\n    if (this.input) {\n      this.input.addEventListener('input', (e) => this.handleInput(e));\n\n      // Prevent form submission if wrapped in form\n      const form = this.input.closest('form');\n      form?.addEventListener('submit', (e) => this.handleSubmit(e));\n    }\n  }\n\n  private handleKeyboard(e: KeyboardEvent): void {\n    // Cmd/Ctrl + K to focus search\n    if ((e.metaKey || e.ctrlKey) && e.key === 'k') {\n      e.preventDefault();\n      this.focus();\n      return;\n    }\n\n    // Escape to blur search when focused\n    if (e.key === 'Escape' && document.activeElement === this.input) {\n      this.blur();\n    }\n  }\n\n  private handleInput(e: Event): void {\n    const target = e.target as HTMLInputElement;\n    const query = target.value.trim();\n\n    // Emit custom event for search\n    document.dispatchEvent(new CustomEvent('app:search', {\n      detail: { query },\n      bubbles: true\n    }));\n  }\n\n  private handleSubmit(e: Event): void {\n    e.preventDefault();\n\n    const query = this.value.trim();\n    if (!query) return;\n\n    // Emit custom event for search submit\n    document.dispatchEvent(new CustomEvent('app:search-submit', {\n      detail: { query },\n      bubbles: true\n    }));\n  }\n}\n", "/**\n * Lock Screen Controller\n *\n * Handles PIN-based lock screen functionality\n */\n\nimport { DrawerController } from './drawers';\n\nexport class LockScreenController {\n  private static readonly CORRECT_PIN = '1234'; // Demo PIN\n\n  private lockScreen: HTMLElement | null = null;\n  private pinInput: HTMLElement | null = null;\n  private pinKeypad: HTMLElement | null = null;\n  private lockTimeEl: HTMLElement | null = null;\n  private pinDigits: NodeListOf<HTMLElement> | null = null;\n  private currentPin = '';\n  private drawers: DrawerController | null = null;\n\n  constructor(drawers?: DrawerController) {\n    this.drawers = drawers ?? null;\n    this.lockScreen = document.getElementById('lockScreen');\n    this.pinInput = document.getElementById('pinInput');\n    this.pinKeypad = document.getElementById('pinKeypad');\n    this.lockTimeEl = document.getElementById('lockTime');\n    this.pinDigits = this.pinInput?.querySelectorAll<HTMLElement>('swp-pin-digit') ?? null;\n\n    this.setupListeners();\n  }\n\n  /**\n   * Check if lock screen is active\n   */\n  get isActive(): boolean {\n    return this.lockScreen?.classList.contains('active') ?? false;\n  }\n\n  /**\n   * Show the lock screen\n   */\n  show(): void {\n    this.drawers?.closeAll();\n\n    if (this.lockScreen) {\n      this.lockScreen.classList.add('active');\n      document.body.style.overflow = 'hidden';\n    }\n\n    this.currentPin = '';\n    this.updateDisplay();\n\n    // Update lock time\n    if (this.lockTimeEl) {\n      this.lockTimeEl.textContent = `L\u00E5st kl. ${this.formatTime()}`;\n    }\n  }\n\n  /**\n   * Hide the lock screen\n   */\n  hide(): void {\n    if (this.lockScreen) {\n      this.lockScreen.classList.remove('active');\n      document.body.style.overflow = '';\n    }\n\n    this.currentPin = '';\n    this.updateDisplay();\n  }\n\n  private formatTime(): string {\n    const now = new Date();\n    const hours = now.getHours().toString().padStart(2, '0');\n    const minutes = now.getMinutes().toString().padStart(2, '0');\n    return `${hours}:${minutes}`;\n  }\n\n  private updateDisplay(): void {\n    if (!this.pinDigits) return;\n\n    this.pinDigits.forEach((digit, index) => {\n      digit.classList.remove('filled', 'error');\n      if (index < this.currentPin.length) {\n        digit.textContent = '\u2022';\n        digit.classList.add('filled');\n      } else {\n        digit.textContent = '';\n      }\n    });\n  }\n\n  private showError(): void {\n    if (!this.pinDigits) return;\n\n    this.pinDigits.forEach(digit => digit.classList.add('error'));\n\n    // Shake animation\n    this.pinInput?.classList.add('shake');\n\n    setTimeout(() => {\n      this.currentPin = '';\n      this.updateDisplay();\n      this.pinInput?.classList.remove('shake');\n    }, 500);\n  }\n\n  private verify(): void {\n    if (this.currentPin === LockScreenController.CORRECT_PIN) {\n      this.hide();\n    } else {\n      this.showError();\n    }\n  }\n\n  private addDigit(digit: string): void {\n    if (this.currentPin.length >= 4) return;\n\n    this.currentPin += digit;\n    this.updateDisplay();\n\n    // Auto-verify when 4 digits entered\n    if (this.currentPin.length === 4) {\n      setTimeout(() => this.verify(), 200);\n    }\n  }\n\n  private removeDigit(): void {\n    if (this.currentPin.length === 0) return;\n    this.currentPin = this.currentPin.slice(0, -1);\n    this.updateDisplay();\n  }\n\n  private clearPin(): void {\n    this.currentPin = '';\n    this.updateDisplay();\n  }\n\n  private setupListeners(): void {\n    // Keypad click handler\n    this.pinKeypad?.addEventListener('click', (e) => this.handleKeypadClick(e));\n\n    // Keyboard input\n    document.addEventListener('keydown', (e) => this.handleKeyboard(e));\n\n    // Lock button in sidebar\n    document.querySelector<HTMLElement>('swp-side-menu-action.lock')\n      ?.addEventListener('click', () => this.show());\n  }\n\n  private handleKeypadClick(e: Event): void {\n    const target = e.target as HTMLElement;\n    const key = target.closest<HTMLElement>('swp-pin-key');\n\n    if (!key) return;\n\n    const digit = key.dataset.digit;\n    const action = key.dataset.action;\n\n    if (digit) {\n      this.addDigit(digit);\n    } else if (action === 'backspace') {\n      this.removeDigit();\n    } else if (action === 'clear') {\n      this.clearPin();\n    }\n  }\n\n  private handleKeyboard(e: KeyboardEvent): void {\n    if (!this.isActive) return;\n\n    // Prevent default to avoid other interactions\n    e.preventDefault();\n\n    if (e.key >= '0' && e.key <= '9') {\n      this.addDigit(e.key);\n    } else if (e.key === 'Backspace') {\n      this.removeDigit();\n    } else if (e.key === 'Escape') {\n      this.clearPin();\n    }\n  }\n}\n", "/**\n * Cash Controller\n *\n * Handles tab switching, cash calculations, and form interactions\n * for the Cash Register page.\n */\n\nexport class CashController {\n  // Base values (from system - would come from server in real app)\n  private readonly startBalance = 2000;\n  private readonly cashSales = 3540;\n\n  constructor() {\n    this.setupTabs();\n    this.setupCashCalculation();\n    this.setupCheckboxSelection();\n    this.setupApprovalCheckbox();\n    this.setupDateFilters();\n    this.setupRowToggle();\n    this.setupDraftRowClick();\n  }\n\n  /**\n   * Setup tab switching functionality\n   */\n  private setupTabs(): void {\n    const tabs = document.querySelectorAll<HTMLElement>('swp-tab[data-tab]');\n\n    tabs.forEach(tab => {\n      tab.addEventListener('click', () => {\n        const targetTab = tab.dataset.tab;\n        if (targetTab) {\n          this.switchToTab(targetTab);\n        }\n      });\n    });\n  }\n\n  /**\n   * Switch to a specific tab by name\n   */\n  private switchToTab(targetTab: string): void {\n    const tabs = document.querySelectorAll<HTMLElement>('swp-tab[data-tab]');\n    const contents = document.querySelectorAll<HTMLElement>('swp-tab-content[data-tab]');\n    const statsBars = document.querySelectorAll<HTMLElement>('swp-cash-stats[data-for-tab]');\n\n    // Update tab states\n    tabs.forEach(t => {\n      if (t.dataset.tab === targetTab) {\n        t.classList.add('active');\n      } else {\n        t.classList.remove('active');\n      }\n    });\n\n    // Update content visibility\n    contents.forEach(content => {\n      if (content.dataset.tab === targetTab) {\n        content.classList.add('active');\n      } else {\n        content.classList.remove('active');\n      }\n    });\n\n    // Update stats bar visibility\n    statsBars.forEach(stats => {\n      if (stats.dataset.forTab === targetTab) {\n        stats.classList.add('active');\n      } else {\n        stats.classList.remove('active');\n      }\n    });\n  }\n\n  /**\n   * Setup cash calculation with real-time updates\n   */\n  private setupCashCalculation(): void {\n    const payoutsInput = document.getElementById('payouts') as HTMLInputElement;\n    const toBankInput = document.getElementById('toBank') as HTMLInputElement;\n    const actualCashInput = document.getElementById('actualCash') as HTMLInputElement;\n\n    if (!payoutsInput || !toBankInput || !actualCashInput) return;\n\n    const calculate = () => this.calculateCash(payoutsInput, toBankInput, actualCashInput);\n\n    payoutsInput.addEventListener('input', calculate);\n    toBankInput.addEventListener('input', calculate);\n    actualCashInput.addEventListener('input', calculate);\n\n    // Initial calculation\n    calculate();\n  }\n\n  /**\n   * Calculate expected cash and difference\n   */\n  private calculateCash(\n    payoutsInput: HTMLInputElement,\n    toBankInput: HTMLInputElement,\n    actualCashInput: HTMLInputElement\n  ): void {\n    const payouts = this.parseNumber(payoutsInput.value);\n    const toBank = this.parseNumber(toBankInput.value);\n    const actual = this.parseNumber(actualCashInput.value);\n\n    // Expected = start + sales - payouts - to bank\n    const expectedCash = this.startBalance + this.cashSales - payouts - toBank;\n\n    const expectedElement = document.getElementById('expectedCash');\n    if (expectedElement) {\n      expectedElement.textContent = this.formatNumber(expectedCash);\n    }\n\n    // Calculate and display difference\n    this.updateDifference(actual, expectedCash, actualCashInput.value);\n  }\n\n  /**\n   * Update difference box with color coding\n   */\n  private updateDifference(actual: number, expected: number, rawValue: string): void {\n    const box = document.getElementById('differenceBox');\n    const value = document.getElementById('differenceValue');\n    if (!box || !value) return;\n\n    const diff = actual - expected;\n\n    // Remove all state classes\n    box.classList.remove('positive', 'negative', 'neutral');\n\n    if (actual === 0 && rawValue === '') {\n      // No input yet\n      value.textContent = '\u2013 kr';\n      box.classList.add('neutral');\n    } else if (diff > 0) {\n      // More cash than expected\n      value.textContent = '+' + this.formatNumber(diff) + ' kr';\n      box.classList.add('positive');\n    } else if (diff < 0) {\n      // Less cash than expected\n      value.textContent = this.formatNumber(diff) + ' kr';\n      box.classList.add('negative');\n    } else {\n      // Exact match\n      value.textContent = '0,00 kr';\n      box.classList.add('neutral');\n    }\n  }\n\n  /**\n   * Setup checkbox selection for table rows\n   */\n  private setupCheckboxSelection(): void {\n    const selectAll = document.getElementById('selectAll') as HTMLInputElement;\n    const rowCheckboxes = document.querySelectorAll<HTMLInputElement>('.row-select');\n    const exportBtn = document.getElementById('exportBtn') as HTMLButtonElement;\n    const selectionCount = document.getElementById('selectionCount');\n\n    if (!selectAll || !exportBtn || !selectionCount) return;\n\n    const updateSelection = () => {\n      const checked = document.querySelectorAll<HTMLInputElement>('.row-select:checked');\n      const count = checked.length;\n\n      selectionCount.textContent = count === 0 ? '0 valgt' : `${count} valgt`;\n      exportBtn.disabled = count === 0;\n\n      // Update select all state\n      selectAll.checked = count === rowCheckboxes.length && count > 0;\n      selectAll.indeterminate = count > 0 && count < rowCheckboxes.length;\n    };\n\n    selectAll.addEventListener('change', () => {\n      rowCheckboxes.forEach(cb => cb.checked = selectAll.checked);\n      updateSelection();\n    });\n\n    rowCheckboxes.forEach(cb => {\n      cb.addEventListener('change', updateSelection);\n      // Stop click from bubbling to row\n      cb.addEventListener('click', e => e.stopPropagation());\n    });\n  }\n\n  /**\n   * Setup approval checkbox to enable/disable approve button\n   */\n  private setupApprovalCheckbox(): void {\n    const checkbox = document.getElementById('confirmCheckbox') as HTMLInputElement;\n    const approveBtn = document.getElementById('approveBtn') as HTMLButtonElement;\n\n    if (!checkbox || !approveBtn) return;\n\n    checkbox.addEventListener('change', () => {\n      approveBtn.disabled = !checkbox.checked;\n    });\n  }\n\n  /**\n   * Setup date filter defaults (last 30 days)\n   */\n  private setupDateFilters(): void {\n    const dateFrom = document.getElementById('dateFrom') as HTMLInputElement;\n    const dateTo = document.getElementById('dateTo') as HTMLInputElement;\n\n    if (!dateFrom || !dateTo) return;\n\n    const today = new Date();\n    const thirtyDaysAgo = new Date(today);\n    thirtyDaysAgo.setDate(today.getDate() - 30);\n\n    dateTo.value = this.formatDateISO(today);\n    dateFrom.value = this.formatDateISO(thirtyDaysAgo);\n  }\n\n  /**\n   * Format number as Danish currency\n   */\n  private formatNumber(num: number): string {\n    return num.toLocaleString('da-DK', {\n      minimumFractionDigits: 2,\n      maximumFractionDigits: 2\n    });\n  }\n\n  /**\n   * Parse Danish number format\n   */\n  private parseNumber(str: string): number {\n    if (!str) return 0;\n    return parseFloat(str.replace(/\\./g, '').replace(',', '.')) || 0;\n  }\n\n  /**\n   * Format date as ISO string (YYYY-MM-DD)\n   */\n  private formatDateISO(date: Date): string {\n    return date.toISOString().split('T')[0];\n  }\n\n  /**\n   * Setup row toggle for expandable details\n   */\n  private setupRowToggle(): void {\n    const rows = document.querySelectorAll<HTMLElement>('swp-cash-table-row[data-id]:not(.draft-row)');\n\n    rows.forEach(row => {\n      const rowId = row.getAttribute('data-id');\n      if (!rowId) return;\n\n      const detail = document.querySelector<HTMLElement>(`swp-cash-row-detail[data-for=\"${rowId}\"]`);\n      if (!detail) return;\n\n      row.addEventListener('click', (e) => {\n        // Don't toggle if clicking on checkbox\n        if ((e.target as HTMLElement).closest('input[type=\"checkbox\"]')) return;\n\n        const icon = row.querySelector('swp-row-toggle i');\n        const isExpanded = row.classList.contains('expanded');\n\n        // Close other expanded rows\n        document.querySelectorAll('swp-cash-table-row.expanded').forEach(r => {\n          if (r !== row) {\n            const otherId = r.getAttribute('data-id');\n            if (otherId) {\n              const otherDetail = document.querySelector<HTMLElement>(`swp-cash-row-detail[data-for=\"${otherId}\"]`);\n              const otherIcon = r.querySelector('swp-row-toggle i');\n              if (otherDetail && otherIcon) {\n                this.collapseRow(r, otherDetail, otherIcon as HTMLElement);\n              }\n            }\n          }\n        });\n\n        // Toggle current row\n        if (isExpanded) {\n          this.collapseRow(row, detail, icon);\n        } else {\n          this.expandRow(row, detail, icon);\n        }\n      });\n    });\n  }\n\n  /**\n   * Expand a row with animation\n   */\n  private expandRow(row: Element, detail: HTMLElement, icon: Element | null): void {\n    row.classList.add('expanded');\n    detail.classList.add('expanded');\n\n    // Animate icon rotation\n    icon?.animate([\n      { transform: 'rotate(0deg)' },\n      { transform: 'rotate(90deg)' }\n    ], {\n      duration: 200,\n      easing: 'ease-out',\n      fill: 'forwards'\n    });\n\n    // Animate detail expansion\n    const content = detail.querySelector('swp-row-detail-content') as HTMLElement;\n    if (content) {\n      const height = content.offsetHeight;\n      detail.animate([\n        { height: '0px', opacity: 0 },\n        { height: `${height}px`, opacity: 1 }\n      ], {\n        duration: 250,\n        easing: 'ease-out',\n        fill: 'forwards'\n      });\n    }\n  }\n\n  /**\n   * Collapse a row with animation\n   */\n  private collapseRow(row: Element, detail: HTMLElement, icon: Element | null): void {\n    // Animate icon rotation\n    icon?.animate([\n      { transform: 'rotate(90deg)' },\n      { transform: 'rotate(0deg)' }\n    ], {\n      duration: 200,\n      easing: 'ease-out',\n      fill: 'forwards'\n    });\n\n    // Animate detail collapse\n    const content = detail.querySelector('swp-row-detail-content') as HTMLElement;\n    if (content) {\n      const height = content.offsetHeight;\n      const animation = detail.animate([\n        { height: `${height}px`, opacity: 1 },\n        { height: '0px', opacity: 0 }\n      ], {\n        duration: 200,\n        easing: 'ease-out',\n        fill: 'forwards'\n      });\n\n      animation.onfinish = () => {\n        row.classList.remove('expanded');\n        detail.classList.remove('expanded');\n      };\n    } else {\n      row.classList.remove('expanded');\n      detail.classList.remove('expanded');\n    }\n  }\n\n  /**\n   * Setup draft row click to navigate to reconciliation tab\n   */\n  private setupDraftRowClick(): void {\n    const draftRow = document.querySelector<HTMLElement>('swp-cash-table-row.draft-row');\n    if (!draftRow) return;\n\n    draftRow.style.cursor = 'pointer';\n    draftRow.addEventListener('click', (e) => {\n      // Don't navigate if clicking on checkbox\n      if ((e.target as HTMLElement).closest('input[type=\"checkbox\"]')) return;\n\n      this.switchToTab('afstemning');\n    });\n  }\n}\n", "/**\n * Employees Controller\n *\n * Handles content swap between list view and detail view,\n * plus tab switching within each view.\n * Uses History API for browser back/forward navigation.\n */\n\nexport class EmployeesController {\n  private ratesSync: RatesSyncController | null = null;\n  private listView: HTMLElement | null = null;\n  private detailView: HTMLElement | null = null;\n\n  constructor() {\n    this.listView = document.getElementById('employees-list-view');\n    this.detailView = document.getElementById('employee-detail-view');\n\n    // Only initialize if we're on the employees page\n    if (!this.listView) return;\n\n    this.setupListTabs();\n    this.setupDetailTabs();\n    this.setupChevronNavigation();\n    this.setupBackNavigation();\n    this.setupHistoryNavigation();\n    this.restoreStateFromUrl();\n    this.ratesSync = new RatesSyncController();\n  }\n\n  /**\n   * Setup popstate listener for browser back/forward\n   */\n  private setupHistoryNavigation(): void {\n    window.addEventListener('popstate', (e: PopStateEvent) => {\n      if (e.state?.employeeKey) {\n        this.showDetailViewInternal(e.state.employeeKey);\n      } else {\n        this.showListViewInternal();\n      }\n    });\n  }\n\n  /**\n   * Restore view state from URL on page load\n   */\n  private restoreStateFromUrl(): void {\n    const hash = window.location.hash;\n    if (hash.startsWith('#employee-')) {\n      const employeeKey = hash.substring(1); // Remove #\n      this.showDetailViewInternal(employeeKey);\n    }\n  }\n\n  /**\n   * Setup tab switching for the list view\n   */\n  private setupListTabs(): void {\n    if (!this.listView) return;\n\n    const tabs = this.listView.querySelectorAll<HTMLElement>('swp-tab-bar > swp-tab[data-tab]');\n\n    tabs.forEach(tab => {\n      tab.addEventListener('click', () => {\n        const targetTab = tab.dataset.tab;\n        if (targetTab) {\n          this.switchTab(this.listView!, targetTab);\n        }\n      });\n    });\n  }\n\n  /**\n   * Setup tab switching for the detail view\n   */\n  private setupDetailTabs(): void {\n    if (!this.detailView) return;\n\n    const tabs = this.detailView.querySelectorAll<HTMLElement>('swp-tab-bar > swp-tab[data-tab]');\n\n    tabs.forEach(tab => {\n      tab.addEventListener('click', () => {\n        const targetTab = tab.dataset.tab;\n        if (targetTab) {\n          this.switchTab(this.detailView!, targetTab);\n        }\n      });\n    });\n  }\n\n  /**\n   * Switch to a specific tab within a container\n   */\n  private switchTab(container: HTMLElement, targetTab: string): void {\n    const tabs = container.querySelectorAll<HTMLElement>('swp-tab-bar > swp-tab[data-tab]');\n    const contents = container.querySelectorAll<HTMLElement>('swp-tab-content[data-tab]');\n\n    tabs.forEach(t => {\n      t.classList.toggle('active', t.dataset.tab === targetTab);\n    });\n\n    contents.forEach(content => {\n      content.classList.toggle('active', content.dataset.tab === targetTab);\n    });\n  }\n\n  /**\n   * Setup row click to show detail view\n   * Ignores clicks on action buttons\n   */\n  private setupChevronNavigation(): void {\n    document.addEventListener('click', (e: Event) => {\n      const target = e.target as HTMLElement;\n\n      // Ignore clicks on action buttons\n      if (target.closest('swp-icon-btn') || target.closest('swp-table-actions')) {\n        return;\n      }\n\n      const row = target.closest<HTMLElement>('swp-data-table-row[data-employee-detail]');\n\n      if (row) {\n        const employeeKey = row.dataset.employeeDetail;\n        if (employeeKey) {\n          this.showDetailView(employeeKey);\n        }\n      }\n    });\n  }\n\n  /**\n   * Setup back button to return to list view\n   */\n  private setupBackNavigation(): void {\n    document.addEventListener('click', (e: Event) => {\n      const target = e.target as HTMLElement;\n      const backLink = target.closest<HTMLElement>('[data-employee-back]');\n\n      if (backLink) {\n        this.showListView();\n      }\n    });\n  }\n\n  /**\n   * Show the detail view and hide list view (with history push)\n   */\n  private showDetailView(employeeKey: string): void {\n    // Push state to history\n    history.pushState(\n      { employeeKey },\n      '',\n      `#${employeeKey}`\n    );\n    this.showDetailViewInternal(employeeKey);\n  }\n\n  /**\n   * Show detail view without modifying history (for popstate)\n   */\n  private showDetailViewInternal(employeeKey: string): void {\n    if (this.listView && this.detailView) {\n      this.listView.style.display = 'none';\n      this.detailView.style.display = 'block';\n      this.detailView.dataset.employee = employeeKey;\n\n      // Reset to first tab\n      this.switchTab(this.detailView, 'general');\n    }\n  }\n\n  /**\n   * Show the list view and hide detail view (with history push)\n   */\n  private showListView(): void {\n    // Push state to history (clear hash)\n    history.pushState(\n      {},\n      '',\n      window.location.pathname\n    );\n    this.showListViewInternal();\n  }\n\n  /**\n   * Show list view without modifying history (for popstate)\n   */\n  private showListViewInternal(): void {\n    if (this.listView && this.detailView) {\n      this.detailView.style.display = 'none';\n      this.listView.style.display = 'block';\n    }\n  }\n}\n\n/**\n * Rates Sync Controller\n *\n * Syncs changes between the rates drawer and the salary tab cards.\n * Uses ID-based lookups:\n * - Checkbox: id=\"rate-{key}-enabled\"\n * - Text input: id=\"rate-{key}\"\n * - Card row: id=\"card-{key}\"\n */\nclass RatesSyncController {\n  private drawer: HTMLElement | null = null;\n\n  constructor() {\n    this.drawer = document.getElementById('rates-drawer');\n\n    if (!this.drawer) return;\n\n    this.setupCheckboxListeners();\n    this.setupInputListeners();\n  }\n\n  /**\n   * Extract rate key from checkbox ID (e.g., \"rate-normal-enabled\" \u2192 \"normal\")\n   */\n  private extractRateKey(checkboxId: string): string | null {\n    const match = checkboxId.match(/^rate-(.+)-enabled$/);\n    return match ? match[1] : null;\n  }\n\n  /**\n   * Setup checkbox change listeners in drawer\n   */\n  private setupCheckboxListeners(): void {\n    if (!this.drawer) return;\n\n    this.drawer.addEventListener('change', (e: Event) => {\n      const target = e.target as HTMLInputElement;\n      if (target.type !== 'checkbox' || !target.id) return;\n\n      const rateKey = this.extractRateKey(target.id);\n      if (!rateKey) return;\n\n      const isChecked = target.checked;\n      const row = target.closest<HTMLElement>('swp-data-row');\n      if (!row) return;\n\n      // Toggle disabled class in drawer row\n      const label = row.querySelector('swp-data-label');\n      const input = row.querySelector('swp-data-input');\n      if (label) label.classList.toggle('disabled', !isChecked);\n      if (input) input.classList.toggle('disabled', !isChecked);\n\n      // Toggle visibility in card\n      this.toggleCardRow(rateKey, isChecked);\n\n      // If enabling, also sync the current value\n      if (isChecked) {\n        const textInput = document.getElementById(`rate-${rateKey}`) as HTMLInputElement | null;\n        if (textInput) {\n          this.syncValueToCard(rateKey, textInput.value);\n        }\n      }\n    });\n  }\n\n  /**\n   * Setup input change listeners in drawer\n   */\n  private setupInputListeners(): void {\n    if (!this.drawer) return;\n\n    this.drawer.addEventListener('input', (e: Event) => {\n      const target = e.target as HTMLInputElement;\n      if (target.type !== 'text' || !target.id) return;\n\n      // Extract rate key from input ID (e.g., \"rate-normal\" \u2192 \"normal\")\n      const match = target.id.match(/^rate-(.+)$/);\n      if (!match) return;\n\n      const rateKey = match[1];\n      // Skip if this matches the checkbox pattern\n      if (rateKey.endsWith('-enabled')) return;\n\n      this.syncValueToCard(rateKey, target.value);\n    });\n  }\n\n  /**\n   * Toggle card row visibility by ID\n   */\n  private toggleCardRow(rateKey: string, visible: boolean): void {\n    const cardRow = document.getElementById(`card-${rateKey}`);\n    if (cardRow) {\n      cardRow.style.display = visible ? '' : 'none';\n    }\n  }\n\n  /**\n   * Format number with 2 decimals using Danish locale (comma as decimal separator)\n   */\n  private formatNumber(value: string): string {\n    // Parse the input (handle both dot and comma as decimal separator)\n    const normalized = value.replace(',', '.');\n    const num = parseFloat(normalized);\n\n    if (isNaN(num)) return value;\n\n    // Format with 2 decimals and comma as decimal separator\n    return num.toFixed(2).replace('.', ',');\n  }\n\n  /**\n   * Sync value from drawer to card by ID\n   */\n  private syncValueToCard(rateKey: string, value: string): void {\n    const cardInput = document.getElementById(`value-${rateKey}`) as HTMLInputElement | null;\n    if (!cardInput) return;\n\n    // Get the unit from drawer input container\n    const textInput = document.getElementById(`rate-${rateKey}`);\n    const inputContainer = textInput?.closest('swp-data-input');\n    const unit = inputContainer?.textContent?.trim().replace(value, '').trim() || 'kr';\n\n    // Format with 2 decimals\n    const formattedValue = this.formatNumber(value);\n    cardInput.value = `${formattedValue} ${unit}`;\n  }\n}\n", "/**\n * Salon OS App\n *\n * Main application class that orchestrates all UI controllers\n */\n\nimport { SidebarController } from './modules/sidebar';\nimport { DrawerController } from './modules/drawers';\nimport { ThemeController } from './modules/theme';\nimport { SearchController } from './modules/search';\nimport { LockScreenController } from './modules/lockscreen';\nimport { CashController } from './modules/cash';\nimport { EmployeesController } from './modules/employees';\n\n/**\n * Main application class\n */\nexport class App {\n  readonly sidebar: SidebarController;\n  readonly drawers: DrawerController;\n  readonly theme: ThemeController;\n  readonly search: SearchController;\n  readonly lockScreen: LockScreenController;\n  readonly cash: CashController;\n  readonly employees: EmployeesController;\n\n  constructor() {\n    // Initialize controllers\n    this.sidebar = new SidebarController();\n    this.drawers = new DrawerController();\n    this.theme = new ThemeController();\n    this.search = new SearchController();\n    this.lockScreen = new LockScreenController(this.drawers);\n    this.cash = new CashController();\n    this.employees = new EmployeesController();\n  }\n}\n\n/**\n * Global app instance\n */\nlet app: App;\n\n/**\n * Initialize the application\n */\nfunction init(): void {\n  app = new App();\n\n  // Expose to window for debugging\n  if (typeof window !== 'undefined') {\n    (window as unknown as { app: App }).app = app;\n  }\n}\n\n// Wait for DOM ready\nif (document.readyState === 'loading') {\n  document.addEventListener('DOMContentLoaded', init);\n} else {\n  init();\n}\n\nexport { app };\nexport default App;\n"],
  "mappings": "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",
  "names": []
}

diff --git a/PlanTempus.Application/wwwroot/ts/modules/employees.ts b/PlanTempus.Application/wwwroot/ts/modules/employees.ts
index 7c575c2..99b87a1 100644
--- a/PlanTempus.Application/wwwroot/ts/modules/employees.ts
+++ b/PlanTempus.Application/wwwroot/ts/modules/employees.ts
@@ -211,6 +211,7 @@ class RatesSyncController {
this.setupCheckboxListeners();
this.setupInputListeners();
+ this.setupDoubleClickToEdit();
}
/**
@@ -319,4 +320,50 @@ class RatesSyncController {
const formattedValue = this.formatNumber(value);
cardInput.value = `${formattedValue} ${unit}`;
}
+
+ /**
+ * Setup double-click on salary card inputs to open drawer and focus field
+ */
+ private setupDoubleClickToEdit(): void {
+ document.addEventListener('dblclick', (e: Event) => {
+ const target = e.target as HTMLElement;
+ const input = target.closest('input[id^="value-"]');
+
+ if (!input || !input.id) return;
+
+ // Extract key from value-{key}
+ const match = input.id.match(/^value-(.+)$/);
+ if (!match) return;
+
+ const rateKey = match[1];
+ this.openDrawerAndFocus(rateKey);
+ });
+ }
+
+ /**
+ * Open drawer and focus the corresponding field with highlight
+ */
+ private openDrawerAndFocus(rateKey: string): void {
+ // Open the drawer
+ const trigger = document.querySelector('[data-drawer-trigger="rates-drawer"]');
+ trigger?.click();
+
+ // Wait for drawer to open, then focus field
+ requestAnimationFrame(() => {
+ const drawerInput = document.getElementById(`rate-${rateKey}`) as HTMLInputElement | null;
+ if (!drawerInput) return;
+
+ // Focus the input
+ drawerInput.focus();
+ drawerInput.select();
+
+ // Add highlight to row
+ const row = drawerInput.closest('swp-data-row');
+ if (row) {
+ row.classList.add('focus-highlight');
+ // Remove class after animation
+ setTimeout(() => row.classList.remove('focus-highlight'), 1000);
+ }
+ });
+ }
}