feat: NAS storage for invoices/offers, code cleanup, date/time fixes

- NAS storage for created invoices (PDF via puppeteer), received invoices,
  and offers with auto-save on create/edit
- Deterministic file paths derived from DB fields (no file_path column needed)
- Separate NAS mount points: NAS_FINANCIALS_PATH, NAS_OFFERS_PATH
- Invoice language field (cs/en) stored per invoice, replaces lang modal
- Invoices list filtered by month/year matching KPI card selection
- Centralized date helpers (src/utils/date.ts) replacing all .toISOString()
  calls that returned UTC instead of local time
- Attendance project switching uses exact time (not rounded)
- Comment cleanup: removed ~100 unnecessary/Czech comments
- Removed as-any casts in orders and attendance
- Prisma migrations: add invoice language, drop received_invoices BLOB columns

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
BOHA
2026-03-26 10:36:39 +01:00
parent 0317ba3168
commit baceb88347
60 changed files with 2475 additions and 563 deletions

View File

@@ -18,7 +18,7 @@
}
.admin-stat-card::before {
content: '';
content: "";
position: absolute;
top: 0;
left: 0;
@@ -28,10 +28,18 @@
border-radius: var(--border-radius) var(--border-radius) 0 0;
}
.admin-stat-card.success::before { background: var(--success); }
.admin-stat-card.warning::before { background: var(--warning); }
.admin-stat-card.danger::before { background: var(--danger); }
.admin-stat-card.info::before { background: var(--info); }
.admin-stat-card.success::before {
background: var(--success);
}
.admin-stat-card.warning::before {
background: var(--warning);
}
.admin-stat-card.danger::before {
background: var(--danger);
}
.admin-stat-card.info::before {
background: var(--info);
}
.admin-stat-icon {
width: 40px;
@@ -73,10 +81,22 @@
color: var(--text-secondary);
}
.admin-stat-icon.danger { background: var(--danger-soft); color: var(--danger); }
.admin-stat-icon.info { background: var(--info-soft); color: var(--info); }
.admin-stat-icon.success { background: var(--success-soft); color: var(--success); }
.admin-stat-icon.warning { background: var(--warning-soft); color: var(--warning); }
.admin-stat-icon.danger {
background: var(--danger-soft);
color: var(--danger);
}
.admin-stat-icon.info {
background: var(--info-soft);
color: var(--info);
}
.admin-stat-icon.success {
background: var(--success-soft);
color: var(--success);
}
.admin-stat-icon.warning {
background: var(--warning-soft);
color: var(--warning);
}
/* ============================================================================
Dashboard
@@ -99,10 +119,19 @@
gap: 0.875rem;
}
.dash-kpi-4 { grid-template-columns: repeat(4, 1fr); }
.dash-kpi-3 { grid-template-columns: repeat(3, 1fr); }
.dash-kpi-2 { grid-template-columns: repeat(2, 1fr); }
.dash-kpi-1 { grid-template-columns: 1fr; max-width: 320px; }
.dash-kpi-4 {
grid-template-columns: repeat(4, 1fr);
}
.dash-kpi-3 {
grid-template-columns: repeat(3, 1fr);
}
.dash-kpi-2 {
grid-template-columns: repeat(2, 1fr);
}
.dash-kpi-1 {
grid-template-columns: 1fr;
max-width: 320px;
}
/* Quick actions */
.dash-quick-actions {
@@ -134,20 +163,44 @@
transform: none !important;
}
.dash-quick-btn-success { background: var(--success-soft); color: var(--success); }
.dash-quick-btn-info { background: var(--info-soft); color: var(--info); }
.dash-quick-btn-warning { background: var(--warning-soft); color: var(--warning); }
.dash-quick-btn-danger { background: var(--danger-soft); color: var(--danger); }
.dash-quick-btn-success {
background: var(--success-soft);
color: var(--success);
}
.dash-quick-btn-info {
background: var(--info-soft);
color: var(--info);
}
.dash-quick-btn-warning {
background: var(--warning-soft);
color: var(--warning);
}
.dash-quick-btn-danger {
background: var(--danger-soft);
color: var(--danger);
}
.dash-quick-btn:hover {
transform: translateY(-1px);
filter: brightness(0.95);
}
[data-theme="light"] .dash-quick-btn-success { background: var(--success); color: #fff; }
[data-theme="light"] .dash-quick-btn-info { background: var(--info); color: #fff; }
[data-theme="light"] .dash-quick-btn-warning { background: var(--warning); color: #fff; }
[data-theme="light"] .dash-quick-btn-danger { background: var(--danger); color: #fff; }
[data-theme="light"] .dash-quick-btn-success {
background: var(--success);
color: #fff;
}
[data-theme="light"] .dash-quick-btn-info {
background: var(--info);
color: #fff;
}
[data-theme="light"] .dash-quick-btn-warning {
background: var(--warning);
color: #fff;
}
[data-theme="light"] .dash-quick-btn-danger {
background: var(--danger);
color: #fff;
}
/* Main content 3-col grid */
.dash-main-grid {
@@ -197,12 +250,30 @@
flex-shrink: 0;
}
.dash-activity-icon.success { background: var(--success-soft); color: var(--success); }
.dash-activity-icon.info { background: var(--info-soft); color: var(--info); }
.dash-activity-icon.warning { background: var(--warning-soft); color: var(--warning); }
.dash-activity-icon.danger { background: var(--danger-soft); color: var(--danger); }
.dash-activity-icon.accent { background: var(--accent-soft); color: var(--accent-color); }
.dash-activity-icon.muted { background: var(--bg-tertiary); color: var(--text-secondary); }
.dash-activity-icon.success {
background: var(--success-soft);
color: var(--success);
}
.dash-activity-icon.info {
background: var(--info-soft);
color: var(--info);
}
.dash-activity-icon.warning {
background: var(--warning-soft);
color: var(--warning);
}
.dash-activity-icon.danger {
background: var(--danger-soft);
color: var(--danger);
}
.dash-activity-icon.accent {
background: var(--accent-soft);
color: var(--accent-color);
}
.dash-activity-icon.muted {
background: var(--bg-tertiary);
color: var(--text-secondary);
}
.dash-activity-main {
flex: 1;
@@ -256,10 +327,22 @@
text-transform: uppercase;
}
.dash-presence-avatar.dash-status-in { background: var(--success-soft); color: var(--success); }
.dash-presence-avatar.dash-status-away { background: var(--warning-soft); color: var(--warning); }
.dash-presence-avatar.dash-status-out { background: var(--bg-tertiary); color: var(--text-muted); }
.dash-presence-avatar.dash-status-leave { background: var(--info-soft); color: var(--info); }
.dash-presence-avatar.dash-status-in {
background: var(--success-soft);
color: var(--success);
}
.dash-presence-avatar.dash-status-away {
background: var(--warning-soft);
color: var(--warning);
}
.dash-presence-avatar.dash-status-out {
background: var(--bg-tertiary);
color: var(--text-muted);
}
.dash-presence-avatar.dash-status-leave {
background: var(--info-soft);
color: var(--info);
}
.dash-status-dot {
width: 8px;
@@ -268,15 +351,31 @@
flex-shrink: 0;
}
.dash-status-dot.dash-status-in { background: var(--success); }
.dash-status-dot.dash-status-away { background: var(--warning); }
.dash-status-dot.dash-status-out { background: var(--text-muted); }
.dash-status-dot.dash-status-leave { background: var(--info); }
.dash-status-dot.dash-status-in {
background: var(--success);
}
.dash-status-dot.dash-status-away {
background: var(--warning);
}
.dash-status-dot.dash-status-out {
background: var(--text-muted);
}
.dash-status-dot.dash-status-leave {
background: var(--info);
}
.dash-presence-label.dash-status-in { color: var(--success); }
.dash-presence-label.dash-status-away { color: var(--warning); }
.dash-presence-label.dash-status-out { color: var(--text-muted); }
.dash-presence-label.dash-status-leave { color: var(--info); }
.dash-presence-label.dash-status-in {
color: var(--success);
}
.dash-presence-label.dash-status-away {
color: var(--warning);
}
.dash-presence-label.dash-status-out {
color: var(--text-muted);
}
.dash-presence-label.dash-status-leave {
color: var(--info);
}
.dash-presence-name {
flex: 1;
@@ -414,12 +513,18 @@
grid-template-columns: 1fr 1fr;
}
.dash-kpi-4 { grid-template-columns: repeat(2, 1fr); }
.dash-kpi-4 {
grid-template-columns: repeat(2, 1fr);
}
}
@media (max-width: 768px) {
.dash-kpi-grid { grid-template-columns: repeat(2, 1fr); }
.dash-quick-actions { grid-template-columns: repeat(2, 1fr); }
.dash-kpi-grid {
grid-template-columns: repeat(2, 1fr);
}
.dash-quick-actions {
grid-template-columns: repeat(2, 1fr);
}
.dash-main-grid {
grid-template-columns: 1fr;
@@ -435,9 +540,15 @@
}
@media (max-width: 480px) {
.dash-quick-actions { grid-template-columns: 1fr 1fr; }
.dash-kpi-grid { grid-template-columns: 1fr; }
.dash-profile-grid { grid-template-columns: 1fr; }
.dash-quick-actions {
grid-template-columns: 1fr 1fr;
}
.dash-kpi-grid {
grid-template-columns: 1fr;
}
.dash-profile-grid {
grid-template-columns: 1fr;
}
}
/* ============================================================================