byos_laravel/resources/views/livewire/devices/configure.blade.php
2025-05-05 22:20:31 +02:00

544 lines
26 KiB
PHP

<?php
use App\Models\Playlist;
use App\Models\PlaylistItem;
use Livewire\Volt\Component;
new class extends Component {
public $device;
public $name;
public $api_key;
public $friendly_id;
public $mac_address;
public $default_refresh_interval;
public $width;
public $height;
// Playlist properties
public $playlists;
public $playlist_name;
public $selected_weekdays = null;
public $active_from;
public $active_until;
public $refresh_time = null;
public function mount(\App\Models\Device $device)
{
abort_unless(auth()->user()->devices->contains($device), 403);
$current_image_uuid = $device->current_screen_image;
$current_image_path = 'images/generated/' . $current_image_uuid . '.png';
$this->device = $device;
$this->name = $device->name;
$this->api_key = $device->api_key;
$this->friendly_id = $device->friendly_id;
$this->mac_address = $device->mac_address;
$this->default_refresh_interval = $device->default_refresh_interval;
$this->width = $device->width;
$this->height = $device->height;
$this->playlists = $device->playlists()->with('items.plugin')->orderBy('created_at')->get();
return view('livewire.devices.configure', [
'image' => ($current_image_uuid) ? url($current_image_path) : null,
]);
}
public function deleteDevice(\App\Models\Device $device)
{
abort_unless(auth()->user()->devices->contains($device), 403);
$device->delete();
redirect()->route('devices');
}
public function updateDevice()
{
abort_unless(auth()->user()->devices->contains($this->device), 403);
$this->validate([
'name' => 'required|string|max:255',
'friendly_id' => 'required|string|max:255',
'mac_address' => 'required|string|max:255',
'default_refresh_interval' => 'required|integer|min:1',
'width' => 'required|integer|min:1',
'height' => 'required|integer|min:1',
]);
$this->device->update([
'name' => $this->name,
'friendly_id' => $this->friendly_id,
'mac_address' => $this->mac_address,
'default_refresh_interval' => $this->default_refresh_interval,
'width' => $this->width,
'height' => $this->height,
]);
Flux::modal('edit-device')->close();
}
public function createPlaylist()
{
$this->validate([
'playlist_name' => 'required|string|max:255',
'selected_weekdays' => 'nullable|array',
'active_from' => 'nullable|date_format:H:i',
'active_until' => 'nullable|date_format:H:i',
'refresh_time' => 'nullable|integer|min:60',
]);
if ($this->refresh_time < 60) {
$this->refresh_time = null;
}
if (empty($this->selected_weekdays)){
$this->selected_weekdays = null;
}
$this->device->playlists()->create([
'name' => $this->playlist_name,
'weekdays' => $this->selected_weekdays,
'active_from' => $this->active_from,
'active_until' => $this->active_until,
'refresh_time' => $this->refresh_time,
'is_active' => true,
]);
$this->playlists = $this->device->playlists()->with('items.plugin')->orderBy('created_at')->get();
$this->reset(['playlist_name', 'selected_weekdays', 'active_from', 'active_until']);
Flux::modal('create-playlist')->close();
}
public function togglePlaylistActive(Playlist $playlist)
{
$playlist->update(['is_active' => !$playlist->is_active]);
$this->playlists = $this->device->playlists()->with('items.plugin')->orderBy('created_at')->get();
}
public function movePlaylistItemUp(PlaylistItem $item)
{
$previousItem = $item->playlist->items()
->where('order', '<', $item->order)
->orderBy('order', 'desc')
->first();
if ($previousItem) {
$tempOrder = $previousItem->order;
$previousItem->update(['order' => $item->order]);
$item->update(['order' => $tempOrder]);
$this->playlists = $this->device->playlists()->with('items.plugin')->orderBy('created_at')->get();
}
}
public function movePlaylistItemDown(PlaylistItem $item)
{
$nextItem = $item->playlist->items()
->where('order', '>', $item->order)
->orderBy('order')
->first();
if ($nextItem) {
$tempOrder = $nextItem->order;
$nextItem->update(['order' => $item->order]);
$item->update(['order' => $tempOrder]);
$this->playlists = $this->device->playlists()->with('items.plugin')->orderBy('created_at')->get();
}
}
public function togglePlaylistItemActive(PlaylistItem $item)
{
$item->update(['is_active' => !$item->is_active]);
$this->playlists = $this->device->playlists()->with('items.plugin')->orderBy('created_at')->get();
}
public function deletePlaylist(Playlist $playlist)
{
abort_unless(auth()->user()->devices->contains($playlist->device), 403);
$playlist->delete();
$this->playlists = $this->device->playlists()->with('items.plugin')->orderBy('created_at')->get();
Flux::modal('delete-playlist-' . $playlist->id)->close();
}
public function deletePlaylistItem(PlaylistItem $item)
{
abort_unless(auth()->user()->devices->contains($item->playlist->device), 403);
$item->delete();
$this->playlists = $this->device->playlists()->with('items.plugin')->orderBy('created_at')->get();
Flux::modal('delete-playlist-item-' . $item->id)->close();
}
public function editPlaylist(Playlist $playlist)
{
$this->validate([
'playlist_name' => 'required|string|max:255',
'selected_weekdays' => 'nullable|array',
'active_from' => 'nullable|date_format:H:i',
'active_until' => 'nullable|date_format:H:i',
'refresh_time' => 'nullable|integer|min:60',
]);
if ($this->refresh_time < 60) {
$this->refresh_time = null;
}
if (empty($this->selected_weekdays)){
$this->selected_weekdays = null;
}
$playlist->update([
'name' => $this->playlist_name,
'weekdays' => $this->selected_weekdays,
'active_from' => $this->active_from,
'active_until' => $this->active_until,
'refresh_time' => $this->refresh_time,
]);
$this->playlists = $this->device->playlists()->with('items.plugin')->orderBy('created_at')->get();
$this->reset(['playlist_name', 'selected_weekdays', 'active_from', 'active_until', 'refresh_time']);
Flux::modal('edit-playlist-' . $playlist->id)->close();
}
public function preparePlaylistEdit(Playlist $playlist)
{
$this->playlist_name = $playlist->name;
$this->selected_weekdays = $playlist->weekdays ?? null;
$this->active_from = optional($playlist->active_from)->format('H:i');
$this->active_until = optional($playlist->active_until)->format('H:i');
$this->refresh_time = $playlist->refresh_time;
}
}
?>
<div class="bg-muted flex flex-col items-center justify-center gap-6 p-6 md:p-10">
<div class="flex flex-col gap-6">
<div
class="rounded-xl border bg-white dark:bg-stone-950 dark:border-stone-800 text-stone-800 shadow-xs">
<div class="px-10 py-8 min-w-lg">
@php
$current_image_uuid =$device->current_screen_image;
if($current_image_uuid) {
$file_extension = file_exists(storage_path('app/public/images/generated/' . $current_image_uuid . '.png')) ? 'png' : 'bmp';
$current_image_path = 'storage/images/generated/' . $current_image_uuid . '.' . $file_extension;
} else {
$current_image_path = 'storage/images/setup-logo.bmp';
}
@endphp
<div class="flex items-center justify-between">
<flux:tooltip content="Friendly ID: {{$device->friendly_id}}" position="bottom">
<h1 class="text-xl font-medium dark:text-zinc-200">{{ $device->name }}</h1>
</flux:tooltip>
<div class="flex gap-2">
<flux:tooltip content="Last update" position="bottom">
<span class="dark:text-zinc-200">{{$device->updated_at->diffForHumans()}}</span>
</flux:tooltip>
<flux:separator vertical/>
<flux:tooltip content="MAC Address" position="bottom">
<span class="dark:text-zinc-200">{{$device->mac_address}}</span>
</flux:tooltip>
@if($device->last_firmware_version)
<flux:separator vertical/>
<flux:tooltip content="Firmware Version" position="bottom">
<span class="dark:text-zinc-200">{{$device->last_firmware_version}}</span>
</flux:tooltip>
@endif
@if($device->wifiStrength)
<flux:separator vertical/>
<x-responsive-icons.wifi :strength="$device->wifiStrength" :rssi="$device->last_rssi_level"
class="dark:text-zinc-200"/>
@endif
@if($device->batteryPercent)
<flux:separator vertical/>
<x-responsive-icons.battery :percent="$device->batteryPercent"/>
@endif
</div>
<div>
<flux:modal.trigger name="edit-device">
<flux:button icon="pencil-square" />
</flux:modal.trigger>
<flux:modal.trigger name="delete-device">
<flux:button icon="trash" variant="subtle"/>
</flux:modal.trigger>
</div>
</div>
<flux:modal name="edit-device" class="md:w-96">
<div class="space-y-6">
<div>
<flux:heading size="lg">Edit TRMNL</flux:heading>
<flux:subheading></flux:subheading>
</div>
<!-- @dump($device) -->
<flux:input label="Name" wire:model="name"/>
<flux:input label="API Key" icon="key" value="{{ $device->api_key }}" type="password"
viewable class="max-w-xs" readonly/>
<flux:input label="Friendly ID" wire:model="friendly_id"/>
<flux:input label="MAC Address" wire:model="mac_address"/>
<div class="flex gap-4">
<flux:input label="Width (px)" wire:model="width" type="number" />
<flux:input label="Height (px)" wire:model="height" type="number"/>
</div>
<flux:input label="Default Refresh Interval (seconds)" wire:model="default_refresh_interval"
type="number"/>
<div class="flex">
<flux:spacer/>
<flux:button type="submit" wire:click="updateDevice" variant="primary">Save changes
</flux:button>
</div>
</div>
</flux:modal>
<flux:modal name="delete-device" class="min-w-[22rem] space-y-6">
<div>
<flux:heading size="lg">Delete {{$device->name}}?</flux:heading>
</div>
<div class="flex gap-2">
<flux:spacer/>
<flux:modal.close>
<flux:button variant="ghost">Cancel</flux:button>
</flux:modal.close>
<flux:button wire:click="deleteDevice({{ $device->id }})" variant="danger">Delete device
</flux:button>
</div>
</flux:modal>
@if(!$device->mirror_device_id)
@if($current_image_path)
<flux:separator class="mt-6 mb-6" text="Next Screen"/>
<img src="{{ asset($current_image_path) }}" alt="Next Image"/>
@endif
<flux:separator class="mt-6 mb-6" text="Playlists"/>
<div class="flex justify-between items-center mb-4">
<h3 class="text-lg font-medium dark:text-zinc-200">Device Playlists</h3>
<flux:modal.trigger name="create-playlist">
<flux:button icon="plus" variant="primary">Create Playlist</flux:button>
</flux:modal.trigger>
</div>
@else
<div class="mt-6 mb-6">
<flux:callout variant="info">
<div class="flex items-center gap-2">
<flux:icon name="link" class="h-5 w-5"/>
<div>
This device is mirrored from
<a href="{{ route('devices.configure', $device->mirrorDevice) }}" class="font-medium hover:underline">
{{ $device->mirrorDevice->name }}
</a>
</div>
</div>
</flux:callout>
</div>
@endif
<flux:modal name="create-playlist" class="md:w-96">
<div class="space-y-6">
<div>
<flux:heading size="lg">Create Playlist</flux:heading>
</div>
<form wire:submit="createPlaylist">
<div class="mb-4">
<flux:input label="Playlist Name" wire:model="playlist_name" required/>
</div>
<div class="mb-4">
<flux:checkbox.group wire:model="selected_weekdays" label="Active Days (optional)">
<flux:checkbox label="Monday" value="1"/>
<flux:checkbox label="Tuesday" value="2"/>
<flux:checkbox label="Wednesday" value="3"/>
<flux:checkbox label="Thursday" value="4"/>
<flux:checkbox label="Friday" value="5"/>
<flux:checkbox label="Saturday" value="6"/>
<flux:checkbox label="Sunday" value="0"/>
</flux:checkbox.group>
</div>
<div class="mb-4">
<flux:input type="time" label="Active From (optional)" wire:model="active_from"/>
</div>
<div class="mb-4">
<flux:input type="time" label="Active Until (optional)" wire:model="active_until"/>
</div>
<div class="mb-4">
<flux:input type="number" label="Refresh Time (seconds)" wire:model="refresh_time" min="1" placeholder="Leave empty to use device default"/>
</div>
<div class="flex">
<flux:spacer/>
<flux:button type="submit" variant="primary">Create Playlist</flux:button>
</div>
</form>
</div>
</flux:modal>
@foreach($playlists as $playlist)
<div class="mb-6 rounded-lg border dark:border-zinc-700 p-4">
<div class="flex items-center justify-between mb-4">
<div class="flex items-center gap-4">
<h4 class="text-lg font-medium dark:text-zinc-200">{{ $playlist->name }}</h4>
<flux:switch wire:model.live="playlist.is_active"
wire:click="togglePlaylistActive({{ $playlist->id }})"
:checked="$playlist->is_active"/>
</div>
<div class="flex items-center gap-4">
<div class="flex items-center gap-2 text-sm text-zinc-500 dark:text-zinc-400">
@if($playlist->weekdays)
<span>{{ implode(', ', collect($playlist->weekdays)->map(fn($day) => ['Sun', 'Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat'][$day])->toArray()) }}</span>
@endif
@if($playlist->active_from && $playlist->active_until)
<flux:separator vertical/>
<span>{{ $playlist->active_from->format('H:i') }} - {{ $playlist->active_until->format('H:i') }}</span>
@endif
</div>
<div class="flex gap-2">
<flux:modal.trigger name="edit-playlist-{{ $playlist->id }}">
<flux:button icon="pencil-square" variant="subtle" size="sm" wire:click="preparePlaylistEdit({{ $playlist->id }})"/>
</flux:modal.trigger>
<flux:modal.trigger name="delete-playlist-{{ $playlist->id }}">
<flux:button icon="trash" size="sm"/>
</flux:modal.trigger>
</div>
</div>
</div>
<flux:modal name="edit-playlist-{{ $playlist->id }}" class="md:w-96">
<div class="space-y-6">
<div>
<flux:heading size="lg">Edit Playlist</flux:heading>
</div>
<form wire:submit="editPlaylist({{ $playlist->id }})">
<div class="mb-4">
<flux:input label="Playlist Name" wire:model="playlist_name" required/>
</div>
<div class="mb-4">
<flux:checkbox.group wire:model="selected_weekdays" label="Active Days (optional)">
<flux:checkbox label="Monday" value="1"/>
<flux:checkbox label="Tuesday" value="2"/>
<flux:checkbox label="Wednesday" value="3"/>
<flux:checkbox label="Thursday" value="4"/>
<flux:checkbox label="Friday" value="5"/>
<flux:checkbox label="Saturday" value="6"/>
<flux:checkbox label="Sunday" value="0"/>
</flux:checkbox.group>
</div>
<div class="mb-4">
<flux:input type="time" label="Active From (optional)" wire:model="active_from"/>
</div>
<div class="mb-4">
<flux:input type="time" label="Active Until (optional)" wire:model="active_until"/>
</div>
<div class="mb-4">
<flux:input type="number" label="Refresh Time (seconds)" wire:model="refresh_time" min="1" placeholder="Leave empty to use device default"/>
</div>
<div class="flex">
<flux:spacer/>
<flux:button type="submit" variant="primary">Save Changes</flux:button>
</div>
</form>
</div>
</flux:modal>
<flux:modal name="delete-playlist-{{ $playlist->id }}" class="min-w-[22rem] space-y-6">
<div>
<flux:heading size="lg">Delete {{ $playlist->name }}?</flux:heading>
<p class="mt-2 text-sm text-zinc-600 dark:text-zinc-400">This will permanently delete this playlist and all its items.</p>
</div>
<div class="flex gap-2">
<flux:spacer/>
<flux:modal.close>
<flux:button variant="ghost">Cancel</flux:button>
</flux:modal.close>
<flux:button wire:click="deletePlaylist({{ $playlist->id }})" variant="danger">Delete playlist</flux:button>
</div>
</flux:modal>
<table class="w-full" data-flux-table>
<thead data-flux-columns>
<tr>
<th class="py-3 px-3 first:pl-0 last:pr-0 text-left text-sm font-medium text-zinc-800 dark:text-white"
data-flux-column>
<div class="whitespace-nowrap flex">Plugin</div>
</th>
<th class="py-3 px-3 first:pl-0 last:pr-0 text-left text-sm font-medium text-zinc-800 dark:text-white"
data-flux-column>
<div class="whitespace-nowrap flex">Status</div>
</th>
<th class="py-3 px-3 first:pl-0 last:pr-0 text-right text-sm font-medium text-zinc-800 dark:text-white"
data-flux-column>
<div class="whitespace-nowrap flex justify-end">Actions</div>
</th>
</tr>
</thead>
<tbody class="divide-y divide-zinc-800/10 dark:divide-white/20" data-flux-rows>
@foreach($playlist->items->sortBy('order') as $item)
<tr data-flux-row>
<td class="py-3 px-3 first:pl-0 last:pr-0 text-sm whitespace-nowrap text-zinc-500 dark:text-zinc-300">
{{ $item->plugin->name }}
</td>
<td class="py-3 px-3 first:pl-0 last:pr-0 text-sm whitespace-nowrap text-zinc-500 dark:text-zinc-300">
<flux:switch wire:model.live="item.is_active"
wire:click="togglePlaylistItemActive({{ $item->id }})"
:checked="$item->is_active"/>
</td>
<td class="py-3 px-3 first:pl-0 last:pr-0 text-sm whitespace-nowrap">
<div class="flex justify-end gap-2">
@if(!$loop->first)
<flux:button wire:click="movePlaylistItemUp({{ $item->id }})"
icon="arrow-up" variant="subtle" size="sm"/>
@endif
@if(!$loop->last)
<flux:button wire:click="movePlaylistItemDown({{ $item->id }})"
icon="arrow-down" variant="subtle" size="sm"/>
@endif
<flux:modal.trigger name="delete-playlist-item-{{ $item->id }}">
<flux:button icon="trash" size="sm"/>
</flux:modal.trigger>
</div>
<flux:modal name="delete-playlist-item-{{ $item->id }}" class="min-w-[22rem] space-y-6">
<div>
<flux:heading size="lg">Delete {{ $item->plugin->name }}?</flux:heading>
<p class="mt-2 text-sm text-zinc-600 dark:text-zinc-400">This will remove this item from the playlist.</p>
</div>
<div class="flex gap-2">
<flux:spacer/>
<flux:modal.close>
<flux:button variant="ghost">Cancel</flux:button>
</flux:modal.close>
<flux:button wire:click="deletePlaylistItem({{ $item->id }})" variant="danger">Delete item</flux:button>
</div>
</flux:modal>
</td>
</tr>
@endforeach
</tbody>
</table>
</div>
@endforeach
</div>
</div>
</div>
</div>