Update from Git Manager GUI
This commit is contained in:
@@ -5,6 +5,30 @@ export default function Settings() {
|
||||
const [giteaToken, setGiteaToken] = useState('');
|
||||
const [giteaURL, setGiteaURL] = useState('');
|
||||
|
||||
function normalizeAndValidateGiteaUrl(rawUrl) {
|
||||
const value = (rawUrl || '').trim();
|
||||
if (!value) return { ok: true, value: '' };
|
||||
|
||||
let parsed;
|
||||
try {
|
||||
parsed = new URL(value);
|
||||
} catch (_) {
|
||||
return {
|
||||
ok: false,
|
||||
error: 'Ungültige Gitea-URL. Beispiel für IPv6: http://[2001:db8::1]:3000'
|
||||
};
|
||||
}
|
||||
|
||||
if (parsed.protocol !== 'http:' && parsed.protocol !== 'https:') {
|
||||
return {
|
||||
ok: false,
|
||||
error: 'Die Gitea-URL muss mit http:// oder https:// beginnen.'
|
||||
};
|
||||
}
|
||||
|
||||
return { ok: true, value: value.replace(/\/$/, '') };
|
||||
}
|
||||
|
||||
useEffect(() => {
|
||||
window.electronAPI.loadCredentials().then(data => {
|
||||
if (data) {
|
||||
@@ -16,7 +40,13 @@ export default function Settings() {
|
||||
}, []);
|
||||
|
||||
const save = () => {
|
||||
window.electronAPI.saveCredentials({ githubToken, giteaToken, giteaURL });
|
||||
const checkedUrl = normalizeAndValidateGiteaUrl(giteaURL);
|
||||
if (!checkedUrl.ok) {
|
||||
alert(checkedUrl.error);
|
||||
return;
|
||||
}
|
||||
|
||||
window.electronAPI.saveCredentials({ githubToken, giteaToken, giteaURL: checkedUrl.value });
|
||||
alert('Settings saved securely!');
|
||||
}
|
||||
|
||||
|
||||
BIN
renderer/icon.png
Normal file
BIN
renderer/icon.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 67 KiB |
@@ -4,30 +4,63 @@
|
||||
<meta charset="utf-8" />
|
||||
<title>Git Manager Explorer Pro</title>
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1">
|
||||
<link rel="icon" type="image/png" href="./icon.png">
|
||||
<link rel="stylesheet" href="style.css">
|
||||
</head>
|
||||
<body>
|
||||
<div id="app">
|
||||
<div id="toolbar">
|
||||
<div class="tool-group">
|
||||
<button id="btnSettings" title="Einstellungen">⚙️ Settings</button>
|
||||
<button id="btnBack" class="secondary hidden" title="Zurück">⬅️ Zurück</button>
|
||||
<button id="btnSelectFolder" class="accent-btn" title="Lokalen Ordner öffnen">📂 Open Local</button>
|
||||
<button id="btnLoadGiteaRepos" class="accent-btn" title="Gitea Repositories laden">🌐 Load Gitea</button>
|
||||
<div class="toolbar-row toolbar-row--top">
|
||||
<div class="toolbar-brand" aria-label="App Kopfbereich">
|
||||
<div class="toolbar-brand-mark">
|
||||
<img src="./icon.png" alt="Git Manager Logo" class="toolbar-brand-logo">
|
||||
</div>
|
||||
<div class="toolbar-brand-copy">
|
||||
<span class="toolbar-kicker">Workspace Control</span>
|
||||
<strong>Git Manager Explorer Pro</strong>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="toolbar-top-actions">
|
||||
<div class="tool-group tool-group--quick-actions">
|
||||
<button id="btnOpenRepoActions" title="Neues Repository erstellen">🚀 New Repo</button>
|
||||
<button id="btnPush" title="Projekt pushen">⬆️ Push</button>
|
||||
</div>
|
||||
|
||||
<div class="tool-group tool-group--utility">
|
||||
<span class="tool-group-title">Steuerung</span>
|
||||
<button id="btnSettings" title="Einstellungen">⚙️ Settings</button>
|
||||
<button id="btnBatchActions" title="Batch-Aktionen">🧩 Batch</button>
|
||||
<button id="btnOpenActivityLog" title="Aktivitätsprotokoll">📝 Activity</button>
|
||||
<button id="btnRetryQueueNow" class="secondary" title="Retry-Queue jetzt verarbeiten">🔁 Queue (0)</button>
|
||||
<button id="btnBack" class="secondary hidden" title="Zurück">⬅️ Zurück</button>
|
||||
</div>
|
||||
|
||||
<div class="toolbar-status-wrap">
|
||||
<span class="status-dot" aria-hidden="true"></span>
|
||||
<span id="status" class="status">Bereit</span>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="tool-group">
|
||||
<select id="platform" title="Plattform auswählen">
|
||||
<option value="gitea" selected>Gitea</option>
|
||||
<option value="github">GitHub</option>
|
||||
</select>
|
||||
<button id="btnOpenRepoActions" title="Neues Repository erstellen">🚀 New Repo</button>
|
||||
<button id="btnPush" title="Projekt pushen">⬆️ Push</button>
|
||||
<button id="btnCommits" class="hidden" title="Commit History anzeigen">📊 Commits</button>
|
||||
<button id="btnReleases" class="hidden" title="Releases anzeigen">📦 Releases</button>
|
||||
|
||||
<div class="toolbar-row toolbar-row--bottom">
|
||||
<div class="tool-group tool-group--workspace">
|
||||
<span class="tool-group-title">Quelle</span>
|
||||
<button id="btnSelectFolder" class="accent-btn" title="Lokalen Ordner öffnen">📂 Open Local</button>
|
||||
<button id="btnLoadGiteaRepos" class="accent-btn" title="Gitea Repositories laden">🌐 Load Gitea</button>
|
||||
</div>
|
||||
|
||||
<div class="tool-group tool-group--repo">
|
||||
<span class="tool-group-title">Repository</span>
|
||||
<input id="platform" type="hidden" value="gitea">
|
||||
<div class="platform-switch" role="tablist" aria-label="Plattform auswählen">
|
||||
<button type="button" class="platform-option active" data-platform="gitea" aria-pressed="true">Gitea</button>
|
||||
<button type="button" class="platform-option" data-platform="github" aria-pressed="false">GitHub</button>
|
||||
</div>
|
||||
<button id="btnCommits" class="hidden" title="Commit History anzeigen">📊 Commits</button>
|
||||
<button id="btnReleases" class="hidden" title="Releases anzeigen">📦 Releases</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<span id="status" class="status">Bereit</span>
|
||||
</div>
|
||||
|
||||
<main id="main">
|
||||
@@ -36,65 +69,155 @@
|
||||
</main>
|
||||
|
||||
<div id="settingsModal" class="modal hidden">
|
||||
<div class="modalContent card">
|
||||
<h2>⚙️ Einstellungen</h2>
|
||||
|
||||
<div class="input-group">
|
||||
<label>GitHub Token</label>
|
||||
<input id="githubToken" type="password" placeholder="ghp_...">
|
||||
<div class="modalContent card settings-modal-content">
|
||||
<button id="btnSettingsWatermark" class="settings-watermark-btn" title="Projektinformationen anzeigen" aria-label="Projektinformationen anzeigen">ⓘ</button>
|
||||
<div id="settingsWatermarkCard" class="settings-watermark-card hidden" role="dialog" aria-label="Projektinformationen">
|
||||
<h4>Projektinformationen</h4>
|
||||
<div class="settings-watermark-row"><span>Ersteller:</span><strong>M_Viper</strong></div>
|
||||
<div class="settings-watermark-row"><span>Webseite:</span><a href="https://m-viper.de" target="_blank" rel="noopener noreferrer">https://m-viper.de</a></div>
|
||||
<div class="settings-watermark-row"><span>Discord:</span><a id="watermarkDiscord" href="https://discord.com/invite/FdRs4BRd8D" target="_blank" rel="noopener noreferrer">discord.com/invite/FdRs4BRd8D</a></div>
|
||||
<div class="settings-watermark-row"><span>E-Mail:</span><a id="watermarkMail" href="mailto:admin@m-viper.de">admin@m-viper.de</a></div>
|
||||
<div class="settings-watermark-row"><span>Version:</span><strong id="watermarkVersion">-</strong></div>
|
||||
<div class="settings-watermark-row"><span>Copyright:</span><strong id="watermarkCopyright">-</strong></div>
|
||||
<div class="settings-watermark-row"><span>Projekt:</span><strong>Git Manager Explorer Pro</strong></div>
|
||||
</div>
|
||||
|
||||
<div class="input-group">
|
||||
<label>Gitea Token</label>
|
||||
<input id="giteaToken" type="password" placeholder="Token hier einfügen">
|
||||
</div>
|
||||
|
||||
<div class="input-group">
|
||||
<label>Gitea URL</label>
|
||||
<input id="giteaURL" type="text" placeholder="https://gitea.example.com">
|
||||
</div>
|
||||
|
||||
<div class="input-group" style="margin-top: 20px; padding-top: 20px; border-top: 1px solid rgba(255,255,255,0.1);">
|
||||
<label style="margin-bottom: 12px;">Übersicht</label>
|
||||
<div style="display: flex; flex-direction: column; gap: 10px;">
|
||||
<label style="display: flex; align-items: center; gap: 10px; cursor: pointer; text-transform: none; letter-spacing: normal; font-weight: normal;">
|
||||
<input type="checkbox" id="settingFavorites" checked>
|
||||
<span>⭐ Favoriten-Bereich anzeigen</span>
|
||||
</label>
|
||||
<label style="display: flex; align-items: center; gap: 10px; cursor: pointer; text-transform: none; letter-spacing: normal; font-weight: normal;">
|
||||
<input type="checkbox" id="settingRecent" checked>
|
||||
<span>🕐 Zuletzt geöffnet anzeigen</span>
|
||||
</label>
|
||||
<label style="display: flex; align-items: center; gap: 10px; cursor: pointer; text-transform: none; letter-spacing: normal; font-weight: normal;">
|
||||
<input type="checkbox" id="settingCompact">
|
||||
<span>⊞ Kompakt-Modus (kleinere Karten)</span>
|
||||
</label>
|
||||
<label style="display: flex; align-items: center; gap: 10px; cursor: pointer; text-transform: none; letter-spacing: normal; font-weight: normal;">
|
||||
<input type="checkbox" id="settingColoredIcons" checked>
|
||||
<span>🎨 Farbige Datei-Icons</span>
|
||||
</label>
|
||||
|
||||
<div class="settings-header">
|
||||
<div>
|
||||
<div class="settings-eyebrow">Konfiguration</div>
|
||||
<h2>⚙️ Einstellungen</h2>
|
||||
<p class="settings-subtitle">Alle wichtigen Optionen auf einer Seite: Zugangsdaten, Verbindungscheck, Darstellung und Updates.</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="input-group" style="margin-top: 30px; padding-top: 20px; border-top: 1px solid rgba(255,255,255,0.1);">
|
||||
<label>App Version</label>
|
||||
<div style="display: flex; gap: 12px; align-items: center;">
|
||||
<input id="appVersion" type="text" readonly style="flex: 1; background: rgba(255,255,255,0.05); cursor: not-allowed;">
|
||||
<button id="btnCheckUpdates" style="
|
||||
background: linear-gradient(135deg, #00d4ff, #8b5cf6);
|
||||
color: #000;
|
||||
border: none;
|
||||
padding: 10px 20px;
|
||||
border-radius: 8px;
|
||||
font-weight: 600;
|
||||
cursor: pointer;
|
||||
white-space: nowrap;
|
||||
">🔄 Nach Updates suchen</button>
|
||||
<div class="settings-layout">
|
||||
<div class="settings-column settings-column--left">
|
||||
<section class="settings-panel settings-panel--credentials">
|
||||
<div class="settings-panel-header">
|
||||
<div>
|
||||
<h3>Zugangsdaten</h3>
|
||||
<p>API-Zugriffe für GitHub und Gitea konfigurieren.</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="settings-fields-grid">
|
||||
<div class="input-group">
|
||||
<label for="githubToken">GitHub Token</label>
|
||||
<input id="githubToken" type="password" placeholder="ghp_...">
|
||||
</div>
|
||||
|
||||
<div class="input-group">
|
||||
<label for="giteaToken">Gitea Token</label>
|
||||
<input id="giteaToken" type="password" placeholder="Token hier einfügen">
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="input-group input-group--wide">
|
||||
<label for="giteaURL">Gitea URL</label>
|
||||
<input id="giteaURL" type="text" placeholder="https://gitea.example.com">
|
||||
<div class="settings-connection-tools">
|
||||
<div id="giteaUrlHint" class="settings-inline-hint">Hinweis: IPv6 mit Klammern eingeben, z.B. http://[2001:db8::1]:3000</div>
|
||||
<button id="btnTestGiteaConnection" class="secondary">🔌 Verbindung testen</button>
|
||||
</div>
|
||||
</div>
|
||||
</section>
|
||||
|
||||
<section class="settings-panel settings-panel--display">
|
||||
<div class="settings-panel-header">
|
||||
<div>
|
||||
<h3>Darstellung</h3>
|
||||
<p>Übersicht und Explorer an deinen Arbeitsstil anpassen.</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="settings-toggle-list">
|
||||
<label class="settings-toggle-row" for="settingFavorites">
|
||||
<span class="settings-toggle-info">
|
||||
<span class="settings-toggle-title">⭐ Favoriten-Bereich anzeigen</span>
|
||||
<span class="settings-toggle-desc">Pinnt wichtige Repositories und Ordner sichtbar im Kopfbereich.</span>
|
||||
</span>
|
||||
<span class="toggle-switch">
|
||||
<input type="checkbox" id="settingFavorites" checked>
|
||||
<span class="toggle-track"></span>
|
||||
</span>
|
||||
</label>
|
||||
|
||||
<label class="settings-toggle-row" for="settingRecent">
|
||||
<span class="settings-toggle-info">
|
||||
<span class="settings-toggle-title">🕐 Zuletzt geöffnet anzeigen</span>
|
||||
<span class="settings-toggle-desc">Zeigt deine letzten Projekte direkt in der Übersicht an.</span>
|
||||
</span>
|
||||
<span class="toggle-switch">
|
||||
<input type="checkbox" id="settingRecent" checked>
|
||||
<span class="toggle-track"></span>
|
||||
</span>
|
||||
</label>
|
||||
|
||||
<label class="settings-toggle-row" for="settingCompact">
|
||||
<span class="settings-toggle-info">
|
||||
<span class="settings-toggle-title">⊞ Kompakt-Modus</span>
|
||||
<span class="settings-toggle-desc">Verdichtet Karten und Abstände für kleinere Fenster.</span>
|
||||
</span>
|
||||
<span class="toggle-switch">
|
||||
<input type="checkbox" id="settingCompact">
|
||||
<span class="toggle-track"></span>
|
||||
</span>
|
||||
</label>
|
||||
|
||||
<label class="settings-toggle-row" for="settingColoredIcons">
|
||||
<span class="settings-toggle-info">
|
||||
<span class="settings-toggle-title">🎨 Farbige Datei-Icons</span>
|
||||
<span class="settings-toggle-desc">Setzt stärkere Dateityp-Farben für schnellere Orientierung.</span>
|
||||
</span>
|
||||
<span class="toggle-switch">
|
||||
<input type="checkbox" id="settingColoredIcons" checked>
|
||||
<span class="toggle-track"></span>
|
||||
</span>
|
||||
</label>
|
||||
</div>
|
||||
</section>
|
||||
</div>
|
||||
|
||||
<div class="settings-column settings-column--right">
|
||||
<section class="settings-panel settings-panel--health">
|
||||
<div class="settings-panel-header">
|
||||
<div>
|
||||
<h3>Verbindungsstatus</h3>
|
||||
<p>Direkt sehen, ob URL, API und Auth sauber antworten.</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="settings-health-box">
|
||||
<div class="settings-health-row"><span>URL</span><strong id="healthUrl">Unbekannt</strong></div>
|
||||
<div class="settings-health-row"><span>API</span><strong id="healthApi">Unbekannt</strong></div>
|
||||
<div class="settings-health-row"><span>Auth</span><strong id="healthAuth">Unbekannt</strong></div>
|
||||
<div class="settings-health-row"><span>Latenz</span><strong id="healthLatency">-</strong></div>
|
||||
<div class="settings-health-row"><span>Server</span><strong id="healthVersion">-</strong></div>
|
||||
<div class="settings-health-row"><span>Letzter Fehler</span><strong id="healthLastError">-</strong></div>
|
||||
</div>
|
||||
</section>
|
||||
|
||||
<section class="settings-panel settings-panel--app">
|
||||
<div class="settings-panel-header">
|
||||
<div>
|
||||
<h3>App & Updates</h3>
|
||||
<p>Version prüfen und neue Releases direkt anstoßen.</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="settings-version-card">
|
||||
<div class="input-group input-group--wide settings-version-field">
|
||||
<label for="appVersion">App Version</label>
|
||||
<input id="appVersion" class="settings-readonly-input" type="text" readonly>
|
||||
</div>
|
||||
<button id="btnCheckUpdates" class="settings-update-btn">🔄 Nach Updates suchen</button>
|
||||
</div>
|
||||
</section>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="modal-buttons">
|
||||
<button id="btnSaveSettings">Speichern</button>
|
||||
|
||||
<div class="modal-buttons settings-modal-actions">
|
||||
<button id="btnSaveSettings" class="accent-btn">Speichern</button>
|
||||
<button id="btnCloseSettings" class="secondary">Abbrechen</button>
|
||||
</div>
|
||||
</div>
|
||||
@@ -147,6 +270,79 @@
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div id="batchActionModal" class="modal hidden">
|
||||
<div class="modalContent card">
|
||||
<h2>🧩 Batch-Aktionen</h2>
|
||||
|
||||
<div class="input-group">
|
||||
<label>Aktion</label>
|
||||
<select id="batchActionType">
|
||||
<option value="refresh">Repos aktualisieren</option>
|
||||
<option value="clone">Repos klonen</option>
|
||||
<option value="create-tag">Tag erstellen</option>
|
||||
<option value="create-release">Release erstellen</option>
|
||||
</select>
|
||||
</div>
|
||||
|
||||
<div class="input-group">
|
||||
<label>Repositories (pro Zeile: owner/repo)</label>
|
||||
<textarea id="batchRepoList" class="batch-textarea" placeholder="M_Viper/ProjektA M_Viper/ProjektB"></textarea>
|
||||
</div>
|
||||
|
||||
<div id="batchCloneGroup" class="input-group hidden">
|
||||
<label>Zielordner für Clone</label>
|
||||
<div class="batch-inline-row">
|
||||
<input id="batchCloneTarget" type="text" readonly placeholder="Bitte Zielordner auswählen">
|
||||
<button id="btnSelectBatchCloneTarget" class="secondary">📁 Wählen</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div id="batchTagGroup" class="input-group hidden">
|
||||
<label>Tag</label>
|
||||
<input id="batchTagName" type="text" placeholder="v1.0.0">
|
||||
</div>
|
||||
|
||||
<div id="batchReleaseNameGroup" class="input-group hidden">
|
||||
<label>Release-Name</label>
|
||||
<input id="batchReleaseName" type="text" placeholder="Release v1.0.0">
|
||||
</div>
|
||||
|
||||
<div id="batchReleaseBodyGroup" class="input-group hidden">
|
||||
<label>Release-Text</label>
|
||||
<textarea id="batchReleaseBody" class="batch-textarea" placeholder="Changelog..."></textarea>
|
||||
</div>
|
||||
|
||||
<div class="modal-buttons">
|
||||
<button id="btnRunBatchAction" class="accent-btn">Ausführen</button>
|
||||
<button id="btnCloseBatchAction" class="secondary">Abbrechen</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div id="activityLogModal" class="modal hidden">
|
||||
<div class="modalContent card">
|
||||
<h2>📝 Aktivitätsprotokoll</h2>
|
||||
|
||||
<div class="activity-toolbar">
|
||||
<select id="activityFilterLevel">
|
||||
<option value="all">Alle</option>
|
||||
<option value="info">Info</option>
|
||||
<option value="warning">Warn</option>
|
||||
<option value="error">Error</option>
|
||||
</select>
|
||||
<button id="btnRetryQueueRefresh" class="secondary">🔁 Queue jetzt retry</button>
|
||||
<button id="btnClearActivityLog" class="secondary">🧹 Log leeren</button>
|
||||
</div>
|
||||
|
||||
<div id="activityQueueInfo" class="activity-queue-info">Retry-Queue: 0</div>
|
||||
<div id="activityLogList" class="activity-log-list"></div>
|
||||
|
||||
<div class="modal-buttons">
|
||||
<button id="btnCloseActivityLog" class="secondary">Schließen</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div id="fileEditorModal" class="modal hidden">
|
||||
<div class="file-editor-card">
|
||||
<div class="file-editor-header">
|
||||
@@ -197,7 +393,7 @@
|
||||
<div style="display: flex; align-items: center; gap: 20px; margin-bottom: 20px;">
|
||||
<div style="font-size: 3rem; filter: drop-shadow(0 0 10px var(--accent-primary));">🚀</div>
|
||||
<div>
|
||||
<h2 style="margin: 0; background: var(--accent-gradient); -webkit-background-clip: text; -webkit-text-fill-color: transparent;">Update verfügbar!</h2>
|
||||
<h2 style="margin: 0; background: var(--accent-gradient); background-clip: text; -webkit-background-clip: text; -webkit-text-fill-color: transparent;">Update verfügbar!</h2>
|
||||
<p id="updateVersionInfo" style="color: var(--text-secondary); margin: 5px 0 0 0; font-family: monospace;"></p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
@@ -63,6 +63,31 @@ function formatRelDate(iso) {
|
||||
return new Date(iso).toLocaleDateString('de-DE');
|
||||
}
|
||||
|
||||
function setPlatformSelection(platform) {
|
||||
const platformInput = $('platform');
|
||||
if (platformInput) {
|
||||
platformInput.value = platform;
|
||||
}
|
||||
|
||||
document.querySelectorAll('.platform-option').forEach(button => {
|
||||
const isActive = button.dataset.platform === platform;
|
||||
button.classList.toggle('active', isActive);
|
||||
button.setAttribute('aria-pressed', isActive ? 'true' : 'false');
|
||||
});
|
||||
}
|
||||
|
||||
function initializePlatformSelection() {
|
||||
const platformInput = $('platform');
|
||||
const initialPlatform = platformInput?.value || 'gitea';
|
||||
setPlatformSelection(initialPlatform);
|
||||
|
||||
document.querySelectorAll('.platform-option').forEach(button => {
|
||||
button.addEventListener('click', () => {
|
||||
setPlatformSelection(button.dataset.platform || 'gitea');
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
/* Rendert Favoriten + Zuletzt-geöffnet-Bereich in ein beliebiges Container-Element */
|
||||
// Collapse-Zustand (wird in Credentials persistiert)
|
||||
const favSectionCollapsed = { favorites: false, recent: false };
|
||||
@@ -299,6 +324,85 @@ let currentState = {
|
||||
path: ''
|
||||
};
|
||||
|
||||
const MAX_ACTIVITY_ITEMS = 300;
|
||||
let activityEntries = [];
|
||||
let retryQueueCount = 0;
|
||||
|
||||
function logActivity(level, message) {
|
||||
const entry = {
|
||||
id: `${Date.now()}-${Math.random().toString(16).slice(2)}`,
|
||||
level: level || 'info',
|
||||
message: String(message || ''),
|
||||
ts: new Date().toISOString()
|
||||
};
|
||||
activityEntries.unshift(entry);
|
||||
if (activityEntries.length > MAX_ACTIVITY_ITEMS) {
|
||||
activityEntries = activityEntries.slice(0, MAX_ACTIVITY_ITEMS);
|
||||
}
|
||||
renderActivityLog();
|
||||
}
|
||||
|
||||
function formatActivityTimestamp(iso) {
|
||||
try {
|
||||
return new Date(iso).toLocaleTimeString('de-DE', { hour12: false });
|
||||
} catch (_) {
|
||||
return '--:--:--';
|
||||
}
|
||||
}
|
||||
|
||||
function renderActivityLog() {
|
||||
const list = $('activityLogList');
|
||||
if (!list) return;
|
||||
|
||||
const filter = ($('activityFilterLevel') && $('activityFilterLevel').value) || 'all';
|
||||
const visible = activityEntries.filter(e => filter === 'all' || e.level === filter);
|
||||
|
||||
if (visible.length === 0) {
|
||||
list.innerHTML = '<div class="activity-log-item info"><span class="activity-log-message">Noch keine Einträge.</span></div>';
|
||||
return;
|
||||
}
|
||||
|
||||
list.innerHTML = visible.map(e => {
|
||||
const lvl = (e.level || 'info').toUpperCase();
|
||||
return `
|
||||
<div class="activity-log-item ${e.level}">
|
||||
<span class="activity-log-time">${formatActivityTimestamp(e.ts)}</span>
|
||||
<span class="activity-log-level">${lvl}</span>
|
||||
<span class="activity-log-message">${escapeHtml(e.message)}</span>
|
||||
</div>
|
||||
`;
|
||||
}).join('');
|
||||
}
|
||||
|
||||
function updateRetryQueueBadge(count) {
|
||||
retryQueueCount = Math.max(0, Number(count || 0));
|
||||
const btn = $('btnRetryQueueNow');
|
||||
if (btn) btn.textContent = `🔁 Queue (${retryQueueCount})`;
|
||||
const info = $('activityQueueInfo');
|
||||
if (info) info.textContent = `Retry-Queue: ${retryQueueCount}`;
|
||||
}
|
||||
|
||||
function parseBatchRepoInput(raw) {
|
||||
return String(raw || '')
|
||||
.split(/\r?\n/)
|
||||
.map(line => line.trim())
|
||||
.filter(Boolean)
|
||||
.filter(line => line.includes('/'));
|
||||
}
|
||||
|
||||
function updateBatchActionFields() {
|
||||
const action = $('batchActionType')?.value || 'refresh';
|
||||
const cloneGroup = $('batchCloneGroup');
|
||||
const tagGroup = $('batchTagGroup');
|
||||
const releaseNameGroup = $('batchReleaseNameGroup');
|
||||
const releaseBodyGroup = $('batchReleaseBodyGroup');
|
||||
|
||||
if (cloneGroup) cloneGroup.classList.toggle('hidden', action !== 'clone');
|
||||
if (tagGroup) tagGroup.classList.toggle('hidden', !(action === 'create-tag' || action === 'create-release'));
|
||||
if (releaseNameGroup) releaseNameGroup.classList.toggle('hidden', action !== 'create-release');
|
||||
if (releaseBodyGroup) releaseBodyGroup.classList.toggle('hidden', !(action === 'create-tag' || action === 'create-release'));
|
||||
}
|
||||
|
||||
// Clipboard für Cut & Paste
|
||||
let clipboard = {
|
||||
item: null, // { path, name, type, owner, repo, isGitea, isLocal, nodePath }
|
||||
@@ -315,6 +419,131 @@ let lastSelectedItem = null; // { type:'gitea', item, owner, repo } | { type:'l
|
||||
// Feature-Flag für farbige Icons
|
||||
let featureColoredIcons = true;
|
||||
|
||||
let settingsHealth = {
|
||||
url: 'Unbekannt',
|
||||
api: 'Unbekannt',
|
||||
auth: 'Unbekannt',
|
||||
latency: '-',
|
||||
version: '-',
|
||||
lastError: '-'
|
||||
};
|
||||
|
||||
function setHealthField(id, value) {
|
||||
const el = $(id);
|
||||
if (!el) return;
|
||||
el.textContent = value;
|
||||
el.classList.remove('health-ok', 'health-warn', 'health-error');
|
||||
const v = (value || '').toLowerCase();
|
||||
if (v === 'ok' || v === 'erreichbar' || v === 'gueltig' || v === 'gültig') {
|
||||
el.classList.add('health-ok');
|
||||
} else if (v === 'fehler' || v === 'ungueltig' || v === 'ungültig') {
|
||||
el.classList.add('health-error');
|
||||
} else if (v === 'unbekannt' || v === 'kein token' || v === 'token vorhanden' || v === 'nicht konfiguriert') {
|
||||
el.classList.add('health-warn');
|
||||
}
|
||||
}
|
||||
|
||||
function renderSettingsHealth() {
|
||||
setHealthField('healthUrl', settingsHealth.url);
|
||||
setHealthField('healthApi', settingsHealth.api);
|
||||
setHealthField('healthAuth', settingsHealth.auth);
|
||||
setHealthField('healthLatency', settingsHealth.latency);
|
||||
setHealthField('healthVersion', settingsHealth.version);
|
||||
setHealthField('healthLastError', settingsHealth.lastError);
|
||||
}
|
||||
|
||||
function syncSettingsPanelHeights() {
|
||||
const credentialsPanel = document.querySelector('#settingsModal .settings-panel--credentials');
|
||||
const healthPanel = document.querySelector('#settingsModal .settings-panel--health');
|
||||
if (!credentialsPanel || !healthPanel) return;
|
||||
|
||||
healthPanel.style.minHeight = '';
|
||||
|
||||
// In der einspaltigen Ansicht sollen die Karten natuerlich fliessen.
|
||||
if (window.matchMedia('(max-width: 1120px)').matches) return;
|
||||
|
||||
const targetHeight = Math.ceil(credentialsPanel.getBoundingClientRect().height);
|
||||
if (targetHeight > 0) {
|
||||
healthPanel.style.minHeight = `${targetHeight}px`;
|
||||
}
|
||||
}
|
||||
|
||||
function updateSettingsHealth(patch) {
|
||||
settingsHealth = { ...settingsHealth, ...patch };
|
||||
renderSettingsHealth();
|
||||
syncSettingsPanelHeights();
|
||||
}
|
||||
|
||||
function normalizeAndValidateGiteaUrl(rawUrl) {
|
||||
const value = (rawUrl || '').trim();
|
||||
if (!value) return { ok: true, value: '' };
|
||||
|
||||
let parsed;
|
||||
try {
|
||||
parsed = new URL(value);
|
||||
} catch (_) {
|
||||
return {
|
||||
ok: false,
|
||||
error: 'Ungültige URL. Beispiel für IPv6: http://[2001:db8::1]:3000'
|
||||
};
|
||||
}
|
||||
|
||||
if (parsed.protocol !== 'http:' && parsed.protocol !== 'https:') {
|
||||
return {
|
||||
ok: false,
|
||||
error: 'URL muss mit http:// oder https:// beginnen.'
|
||||
};
|
||||
}
|
||||
|
||||
return { ok: true, value: value.replace(/\/$/, '') };
|
||||
}
|
||||
|
||||
function renderGiteaUrlHint(rawValue) {
|
||||
const hint = $('giteaUrlHint');
|
||||
if (!hint) return;
|
||||
|
||||
const result = normalizeAndValidateGiteaUrl(rawValue);
|
||||
if (!rawValue || !rawValue.trim()) {
|
||||
hint.className = 'settings-inline-hint';
|
||||
hint.textContent = 'Hinweis: IPv6 mit Klammern eingeben, z.B. http://[2001:db8::1]:3000';
|
||||
return;
|
||||
}
|
||||
|
||||
if (!result.ok) {
|
||||
hint.className = 'settings-inline-hint error';
|
||||
hint.textContent = result.error;
|
||||
return;
|
||||
}
|
||||
|
||||
hint.className = 'settings-inline-hint success';
|
||||
hint.textContent = `Gültige URL: ${result.value}`;
|
||||
}
|
||||
|
||||
function mapErrorMessage(message) {
|
||||
const raw = String(message || '').toLowerCase();
|
||||
if (!raw) return 'Unbekannter Fehler';
|
||||
|
||||
if (raw.includes('401') || raw.includes('unauthorized') || raw.includes('authentifizierung')) {
|
||||
return 'Authentifizierung fehlgeschlagen. Bitte Token prüfen.';
|
||||
}
|
||||
if (raw.includes('403') || raw.includes('forbidden') || raw.includes('zugriff verweigert')) {
|
||||
return 'Zugriff verweigert. Bitte Token-Berechtigungen prüfen.';
|
||||
}
|
||||
if (raw.includes('404') || raw.includes('not found') || raw.includes('nicht gefunden')) {
|
||||
return 'Server oder Ressource nicht gefunden. URL/Repo prüfen.';
|
||||
}
|
||||
if (raw.includes('econnrefused') || raw.includes('enotfound') || raw.includes('eai_again') || raw.includes('getaddrinfo')) {
|
||||
return 'Server nicht erreichbar. DNS, IPv4/IPv6 und Port prüfen.';
|
||||
}
|
||||
if (raw.includes('timeout') || raw.includes('econnaborted') || raw.includes('zeitueberschreitung') || raw.includes('zeitüberschreitung')) {
|
||||
return 'Zeitüberschreitung bei der Verbindung. Bitte erneut versuchen.';
|
||||
}
|
||||
if (raw.includes('ungueltige') || raw.includes('ungültige') || raw.includes('invalid') || raw.includes('url')) {
|
||||
return 'Ungültige URL. Beispiel für IPv6: http://[2001:db8::1]:3000';
|
||||
}
|
||||
return String(message);
|
||||
}
|
||||
|
||||
function setStatus(txt) {
|
||||
const s = $('status');
|
||||
if (s) s.innerText = txt || '';
|
||||
@@ -404,9 +633,15 @@ function showToast(message, type = 'info', duration = 4000) {
|
||||
}
|
||||
|
||||
// Kurzformen
|
||||
function showError(msg) { setStatus(msg); showToast(msg, 'error'); }
|
||||
function showSuccess(msg) { setStatus(msg); showToast(msg, 'success', 3000); }
|
||||
function showWarning(msg) { setStatus(msg); showToast(msg, 'warning'); }
|
||||
function showError(msg) {
|
||||
const friendly = mapErrorMessage(msg);
|
||||
updateSettingsHealth({ lastError: friendly });
|
||||
setStatus(friendly);
|
||||
showToast(friendly, 'error');
|
||||
logActivity('error', friendly);
|
||||
}
|
||||
function showSuccess(msg) { setStatus(msg); showToast(msg, 'success', 3000); logActivity('info', msg); }
|
||||
function showWarning(msg) { setStatus(msg); showToast(msg, 'warning'); logActivity('warning', msg); }
|
||||
|
||||
// Löschen-Bestätigung als Toast (ersetzt confirm())
|
||||
function showDeleteConfirm(message, onConfirm) {
|
||||
@@ -1230,7 +1465,7 @@ async function saveCurrentFile(isAutoSave = false) {
|
||||
});
|
||||
}
|
||||
|
||||
if (response.ok) {
|
||||
if (response.ok && !response.queued) {
|
||||
tab.originalContent = content;
|
||||
tab.dirty = false;
|
||||
// Push current state to history
|
||||
@@ -1243,6 +1478,13 @@ async function saveCurrentFile(isAutoSave = false) {
|
||||
setStatus(`✓ Gespeichert: ${tab.name}`);
|
||||
}
|
||||
console.log('✅ File saved');
|
||||
} else if (response.ok && response.queued) {
|
||||
tab.originalContent = content;
|
||||
tab.dirty = false;
|
||||
pushToHistory(content);
|
||||
renderTabs();
|
||||
showWarning(response.message || 'Änderung in Retry-Queue gelegt und wird später hochgeladen.');
|
||||
updateRetryQueueBadge(retryQueueCount + 1);
|
||||
} else {
|
||||
alert(`Fehler: ${response.error}`);
|
||||
}
|
||||
@@ -1363,13 +1605,17 @@ async function loadGiteaRepos() {
|
||||
}
|
||||
|
||||
setStatus('Loading Gitea repos...');
|
||||
updateSettingsHealth({ lastError: '-' });
|
||||
|
||||
try {
|
||||
const res = await window.electronAPI.listGiteaRepos();
|
||||
if (!res.ok) {
|
||||
showError('Failed to load repos: ' + (res.error || 'Unknown error'));
|
||||
updateSettingsHealth({ api: 'Fehler', auth: 'Fehler' });
|
||||
return;
|
||||
}
|
||||
|
||||
updateSettingsHealth({ api: 'Erreichbar', auth: 'OK', lastError: '-' });
|
||||
|
||||
const grid = $('explorerGrid');
|
||||
if (!grid) return;
|
||||
@@ -1586,6 +1832,7 @@ async function loadGiteaRepos() {
|
||||
} catch (error) {
|
||||
console.error('Error loading repos:', error);
|
||||
showError('Error loading repositories');
|
||||
updateSettingsHealth({ api: 'Fehler', auth: 'Unbekannt' });
|
||||
}
|
||||
}
|
||||
|
||||
@@ -2802,8 +3049,11 @@ function setupGlobalDropZone() {
|
||||
INITIALISIERUNG
|
||||
------------------------- */
|
||||
window.addEventListener('DOMContentLoaded', async () => {
|
||||
initializePlatformSelection();
|
||||
|
||||
// Favoriten & Verlauf vorladen
|
||||
await loadFavoritesAndRecent();
|
||||
renderSettingsHealth();
|
||||
// Prevent default drag/drop on document (except in repo view)
|
||||
document.addEventListener('dragover', e => {
|
||||
if (currentState.view !== 'gitea-repo') {
|
||||
@@ -2825,6 +3075,16 @@ window.addEventListener('DOMContentLoaded', async () => {
|
||||
if ($('githubToken')) $('githubToken').value = creds.githubToken || '';
|
||||
if ($('giteaToken')) $('giteaToken').value = creds.giteaToken || '';
|
||||
if ($('giteaURL')) $('giteaURL').value = creds.giteaURL || '';
|
||||
renderGiteaUrlHint(creds.giteaURL || '');
|
||||
|
||||
const checkedUrl = normalizeAndValidateGiteaUrl(creds.giteaURL || '');
|
||||
updateSettingsHealth({
|
||||
url: checkedUrl.ok && checkedUrl.value ? 'Gültig' : (checkedUrl.ok ? 'Leer' : 'Ungültig'),
|
||||
api: creds.giteaURL ? 'Unbekannt' : 'Nicht konfiguriert',
|
||||
auth: creds.giteaToken ? 'Token vorhanden' : 'Kein Token',
|
||||
latency: '-',
|
||||
version: '-'
|
||||
});
|
||||
|
||||
// Feature-Flags aus gespeicherten Einstellungen
|
||||
if (typeof creds.featureFavorites === 'boolean') featureFavorites = creds.featureFavorites;
|
||||
@@ -2863,6 +3123,15 @@ window.addEventListener('DOMContentLoaded', async () => {
|
||||
} else {
|
||||
console.log('ℹ️ Keine Credentials gespeichert');
|
||||
setStatus('Bereit - bitte Settings konfigurieren');
|
||||
renderGiteaUrlHint('');
|
||||
updateSettingsHealth({
|
||||
url: 'Nicht konfiguriert',
|
||||
api: 'Nicht konfiguriert',
|
||||
auth: 'Kein Token',
|
||||
latency: '-',
|
||||
version: '-',
|
||||
lastError: '-'
|
||||
});
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Error loading credentials:', error);
|
||||
@@ -2906,11 +3175,249 @@ window.addEventListener('DOMContentLoaded', async () => {
|
||||
if ($('btnSettings')) {
|
||||
$('btnSettings').onclick = () => {
|
||||
$('settingsModal').classList.remove('hidden');
|
||||
$('settingsWatermarkCard')?.classList.add('hidden');
|
||||
renderSettingsHealth();
|
||||
requestAnimationFrame(syncSettingsPanelHeights);
|
||||
};
|
||||
}
|
||||
|
||||
if ($('btnSettingsWatermark') && $('settingsWatermarkCard')) {
|
||||
$('btnSettingsWatermark').onclick = (e) => {
|
||||
e.stopPropagation();
|
||||
$('settingsWatermarkCard').classList.toggle('hidden');
|
||||
};
|
||||
|
||||
$('settingsWatermarkCard').addEventListener('click', (e) => {
|
||||
e.stopPropagation();
|
||||
});
|
||||
|
||||
document.addEventListener('click', (e) => {
|
||||
if ($('settingsModal')?.classList.contains('hidden')) return;
|
||||
if ($('settingsWatermarkCard')?.classList.contains('hidden')) return;
|
||||
const target = e.target;
|
||||
if ($('btnSettingsWatermark')?.contains(target)) return;
|
||||
if ($('settingsWatermarkCard')?.contains(target)) return;
|
||||
$('settingsWatermarkCard')?.classList.add('hidden');
|
||||
});
|
||||
}
|
||||
|
||||
window.addEventListener('resize', syncSettingsPanelHeights);
|
||||
|
||||
if ($('btnBatchActions')) {
|
||||
$('btnBatchActions').onclick = () => {
|
||||
$('batchActionModal')?.classList.remove('hidden');
|
||||
updateBatchActionFields();
|
||||
};
|
||||
}
|
||||
|
||||
if ($('btnOpenActivityLog')) {
|
||||
$('btnOpenActivityLog').onclick = () => {
|
||||
$('activityLogModal')?.classList.remove('hidden');
|
||||
renderActivityLog();
|
||||
};
|
||||
}
|
||||
|
||||
if ($('btnCloseActivityLog')) {
|
||||
$('btnCloseActivityLog').onclick = () => $('activityLogModal')?.classList.add('hidden');
|
||||
}
|
||||
|
||||
if ($('activityFilterLevel')) {
|
||||
$('activityFilterLevel').addEventListener('change', renderActivityLog);
|
||||
}
|
||||
|
||||
if ($('btnClearActivityLog')) {
|
||||
$('btnClearActivityLog').onclick = () => {
|
||||
activityEntries = [];
|
||||
renderActivityLog();
|
||||
};
|
||||
}
|
||||
|
||||
if ($('btnCloseBatchAction')) {
|
||||
$('btnCloseBatchAction').onclick = () => $('batchActionModal')?.classList.add('hidden');
|
||||
}
|
||||
|
||||
if ($('batchActionType')) {
|
||||
$('batchActionType').addEventListener('change', updateBatchActionFields);
|
||||
updateBatchActionFields();
|
||||
}
|
||||
|
||||
if ($('btnSelectBatchCloneTarget')) {
|
||||
$('btnSelectBatchCloneTarget').onclick = async () => {
|
||||
const folder = await window.electronAPI.selectFolder();
|
||||
if (folder && $('batchCloneTarget')) {
|
||||
$('batchCloneTarget').value = folder;
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
if ($('btnRunBatchAction')) {
|
||||
$('btnRunBatchAction').onclick = async () => {
|
||||
const action = $('batchActionType')?.value || 'refresh';
|
||||
const repos = parseBatchRepoInput($('batchRepoList')?.value || '');
|
||||
if (repos.length === 0) {
|
||||
showWarning('Bitte mindestens ein Repository im Format owner/repo eintragen.');
|
||||
return;
|
||||
}
|
||||
|
||||
const options = {
|
||||
cloneTargetDir: $('batchCloneTarget')?.value || '',
|
||||
tag: $('batchTagName')?.value || '',
|
||||
name: $('batchReleaseName')?.value || '',
|
||||
body: $('batchReleaseBody')?.value || ''
|
||||
};
|
||||
|
||||
if (action === 'clone' && !options.cloneTargetDir) {
|
||||
showWarning('Bitte zuerst einen Zielordner für Clone wählen.');
|
||||
return;
|
||||
}
|
||||
|
||||
if ((action === 'create-tag' || action === 'create-release') && !String(options.tag).trim()) {
|
||||
showWarning('Bitte einen Tag eintragen.');
|
||||
return;
|
||||
}
|
||||
|
||||
const btn = $('btnRunBatchAction');
|
||||
const old = btn.textContent;
|
||||
btn.disabled = true;
|
||||
btn.textContent = 'Läuft...';
|
||||
logActivity('info', `Batch gestartet: ${action} (${repos.length} Repos)`);
|
||||
|
||||
try {
|
||||
const res = await window.electronAPI.runBatchRepoAction({ action, repos, options });
|
||||
if (!res.ok) {
|
||||
showError(res.error || 'Batch-Aktion fehlgeschlagen');
|
||||
return;
|
||||
}
|
||||
|
||||
const summary = res.summary || { total: repos.length, success: 0, failed: 0 };
|
||||
if (summary.failed > 0) {
|
||||
showWarning(`Batch beendet: ${summary.success}/${summary.total} erfolgreich, ${summary.failed} fehlgeschlagen.`);
|
||||
} else {
|
||||
showSuccess(`Batch erfolgreich: ${summary.success}/${summary.total}`);
|
||||
}
|
||||
|
||||
(res.results || []).forEach(r => {
|
||||
if (r.ok) logActivity('info', `${r.repo}: ${r.message || 'OK'}`);
|
||||
else logActivity('error', `${r.repo}: ${r.error || 'Fehler'}`);
|
||||
});
|
||||
} catch (error) {
|
||||
showError(error && error.message ? error.message : String(error));
|
||||
} finally {
|
||||
btn.disabled = false;
|
||||
btn.textContent = old;
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
if ($('btnRetryQueueNow')) {
|
||||
$('btnRetryQueueNow').onclick = async () => {
|
||||
try {
|
||||
const res = await window.electronAPI.processRetryQueueNow();
|
||||
if (res.ok) {
|
||||
showSuccess(`Queue verarbeitet: ${res.succeeded || 0} erfolgreich, ${res.failed || 0} verworfen.`);
|
||||
} else {
|
||||
showWarning(res.error || 'Queue konnte nicht verarbeitet werden');
|
||||
}
|
||||
} catch (e) {
|
||||
showError(e && e.message ? e.message : String(e));
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
if ($('btnRetryQueueRefresh')) {
|
||||
$('btnRetryQueueRefresh').onclick = async () => {
|
||||
try {
|
||||
const res = await window.electronAPI.processRetryQueueNow();
|
||||
if (res.ok) {
|
||||
showSuccess(`Queue verarbeitet: ${res.succeeded || 0} erfolgreich, ${res.failed || 0} verworfen.`);
|
||||
} else {
|
||||
showWarning(res.error || 'Queue konnte nicht verarbeitet werden');
|
||||
}
|
||||
} catch (e) {
|
||||
showError(e && e.message ? e.message : String(e));
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
if ($('giteaURL')) {
|
||||
$('giteaURL').addEventListener('input', (e) => {
|
||||
const raw = e.target.value;
|
||||
renderGiteaUrlHint(raw);
|
||||
const checked = normalizeAndValidateGiteaUrl(raw);
|
||||
updateSettingsHealth({
|
||||
url: checked.ok && checked.value ? 'Gültig' : (checked.ok ? 'Leer' : 'Ungültig')
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
if ($('btnTestGiteaConnection')) {
|
||||
$('btnTestGiteaConnection').onclick = async () => {
|
||||
const token = $('giteaToken')?.value || '';
|
||||
const rawUrl = $('giteaURL')?.value || '';
|
||||
const checked = normalizeAndValidateGiteaUrl(rawUrl);
|
||||
if (!checked.ok) {
|
||||
showError(checked.error);
|
||||
updateSettingsHealth({ url: 'Ungültig', api: 'Unbekannt', auth: token ? 'Token vorhanden' : 'Kein Token' });
|
||||
return;
|
||||
}
|
||||
if (!checked.value) {
|
||||
showWarning('Bitte zuerst eine Gitea URL eintragen.');
|
||||
return;
|
||||
}
|
||||
|
||||
setStatus('Teste Gitea-Verbindung...');
|
||||
const btn = $('btnTestGiteaConnection');
|
||||
const oldText = btn.textContent;
|
||||
btn.disabled = true;
|
||||
btn.textContent = 'Teste...';
|
||||
try {
|
||||
const res = await window.electronAPI.testGiteaConnection({
|
||||
token,
|
||||
url: checked.value,
|
||||
timeout: 8000
|
||||
});
|
||||
|
||||
if (!res.ok) {
|
||||
showError(res.error || 'Verbindungstest fehlgeschlagen');
|
||||
updateSettingsHealth({
|
||||
url: 'Gültig',
|
||||
api: 'Fehler',
|
||||
auth: token ? 'Fehler' : 'Kein Token',
|
||||
latency: '-',
|
||||
version: '-'
|
||||
});
|
||||
return;
|
||||
}
|
||||
|
||||
const result = res.result || {};
|
||||
const checks = result.checks || {};
|
||||
const metrics = result.metrics || {};
|
||||
const server = result.server || {};
|
||||
|
||||
updateSettingsHealth({
|
||||
url: 'Gültig',
|
||||
api: checks.apiReachable ? 'Erreichbar' : 'Fehler',
|
||||
auth: checks.authProvided ? (checks.authOk ? 'OK' : 'Fehler') : 'Kein Token',
|
||||
latency: metrics.latencyMs ? `${metrics.latencyMs} ms` : '-',
|
||||
version: server.version || '-',
|
||||
lastError: '-'
|
||||
});
|
||||
|
||||
if (result.ok) showSuccess('Verbindung erfolgreich getestet');
|
||||
else showWarning('Server erreichbar, aber Auth/Teilcheck fehlgeschlagen');
|
||||
} catch (error) {
|
||||
console.error('test-gitea-connection error:', error);
|
||||
showError(error && error.message ? error.message : String(error));
|
||||
} finally {
|
||||
btn.disabled = false;
|
||||
btn.textContent = oldText;
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
if ($('btnCloseSettings')) {
|
||||
$('btnCloseSettings').onclick = () => {
|
||||
$('settingsWatermarkCard')?.classList.add('hidden');
|
||||
$('settingsModal').classList.add('hidden');
|
||||
};
|
||||
}
|
||||
@@ -2926,6 +3433,14 @@ window.addEventListener('DOMContentLoaded', async () => {
|
||||
$('repoActionModal').classList.add('hidden');
|
||||
};
|
||||
}
|
||||
|
||||
try {
|
||||
const queue = await window.electronAPI.getRetryQueue();
|
||||
if (queue && queue.ok) {
|
||||
updateRetryQueueBadge(queue.size || 0);
|
||||
logActivity('info', `Retry-Queue geladen (${queue.size || 0} Einträge)`);
|
||||
}
|
||||
} catch (_) {}
|
||||
|
||||
if ($('btnSaveSettings')) {
|
||||
$('btnSaveSettings').onclick = async () => {
|
||||
@@ -2941,10 +3456,16 @@ window.addEventListener('DOMContentLoaded', async () => {
|
||||
featureColoredIcons = cbColorIcons2 ? cbColorIcons2.checked : true;
|
||||
document.body.classList.toggle('compact-mode', compactMode);
|
||||
|
||||
const checkedUrl = normalizeAndValidateGiteaUrl($('giteaURL').value);
|
||||
if (!checkedUrl.ok) {
|
||||
showError(checkedUrl.error);
|
||||
return;
|
||||
}
|
||||
|
||||
const data = {
|
||||
githubToken: $('githubToken').value,
|
||||
giteaToken: $('giteaToken').value,
|
||||
giteaURL: $('giteaURL').value,
|
||||
giteaURL: checkedUrl.value,
|
||||
featureFavorites,
|
||||
featureRecent,
|
||||
compactMode,
|
||||
@@ -2955,6 +3476,12 @@ window.addEventListener('DOMContentLoaded', async () => {
|
||||
await window.electronAPI.saveCredentials(data);
|
||||
$('settingsModal').classList.add('hidden');
|
||||
showSuccess('Settings saved');
|
||||
renderGiteaUrlHint(checkedUrl.value);
|
||||
updateSettingsHealth({
|
||||
url: checkedUrl.value ? 'Gültig' : 'Leer',
|
||||
auth: data.giteaToken ? 'Token vorhanden' : 'Kein Token',
|
||||
lastError: '-'
|
||||
});
|
||||
// Ansicht aktualisieren falls Feature-Flags geändert
|
||||
loadGiteaRepos();
|
||||
} catch (error) {
|
||||
@@ -3087,6 +3614,30 @@ window.addEventListener('DOMContentLoaded', async () => {
|
||||
showProgress(p.percent, `Download: ${p.processed}/${p.total}`);
|
||||
});
|
||||
|
||||
if (window.electronAPI.onRetryQueueUpdated) {
|
||||
window.electronAPI.onRetryQueueUpdated((payload) => {
|
||||
const size = payload && typeof payload.size === 'number' ? payload.size : 0;
|
||||
updateRetryQueueBadge(size);
|
||||
if (payload && payload.event === 'queued' && payload.item) {
|
||||
const p = payload.item.payload || {};
|
||||
logActivity('warning', `Queue: ${p.owner}/${p.repo}/${p.path} wurde eingeplant`);
|
||||
} else if (payload && payload.event === 'processed') {
|
||||
logActivity('info', `Queue-Retry: ${payload.succeeded || 0} erfolgreich, ${payload.failed || 0} verworfen`);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
if (window.electronAPI.onBatchActionProgress) {
|
||||
window.electronAPI.onBatchActionProgress((payload) => {
|
||||
if (!payload) return;
|
||||
if (payload.status === 'running') {
|
||||
logActivity('info', `Batch ${payload.action}: ${payload.repo} (${payload.index}/${payload.total})`);
|
||||
} else if (payload.status === 'error') {
|
||||
logActivity('error', `Batch ${payload.action}: ${payload.repo} - ${payload.error || 'Fehler'}`);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
// Setup globalen Drop-Handler für Repo-Ansicht
|
||||
setupGlobalDropZone();
|
||||
setupBackgroundContextMenu();
|
||||
@@ -4424,9 +4975,16 @@ async function initUpdater() {
|
||||
if (versionRes && versionRes.ok && $('appVersion')) {
|
||||
$('appVersion').value = versionRes.version;
|
||||
}
|
||||
if (versionRes && versionRes.ok && $('watermarkVersion')) {
|
||||
$('watermarkVersion').textContent = versionRes.version;
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('[Renderer] Fehler beim Laden der Version:', error);
|
||||
}
|
||||
|
||||
if ($('watermarkCopyright')) {
|
||||
$('watermarkCopyright').textContent = `© ${new Date().getFullYear()} M_Viper`;
|
||||
}
|
||||
|
||||
// Manueller Check Button in Settings
|
||||
if ($('btnCheckUpdates')) {
|
||||
|
||||
1079
renderer/style.css
1079
renderer/style.css
File diff suppressed because it is too large
Load Diff
Reference in New Issue
Block a user