diff --git a/config/urls.py b/config/urls.py
index e3e911e..daa6e4d 100644
--- a/config/urls.py
+++ b/config/urls.py
@@ -4,8 +4,8 @@ from django.contrib import admin
from django.urls import include, path
urlpatterns = [
- path('', include(('gallery.urls', 'gallery'), namespace='gallery')),
path('admin/', admin.site.urls),
+ path('', include(('gallery.urls', 'gallery'), namespace='gallery')),
]
if settings.DEBUG:
diff --git a/gallery/admin.py b/gallery/admin.py
index 8036d70..65c3b58 100644
--- a/gallery/admin.py
+++ b/gallery/admin.py
@@ -5,7 +5,7 @@ from imagekit.admin import AdminThumbnail
from imagekit.cachefiles import ImageCacheFile
from imagekit.processors import ResizeToFit
-from gallery.models import Album, City, Location, Photo
+from gallery.models import Album, City, Location, Photo, Redir
class AdminThumbnailSpec(ImageSpec):
@@ -20,6 +20,26 @@ def cached_admin_thumb(instance):
return cached
+class RedirAdmin(admin.ModelAdmin):
+ list_display = ('path', 'album', 'test_url')
+ search_fields = ('path',)
+ ordering = ('path',)
+ list_per_page = 10
+ list_editable = ('album',)
+
+ def formfield_for_foreignkey(self, db_field, request, **kwargs):
+ if db_field.name == "album":
+ kwargs["queryset"] = Album.objects.all().order_by('name') # Järjestä albumit aakkosjärjestykseen
+ return super().formfield_for_foreignkey(db_field, request, **kwargs)
+
+ def test_url(self, obj):
+ return format_html(
+ 'http://nyymix.net/{}',
+ obj.path, obj.path
+ )
+ test_url.short_description = "Test redirection"
+
+
class CityAdmin(admin.ModelAdmin):
list_display = ('name',)
search_fields = ('name',)
@@ -37,39 +57,53 @@ class LocationAdmin(admin.ModelAdmin):
class AlbumAdmin(admin.ModelAdmin):
prepopulated_fields = {'slug': ('name',)}
- list_display = ('__str__', 'location', 'album_date', 'is_public', 'thumbnail', )
+ list_display = ('__str__', 'name', 'location', 'album_date', 'is_public', 'thumbnail', )
search_fields = ('name',)
ordering = ('-album_date',)
- list_per_page = 30
- list_editable = ('is_public', 'location')
+ list_per_page = 20
+ list_editable = ('name', 'is_public', 'location')
+ readonly_fields = ['cover_preview'] # Lisätään esikatselukuva readonly_fieldsiin
+
+ def cover_preview(self, obj):
+ if obj.cover and obj.cover.photo:
+ return format_html(
+ '',
+ obj.cover.photo.url,
+ )
+ return "No cover image available"
+ cover_preview.short_description = "Cover Preview"
def formfield_for_foreignkey(self, db_field, request, **kwargs):
if db_field.name == "cover":
if hasattr(request, 'resolver_match') and request.resolver_match.kwargs.get('object_id'):
album_id = request.resolver_match.kwargs['object_id']
kwargs["queryset"] = Photo.objects.filter(album_id=album_id)
+
return super().formfield_for_foreignkey(db_field, request, **kwargs)
-
+
def thumbnail(self, obj):
if obj.cover and obj.cover.photo:
return format_html(
- '',
+ '',
obj.cover.photo.url,
)
return "-"
thumbnail.short_description = "Thumbnail"
-
class PhotoAdmin(admin.ModelAdmin):
- list_display = ('slug', 'album', 'admin_thumbnail',)
+ list_display = ('slug', 'album', 'is_favorite', 'admin_thumbnail',)
list_display_links = ('slug',)
- search_fields = ('slug',)
+ search_fields = ('slug', 'photo',)
readonly_fields = ['slug', 'taken_at', 'height', 'width', 'exif', ]
admin_thumbnail = AdminThumbnail(image_field=cached_admin_thumb)
+ list_filter = ('album',)
+ list_per_page = 30
+ list_editable = ('is_favorite',)
admin.site.register(City, CityAdmin)
admin.site.register(Location, LocationAdmin)
admin.site.register(Album, AlbumAdmin)
admin.site.register(Photo, PhotoAdmin)
+admin.site.register(Redir, RedirAdmin)
diff --git a/gallery/migrations/0001_initial.py b/gallery/migrations/0001_initial.py
index e8626ae..94c7531 100644
--- a/gallery/migrations/0001_initial.py
+++ b/gallery/migrations/0001_initial.py
@@ -1,4 +1,4 @@
-# Generated by Django 5.1.4 on 2025-01-18 07:39
+# Generated by Django 5.1.4 on 2025-01-20 19:09
import datetime
import django.db.models.deletion
@@ -33,7 +33,7 @@ class Migration(migrations.Migration):
],
options={
'verbose_name_plural': 'Locations',
- 'unique_together': {('place', 'city')},
+ 'unique_together': {('city', 'place')},
},
),
migrations.CreateModel(
@@ -44,7 +44,6 @@ class Migration(migrations.Migration):
('slug', models.SlugField(max_length=150, unique=True, verbose_name='Slug')),
('album_date', models.DateField(default=datetime.datetime.now, verbose_name='Album Date')),
('is_public', models.BooleanField(default=False, verbose_name='Published')),
- ('parent', models.ForeignKey(blank=True, limit_choices_to={'parent__isnull': True}, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='children', to='gallery.album', verbose_name='Parent Album')),
('location', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='albums', to='gallery.location', verbose_name='Location')),
],
options={
@@ -77,4 +76,15 @@ class Migration(migrations.Migration):
name='cover',
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='cover_to', to='gallery.photo', verbose_name='Album cover'),
),
+ migrations.CreateModel(
+ name='Redir',
+ fields=[
+ ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
+ ('path', models.CharField(db_index=True, max_length=255, unique=True, verbose_name='Path')),
+ ('album', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='pathurl', to='gallery.album', verbose_name='Album')),
+ ],
+ options={
+ 'verbose_name_plural': 'Redirs',
+ },
+ ),
]
diff --git a/gallery/models/__init__.py b/gallery/models/__init__.py
index 0721982..9b14cb2 100644
--- a/gallery/models/__init__.py
+++ b/gallery/models/__init__.py
@@ -1,3 +1,4 @@
from .album import *
from .location import *
from .photo import *
+from .redir import *
diff --git a/gallery/models/album.py b/gallery/models/album.py
index 66a7935..bfc0e4b 100644
--- a/gallery/models/album.py
+++ b/gallery/models/album.py
@@ -12,7 +12,6 @@ from gallery.models.location import Location
class Album(models.Model):
name = models.CharField(max_length=150, unique=True, verbose_name="Album")
slug = models.SlugField(max_length=150, unique=True, verbose_name="Slug")
- parent = models.ForeignKey('self', on_delete=models.SET_NULL, blank=True, null=True, limit_choices_to={'parent__isnull': True}, related_name='children', verbose_name="Parent Album")
location = models.ForeignKey(Location, blank=True, null=True, on_delete=models.SET_NULL, related_name='albums', verbose_name="Location")
album_date = models.DateField(default=datetime.now, verbose_name="Album Date")
cover = models.ForeignKey("Photo", blank=True, null=True, on_delete=models.SET_NULL, related_name='cover_to', verbose_name="Album cover")
@@ -21,12 +20,12 @@ class Album(models.Model):
@property
def photos_in_album(self):
return self.photos.count()
-
+
@property
def photos_views(self):
total_views = sum(self.photos.views())
return total_views
-
+
@property
def cover_url(self):
if self.cover:
@@ -35,11 +34,6 @@ class Album(models.Model):
if random_cover:
return random_cover.photo_md.url
return static('img/placeholder.png')
-
- def clean(self):
- super().clean()
- if self.parent and self.parent == self:
- raise ValidationError({'parent': "An album cannot be its own parent."})
def save(self, *args, **kwargs):
if not self.slug:
diff --git a/gallery/models/location.py b/gallery/models/location.py
index 9790efe..7338d60 100644
--- a/gallery/models/location.py
+++ b/gallery/models/location.py
@@ -18,6 +18,8 @@ class Location(models.Model):
class Meta:
verbose_name_plural = "Locations"
unique_together = ('place', "city")
+ ordering = ['city']
+
def __str__(self):
if self.place:
diff --git a/gallery/models/redir.py b/gallery/models/redir.py
new file mode 100644
index 0000000..3473b6a
--- /dev/null
+++ b/gallery/models/redir.py
@@ -0,0 +1,14 @@
+from django.db import models
+
+from gallery.models import Album
+
+
+class Redir(models.Model):
+ path = models.CharField(max_length=255, db_index=True, unique=True, verbose_name="Path")
+ album = models.ForeignKey(Album, on_delete=models.CASCADE, related_name='pathurl', verbose_name="Album")
+
+ class Meta:
+ verbose_name_plural = "Redirs"
+
+ def __str__(self):
+ return '{0} - {1}'.format(self.path, self.album.slug)
diff --git a/gallery/urls.py b/gallery/urls.py
index 1ca5714..884fe21 100644
--- a/gallery/urls.py
+++ b/gallery/urls.py
@@ -11,5 +11,5 @@ urlpatterns = [
path('albums///', views.PhotoDetail.as_view(), name='photo_url'),
path('albums//', views.AlbumDetail.as_view(), name='album_url'),
path('albums/', views.AlbumsList.as_view(), name='albums_url'),
-
+ path('/', views.redirect_to_album, name='redirect_to_album'),
]
diff --git a/gallery/views.py b/gallery/views.py
index 2f64dbb..7298d39 100644
--- a/gallery/views.py
+++ b/gallery/views.py
@@ -2,7 +2,7 @@ from django.core.paginator import Paginator
from django.shortcuts import get_object_or_404, redirect, render
from django.views.generic import DetailView, ListView, TemplateView
-from .models import Album, Photo
+from .models import Album, Photo, Redir
class Main(TemplateView):
@@ -86,3 +86,9 @@ class PhotoDetail(DetailView):
photo.save()
return redirect('gallery:photo_url', album_slug=self.kwargs['album_slug'], photo_slug=self.kwargs['photo_slug'])
+
+
+
+def redirect_to_album(request, redir_path):
+ redir = get_object_or_404(Redir, path=redir_path)
+ return redirect('gallery:album_url', album_slug=redir.album.slug)
\ No newline at end of file