- Installed p7h/nas-file-manager package via private VCS repo - Published config/nas-file-manager.php with super_admin middleware restriction - Added NAS env vars to .env.example - Created admin/nas-storage page with connection info panel and file browser widget - Added NAS Storage link to admin sidebar (super_admin only) - Added SuperAdminController@nasStorage method and admin.nas-storage route - Includes all accumulated branch changes: profile wall, 2FA, audit logs, settings panel, country/phone/timezone components, posts, slideshow, playlist shares, video downloads/shares, comment likes, notifications, social links, and more Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
292 lines
14 KiB
PHP
292 lines
14 KiB
PHP
@extends('admin.layout')
|
||
|
||
@section('title', 'Videos')
|
||
|
||
@section('content')
|
||
|
||
{{-- ── Page header ──────────────────────────────────────────────────── --}}
|
||
<div class="adm-page-header">
|
||
<h1 class="adm-page-title"><i class="bi bi-play-circle"></i> Videos</h1>
|
||
</div>
|
||
|
||
{{-- ── Alerts ───────────────────────────────────────────────────────── --}}
|
||
@if(session('success'))
|
||
<div class="adm-alert adm-alert-success">
|
||
<i class="bi bi-check-circle-fill"></i>
|
||
<span>{{ session('success') }}</span>
|
||
<button class="adm-alert-close"><i class="bi bi-x"></i></button>
|
||
</div>
|
||
@endif
|
||
@if(session('error'))
|
||
<div class="adm-alert adm-alert-error">
|
||
<i class="bi bi-exclamation-circle-fill"></i>
|
||
<span>{{ session('error') }}</span>
|
||
<button class="adm-alert-close"><i class="bi bi-x"></i></button>
|
||
</div>
|
||
@endif
|
||
|
||
{{-- ── Filter card ──────────────────────────────────────────────────── --}}
|
||
<div class="adm-card">
|
||
<div class="adm-card-body" style="padding:16px 20px;">
|
||
<form method="GET" action="{{ route('admin.videos') }}" class="adm-filter-form">
|
||
|
||
<div class="adm-filter-search">
|
||
<i class="bi bi-search"></i>
|
||
<input type="text" name="search" class="adm-input"
|
||
placeholder="Search title or description…"
|
||
value="{{ request('search') }}" autocomplete="off">
|
||
</div>
|
||
|
||
<select name="status" class="adm-select">
|
||
<option value="">All Status</option>
|
||
<option value="ready" {{ request('status') === 'ready' ? 'selected' : '' }}>Ready</option>
|
||
<option value="processing" {{ request('status') === 'processing' ? 'selected' : '' }}>Processing</option>
|
||
<option value="pending" {{ request('status') === 'pending' ? 'selected' : '' }}>Pending</option>
|
||
<option value="failed" {{ request('status') === 'failed' ? 'selected' : '' }}>Failed</option>
|
||
</select>
|
||
|
||
<select name="visibility" class="adm-select">
|
||
<option value="">All Visibility</option>
|
||
<option value="public" {{ request('visibility') === 'public' ? 'selected' : '' }}>Public</option>
|
||
<option value="unlisted" {{ request('visibility') === 'unlisted' ? 'selected' : '' }}>Unlisted</option>
|
||
<option value="private" {{ request('visibility') === 'private' ? 'selected' : '' }}>Private</option>
|
||
</select>
|
||
|
||
<select name="type" class="adm-select">
|
||
<option value="">All Types</option>
|
||
<option value="generic" {{ request('type') === 'generic' ? 'selected' : '' }}>Generic</option>
|
||
<option value="music" {{ request('type') === 'music' ? 'selected' : '' }}>Music</option>
|
||
<option value="match" {{ request('type') === 'match' ? 'selected' : '' }}>Match</option>
|
||
</select>
|
||
|
||
<select name="sort" class="adm-select">
|
||
<option value="latest" {{ request('sort', 'latest') === 'latest' ? 'selected' : '' }}>Newest first</option>
|
||
<option value="oldest" {{ request('sort') === 'oldest' ? 'selected' : '' }}>Oldest first</option>
|
||
<option value="title_asc" {{ request('sort') === 'title_asc' ? 'selected' : '' }}>Title A–Z</option>
|
||
<option value="title_desc" {{ request('sort') === 'title_desc' ? 'selected' : '' }}>Title Z–A</option>
|
||
</select>
|
||
|
||
<button type="submit" class="adm-btn adm-btn-primary">
|
||
<i class="bi bi-funnel"></i> Filter
|
||
</button>
|
||
@if(request()->hasAny(['search','status','visibility','type','sort']))
|
||
<a href="{{ route('admin.videos') }}" class="adm-btn">
|
||
<i class="bi bi-x-lg"></i> Clear
|
||
</a>
|
||
@endif
|
||
|
||
</form>
|
||
</div>
|
||
</div>
|
||
|
||
{{-- ── Videos table ─────────────────────────────────────────────────── --}}
|
||
<div class="adm-card">
|
||
<div class="adm-card-header">
|
||
<div class="adm-card-title">
|
||
<i class="bi bi-play-circle"></i>
|
||
All Videos
|
||
<span class="adm-badge adm-badge-user">{{ $videos->total() ?? $videos->count() }}</span>
|
||
</div>
|
||
</div>
|
||
|
||
<div class="adm-table-wrap">
|
||
<table class="adm-table">
|
||
<thead>
|
||
<tr>
|
||
<th>Video</th>
|
||
<th>Owner</th>
|
||
<th>Status</th>
|
||
<th>Visibility</th>
|
||
<th>Type</th>
|
||
<th>Views</th>
|
||
<th>Likes</th>
|
||
<th>Uploaded</th>
|
||
<th style="width:110px; text-align:right;">Actions</th>
|
||
</tr>
|
||
</thead>
|
||
<tbody>
|
||
@forelse($videos as $video)
|
||
<tr>
|
||
{{-- Thumbnail + title --}}
|
||
<td>
|
||
<div style="display:flex; align-items:center; gap:12px;">
|
||
@if($video->thumbnail)
|
||
<img src="{{ asset('storage/thumbnails/' . $video->thumbnail) }}"
|
||
alt="" style="width:72px;height:44px;object-fit:cover;border-radius:6px;flex-shrink:0;border:1px solid var(--border);">
|
||
@else
|
||
<div style="width:72px;height:44px;border-radius:6px;background:var(--bg-card2);border:1px solid var(--border);display:flex;align-items:center;justify-content:center;flex-shrink:0;">
|
||
<i class="bi bi-play-circle" style="color:var(--text-3);font-size:18px;"></i>
|
||
</div>
|
||
@endif
|
||
<div style="min-width:0;">
|
||
<div style="font-weight:500;font-size:13px;white-space:nowrap;overflow:hidden;text-overflow:ellipsis;max-width:180px;">
|
||
{{ $video->title }}
|
||
</div>
|
||
@if($video->description)
|
||
<div style="font-size:11px;color:var(--text-2);margin-top:2px;white-space:nowrap;overflow:hidden;text-overflow:ellipsis;max-width:180px;">
|
||
{{ Str::limit($video->description, 55) }}
|
||
</div>
|
||
@endif
|
||
</div>
|
||
</div>
|
||
</td>
|
||
|
||
{{-- Owner --}}
|
||
<td>
|
||
<a href="{{ route('channel', $video->user->channel) }}" target="_blank"
|
||
style="font-size:13px;color:var(--text);text-decoration:none;display:flex;align-items:center;gap:4px;">
|
||
{{ $video->user->name }}
|
||
<i class="bi bi-box-arrow-up-right" style="font-size:10px;opacity:.4;"></i>
|
||
</a>
|
||
</td>
|
||
|
||
{{-- Status --}}
|
||
<td>
|
||
@php
|
||
$statusMap = [
|
||
'ready' => ['adm-badge-ready', 'bi-check-circle-fill', 'Ready'],
|
||
'processing' => ['adm-badge-unlisted','bi-arrow-repeat', 'Processing'],
|
||
'pending' => ['adm-badge-unverified','bi-clock', 'Pending'],
|
||
'failed' => ['adm-badge-superadmin','bi-x-circle-fill', 'Failed'],
|
||
];
|
||
[$cls, $ico, $lbl] = $statusMap[$video->status] ?? ['adm-badge-user','bi-dash','Unknown'];
|
||
@endphp
|
||
<span class="adm-badge {{ $cls }}"><i class="bi {{ $ico }}"></i> {{ $lbl }}</span>
|
||
</td>
|
||
|
||
{{-- Visibility --}}
|
||
<td>
|
||
@php
|
||
$visMap = [
|
||
'public' => ['adm-badge-ready', 'bi-globe2', 'Public'],
|
||
'unlisted' => ['adm-badge-unlisted','bi-link-45deg', 'Unlisted'],
|
||
'private' => ['adm-badge-private', 'bi-lock-fill', 'Private'],
|
||
];
|
||
[$vcls, $vico, $vlbl] = $visMap[$video->visibility] ?? ['adm-badge-user','bi-question','—'];
|
||
@endphp
|
||
<span class="adm-badge {{ $vcls }}"><i class="bi {{ $vico }}"></i> {{ $vlbl }}</span>
|
||
</td>
|
||
|
||
{{-- Type --}}
|
||
<td>
|
||
@php
|
||
$typeMap = [
|
||
'generic' => ['bi-play-circle','#94a3b8'],
|
||
'music' => ['bi-music-note-beamed','#a78bfa'],
|
||
'match' => ['bi-trophy','#fbbf24'],
|
||
];
|
||
[$tico, $tcol] = $typeMap[$video->type] ?? ['bi-play-circle','#94a3b8'];
|
||
@endphp
|
||
<span style="display:inline-flex;align-items:center;gap:5px;font-size:12px;color:{{ $tcol }};">
|
||
<i class="bi {{ $tico }}"></i>
|
||
{{ ucfirst($video->type) }}
|
||
</span>
|
||
</td>
|
||
|
||
{{-- Views --}}
|
||
<td style="font-size:13px;color:var(--text-2);">
|
||
{{ number_format(\DB::table('video_views')->where('video_id',$video->id)->count()) }}
|
||
</td>
|
||
|
||
{{-- Likes --}}
|
||
<td style="font-size:13px;color:var(--text-2);">
|
||
{{ number_format(\DB::table('video_likes')->where('video_id',$video->id)->count()) }}
|
||
</td>
|
||
|
||
{{-- Date --}}
|
||
<td class="text-muted-sm">{{ $video->created_at->format('M d, Y') }}</td>
|
||
|
||
{{-- Actions --}}
|
||
<td>
|
||
<div class="adm-row-actions" style="justify-content:flex-end;">
|
||
<a href="{{ route('videos.show', $video) }}" target="_blank"
|
||
class="adm-btn adm-btn-sm" title="Watch">
|
||
<i class="bi bi-play"></i>
|
||
</a>
|
||
<a href="{{ route('admin.videos.edit', $video) }}"
|
||
class="adm-btn adm-btn-sm" title="Edit">
|
||
<i class="bi bi-pencil"></i>
|
||
</a>
|
||
<button type="button"
|
||
class="adm-btn adm-btn-sm adm-btn-danger"
|
||
title="Delete"
|
||
onclick="openDeleteDialog('{{ $video->getRouteKey() }}', '{{ addslashes($video->title) }}')">
|
||
<i class="bi bi-trash"></i>
|
||
</button>
|
||
</div>
|
||
</td>
|
||
</tr>
|
||
@empty
|
||
<tr>
|
||
<td colspan="9">
|
||
<div class="empty-state">
|
||
<i class="bi bi-play-circle"></i>
|
||
<p>No videos found</p>
|
||
</div>
|
||
</td>
|
||
</tr>
|
||
@endforelse
|
||
</tbody>
|
||
</table>
|
||
</div>
|
||
|
||
{{-- Pagination --}}
|
||
@if($videos instanceof \Illuminate\Pagination\LengthAwarePaginator && $videos->hasPages())
|
||
<div style="padding:16px 20px; border-top:1px solid var(--border);">
|
||
{{ $videos->onEachSide(1)->links() }}
|
||
</div>
|
||
@endif
|
||
</div>
|
||
|
||
{{-- ── Delete confirmation dialog ───────────────────────────────────── --}}
|
||
<div class="adm-dialog-overlay" id="deleteDialog">
|
||
<div class="adm-dialog">
|
||
<div class="adm-dialog-header">
|
||
<div class="adm-dialog-title">
|
||
<i class="bi bi-exclamation-triangle-fill"></i> Delete Video
|
||
</div>
|
||
<button type="button" class="adm-btn adm-btn-sm" onclick="closeDeleteDialog()"
|
||
style="border:none;background:none;color:var(--text-2);">
|
||
<i class="bi bi-x-lg"></i>
|
||
</button>
|
||
</div>
|
||
<div class="adm-dialog-body">
|
||
<p>You are about to permanently delete <strong id="dlgVideoTitle"></strong>.</p>
|
||
<div class="adm-dialog-warning">
|
||
<i class="bi bi-info-circle-fill" style="flex-shrink:0;margin-top:1px;"></i>
|
||
All views, likes, comments, and HLS files for this video will also be deleted.
|
||
</div>
|
||
</div>
|
||
<div class="adm-dialog-footer">
|
||
<button type="button" class="adm-btn" onclick="closeDeleteDialog()">Cancel</button>
|
||
<form id="deleteForm" method="POST">
|
||
@csrf @method('DELETE')
|
||
<button type="submit" class="adm-btn adm-btn-danger" style="height:36px;">
|
||
<i class="bi bi-trash"></i> Delete Video
|
||
</button>
|
||
</form>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
@endsection
|
||
|
||
@section('scripts')
|
||
<script>
|
||
function openDeleteDialog(videoId, videoTitle) {
|
||
document.getElementById('dlgVideoTitle').textContent = videoTitle;
|
||
document.getElementById('deleteForm').action = '/admin/videos/' + videoId;
|
||
document.getElementById('deleteDialog').classList.add('open');
|
||
}
|
||
function closeDeleteDialog() {
|
||
document.getElementById('deleteDialog').classList.remove('open');
|
||
}
|
||
document.getElementById('deleteDialog').addEventListener('click', function(e) {
|
||
if (e.target === this) closeDeleteDialog();
|
||
});
|
||
document.addEventListener('keydown', function(e) {
|
||
if (e.key === 'Escape') closeDeleteDialog();
|
||
});
|
||
</script>
|
||
@endsection
|