mirror of
https://github.com/pacnpal/thrillwiki_django_no_react.git
synced 2025-12-20 07:11:08 -05:00
143 lines
6.6 KiB
HTML
143 lines
6.6 KiB
HTML
{% load static %}
|
|
|
|
<div x-data="photoDisplay({
|
|
photos: [
|
|
{% for photo in photos %}
|
|
{
|
|
id: {{ photo.id }},
|
|
url: '{{ photo.image.url }}',
|
|
caption: '{{ photo.caption|default:""|escapejs }}',
|
|
date_taken: '{{ photo.date_taken|date:"F j, Y g:i A"|default:""|escapejs }}',
|
|
uploaded_by: '{{ photo.uploaded_by.username|default:""|escapejs }}'
|
|
}{% if not forloop.last %},{% endif %}
|
|
{% endfor %}
|
|
],
|
|
contentType: '{{ content_type }}',
|
|
objectId: {{ object_id }},
|
|
csrfToken: '{{ csrf_token }}',
|
|
uploadUrl: '{% url "photos:upload" %}'
|
|
})" class="w-full">
|
|
<!-- Photo Grid - Adaptive Layout -->
|
|
<div class="relative">
|
|
|
|
<!-- Success Message -->
|
|
<div x-show="showSuccess"
|
|
x-transition:enter="transition ease-out duration-300"
|
|
x-transition:enter-start="opacity-0 transform -translate-y-2"
|
|
x-transition:enter-end="opacity-100 transform translate-y-0"
|
|
x-transition:leave="transition ease-in duration-200"
|
|
x-transition:leave-start="opacity-100 transform translate-y-0"
|
|
x-transition:leave-end="opacity-0 transform -translate-y-2"
|
|
class="absolute top-0 left-0 right-0 z-20 px-4 py-2 mx-auto mt-2 text-sm text-center text-green-800 transform -translate-y-full bg-green-100 rounded-lg w-fit dark:bg-green-200 dark:text-green-900">
|
|
Photo uploaded successfully!
|
|
</div>
|
|
|
|
<!-- Upload Progress -->
|
|
<template x-if="uploading">
|
|
<div class="absolute top-0 left-0 right-0 z-20 p-4 mx-auto mt-2 bg-white rounded-lg shadow-lg w-fit dark:bg-gray-800">
|
|
<div class="flex items-center justify-between mb-1">
|
|
<span class="text-sm text-gray-700 dark:text-gray-300">Uploading...</span>
|
|
<span class="text-sm text-gray-700 dark:text-gray-300" x-text="uploadProgress + '%'"></span>
|
|
</div>
|
|
<div class="w-64 h-2 bg-gray-200 rounded-full dark:bg-gray-700">
|
|
<div class="h-2 transition-all duration-300 bg-blue-600 rounded-full"
|
|
:style="'width: ' + uploadProgress + '%'"></div>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<!-- Error Message -->
|
|
<template x-if="error">
|
|
<div class="absolute top-0 left-0 right-0 z-20 px-4 py-2 mx-auto mt-2 text-sm text-center text-red-800 bg-red-100 rounded-lg w-fit dark:bg-red-200 dark:text-red-900"
|
|
x-text="error"></div>
|
|
</template>
|
|
|
|
<!-- Photo Grid -->
|
|
<div :class="{
|
|
'grid gap-4': true,
|
|
'grid-cols-1 max-w-2xl mx-auto': photos.length === 1,
|
|
'grid-cols-2 max-w-3xl mx-auto': photos.length === 2,
|
|
'grid-cols-2 md:grid-cols-3 lg:grid-cols-4': photos.length > 2
|
|
}">
|
|
<template x-for="photo in photos" :key="photo.id">
|
|
<div class="relative cursor-pointer group aspect-w-16 aspect-h-9" @click="showFullscreen(photo)">
|
|
<img :src="photo.url"
|
|
:alt="photo.caption || ''"
|
|
class="object-cover transition-transform duration-300 rounded-lg group-hover:scale-105">
|
|
</div>
|
|
</template>
|
|
</div>
|
|
|
|
<!-- No Photos Message -->
|
|
<template x-if="photos.length === 0">
|
|
<div class="flex flex-col items-center justify-center py-12 text-center">
|
|
<i class="mb-4 text-4xl text-gray-400 fas fa-camera dark:text-gray-600"></i>
|
|
<p class="mb-4 text-gray-500 dark:text-gray-400">No photos available yet.</p>
|
|
{% if user.is_authenticated and perms.media.add_photo %}
|
|
<p class="text-sm text-gray-400 dark:text-gray-500">Click the upload button to add the first photo!</p>
|
|
{% endif %}
|
|
</div>
|
|
</template>
|
|
</div>
|
|
|
|
<!-- Fullscreen Photo Modal -->
|
|
<div x-show="fullscreenPhoto"
|
|
class="fixed inset-0 z-50 flex items-center justify-center bg-black/90"
|
|
@click.self="fullscreenPhoto = null"
|
|
@keydown.escape.window="fullscreenPhoto = null">
|
|
<div class="relative p-4 mx-auto max-w-7xl">
|
|
<!-- Close Button -->
|
|
<button @click="fullscreenPhoto = null"
|
|
class="absolute text-white top-4 right-4 hover:text-gray-300">
|
|
<i class="text-2xl fas fa-times"></i>
|
|
</button>
|
|
|
|
<!-- Photo Container -->
|
|
<div class="relative">
|
|
<img :src="fullscreenPhoto?.url"
|
|
:alt="fullscreenPhoto?.caption || ''"
|
|
class="max-h-[90vh] w-auto mx-auto rounded-lg">
|
|
|
|
<!-- Photo Info Overlay -->
|
|
<div class="absolute bottom-0 left-0 right-0 p-4 text-white bg-black/50 rounded-b-lg">
|
|
<!-- Caption -->
|
|
<div x-show="fullscreenPhoto?.caption"
|
|
class="mb-2 text-lg font-medium"
|
|
x-text="fullscreenPhoto?.caption">
|
|
</div>
|
|
|
|
<!-- Photo Details -->
|
|
<div class="flex flex-wrap gap-4 text-sm">
|
|
<!-- Uploaded By -->
|
|
<div x-show="fullscreenPhoto?.uploaded_by" class="flex items-center">
|
|
<i class="mr-2 fas fa-user"></i>
|
|
<span x-text="fullscreenPhoto?.uploaded_by"></span>
|
|
</div>
|
|
|
|
<!-- Date Taken -->
|
|
<div x-show="fullscreenPhoto?.date_taken" class="flex items-center">
|
|
<i class="mr-2 fas fa-calendar"></i>
|
|
<span x-text="fullscreenPhoto?.date_taken"></span>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Actions -->
|
|
<div class="absolute flex gap-2 bottom-4 right-4">
|
|
<a :href="fullscreenPhoto?.url"
|
|
download
|
|
class="p-2 text-white rounded-full bg-white/10 hover:bg-white/20"
|
|
title="Download">
|
|
<i class="fas fa-download"></i>
|
|
</a>
|
|
<button @click="sharePhoto(fullscreenPhoto)"
|
|
class="p-2 text-white rounded-full bg-white/10 hover:bg-white/20"
|
|
title="Share">
|
|
<i class="fas fa-share-alt"></i>
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|