from django.contrib import admin from django.utils.html import format_html from django.urls import reverse from django.db.models import Sum, Count from django.utils.safestring import mark_safe from .models import ( Person, Paechter, Destinataer, DokumentLink, Foerderung, Land, Verpachtung, CSVImport, Rentmeister, StiftungsKonto, Verwaltungskosten, BankTransaction, AuditLog, BackupJob ) @admin.register(CSVImport) class CSVImportAdmin(admin.ModelAdmin): list_display = ['import_type', 'filename', 'status', 'total_rows', 'imported_rows', 'failed_rows', 'created_by', 'started_at', 'duration_display'] list_filter = ['import_type', 'status', 'started_at'] search_fields = ['filename', 'created_by'] readonly_fields = ['id', 'started_at', 'completed_at', 'get_success_rate'] ordering = ['-started_at'] fieldsets = ( ('Grundinformationen', { 'fields': ('import_type', 'filename', 'file_size', 'status') }), ('Ergebnisse', { 'fields': ('total_rows', 'imported_rows', 'failed_rows', 'get_success_rate', 'error_log') }), ('Metadaten', { 'fields': ('created_by', 'started_at', 'completed_at') }), ) def duration_display(self, obj): duration = obj.get_duration() if duration: return f"{duration.total_seconds():.1f}s" return "-" duration_display.short_description = "Dauer" def get_success_rate(self, obj): rate = obj.get_success_rate() if rate >= 90: color = "success" elif rate >= 70: color = "warning" else: color = "danger" return format_html('{:.1f}%', color, rate) get_success_rate.short_description = "Erfolgsrate" @admin.register(Person) class PersonAdmin(admin.ModelAdmin): list_display = ['nachname', 'vorname', 'familienzweig', 'geburtsdatum', 'email', 'iban_display'] list_filter = ['familienzweig', 'geburtsdatum'] search_fields = ['nachname', 'vorname', 'email', 'familienzweig'] ordering = ['nachname', 'vorname'] readonly_fields = ['id'] fieldsets = ( ('Persönliche Daten', { 'fields': ('vorname', 'nachname', 'geburtsdatum', 'email', 'telefon') }), ('Stiftungsdaten', { 'fields': ('familienzweig', 'iban', 'adresse') }), ('Zusätzlich', { 'fields': ('notizen', 'aktiv') }), ('System', { 'fields': ('id',), 'classes': ('collapse',) }), ) def iban_display(self, obj): if obj.iban: return format_html('{}', obj.iban) return '-' iban_display.short_description = 'IBAN' def get_queryset(self, request): return super().get_queryset(request).annotate( total_foerderungen=Sum('foerderung__betrag') ) @admin.register(Paechter) class PaechterAdmin(admin.ModelAdmin): list_display = ['nachname', 'vorname', 'pachtnummer', 'pachtzins_aktuell', 'landwirtschaftliche_ausbildung', 'aktiv'] list_filter = ['landwirtschaftliche_ausbildung', 'aktiv'] search_fields = ['nachname', 'vorname', 'email', 'pachtnummer'] ordering = ['nachname', 'vorname'] readonly_fields = ['id'] fieldsets = ( ('Persönliche Daten', { 'fields': ('vorname', 'nachname', 'geburtsdatum', 'email', 'telefon') }), ('Pacht-Informationen', { 'fields': ('pachtnummer', 'pachtbeginn_erste', 'pachtende_letzte', 'pachtzins_aktuell') }), ('Landwirtschaftliche Qualifikation', { 'fields': ('landwirtschaftliche_ausbildung', 'berufserfahrung_jahre', 'spezialisierung') }), ('Kontaktdaten', { 'fields': ('iban', 'strasse', 'plz', 'ort') }), ('Pächter-Typ', { 'fields': ('personentyp',) }), ('Zusätzlich', { 'fields': ('notizen', 'aktiv') }), ('System', { 'fields': ('id',), 'classes': ('collapse',) }), ) def iban_display(self, obj): if obj.iban: return format_html('{}', obj.iban) return '-' iban_display.short_description = 'IBAN' @admin.register(Destinataer) class DestinataerAdmin(admin.ModelAdmin): list_display = ['nachname', 'vorname', 'familienzweig', 'berufsgruppe', 'institution', 'finanzielle_notlage', 'aktiv'] list_filter = ['familienzweig', 'berufsgruppe', 'finanzielle_notlage', 'aktiv'] search_fields = ['nachname', 'vorname', 'email', 'institution', 'familienzweig'] ordering = ['nachname', 'vorname'] readonly_fields = ['id'] fieldsets = ( ('Persönliche Daten', { 'fields': ('vorname', 'nachname', 'geburtsdatum', 'email', 'telefon') }), ('Berufliche Informationen', { 'fields': ('berufsgruppe', 'ausbildungsstand', 'institution') }), ('Projekt & Finanzen', { 'fields': ('projekt_beschreibung', 'jaehrliches_einkommen', 'finanzielle_notlage') }), ('Stiftungsdaten', { 'fields': ('familienzweig', 'iban', 'strasse', 'plz', 'ort') }), ('Zusätzlich', { 'fields': ('notizen', 'aktiv') }), ('System', { 'fields': ('id',), 'classes': ('collapse',) }), ) def iban_display(self, obj): if obj.iban: return format_html('{}', obj.iban) return '-' iban_display.short_description = 'IBAN' @admin.register(Land) class LandAdmin(admin.ModelAdmin): list_display = [ 'lfd_nr', 'gemeinde', 'gemarkung', 'flur', 'flurstueck', 'groesse_qm', 'verp_flaeche_aktuell', 'verpachtungsgrad_display', 'aktiv' ] list_filter = ['gemeinde', 'gemarkung', 'aktiv'] search_fields = ['lfd_nr', 'gemeinde', 'gemarkung', 'flur', 'flurstueck'] ordering = ['gemeinde', 'gemarkung', 'flur', 'flurstueck'] readonly_fields = ['id', 'gesamtflaeche_berechnet', 'verpachtungsgrad_berechnet'] fieldsets = ( ('Identifikation', { 'fields': ('lfd_nr', 'ew_nummer') }), ('Gerichtliche Zuständigkeit', { 'fields': ('amtsgericht',) }), ('Verwaltungsstruktur', { 'fields': ('gemeinde', 'gemarkung', 'flur', 'flurstueck') }), ('Flächenangaben', { 'fields': ('groesse_qm', 'gruenland_qm', 'acker_qm', 'wald_qm', 'sonstiges_qm') }), ('Verpachtung', { 'fields': ('verpachtete_gesamtflaeche', 'flaeche_alte_liste', 'verp_flaeche_aktuell') }), ('Steuern und Abgaben', { 'fields': ('anteil_grundsteuer', 'anteil_lwk') }), ('Status', { 'fields': ('aktiv', 'notizen') }), ('System', { 'fields': ('id', 'erstellt_am', 'aktualisiert_am'), 'classes': ('collapse',) }), ) def verpachtungsgrad_display(self, obj): grad = obj.get_verpachtungsgrad() if grad > 90: color = 'green' elif grad > 70: color = 'orange' else: color = 'red' return format_html('{:.1f}%', color, grad) verpachtungsgrad_display.short_description = 'Verpachtungsgrad' def gesamtflaeche_berechnet(self, obj): return f"{obj.get_gesamtflaeche():.2f} qm" gesamtflaeche_berechnet.short_description = 'Berechnete Gesamtfläche' def verpachtungsgrad_berechnet(self, obj): return f"{obj.get_verpachtungsgrad():.1f}%" verpachtungsgrad_berechnet.short_description = 'Verpachtungsgrad' @admin.register(Verpachtung) class VerpachtungAdmin(admin.ModelAdmin): list_display = [ 'vertragsnummer', 'land', 'paechter', 'pachtbeginn', 'pachtende', 'pachtzins_jaehrlich', 'verpachtete_flaeche', 'status', 'restlaufzeit' ] list_filter = ['status', 'pachtbeginn', 'pachtende', 'land__gemeinde'] search_fields = ['vertragsnummer', 'land__gemeinde', 'paechter__nachname', 'paechter__vorname'] ordering = ['-pachtbeginn'] readonly_fields = ['id', 'vertragsdauer_tage', 'restlaufzeit_tage', 'is_aktiv_status'] fieldsets = ( ('Vertragsdaten', { 'fields': ('vertragsnummer', 'land', 'paechter', 'pachtbeginn', 'pachtende', 'verlaengerung') }), ('Finanzielle Bedingungen', { 'fields': ('pachtzins_pro_qm', 'pachtzins_jaehrlich') }), ('Flächenangaben', { 'fields': ('verpachtete_flaeche',) }), ('Status', { 'fields': ('status',) }), ('Dokumentation', { 'fields': ('verwendungsnachweis', 'bemerkungen') }), ('System', { 'fields': ('id', 'erstellt_am', 'aktualisiert_am'), 'classes': ('collapse',) }), ) def restlaufzeit(self, obj): tage = obj.get_restlaufzeit_tage() if tage > 0: if tage < 30: color = 'red' elif tage < 90: color = 'orange' else: color = 'green' return format_html('{} Tage', color, tage) return 'Abgelaufen' restlaufzeit.short_description = 'Restlaufzeit' def vertragsdauer_tage(self, obj): return f"{obj.get_vertragsdauer_tage()} Tage" vertragsdauer_tage.short_description = 'Vertragsdauer' def restlaufzeit_tage(self, obj): return f"{obj.get_restlaufzeit_tage()} Tage" restlaufzeit_tage.short_description = 'Restlaufzeit' def is_aktiv_status(self, obj): if obj.is_aktiv(): return format_html('✓ Aktiv') return format_html('✗ Inaktiv') is_aktiv_status.short_description = 'Aktueller Status' @admin.register(DokumentLink) class DokumentLinkAdmin(admin.ModelAdmin): list_display = ['titel', 'kontext', 'paperless_document_id'] list_filter = ['kontext'] search_fields = ['titel', 'kontext'] ordering = ['titel'] readonly_fields = ['id'] fieldsets = ( ('Dokument', { 'fields': ('titel', 'kontext', 'paperless_document_id', 'beschreibung') }), ('System', { 'fields': ('id',), 'classes': ('collapse',) }), ) @admin.register(Foerderung) class FoerderungAdmin(admin.ModelAdmin): list_display = ['destinataer', 'jahr', 'betrag', 'verwendungsnachweis_link', 'total_for_destinataer'] list_filter = ['jahr', 'destinataer__familienzweig'] search_fields = ['destinataer__nachname', 'destinataer__vorname', 'destinataer__familienzweig'] ordering = ['-jahr', '-betrag'] readonly_fields = ['id'] fieldsets = ( ('Förderung', { 'fields': ('destinataer', 'person', 'jahr', 'betrag', 'kategorie', 'status') }), ('Dokumentation', { 'fields': ('verwendungsnachweis', 'bemerkungen') }), ('Daten', { 'fields': ('antragsdatum', 'entscheidungsdatum') }), ('System', { 'fields': ('id',), 'classes': ('collapse',) }), ) def verwendungsnachweis_link(self, obj): if obj.verwendungsnachweis: return format_html( '{}', reverse('admin:stiftung_dokumentlink_change', args=[obj.verwendungsnachweis.id]), obj.verwendungsnachweis.titel ) return '-' verwendungsnachweis_link.short_description = 'Verwendungsnachweis' def total_for_destinataer(self, obj): total = Foerderung.objects.filter(destinataer=obj.destinataer).aggregate(Sum('betrag'))['betrag__sum'] or 0 return f"€{total:,.2f}" total_for_destinataer.short_description = 'Gesamt für Destinatär' @admin.register(Rentmeister) class RentmeisterAdmin(admin.ModelAdmin): list_display = ['__str__', 'email', 'telefon', 'seit_datum', 'bis_datum', 'aktiv', 'monatliche_verguetung'] list_filter = ['aktiv', 'seit_datum', 'anrede'] search_fields = ['vorname', 'nachname', 'email', 'telefon', 'ort'] ordering = ['nachname', 'vorname'] readonly_fields = ['id', 'erstellt_am', 'aktualisiert_am'] fieldsets = ( ('Persönliche Daten', { 'fields': ('anrede', 'vorname', 'nachname', 'titel') }), ('Kontaktdaten', { 'fields': ('email', 'telefon', 'mobil', 'strasse', 'plz', 'ort') }), ('Bankdaten', { 'fields': ('iban', 'bic', 'bank_name'), 'classes': ['collapse'] }), ('Stiftungsdaten', { 'fields': ('seit_datum', 'bis_datum', 'aktiv', 'monatliche_verguetung', 'km_pauschale') }), ('Zusätzliche Informationen', { 'fields': ('notizen',), 'classes': ['collapse'] }), ('System', { 'fields': ('id', 'erstellt_am', 'aktualisiert_am'), 'classes': ['collapse'] }) ) @admin.register(StiftungsKonto) class StiftungsKontoAdmin(admin.ModelAdmin): list_display = ['kontoname', 'bank_name', 'konto_typ', 'saldo', 'saldo_datum', 'aktiv'] list_filter = ['konto_typ', 'aktiv', 'bank_name'] search_fields = ['kontoname', 'bank_name', 'iban'] ordering = ['bank_name', 'kontoname'] readonly_fields = ['id', 'erstellt_am', 'aktualisiert_am'] fieldsets = ( ('Kontodaten', { 'fields': ('kontoname', 'bank_name', 'iban', 'bic', 'konto_typ') }), ('Finanzdaten', { 'fields': ('saldo', 'saldo_datum', 'zinssatz', 'laufzeit_bis') }), ('Status', { 'fields': ('aktiv', 'notizen') }), ('System', { 'fields': ('id', 'erstellt_am', 'aktualisiert_am'), 'classes': ['collapse'] }) ) @admin.register(Verwaltungskosten) class VerwaltungskostenAdmin(admin.ModelAdmin): list_display = ['bezeichnung', 'kategorie', 'betrag', 'datum', 'status', 'rentmeister', 'konto'] list_filter = ['kategorie', 'status', 'datum', 'rentmeister', 'konto'] search_fields = ['bezeichnung', 'lieferant_firma', 'rechnungsnummer', 'beschreibung'] ordering = ['-datum', '-erstellt_am'] readonly_fields = ['id', 'erstellt_am', 'aktualisiert_am'] date_hierarchy = 'datum' fieldsets = ( ('Grunddaten', { 'fields': ('bezeichnung', 'kategorie', 'betrag', 'datum', 'status') }), ('Zuordnung', { 'fields': ('rentmeister', 'konto') }), ('Lieferant/Rechnung', { 'fields': ('lieferant_firma', 'rechnungsnummer'), 'classes': ['collapse'] }), ('Fahrtkosten', { 'fields': ('km_anzahl', 'km_satz', 'von_ort', 'nach_ort', 'zweck'), 'classes': ['collapse'], 'description': 'Nur für Kategorie "Fahrtkosten" relevant' }), ('Zusätzliche Informationen', { 'fields': ('beschreibung', 'notizen'), 'classes': ['collapse'] }), ('System', { 'fields': ('id', 'erstellt_am', 'aktualisiert_am'), 'classes': ['collapse'] }) ) @admin.register(BankTransaction) class BankTransactionAdmin(admin.ModelAdmin): list_display = [ 'datum', 'konto', 'betrag', 'empfaenger_zahlungspflichtiger', 'transaction_type', 'status', 'verwaltungskosten' ] list_filter = [ 'konto', 'transaction_type', 'status', 'datum', 'importiert_am' ] search_fields = [ 'verwendungszweck', 'empfaenger_zahlungspflichtiger', 'referenz' ] readonly_fields = ['importiert_am', 'import_datei'] ordering = ['-datum', '-importiert_am'] fieldsets = ( ('Basisdaten', { 'fields': ('konto', 'datum', 'valuta', 'betrag', 'waehrung') }), ('Transaktionsdetails', { 'fields': ('verwendungszweck', 'empfaenger_zahlungspflichtiger', 'iban_gegenpartei', 'bic_gegenpartei', 'referenz', 'transaction_type') }), ('Verwaltung', { 'fields': ('status', 'kommentare', 'verwaltungskosten') }), ('Import-Information', { 'fields': ('import_datei', 'importiert_am', 'saldo_nach_buchung'), 'classes': ('collapse',) }), ) def get_queryset(self, request): return super().get_queryset(request).select_related('konto', 'verwaltungskosten') @admin.register(AuditLog) class AuditLogAdmin(admin.ModelAdmin): list_display = ['timestamp', 'username', 'action', 'entity_type', 'entity_name', 'ip_address'] list_filter = ['action', 'entity_type', 'timestamp', 'username'] search_fields = ['username', 'entity_name', 'description', 'ip_address'] readonly_fields = ['id', 'timestamp', 'user', 'username', 'action', 'entity_type', 'entity_id', 'entity_name', 'description', 'changes', 'ip_address', 'user_agent', 'session_key'] ordering = ['-timestamp'] date_hierarchy = 'timestamp' fieldsets = ( ('Benutzer und Zeit', { 'fields': ('timestamp', 'user', 'username', 'session_key') }), ('Aktion', { 'fields': ('action', 'entity_type', 'entity_id', 'entity_name', 'description') }), ('Änderungsdetails', { 'fields': ('changes',), 'classes': ['collapse'] }), ('Request-Informationen', { 'fields': ('ip_address', 'user_agent'), 'classes': ['collapse'] }), ('System', { 'fields': ('id',), 'classes': ['collapse'] }) ) def has_add_permission(self, request): return False # Don't allow manual creation def has_change_permission(self, request, obj=None): return False # Don't allow editing @admin.register(BackupJob) class BackupJobAdmin(admin.ModelAdmin): list_display = ['created_at', 'backup_type', 'status', 'backup_size_display', 'duration_display', 'created_by'] list_filter = ['backup_type', 'status', 'created_at'] search_fields = ['backup_filename', 'created_by__username'] readonly_fields = ['id', 'created_at', 'started_at', 'completed_at', 'backup_size', 'get_duration'] ordering = ['-created_at'] fieldsets = ( ('Job-Details', { 'fields': ('backup_type', 'status', 'created_by') }), ('Zeitpunkte', { 'fields': ('created_at', 'started_at', 'completed_at', 'get_duration') }), ('Ergebnis', { 'fields': ('backup_filename', 'backup_size', 'database_size', 'files_count') }), ('Fehlerbehandlung', { 'fields': ('error_message',), 'classes': ['collapse'] }), ('System', { 'fields': ('id',), 'classes': ['collapse'] }) ) def backup_size_display(self, obj): return obj.get_size_display() backup_size_display.short_description = 'Backup-Größe' def duration_display(self, obj): duration = obj.get_duration() if duration: return f"{duration.total_seconds():.1f}s" return "-" duration_display.short_description = "Dauer" def has_add_permission(self, request): return False # Use the web interface for creating backups # Customize admin site admin.site.site_header = "Stiftungsverwaltung Administration" admin.site.site_title = "Stiftungsverwaltung Admin" admin.site.index_title = "Willkommen zur Stiftungsverwaltung"