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:
@@ -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;
|
||||
}
|
||||
}
|
||||
|
||||
/* ============================================================================
|
||||
|
||||
Reference in New Issue
Block a user