Add radio embed widget, SSE real-time, song history, moderation panel, and Auto DJ

- Embed widget: standalone iframe player with dark/light/transparent themes, copy-paste embed code admin page
- Real-time SSE: streaming now-playing/listeners/dj events, replaces polling in radio-player and embed
- Song history: auto-records song changes to radio_song_plays table, Filament resource to view
- DJ moderation: unified panel for shouts approval, song request queue, DJ applications
- Auto DJ: playlist management with round-robin playback when no DJ is live
- Refactored radio-player Alpine component to use EventSource API with auto-reconnect
This commit is contained in:
root
2026-05-24 14:07:32 +02:00
parent 5476dce882
commit 0c6c558a59
32 changed files with 2236 additions and 29 deletions
@@ -0,0 +1,66 @@
<?php
declare(strict_types=1);
namespace App\Http\Controllers\Radio;
use App\Enums\RadioSettings;
use App\Http\Controllers\Controller;
use App\Models\Miscellaneous\WebsiteSetting;
use App\Services\Community\RadioStreamService;
use Illuminate\Http\JsonResponse;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\Cache;
use Illuminate\View\View;
class EmbedController extends Controller
{
public function __construct(
private readonly RadioStreamService $streamService,
) {}
public function show(Request $request): View
{
$theme = $request->query('theme', $this->getSetting(RadioSettings::EmbedTheme, 'dark'));
$settings = [
'streamUrl' => $this->streamService->formatStreamUrl($this->getSetting(RadioSettings::StreamUrl, '')),
'theme' => in_array($theme, ['dark', 'light', 'transparent']) ? $theme : 'dark',
'autoPlay' => $request->query('autoplay', $this->getSetting(RadioSettings::EmbedAutoPlay, '0')) === '1',
'primaryColor' => $this->getSetting('radio_player_color_primary', '#eeb425'),
'secondaryColor' => $this->getSetting('radio_player_color_secondary', '#1a1a2e'),
'textColor' => $this->getSetting('radio_player_color_text', '#ffffff'),
'accentColor' => $this->getSetting('radio_player_color_accent', '#eeb425'),
];
return view('radio.embed', compact('settings'));
}
public function config(): JsonResponse
{
return response()->json([
'enabled' => (bool) $this->getSetting(RadioSettings::EmbedEnabled, '0'),
'stream_url' => $this->streamService->formatStreamUrl($this->getSetting(RadioSettings::StreamUrl, '')),
'theme' => $this->getSetting(RadioSettings::EmbedTheme, 'dark'),
'auto_play' => (bool) $this->getSetting(RadioSettings::EmbedAutoPlay, '0'),
'primary_color' => $this->getSetting('radio_player_color_primary', '#eeb425'),
'secondary_color' => $this->getSetting('radio_player_color_secondary', '#1a1a2e'),
'text_color' => $this->getSetting('radio_player_color_text', '#ffffff'),
'accent_color' => $this->getSetting('radio_player_color_accent', '#eeb425'),
'height' => (int) $this->getSetting(RadioSettings::EmbedHeight, '400'),
'width' => $this->getSetting(RadioSettings::EmbedWidth, '100%'),
'allowed_domains' => $this->getSetting(RadioSettings::EmbedAllowedDomains, ''),
]);
}
private function getSetting(RadioSettings|string $key, mixed $default = null): mixed
{
$keyStr = $key instanceof RadioSettings ? $key->value : $key;
return Cache::remember("setting_{$keyStr}", 60, function () use ($keyStr, $default): mixed {
$websiteSetting = WebsiteSetting::where('key', $keyStr)->first();
return $websiteSetting?->value ?? $default;
});
}
}