feat(dashboard): Logbuch-Titel per Inline-Bearbeitung umbenennen

Ersetzt Umbenennen-Button und Modal durch Klick auf den Kartentitel.

Co-authored-by: Cursor <cursoragent@cursor.com>
This commit is contained in:
2026-05-31 10:48:22 +02:00
parent 917fb92d85
commit 257bca14d1
3 changed files with 89 additions and 167 deletions
+20 -26
View File
@@ -1298,6 +1298,26 @@ html.scheme-dark .themed-select-option.is-selected {
margin: 0;
}
.logbook-title-editable {
cursor: text;
border-radius: 4px;
transition: background-color 0.15s ease;
}
.logbook-title-editable:hover {
background: var(--app-accent-bg);
}
.logbook-title-inline-edit {
flex: 1;
min-width: 0;
margin: 0;
padding: 2px 8px;
font-size: 16px;
font-weight: 600;
line-height: 1.4;
}
.card-icon {
background: var(--app-accent-bg);
color: var(--app-accent-light);
@@ -1418,32 +1438,6 @@ html.scheme-dark .themed-select-option.is-selected {
background: rgba(244, 63, 94, 0.1);
}
.btn-edit {
background: none;
border: none;
color: #475569;
cursor: pointer;
padding: 6px;
border-radius: 6px;
display: flex;
align-items: center;
justify-content: center;
opacity: 0;
transition: all 0.2s ease;
position: absolute;
top: 12px;
right: 48px;
}
.logbook-card:hover .btn-edit {
opacity: 1;
}
.btn-edit:hover {
color: var(--app-accent-light);
background: var(--app-accent-bg);
}
.btn-pdf {
background: none;
border: none;
-108
View File
@@ -1,108 +0,0 @@
import React, { useState, useEffect } from 'react'
import { useTranslation } from 'react-i18next'
import { Edit2, X } from 'lucide-react'
interface EditLogbookModalProps {
open: boolean
onClose: () => void
currentTitle: string
onSave: (newTitle: string) => Promise<void>
}
export default function EditLogbookModal({ open, onClose, currentTitle, onSave }: EditLogbookModalProps) {
const { t } = useTranslation()
const [title, setTitle] = useState(currentTitle)
const [loading, setLoading] = useState(false)
const [error, setError] = useState<string | null>(null)
useEffect(() => {
if (open) {
setTitle(currentTitle)
setError(null)
}
}, [open, currentTitle])
if (!open) return null
const handleSubmit = async (e: React.FormEvent) => {
e.preventDefault()
const trimmedTitle = title.trim()
if (!trimmedTitle) return
if (trimmedTitle === currentTitle.trim()) {
onClose()
return
}
setLoading(true)
setError(null)
try {
await onSave(trimmedTitle)
onClose()
} catch (err: any) {
setError(err.message || 'Failed to rename logbook')
} finally {
setLoading(false)
}
}
return (
<div className="custom-dialog-overlay" onClick={onClose}>
<div className="custom-dialog-card glass scale-in" onClick={(e) => e.stopPropagation()}>
<button
type="button"
className="registration-disclaimer__close feedback-modal__close"
onClick={onClose}
disabled={loading}
aria-label={t('feedback.cancel')}
style={{ position: 'absolute', top: '12px', right: '12px' }}
>
<X size={18} />
</button>
<h3 className="custom-dialog-title" style={{ display: 'flex', alignItems: 'center', gap: '8px', justifyContent: 'center' }}>
<Edit2 size={20} />
{t('dashboard.edit_title')}
</h3>
<form onSubmit={handleSubmit} style={{ width: '100%', marginTop: '16px' }}>
<div className="input-group" style={{ marginBottom: '20px' }}>
<input
type="text"
className="input-text"
placeholder={t('dashboard.edit_placeholder')}
value={title}
onChange={(e) => setTitle(e.target.value)}
disabled={loading}
required
autoFocus
style={{ width: '100%', textAlign: 'left' }}
/>
</div>
{error && <div className="auth-error" style={{ marginBottom: '16px', fontSize: '13px' }}>{error}</div>}
<div className="custom-dialog-actions">
<button
type="button"
className="btn secondary"
onClick={onClose}
disabled={loading}
style={{ width: 'auto', padding: '8px 20px', margin: 0 }}
>
{t('logs.cancel_event_edit')}
</button>
<button
type="submit"
className="btn primary"
disabled={loading || !title.trim()}
style={{ width: 'auto', minWidth: '80px', padding: '8px 20px', margin: 0 }}
>
{t('dashboard.edit_btn')}
</button>
</div>
</form>
</div>
</div>
)
}
+69 -33
View File
@@ -1,4 +1,4 @@
import React, { useState, useEffect } from 'react'
import React, { useState, useEffect, useRef } from 'react'
import { useTranslation } from 'react-i18next'
import { useLiveQuery } from 'dexie-react-hooks'
import { db } from '../services/db.js'
@@ -8,10 +8,9 @@ import BetaBadge from './BetaBadge.tsx'
import { PlausibleEvents, trackPlausibleEvent } from '../services/analytics.js'
import { logoutUser } from '../services/auth.js'
import { useDialog } from './ModalDialog.tsx'
import { BookOpen, Plus, Trash2, Edit2, LogOut, Languages, RefreshCw, Ship, User, Wifi, WifiOff } from 'lucide-react'
import { BookOpen, Plus, Trash2, LogOut, Languages, RefreshCw, Ship, User, Wifi, WifiOff } from 'lucide-react'
import DisclaimerHeaderButton from './DisclaimerHeaderButton.tsx'
import FeedbackHeaderButton from './FeedbackHeaderButton.tsx'
import EditLogbookModal from './EditLogbookModal.tsx'
interface LogbookDashboardProps {
onSelectLogbook: (id: string, title: string) => void
@@ -24,7 +23,9 @@ export default function LogbookDashboard({ onSelectLogbook, onLogout, onOpenProf
const { showConfirm } = useDialog()
const [logbooks, setLogbooks] = useState<DecryptedLogbook[]>([])
const [newTitle, setNewTitle] = useState('')
const [editingLogbook, setEditingLogbook] = useState<DecryptedLogbook | null>(null)
const [editingLogbookId, setEditingLogbookId] = useState<string | null>(null)
const [editingTitleDraft, setEditingTitleDraft] = useState('')
const titleInputRef = useRef<HTMLInputElement>(null)
const [loading, setLoading] = useState(false)
const [refreshing, setRefreshing] = useState(false)
const [error, setError] = useState<string | null>(null)
@@ -101,25 +102,44 @@ export default function LogbookDashboard({ onSelectLogbook, onLogout, onOpenProf
}
}
const handleEdit = (lb: DecryptedLogbook, e: React.MouseEvent) => {
e.stopPropagation() // Prevent selecting the logbook when clicking edit
setEditingLogbook(lb)
useEffect(() => {
if (editingLogbookId) {
titleInputRef.current?.focus()
titleInputRef.current?.select()
}
}, [editingLogbookId])
const startTitleEdit = (lb: DecryptedLogbook, e: React.MouseEvent) => {
e.stopPropagation()
setEditingLogbookId(lb.id)
setEditingTitleDraft(lb.title)
}
const handleSaveTitle = async (newTitle: string) => {
if (!editingLogbook) return
const cancelTitleEdit = () => {
setEditingLogbookId(null)
setEditingTitleDraft('')
}
const commitTitleEdit = async (id: string) => {
if (editingLogbookId !== id) return
const lb = logbooks.find((item) => item.id === id)
const trimmedTitle = editingTitleDraft.trim()
cancelTitleEdit()
if (!lb || !trimmedTitle || trimmedTitle === lb.title.trim()) return
setLoading(true)
setError(null)
try {
await updateLogbookTitle(editingLogbook.id, newTitle)
await updateLogbookTitle(id, trimmedTitle)
setLogbooks((prev) =>
prev.map((lb) => (lb.id === editingLogbook.id ? { ...lb, title: newTitle, updatedAt: new Date().toISOString() } : lb))
prev.map((item) =>
item.id === id ? { ...item, title: trimmedTitle, updatedAt: new Date().toISOString() } : item
)
)
setEditingLogbook(null)
} catch (err: any) {
setError(err.message || 'Failed to update logbook title')
throw err
} finally {
setLoading(false)
}
@@ -138,7 +158,10 @@ export default function LogbookDashboard({ onSelectLogbook, onLogout, onOpenProf
const ownedLogbooks = logbooks.filter((lb) => !lb.isShared)
const sharedLogbooks = logbooks.filter((lb) => lb.isShared)
const renderLogbookCard = (lb: DecryptedLogbook) => (
const renderLogbookCard = (lb: DecryptedLogbook) => {
const isEditingTitle = editingLogbookId === lb.id
return (
<div
key={lb.id}
className={`logbook-card glass${lb.isShared ? ' logbook-card--shared' : ''}`}
@@ -150,7 +173,36 @@ export default function LogbookDashboard({ onSelectLogbook, onLogout, onOpenProf
<div className="card-info">
<div className="card-title-row">
<h3>{lb.title}</h3>
{isEditingTitle ? (
<input
ref={titleInputRef}
type="text"
className="logbook-title-inline-edit input-text"
value={editingTitleDraft}
onChange={(e) => setEditingTitleDraft(e.target.value)}
onClick={(e) => e.stopPropagation()}
onKeyDown={(e) => {
if (e.key === 'Enter') {
e.preventDefault()
void commitTitleEdit(lb.id)
} else if (e.key === 'Escape') {
e.preventDefault()
cancelTitleEdit()
}
}}
onBlur={() => void commitTitleEdit(lb.id)}
disabled={loading}
aria-label={t('dashboard.edit_title')}
/>
) : (
<h3
className={lb.isShared ? undefined : 'logbook-title-editable'}
onClick={lb.isShared ? undefined : (e) => startTitleEdit(lb, e)}
title={lb.isShared ? undefined : t('dashboard.edit_title')}
>
{lb.title}
</h3>
)}
<LogbookRoleBadge role={lb.accessRole} />
</div>
<div className="card-meta">
@@ -170,15 +222,6 @@ export default function LogbookDashboard({ onSelectLogbook, onLogout, onOpenProf
</div>
</div>
<button
className="btn-edit"
onClick={(e) => handleEdit(lb, e)}
title={t('dashboard.edit_title')}
style={{ visibility: lb.isShared ? 'hidden' : 'visible' }}
>
<Edit2 size={18} />
</button>
<button
className="btn-delete"
onClick={(e) => handleDelete(lb.id, e)}
@@ -188,7 +231,8 @@ export default function LogbookDashboard({ onSelectLogbook, onLogout, onOpenProf
<Trash2 size={18} />
</button>
</div>
)
)
}
const renderLogbookSection = (
title: string,
@@ -326,14 +370,6 @@ export default function LogbookDashboard({ onSelectLogbook, onLogout, onOpenProf
)}
</section>
</main>
{/* Edit Logbook Title Modal */}
<EditLogbookModal
open={editingLogbook !== null}
onClose={() => setEditingLogbook(null)}
currentTitle={editingLogbook?.title || ''}
onSave={handleSaveTitle}
/>
</div>
)
}