/* Backup Page — restic-engine UI */ .backup-layout { display: flex; min-height: calc(100vh - var(--topbar-height, 60px)); } .backup-layout .main { flex: 1; min-width: 0; overflow-y: auto; } .backup-page { color: var(--text-primary); width: 100%; padding-bottom: 48px; } /* The whole backup page is one .config-section card containing both the .page-header and the body. Remove the card's inner padding so the .page-header sits flush at the top and its border-bottom acts as a full-width divider; the body gets its own padding. */ .backup-page-section { padding: 0; overflow: hidden; } .backup-page-section > .page-header { margin-bottom: 0; } .backup-page-body { padding: 22px; } /* Configuration tab embeds /config's renderConfig, which emits its own .page-header. The outer backup page already has one, so suppress the embedded one to avoid the duplicate "Backup" heading. */ .backup-embedded-config > .page-header { display: none; } /* SVG icon slot inside the shared .page-header (defined in config.css). */ .page-header-icon-slot { width: 36px; height: 36px; flex-shrink: 0; color: var(--accent); display: flex; align-items: center; justify-content: center; } .page-header-icon-slot svg { width: 32px; height: 32px; } .backup-engine-badge { background: linear-gradient(135deg, var(--accent), var(--accent-hover)); color: var(--text-primary); padding: 4px 10px; border-radius: 999px; font-size: 0.7rem; font-weight: 600; letter-spacing: 0.05em; text-transform: uppercase; } .backup-primary-btn, .backup-secondary-btn, .backup-danger-btn, .backup-refresh-btn { display: inline-flex; align-items: center; gap: 8px; padding: 9px 16px; border-radius: 8px; border: none; font-size: 0.875rem; font-weight: 500; cursor: pointer; transition: transform 0.12s ease, background 0.12s ease, box-shadow 0.12s ease; } .backup-primary-btn { background: linear-gradient(135deg, var(--accent), var(--accent-hover)); color: var(--text-primary); box-shadow: 0 4px 12px rgba(var(--accent-rgb), 0.25); } .backup-primary-btn:hover { transform: translateY(-1px); box-shadow: 0 6px 18px rgba(var(--accent-rgb), 0.35); } .backup-secondary-btn, .backup-refresh-btn { background: rgba(var(--text-rgb), 0.06); color: var(--text-primary); border: 1px solid rgba(var(--text-rgb), 0.12); } .backup-secondary-btn:hover, .backup-refresh-btn:hover { background: rgba(var(--text-rgb), 0.1); } .backup-danger-btn { background: linear-gradient(135deg, #dc2626, #b91c1c); color: #fff; } .backup-danger-btn:hover { transform: translateY(-1px); box-shadow: 0 6px 18px rgba(220, 38, 38, 0.35); } .backup-tabpanel { display: none; } .backup-tabpanel.active { display: block; animation: backupFadeIn 0.25s ease; } @keyframes backupFadeIn { from { opacity: 0; transform: translateY(4px); } to { opacity: 1; transform: translateY(0); } } .backup-summary-row { display: grid; grid-template-columns: repeat(auto-fit, minmax(220px, 1fr)); gap: 16px; margin-bottom: 24px; } .backup-summary-tile { background: rgba(var(--text-rgb), 0.04); border: 1px solid rgba(var(--text-rgb), 0.08); border-radius: 14px; padding: 18px 20px; transition: transform 0.15s ease, border-color 0.15s ease; } .backup-summary-tile:hover { transform: translateY(-2px); border-color: rgba(var(--accent-rgb), 0.35); } .backup-summary-tile-label { font-size: 0.75rem; font-weight: 600; letter-spacing: 0.08em; text-transform: uppercase; color: var(--text-secondary, rgba(var(--text-rgb), 0.6)); margin-bottom: 8px; } .backup-summary-tile-value { font-size: 1.6rem; font-weight: 600; color: var(--text-primary); letter-spacing: -0.02em; } .backup-summary-tile-detail { font-size: 0.78rem; color: var(--text-secondary, rgba(var(--text-rgb), 0.55)); margin-top: 4px; } .backup-cards-row { display: grid; grid-template-columns: 2fr 1fr; gap: 20px; } @media (max-width: 980px) { .backup-cards-row { grid-template-columns: 1fr; } } .backup-card { background: rgba(var(--text-rgb), 0.04); border: 1px solid rgba(var(--text-rgb), 0.08); border-radius: 14px; padding: 18px 22px; } .backup-card-header { display: flex; align-items: baseline; justify-content: space-between; gap: 12px; margin-bottom: 16px; flex-wrap: wrap; } .backup-card-header h2 { margin: 0; font-size: 1.05rem; font-weight: 600; letter-spacing: -0.01em; } .backup-card-hint { font-size: 0.78rem; color: var(--text-secondary, rgba(var(--text-rgb), 0.55)); } .backup-app-grid { display: grid; grid-template-columns: repeat(auto-fill, minmax(220px, 1fr)); gap: 12px; } .backup-app-tile { background: rgba(var(--text-rgb), 0.05); border: 1px solid rgba(var(--text-rgb), 0.08); border-radius: 10px; padding: 12px 14px; display: flex; align-items: center; gap: 12px; cursor: pointer; transition: all 0.15s ease; } .backup-app-tile:hover { border-color: rgba(var(--accent-rgb), 0.4); transform: translateY(-1px); } .backup-app-tile-icon { width: 36px; height: 36px; flex-shrink: 0; border-radius: 8px; object-fit: cover; background: rgba(var(--text-rgb), 0.05); } .backup-app-tile-text { display: flex; flex-direction: column; gap: 4px; min-width: 0; flex: 1; } .backup-app-tile-name { font-weight: 600; font-size: 0.95rem; overflow: hidden; text-overflow: ellipsis; white-space: nowrap; } .backup-app-tile-meta { display: flex; align-items: center; gap: 8px; font-size: 0.78rem; color: var(--text-secondary, rgba(var(--text-rgb), 0.6)); } .backup-status-dot { width: 8px; height: 8px; border-radius: 50%; flex-shrink: 0; } .backup-status-dot.ok { background: #22c55e; box-shadow: 0 0 0 3px rgba(34, 197, 94, 0.15); } .backup-status-dot.warn { background: #f59e0b; box-shadow: 0 0 0 3px rgba(245, 158, 11, 0.15); } .backup-status-dot.fail { background: #ef4444; box-shadow: 0 0 0 3px rgba(239, 68, 68, 0.15); } .backup-status-dot.none { background: rgba(var(--text-rgb), 0.25); } .backup-repo-list { display: flex; flex-direction: column; gap: 10px; } .backup-repo-row { display: flex; align-items: center; justify-content: space-between; padding: 10px 12px; background: rgba(var(--text-rgb), 0.05); border-radius: 10px; gap: 12px; } .backup-repo-row-name { font-weight: 600; font-size: 0.9rem; display: flex; align-items: center; gap: 8px; } .backup-repo-row-meta { font-size: 0.75rem; color: var(--text-secondary, rgba(var(--text-rgb), 0.55)); text-align: right; } .backup-repo-type-pill { background: rgba(var(--accent-rgb), 0.12); color: var(--accent); padding: 2px 8px; border-radius: 999px; font-size: 0.68rem; font-weight: 600; text-transform: uppercase; letter-spacing: 0.04em; } .backup-repo-cards { display: grid; grid-template-columns: repeat(auto-fit, minmax(320px, 1fr)); gap: 16px; } /* Locations list — expandable rows mirroring the Tasks page .task-item shell */ .backup-location-list { display: flex; flex-direction: column; gap: 10px; } .backup-location-row.task-item { margin-bottom: 0; } .backup-location-header { padding: 18px 22px; gap: 14px; user-select: none; min-height: 64px; } .backup-location-row-type-icon { flex-shrink: 0; width: 28px; height: 28px; display: inline-flex; align-items: center; justify-content: center; color: var(--accent); } .backup-location-row-type-icon[data-type="sftp"] { color: #818cf8; } .backup-location-row-type-icon[data-type="rest"], .backup-location-row-type-icon[data-type="s3"], .backup-location-row-type-icon[data-type="b2"], .backup-location-row-type-icon[data-type="gs"], .backup-location-row-type-icon[data-type="azure"], .backup-location-row-type-icon[data-type="rclone"] { color: #38bdf8; } /* Status pill — mirrors the task-status pill on the Tasks page so the visual language is consistent. */ .task-status.backup-loc-status { padding: 3px 8px; border-radius: 999px; font-size: 0.7rem; text-transform: uppercase; letter-spacing: 0.05em; border: 1px solid transparent; line-height: 1.3; } .task-status.backup-loc-status.status-ready { background: rgba(var(--status-success-rgb), 0.30); border-color: rgba(var(--status-success-rgb), 0.65); color: #86efac; } .task-status.backup-loc-status.status-init { background: rgba(var(--status-warning-rgb), 0.22); border-color: rgba(var(--status-warning-rgb), 0.60); color: #fcd34d; } .task-status.backup-loc-status.status-disabled { background: rgba(var(--text-rgb), 0.08); border-color: rgba(var(--text-rgb), 0.18); color: var(--text-secondary, rgba(var(--text-rgb), 0.6)); } .backup-location-row-info { display: flex; align-items: center; gap: 10px; flex: 1; min-width: 0; overflow: hidden; } .backup-location-row-name { font-size: 1rem; font-weight: 600; color: var(--text-primary); white-space: nowrap; overflow: hidden; text-overflow: ellipsis; max-width: 240px; } .backup-location-row-status-pill { font-size: 0.78rem; font-weight: 600; color: var(--text-primary); } .backup-location-row-stat { font-size: 0.78rem; color: var(--text-secondary, rgba(var(--text-rgb), 0.6)); white-space: nowrap; } .backup-location-row-sep { font-size: 0.78rem; color: var(--text-secondary, rgba(var(--text-rgb), 0.35)); } .backup-pill-mini { background: rgba(245, 158, 11, 0.15); color: #f59e0b; padding: 2px 8px; border-radius: 999px; font-size: 0.65rem; font-weight: 600; text-transform: uppercase; letter-spacing: 0.05em; } /* Hide separator + stat tokens on narrow screens to keep the row from wrapping — the expanded details still show full info. */ @media (max-width: 720px) { .backup-location-row-sep, .backup-location-row-stat { display: none; } } .backup-location-chevron { flex-shrink: 0; color: var(--text-secondary, rgba(var(--text-rgb), 0.55)); transition: transform 0.2s ease; } .backup-location-row.expanded .backup-location-chevron { transform: rotate(180deg); } .backup-location-details { padding: 16px 20px 20px; } .backup-location-details > .config-category:first-of-type { margin-top: 0; } .backup-location-actions { display: flex; justify-content: space-between; align-items: center; gap: 10px; margin-top: 16px; padding-top: 14px; border-top: 1px solid rgba(var(--text-rgb), 0.06); } .backup-modal-wide .backup-modal-inner, .backup-modal-inner.backup-modal-wide { max-width: 640px; } .backup-modal-wide { /* selector also catches when class is on inner */ } .backup-modal-inner.backup-modal-wide, #backup-location-modal .backup-modal-inner { max-width: 640px; width: min(92vw, 640px); } .backup-repo-card { background: rgba(var(--text-rgb), 0.04); border: 1px solid rgba(var(--text-rgb), 0.08); border-radius: 14px; padding: 18px 22px; } .backup-repo-card-header { display: flex; align-items: center; justify-content: space-between; margin-bottom: 12px; } .backup-repo-card-title { font-weight: 600; font-size: 1.05rem; display: flex; align-items: center; gap: 10px; } .backup-repo-disabled-pill { background: rgba(var(--text-rgb), 0.1); color: var(--text-secondary, rgba(var(--text-rgb), 0.6)); padding: 3px 10px; border-radius: 999px; font-size: 0.7rem; font-weight: 600; text-transform: uppercase; } .backup-repo-enabled-pill { background: rgba(34, 197, 94, 0.15); color: #16a34a; padding: 3px 10px; border-radius: 999px; font-size: 0.7rem; font-weight: 600; text-transform: uppercase; } .backup-repo-detail { display: grid; grid-template-columns: 110px 1fr; gap: 6px 12px; font-size: 0.82rem; margin-bottom: 12px; } .backup-repo-detail-key { color: var(--text-secondary, rgba(var(--text-rgb), 0.55)); } .backup-repo-detail-value { color: var(--text-primary); word-break: break-all; } .backup-snapshot-table-wrap { background: rgba(var(--text-rgb), 0.04); border: 1px solid rgba(var(--text-rgb), 0.08); border-radius: 14px; overflow: hidden; } .backup-snapshot-table { width: 100%; border-collapse: collapse; } .backup-snapshot-table th, .backup-snapshot-table td { padding: 12px 16px; text-align: left; font-size: 0.875rem; border-bottom: 1px solid rgba(var(--text-rgb), 0.06); } .backup-snapshot-table th { background: rgba(var(--text-rgb), 0.04); font-weight: 600; font-size: 0.78rem; text-transform: uppercase; letter-spacing: 0.05em; color: var(--text-secondary, rgba(var(--text-rgb), 0.6)); } .backup-snapshot-table tbody tr:hover { background: rgba(var(--accent-rgb), 0.04); } .backup-col-actions { width: 180px; text-align: right; } .backup-snapshot-id { font-family: ui-monospace, SFMono-Regular, monospace; font-size: 0.82rem; color: var(--text-secondary, rgba(var(--text-rgb), 0.7)); } .backup-row-action-btn { background: rgba(var(--text-rgb), 0.06); border: 1px solid rgba(var(--text-rgb), 0.1); color: var(--text-primary); border-radius: 6px; padding: 5px 10px; font-size: 0.78rem; cursor: pointer; margin-left: 6px; transition: all 0.15s ease; } .backup-row-action-btn:hover { background: rgba(var(--accent-rgb), 0.12); border-color: var(--accent); color: var(--accent); } .backup-row-action-btn.danger:hover { background: rgba(239, 68, 68, 0.12); border-color: #ef4444; color: #ef4444; } .backup-filters { display: grid; grid-template-columns: 1fr 1fr; gap: 12px; margin-bottom: 16px; } @media (max-width: 600px) { .backup-filters { grid-template-columns: 1fr; } } .backup-filter-input, .backup-filter-select { background: rgba(var(--text-rgb), 0.04); border: 1px solid rgba(var(--text-rgb), 0.1); border-radius: 8px; padding: 9px 12px; color: var(--text-primary); font-size: 0.875rem; width: 100%; box-sizing: border-box; min-width: 0; } .backup-filter-input:focus, .backup-filter-select:focus { outline: none; border-color: var(--accent); box-shadow: 0 0 0 3px rgba(var(--accent-rgb), 0.15); } /* Inline forms inside backup cards */ .backup-form-grid { display: grid; grid-template-columns: repeat(auto-fit, minmax(280px, 1fr)); gap: 14px 20px; margin-top: 4px; } .backup-form-row { display: flex; flex-direction: column; gap: 6px; font-size: 0.875rem; } .backup-form-row-toggle { flex-direction: row; align-items: center; justify-content: space-between; gap: 16px; padding: 8px 0; } .backup-form-label { color: var(--text-secondary, rgba(var(--text-rgb), 0.7)); font-weight: 500; font-size: 0.82rem; } .backup-form-readonly { font-family: ui-monospace, SFMono-Regular, monospace; color: var(--text-primary); background: rgba(var(--text-rgb), 0.04); border: 1px solid rgba(var(--text-rgb), 0.08); border-radius: 8px; padding: 9px 12px; font-size: 0.85rem; } .backup-form-section-title { margin-top: 22px; margin-bottom: 10px; font-weight: 600; font-size: 0.85rem; color: var(--text-primary); text-transform: uppercase; letter-spacing: 0.06em; } .backup-form-section-title .backup-card-hint { text-transform: none; letter-spacing: normal; margin-left: 8px; font-weight: 400; } .backup-retention-block { grid-template-columns: 1fr; } .backup-retention-preset-block { grid-column: 1 / -1; display: flex; flex-direction: column; gap: 6px; padding-bottom: 14px; margin-bottom: 6px; border-bottom: 1px solid rgba(var(--text-rgb), 0.08); } .backup-retention-hint { margin-top: -4px; font-style: italic; } .backup-retention-advanced[hidden] { display: none; } .backup-retention-advanced { margin-top: 12px; padding-top: 12px; border-top: 1px dashed rgba(var(--text-rgb), 0.08); } .backup-form-footer { display: flex; justify-content: flex-end; margin-top: 18px; padding-top: 14px; border-top: 1px solid rgba(var(--text-rgb), 0.06); } /* iOS-style toggle */ .backup-toggle { position: relative; width: 38px; height: 22px; flex-shrink: 0; } .backup-toggle input { opacity: 0; width: 100%; height: 100%; cursor: pointer; position: absolute; inset: 0; margin: 0; z-index: 1; } .backup-toggle-slider { position: absolute; inset: 0; background: rgba(var(--text-rgb), 0.15); border-radius: 999px; transition: background 0.18s ease; } .backup-toggle-slider::after { content: ""; position: absolute; top: 2px; left: 2px; width: 18px; height: 18px; background: #fff; border-radius: 50%; transition: transform 0.18s ease; box-shadow: 0 1px 3px rgba(0, 0, 0, 0.25); } .backup-toggle input:checked + .backup-toggle-slider { background: var(--accent); } .backup-toggle input:checked + .backup-toggle-slider::after { transform: translateX(16px); } /* Enable/disable toggle in a location row header — sits between the row info and the expand chevron, controlling the location without expanding it. */ .backup-loc-enable-toggle { margin-right: 6px; } /* Repo card extras */ .backup-repo-stats { display: flex; gap: 18px; flex-wrap: wrap; padding: 10px 14px; background: rgba(var(--text-rgb), 0.04); border-radius: 10px; margin-bottom: 16px; font-size: 0.82rem; } .backup-repo-stat-label { color: var(--text-secondary, rgba(var(--text-rgb), 0.55)); margin-right: 4px; text-transform: uppercase; letter-spacing: 0.05em; font-size: 0.7rem; } .backup-warning-banner { margin-top: 18px; padding: 14px 16px; background: rgba(245, 158, 11, 0.08); border: 1px solid rgba(245, 158, 11, 0.25); border-radius: 10px; font-size: 0.85rem; display: flex; align-items: center; gap: 16px; } .backup-warning-banner-text { display: flex; flex-direction: column; gap: 4px; flex: 1; min-width: 0; } .backup-warning-banner [data-action="export-passwords"] { flex-shrink: 0; white-space: nowrap; } .backup-warning-banner [data-action="export-passwords"][data-busy="1"] { opacity: 0.6; cursor: progress; } .backup-warning-banner strong { color: #f59e0b; } .backup-warning-banner code { background: rgba(var(--text-rgb), 0.06); padding: 4px 8px; border-radius: 6px; font-size: 0.8rem; color: var(--text-primary); display: inline-block; margin-top: 4px; } .backup-modal { display: none; position: fixed; inset: 0; background: rgba(0, 0, 0, 0.6); backdrop-filter: blur(6px); z-index: 1000; align-items: center; justify-content: center; } .backup-modal.open { display: flex; } /* Slightly translucent — backdrop blur still shows through, but enough alpha to keep the modal legible on busy gradients. */ .backup-modal-inner { background: color-mix(in srgb, var(--surface-bg-solid, #1a1d24) 78%, transparent); backdrop-filter: blur(12px); -webkit-backdrop-filter: blur(12px); border: 1px solid rgba(var(--text-rgb), 0.14); border-radius: 14px; width: 90%; max-width: 460px; overflow: hidden; box-shadow: 0 24px 60px rgba(0, 0, 0, 0.55); } .backup-modal-header { display: flex; align-items: center; justify-content: space-between; padding: 16px 20px; border-bottom: 1px solid rgba(var(--text-rgb), 0.08); } .backup-modal-header h3 { margin: 0; font-size: 1rem; font-weight: 600; } .backup-modal-close { background: none; border: none; color: var(--text-secondary, rgba(var(--text-rgb), 0.55)); font-size: 1.4rem; cursor: pointer; line-height: 1; padding: 0 6px; } .backup-modal-body { padding: 18px 20px; font-size: 0.9rem; color: var(--text-primary); max-height: 60vh; overflow-y: auto; } .backup-modal-footer { display: flex; justify-content: flex-end; gap: 10px; padding: 14px 20px; border-top: 1px solid rgba(var(--text-rgb), 0.08); } .backup-empty-state { padding: 40px 20px; text-align: center; color: var(--text-secondary, rgba(var(--text-rgb), 0.55)); font-size: 0.9rem; } .backup-engine-input-row { display: flex; flex-wrap: nowrap; align-items: stretch; gap: 10px; width: 100%; } .backup-engine-input-row > *:not(.backup-engine-details-btn) { flex: 1 1 0%; width: auto; min-width: 0; max-width: none; } .backup-engine-input-row > .custom-select { display: block; } .backup-engine-details-btn { display: inline-flex; align-items: center; justify-content: center; gap: 6px; flex: 0 0 auto; padding: 0 16px; min-height: 44px; white-space: nowrap; line-height: 1; } /* Engine name pill next to the type pill on each location row. */ .backup-engine-pill { background: rgba(var(--accent-rgb), 0.16); color: var(--accent); padding: 2px 8px; border-radius: 999px; font-size: 0.68rem; font-weight: 600; text-transform: uppercase; letter-spacing: 0.04em; } .backup-engine-pill[data-engine="borg"] { background: rgba(245, 158, 11, 0.18); color: #f59e0b; } .backup-engine-pill[data-engine="kopia"] { background: rgba(99, 102, 241, 0.18); color: #818cf8; } /* Engine details modal. */ .backup-engine-modal-head { display: flex; align-items: center; gap: 14px; margin-bottom: 16px; } .backup-engine-logo { width: 36px; height: 36px; flex-shrink: 0; color: var(--accent); } .backup-engine-modal-head h4 { margin: 0 0 4px 0; font-size: 1.1rem; font-weight: 600; } .backup-engine-props { width: 100%; border-collapse: collapse; margin-bottom: 18px; } .backup-engine-props th, .backup-engine-props td { padding: 8px 10px; text-align: left; font-size: 0.85rem; border-bottom: 1px solid rgba(var(--text-rgb), 0.06); } .backup-engine-props th { width: 35%; color: var(--text-secondary, rgba(var(--text-rgb), 0.6)); font-weight: 500; } .backup-engine-features { margin: 6px 0 18px; padding-left: 22px; font-size: 0.88rem; line-height: 1.55; } .backup-engine-features li { margin-bottom: 4px; } .backup-engine-docs-link { color: var(--accent); text-decoration: none; font-family: ui-monospace, SFMono-Regular, monospace; font-size: 0.85rem; } .backup-engine-docs-link:hover { text-decoration: underline; } /* Per-app backup status badge (used on app detail page) */ .backup-app-badge { display: inline-flex; align-items: center; gap: 8px; padding: 6px 10px; background: rgba(var(--text-rgb), 0.05); border-radius: 999px; font-size: 0.78rem; color: var(--text-secondary, rgba(var(--text-rgb), 0.65)); }