Fix: Mailbox Stats über Dovecot mit config/mailpool.php

main v1.0.134
boban 2025-11-04 18:40:39 +01:00
parent afb8d09db3
commit 8e68051fde
15 changed files with 351 additions and 2039 deletions

View File

@ -53,3 +53,18 @@ if (!function_exists('mta_host')) {
return domain_host('mx');
}
}
if (! function_exists('countryFlag')) {
/**
* Gibt das Flag-Emoji eines Landes zurück anhand des ISO-Codes (z. B. "de" 🇩🇪).
*/
function countryFlag(string $code): string
{
$code = strtoupper($code);
// Unicode-Magic: A -> 🇦, B -> 🇧 etc.
return implode('', array_map(
fn($char) => mb_chr(ord($char) + 127397, 'UTF-8'),
str_split($code)
));
}
}

File diff suppressed because it is too large Load Diff

View File

@ -92,7 +92,7 @@ class MailboxCreateModal extends ModalComponent
{
// alle Nicht-System-Domains in Select
$this->domains = Domain::query()
->where('is_system', false)
->where('is_system', false)->where('is_server', false)
->orderBy('domain')->get(['id', 'domain'])->toArray();
// vorselektieren falls mitgegeben, sonst 1. Domain (falls vorhanden)
@ -291,251 +291,3 @@ class MailboxCreateModal extends ModalComponent
return view('livewire.ui.mail.modal.mailbox-create-modal');
}
}
//namespace App\Livewire\Ui\Mail\Modal;
//
//use App\Models\Domain;
//use App\Models\MailUser;
//use Illuminate\Database\QueryException;
//use Illuminate\Support\Facades\Hash;
//use Illuminate\Validation\Rule;
//use Livewire\Attributes\On;
//use LivewireUI\Modal\ModalComponent;
//
//class MailboxCreateModal extends ModalComponent
//{
// // optional vorselektierte Domain
// public ?int $domain_id = null;
//
// // Anzeige
// public string $domain_name = '';
// /** @var array<int,array{id:int,domain:string}> */
// public array $domains = [];
// public string $email_preview = '';
//
// public string $localpart = '';
// public ?string $display_name = null;
// public ?string $password = null;
// public int $quota_mb = 0;
// public ?int $rate_limit_per_hour = null;
// public bool $is_active = true;
// public bool $must_change_pw = true;
//
// // Limits / Status
// public ?int $limit_max_mailboxes = null;
// public ?int $limit_default_quota_mb = null;
// public ?int $limit_max_quota_per_mb = null;
// public ?int $limit_total_quota_mb = null; // 0 = unlimitiert
// public ?int $limit_domain_rate_per_hour = null;
// public bool $allow_rate_limit_override = false;
//
// public int $mailbox_count_used = 0;
// public int $domain_storage_used_mb = 0;
//
// // Hints/Flags
// public string $quota_hint = '';
// public bool $rate_limit_readonly = false;
// public bool $no_mailbox_slots = false;
// public bool $no_storage_left = false;
// public bool $can_create = true;
// public string $block_reason = '';
//
// /* ---------- Validation ---------- */
// protected function rules(): array
// {
// $maxPerMailbox = $this->limit_max_quota_per_mb ?? PHP_INT_MAX;
// $remainingByTotal = (is_null($this->limit_total_quota_mb) || (int)$this->limit_total_quota_mb === 0)
// ? PHP_INT_MAX
// : max(0, (int)$this->limit_total_quota_mb - (int)$this->domain_storage_used_mb);
// $cap = min($maxPerMailbox, $remainingByTotal);
//
// return [
// 'domain_id' => ['required', Rule::exists('domains', 'id')],
// 'localpart' => [
// 'required', 'max:191', 'regex:/^[A-Za-z0-9._%+-]+$/',
// Rule::unique('mail_users', 'localpart')->where(fn($q) => $q->where('domain_id', $this->domain_id)),
// ],
// 'display_name' => ['nullable', 'max:191'],
// 'password' => ['nullable', 'min:8'],
// 'quota_mb' => ['required', 'integer', 'min:0', 'max:' . $cap],
// 'rate_limit_per_hour' => ['nullable', 'integer', 'min:1'],
// 'is_active' => ['boolean'],
// 'must_change_pw' => ['boolean'],
// ];
// }
//
// /* ---------- Lifecycle ---------- */
// public function mount(?int $domainId = null): void
// {
// // alle Nicht-System-Domains in Select
// $this->domains = Domain::query()
// ->where('is_system', false)
// ->orderBy('domain')->get(['id', 'domain'])->toArray();
//
// // vorselektieren falls mitgegeben, sonst 1. Domain (falls vorhanden)
// $this->domain_id = $domainId ?: ($this->domains[0]['id'] ?? null);
//
// // Limits + Anzeige laden
// $this->syncDomainContext();
// }
//
// public function updatedDomainId(): void
// {
// $this->resetErrorBag(); // scoped unique etc.
// $this->syncDomainContext();
// }
//
// public function updatedLocalpart(): void
// {
// $this->localpart = strtolower(trim($this->localpart));
// $this->rebuildEmailPreview();
// }
//
// public function updatedQuotaMb(): void
// {
// $this->recomputeQuotaHints();
// $this->recomputeBlockers();
// }
//
// /* ---------- Helpers ---------- */
// private function syncDomainContext(): void
// {
// if (!$this->domain_id) return;
//
// $d = Domain::query()
// ->withCount('mailUsers')
// ->withSum('mailUsers as used_storage_mb', 'quota_mb')
// ->findOrFail($this->domain_id);
//
// $this->domain_name = $d->domain;
// $this->limit_max_mailboxes = (int)$d->max_mailboxes;
// $this->limit_default_quota_mb = (int)$d->default_quota_mb;
// $this->limit_max_quota_per_mb = $d->max_quota_per_mailbox_mb !== null ? (int)$d->max_quota_per_mailbox_mb : null;
// $this->limit_total_quota_mb = (int)$d->total_quota_mb; // 0 = unlimitiert
// $this->limit_domain_rate_per_hour = $d->rate_limit_per_hour !== null ? (int)$d->rate_limit_per_hour : null;
// $this->allow_rate_limit_override = (bool)$d->rate_limit_override;
//
// $this->mailbox_count_used = (int)$d->mail_users_count;
// $this->domain_storage_used_mb = (int)($d->used_storage_mb ?? 0);
//
// // Defaults
// $this->quota_mb = $this->limit_default_quota_mb ?? 0;
// if (!$this->allow_rate_limit_override) {
// $this->rate_limit_per_hour = $this->limit_domain_rate_per_hour;
// $this->rate_limit_readonly = true;
// } else {
// $this->rate_limit_per_hour = $this->limit_domain_rate_per_hour;
// $this->rate_limit_readonly = false;
// }
//
// $this->rebuildEmailPreview();
// $this->recomputeQuotaHints();
// $this->recomputeBlockers();
// }
//
// private function rebuildEmailPreview(): void
// {
// $this->email_preview = $this->localpart && $this->domain_name
// ? ($this->localpart . '@' . $this->domain_name) : '';
// }
//
// private function recomputeQuotaHints(): void
// {
// $parts = [];
//
// if (!is_null($this->limit_total_quota_mb) && (int)$this->limit_total_quota_mb > 0) {
// $remainingNow = max(0, (int)$this->limit_total_quota_mb - (int)$this->domain_storage_used_mb);
// $remainingAfter = max(0, $remainingNow - max(0, (int)$this->quota_mb));
// $parts[] = "Verbleibend jetzt: {$remainingNow} MiB";
// $parts[] = "nach Speichern: {$remainingAfter} MiB";
// }
// if (!is_null($this->limit_max_quota_per_mb)) $parts[] = "Max {$this->limit_max_quota_per_mb} MiB pro Postfach";
// if (!is_null($this->limit_default_quota_mb)) $parts[] = "Standard: {$this->limit_default_quota_mb} MiB";
//
// $this->quota_hint = implode(' · ', $parts);
// }
//
// private function recomputeBlockers(): void
// {
// // Slots
// $this->no_mailbox_slots = false;
// if (!is_null($this->limit_max_mailboxes)) {
// $free = (int)$this->limit_max_mailboxes - (int)$this->mailbox_count_used;
// if ($free <= 0) $this->no_mailbox_slots = true;
// }
//
// // Speicher
// $this->no_storage_left = false;
// if (!is_null($this->limit_total_quota_mb) && (int)$this->limit_total_quota_mb > 0) {
// $remaining = (int)$this->limit_total_quota_mb - (int)$this->domain_storage_used_mb;
// if ($remaining <= 0) $this->no_storage_left = true;
// }
//
// $reasons = [];
// if ($this->no_mailbox_slots) $reasons[] = 'Keine freien Postfach-Slots in dieser Domain.';
// if ($this->no_storage_left) $reasons[] = 'Kein Domain-Speicher mehr verfügbar.';
// $this->block_reason = implode(' ', $reasons);
// $this->can_create = !($this->no_mailbox_slots || $this->no_storage_left);
// }
//
// /* ---------- Save ---------- */
// #[On('mailbox:create')]
// public function save(): void
// {
// $this->recomputeBlockers();
// if (!$this->can_create) {
// $this->addError('domain_id', $this->block_reason ?: 'Erstellung aktuell nicht möglich.');
// return;
// }
//
// $data = $this->validate();
// $email = $data['localpart'] . '@' . $this->domain_name;
//
// try {
// $u = new MailUser();
// $u->domain_id = $data['domain_id'];
// $u->localpart = $data['localpart'];
// $u->email = $email;
// $u->display_name = $this->display_name ?: null;
// $u->password_hash = $this->password ? Hash::make($this->password) : null;
// $u->is_system = false;
// $u->is_active = (bool)$data['is_active'];
// $u->must_change_pw = (bool)$data['must_change_pw'];
// $u->quota_mb = (int)$data['quota_mb'];
// $u->rate_limit_per_hour = $data['rate_limit_per_hour'];
// $u->save();
// } catch (QueryException $e) {
// $msg = strtolower($e->getMessage());
// if (str_contains($msg, 'mail_users_domain_localpart_unique')) {
// $this->addError('localpart', 'Dieses Postfach existiert in dieser Domain bereits.');
// return;
// }
// if (str_contains($msg, 'mail_users_email_unique')) {
// $this->addError('localpart', 'Diese E-Mail-Adresse ist bereits vergeben.');
// return;
// }
// throw $e;
// }
//
// $this->dispatch('mailbox:created');
// $this->dispatch('closeModal');
// $this->dispatch('toast',
// type: 'done',
// badge: 'Postfach',
// title: 'Postfach angelegt',
// text: 'Das Postfach <b>' . e($email) . '</b> wurde erfolgreich angelegt.',
// duration: 6000
// );
//
// }
//
// public static function modalMaxWidth(): string
// {
// return '3xl';
// }
//
// public function render()
// {
// return view('livewire.ui.mail.modal.mailbox-create-modal');
// }
//}

View File

@ -0,0 +1,49 @@
<?php
namespace App\Livewire\Ui\System\Form;
use App\Models\Setting;
use Livewire\Component;
class DomainsSslForm extends Component
{
// fix / readonly aus ENV oder config
public string $mail_domain_readonly = '';
// editierbar
public string $ui_domain = '';
public string $webmail_domain = '';
protected function rules(): array
{
return [
'ui_domain' => 'nullable|string|max:190',
'webmail_domain' => 'nullable|string|max:190',
];
}
public function mount(): void
{
$this->mail_domain_readonly = (string) config('mailwolt.domain.mail', 'mx');
$this->ui_domain = Setting::get('ui_domain', $this->ui_domain);
$this->webmail_domain = Setting::get('webmail_domain', $this->webmail_domain);
}
public function save(): void
{
$this->validate();
Setting::put('ui_domain', $this->ui_domain);
Setting::put('webmail_domain', $this->webmail_domain);
$this->dispatch('toast',
type: 'done',
badge: 'System',
title: 'Domains gespeichert',
text: 'UI- und Webmail-Domain wurden übernommen.',
duration: 5000,
);
}
public function render() { return view('livewire.ui.system.form.domains-ssl-form'); }
}

View File

@ -0,0 +1,79 @@
<?php
namespace App\Livewire\Ui\System\Form;
use App\Models\Setting;
use Livewire\Component;
class GeneralForm extends Component
{
public string $locale = 'de';
public string $timezone = 'Europe/Berlin';
protected function rules(): array
{
return [
'locale' => 'required|string|max:10',
'timezone' => 'required|string|max:64',
];
}
public function mount(): void
{
// Defaults aus ENV nur für den allerersten Seed in Settings (Redis/DB)
$envLocale = env('APP_LOCALE') ?? env('APP_FALLBACK_LOCALE') ?? $this->locale;
$envTimezone = env('APP_TIMEZONE') ?? $this->timezone;
// Wenn (noch) nichts in Settings liegt, einmalig mit ENV-Werten befüllen
if (Setting::get('locale', null) === null) {
Setting::set('locale', $envLocale);
}
if (Setting::get('timezone', null) === null) {
Setting::set('timezone', $envTimezone);
}
// Ab hier ausschließlich aus Settings lesen (Redis → DB Fallback)
$this->locale = (string) Setting::get('locale', $envLocale);
$this->timezone = (string) Setting::get('timezone', $envTimezone);
// Sofort für die aktuelle Request anwenden
app()->setLocale($this->locale);
@date_default_timezone_set($this->timezone);
config([
'app.locale' => $this->locale,
'app.fallback_locale' => $this->locale,
'app.timezone' => $this->timezone,
]);
}
public function save(): void
{
$this->validate();
// Persistieren: DB → Redis (siehe Setting::set)
Setting::set('locale', $this->locale);
Setting::set('timezone', $this->timezone);
// Direkt in der laufenden Request aktivieren
app()->setLocale($this->locale);
@date_default_timezone_set($this->timezone);
config([
'app.locale' => $this->locale,
'app.fallback_locale' => $this->locale, // optional
'app.timezone' => $this->timezone,
]);
$this->dispatch('toast',
type: 'done',
badge: 'System',
title: 'Allgemein gespeichert',
text: 'Sprache und Zeitzone wurden übernommen.',
duration: 5000,
);
}
public function render()
{
return view('livewire.ui.system.form.general-form');
}
}

View File

@ -0,0 +1,48 @@
<?php
namespace App\Livewire\Ui\System\Form;
use App\Models\Setting;
use Livewire\Component;
class SecurityForm extends Component
{
public bool $twofa_enabled = false;
public ?int $rate_limit = 5;
public ?int $password_min = 10;
protected function rules(): array
{
return [
'twofa_enabled' => 'boolean',
'rate_limit' => 'nullable|integer|min:1|max:100',
'password_min' => 'nullable|integer|min:6|max:128',
];
}
public function mount(): void
{
$this->twofa_enabled = (bool) Setting::get('twofa_enabled', $this->twofa_enabled);
$this->rate_limit = (int) Setting::get('rate_limit', $this->rate_limit);
$this->password_min = (int) Setting::get('password_min', $this->password_min);
}
public function save(): void
{
$this->validate();
Setting::put('twofa_enabled', $this->twofa_enabled);
Setting::put('rate_limit', $this->rate_limit);
Setting::put('password_min', $this->password_min);
$this->dispatch('toast',
type: 'done',
badge: 'Sicherheit',
title: 'Sicherheit gespeichert',
text: '2FA/Rate-Limits/Passwortregeln wurden übernommen.',
duration: 5000,
);
}
public function render() { return view('livewire.ui.system.form.security-form'); }
}

View File

@ -1,6 +1,30 @@
<?php
return [
'domain' => [
'base' => env('BASE_DOMAIN'),
'mail' => env('MTA_SUB'),
'ui' => env('UI_SUB'),
'webmail' => env('WEBMAIL_SUB'),
],
'language' => [
'de' => [
'label' => 'Deutsch',
'locale' => 'de',
'fallback_locale' => 'de',
'flag' => 'de',
],
'en' => [
'label' => 'English',
'locale' => 'en',
'fallback_locale' => 'en',
'flag' => 'gb',
],
],
'units' => [
['name' => 'nginx', 'action' => 'reload'],
['name' => 'postfix', 'action' => 'try-reload-or-restart'],

View File

@ -10,8 +10,8 @@ return [
'label' => 'Mail', 'icon' => 'ph-envelope', 'items' => [
['label' => 'Postfächer', 'route' => 'ui.mail.mailboxes.index'],
['label' => 'Aliasse', 'route' => 'ui.mail.aliases.index'],
['label' => 'Gruppen', 'route' => 'ui.mail.groups.index'],
['label' => 'Filter', 'route' => 'ui.mail.filters.index'],
// ['label' => 'Gruppen', 'route' => 'ui.mail.groups.index'],
// ['label' => 'Filter', 'route' => 'ui.mail.filters.index'],
['label' => 'Quarantäne', 'route' => 'ui.mail.quarantine.index'],
['label' => 'Queues', 'route' => 'ui.mail.queues.index'],
],
@ -19,45 +19,59 @@ return [
[
'label' => 'Domains', 'icon' => 'ph-globe', 'items' => [
['label' => 'Übersicht', 'route' => 'ui.domains.index'],
['label' => 'DNS-Assistent', 'route' => 'ui.domains.dns'],
['label' => 'Zertifikate', 'route' => 'ui.domains.certificates'],
// ['label' => 'DNS-Assistent', 'route' => 'ui.domains.dns'],
// ['label' => 'Zertifikate', 'route' => 'ui.domains.certificates'],
],
],
[
'label' => 'Webmail', 'icon' => 'ph-browser', 'items' => [
['label' => 'Allgemein', 'route' => 'ui.webmail.settings'],
['label' => 'Plugins', 'route' => 'ui.webmail.plugins'],
['label' => 'Allgemein', 'route' => 'ui.logout'],
['label' => 'Plugins', 'route' => 'ui.logout'],
// ['label' => 'Allgemein', 'route' => 'ui.webmail.settings'],
// ['label' => 'Plugins', 'route' => 'ui.webmail.plugins'],
],
],
[
'label' => 'Benutzer', 'icon' => 'ph-users', 'items' => [
['label' => 'Benutzer', 'route' => 'ui.users.index'],
['label' => 'Rollen & Rechte', 'route' => 'ui.users.roles'],
['label' => 'Anmeldesicherheit', 'route' => 'ui.users.security'],
['label' => 'Benutzer', 'route' => 'ui.logout'],
['label' => 'Rollen & Rechte', 'route' => 'ui.logout'],
['label' => 'Anmeldesicherheit', 'route' => 'ui.logout'],
// ['label' => 'Benutzer', 'route' => 'ui.users.index'],
// ['label' => 'Rollen & Rechte', 'route' => 'ui.users.roles'],
// ['label' => 'Anmeldesicherheit', 'route' => 'ui.users.security'],
],
],
[
'label' => 'Sicherheit', 'icon' => 'ph-shield', 'items' => [
['label' => 'TLS & Ciphers', 'route' => 'ui.security.tls'],
['label' => 'Ratelimits', 'route' => 'ui.security.abuse'],
['label' => 'Fail2Ban', 'route' => 'ui.security.fail2ban'],
['label' => 'Audit-Logs', 'route' => 'ui.security.audit'],
['label' => 'Rspamd', 'route' => 'ui.security.rspamd'],
['label' => 'SSL', 'route' => 'ui.security.ssl'],
// ['label' => 'Ratelimits', 'route' => 'ui.security.audit'],
// ['label' => 'TLS & Ciphers', 'route' => 'ui.security.tls'],
// ['label' => 'Ratelimits', 'route' => 'ui.security.abuse'],
// ['label' => 'Audit-Logs', 'route' => 'ui.security.audit'],
],
],
[
'label' => 'System', 'icon' => 'ph-gear-six', 'items' => [
['label' => 'Einstellungen', 'route' => 'ui.system.settings'],
['label' => 'Dienste & Status', 'route' => 'ui.system.services'],
['label' => 'Jobs & Queues', 'route' => 'ui.system.jobs'],
['label' => 'Logs', 'route' => 'ui.system.logs'],
['label' => 'Speicher', 'route' => 'ui.system.storage'],
['label' => 'Über', 'route' => 'ui.system.about'],
// ['label' => 'Dienste & Status', 'route' => 'ui.system.services'],
// ['label' => 'Jobs & Queues', 'route' => 'ui.system.jobs'],
// ['label' => 'Logs', 'route' => 'ui.system.logs'],
// ['label' => 'Speicher', 'route' => 'ui.system.storage'],
// ['label' => 'Über', 'route' => 'ui.system.about'],
],
],
[
'label' => 'Entwickler', 'icon' => 'ph-brackets-curly', 'items' => [
['label' => 'API-Schlüssel', 'route' => 'ui.dev.tokens'],
['label' => 'Webhooks', 'route' => 'ui.dev.webhooks'],
['label' => 'Sandbox', 'route' => 'ui.dev.sandbox'],
['label' => 'API-Schlüssel', 'route' => 'ui.logout'],
['label' => 'Webhooks', 'route' => 'ui.logout'],
['label' => 'Sandbox', 'route' => 'ui.logout'],
// ['label' => 'API-Schlüssel', 'route' => 'ui.dev.tokens'],
// ['label' => 'Webhooks', 'route' => 'ui.dev.webhooks'],
// ['label' => 'Sandbox', 'route' => 'ui.dev.sandbox'],
],
],
];

View File

@ -62,8 +62,8 @@
$itemIcon = $item['icon'] ?? 'ph-dot-outline';
@endphp
<li>
{{-- <a href="{{ isset($item['route']) ? route($item['route']) : '#' }}"--}}
<a href="#"
<a href="{{ isset($item['route']) ? route($item['route']) : '#' }}"
{{-- <a href="#"--}}
class="sidebar-link group relative flex items-center gap-2 px-2 py-2 rounded-lg
border border-transparent transition-colors
hover:bg-gradient-to-t hover:from-white/10 hover:to-transparent

View File

@ -32,7 +32,6 @@
{{-- Row 1: Domain + Typ --}}
<div class="grid grid-cols-1 md:grid-cols-2 gap-3">
{{-- DOMAIN (TailwindPlus Elements) --}}
<div class="relative" wire:ignore id="domain-select-{{ $this->getId() }}">
<label class="block text-xs text-white/60 mb-1">Domain</label>

View File

@ -0,0 +1,33 @@
<div class="space-y-4">
<div>
<label class="block text-white/60 text-sm mb-1">Mailserver-Domain (fix)</label>
<input type="text" value="{{ $mail_domain_readonly }}" disabled
class="w-full h-11 rounded-xl border border-white/10 bg-white/[0.06] px-3 text-white/60 cursor-not-allowed">
<p class="mt-1 text-xs text-white/45">Wird aus ENV/Config gelesen und ist nicht änderbar.</p>
</div>
<div>
<label class="block text-white/60 text-sm mb-1">UI-Domain</label>
<input type="text" wire:model.defer="ui_domain" placeholder="z. B. ui.deinedomain.tld"
class="w-full h-11 rounded-xl border border-white/10 bg-white/[0.04] px-3 text-white/90">
@error('ui_domain') <p class="text-xs text-rose-400 mt-1">{{ $message }}</p> @enderror
</div>
<div>
<label class="block text-white/60 text-sm mb-1">Webmail-Domain</label>
<input type="text" wire:model.defer="webmail_domain" placeholder="z. B. mail.deinedomain.tld"
class="w-full h-11 rounded-xl border border-white/10 bg-white/[0.04] px-3 text-white/90">
@error('webmail_domain') <p class="text-xs text-rose-400 mt-1">{{ $message }}</p> @enderror
</div>
<div class="flex justify-end">
<button wire:click="save"
class="inline-flex items-center gap-2 rounded-xl border border-white/10 bg-white/5 px-3 py-1.5 text-white/80 hover:text-white hover:border-white/20">
Speichern
</button>
</div>
<div class="mt-3 text-xs text-white/45">
TLS/Redirect ist systemweit immer erzwungen (HTTPS). ACME/Zertifikate haben ihren eigenen Reiter.
</div>
</div>

View File

@ -0,0 +1,35 @@
<div class="grid grid-cols-1 md:grid-cols-2 gap-4">
{{-- Sprache --}}
<div>
<label class="block text-white/60 text-sm mb-1">Sprache</label>
<select wire:model.defer="locale"
class="w-full h-11 rounded-xl border border-white/10 bg-white/[0.04] px-3 text-white/90">
@foreach (config('mailwolt.language') as $key => $lang)
<option value="{{ $lang['locale'] }}">
{{ countryFlag($lang['flag']) }} {{ $lang['label'] }}
</option>
@endforeach
</select>
@error('locale') <p class="text-xs text-rose-400 mt-1">{{ $message }}</p> @enderror
</div>
{{-- Zeitzone --}}
<div>
<label class="block text-white/60 text-sm mb-1">Zeitzone</label>
<select wire:model.defer="timezone"
class="w-full h-11 rounded-xl border border-white/10 bg-white/[0.04] px-3 text-white/90">
@foreach (DateTimeZone::listIdentifiers() as $tz)
<option value="{{ $tz }}">{{ $tz }}</option>
@endforeach
</select>
@error('timezone') <p class="text-xs text-rose-400 mt-1">{{ $message }}</p> @enderror
</div>
{{-- Actions: immer unten rechts, volle Breite, rechts ausgerichtet --}}
<div class="md:col-span-2 flex justify-end">
<button wire:click="save"
class="inline-flex items-center gap-2 rounded-xl border border-white/10 bg-white/5 px-3 py-1.5 text-white/80 hover:text-white hover:border-white/20">
Speichern
</button>
</div>
</div>

View File

@ -0,0 +1,27 @@
<div class="space-y-4">
<label class="flex items-center gap-3">
<input type="checkbox" wire:model.defer="twofa_enabled" class="h-4 w-4">
<span class="text-white/80">Zwei-Faktor-Authentifizierung aktivieren</span>
</label>
<div>
<label class="block text-white/60 text-sm mb-1">Login-Rate-Limit (Versuche/Minute)</label>
<input type="number" min="1" max="100" wire:model.defer="rate_limit"
class="w-full h-11 rounded-xl border border-white/10 bg-white/[0.04] px-3 text-white/90">
@error('rate_limit') <p class="text-xs text-rose-400 mt-1">{{ $message }}</p> @enderror
</div>
<div>
<label class="block text-white/60 text-sm mb-1">Minimale Passwortlänge</label>
<input type="number" min="6" max="128" wire:model.defer="password_min"
class="w-full h-11 rounded-xl border border-white/10 bg-white/[0.04] px-3 text-white/90">
@error('password_min') <p class="text-xs text-rose-400 mt-1">{{ $message }}</p> @enderror
</div>
<div class="flex justify-end">
<button wire:click="save"
class="inline-flex items-center gap-2 rounded-xl border border-white/10 bg-white/5 px-3 py-1.5 text-white/80 hover:text-white hover:border-white/20">
Speichern
</button>
</div>
</div>

View File

@ -1,14 +1,3 @@
{{-- resources/views/ui/system/settings.blade.php --}}
{{--@extends('layouts.app')--}}
{{--@section('title', 'System · Einstellungen')--}}
{{--@section('header_title', 'System · Einstellungen')--}}
{{--@section('content')--}}
{{-- <div class="glass-card p-5">--}}
{{-- <livewire:ui.system.settings-form />--}}
{{-- </div>--}}
{{--@endsection--}}
{{-- resources/views/ui/system/settings/index.blade.php --}}
@extends('layouts.app')
@ -55,7 +44,7 @@
</div>
{{-- Livewire-Form (Allgemein) --}}
<livewire:ui.system.general-form />
<livewire:ui.system.form.general-form />
</div>
</section>

View File

@ -37,14 +37,16 @@ Route::middleware('auth.user')->name('ui.')->group(function () {
});
#DOMAIN ROUTES
Route::name('domain.')->group(function () {
Route::name('domains.')->group(function () {
Route::get('/domains', [DomainDnsController::class, 'index'])->name('index');
});
#MAIL ROUTES
Route::name('mail.')->group(function () {
Route::get('/mailboxes', [MailboxController::class, 'index'])->name('mailbox.index');
Route::get('/mailboxes', [MailboxController::class, 'index'])->name('mailboxes.index');
Route::get('/aliases', [AliasController::class, 'index'])->name('aliases.index');
Route::get('/quarantine', function () {return 'Quarantäne';})->name('quarantine.index');
Route::get('/queues', function () {return 'Queues';})->name('queues.index');
});
#LOGOUT ROUTE