2020-09-22 21:56:51 +02:00
|
|
|
import os
|
2021-12-30 23:20:05 +01:00
|
|
|
from math import floor
|
2020-09-22 21:56:51 +02:00
|
|
|
|
|
|
|
from django.db import models
|
|
|
|
from django.urls import reverse
|
|
|
|
from django.dispatch import receiver
|
|
|
|
from django.db.models.signals import post_delete, pre_save
|
2020-10-06 17:01:29 +02:00
|
|
|
from django.utils.translation import ugettext_lazy as _
|
2020-09-22 21:56:51 +02:00
|
|
|
|
2021-12-30 23:20:05 +01:00
|
|
|
from lostplaces.models.abstract_models import Submittable, Taggable, Mapable, Expireable
|
2020-09-22 21:56:51 +02:00
|
|
|
|
|
|
|
from easy_thumbnails.fields import ThumbnailerImageField
|
|
|
|
from easy_thumbnails.files import get_thumbnailer
|
|
|
|
|
2021-10-01 09:40:42 +02:00
|
|
|
PLACE_LEVELS = (
|
|
|
|
(1, 'Ruin'),
|
|
|
|
(2, 'Vandalized'),
|
|
|
|
(3, 'Natures Treasure'),
|
2021-12-30 23:20:05 +01:00
|
|
|
(4, 'Lost in History'),
|
2021-10-01 09:40:42 +02:00
|
|
|
(5, 'Time Capsule')
|
|
|
|
)
|
2021-04-10 08:23:36 +02:00
|
|
|
|
2020-09-22 21:56:51 +02:00
|
|
|
class Place(Submittable, Taggable, Mapable):
|
|
|
|
"""
|
|
|
|
Place defines a lost place (location, name, description etc.).
|
|
|
|
"""
|
|
|
|
|
2020-10-11 02:07:14 +02:00
|
|
|
location = models.CharField(
|
|
|
|
max_length=50,
|
|
|
|
verbose_name=_('Location'),
|
|
|
|
)
|
|
|
|
description = models.TextField(
|
|
|
|
help_text=_('Description of the place: e.g. how to get there, where to be careful, the place\'s history...'),
|
|
|
|
verbose_name=_('Description'),
|
|
|
|
)
|
2020-09-22 21:56:51 +02:00
|
|
|
|
2021-04-10 08:23:36 +02:00
|
|
|
hero = models.ForeignKey(
|
|
|
|
'PlaceImage',
|
|
|
|
on_delete=models.SET_NULL,
|
|
|
|
null=True,
|
|
|
|
blank=True,
|
|
|
|
related_name='place_heros'
|
|
|
|
)
|
|
|
|
|
2021-10-01 09:40:42 +02:00
|
|
|
level = models.IntegerField(
|
|
|
|
default=5,
|
|
|
|
choices=PLACE_LEVELS
|
|
|
|
)
|
|
|
|
|
2021-04-10 08:23:36 +02:00
|
|
|
def get_hero_image(self):
|
|
|
|
if self.hero:
|
|
|
|
return self.hero
|
|
|
|
elif len(self.placeimages.all()) > 0:
|
|
|
|
return self.placeimages.first()
|
|
|
|
else:
|
|
|
|
return None
|
|
|
|
|
2020-09-22 21:56:51 +02:00
|
|
|
def get_absolute_url(self):
|
|
|
|
return reverse('place_detail', kwargs={'pk': self.pk})
|
2021-04-10 08:23:36 +02:00
|
|
|
|
|
|
|
def get_hero_index_in_queryset(self):
|
2021-12-30 23:20:05 +01:00
|
|
|
'''
|
|
|
|
Calculates the index of the hero image within
|
|
|
|
the list / queryset of images. Necessary for
|
|
|
|
the lightbox.
|
|
|
|
'''
|
2021-04-10 08:23:36 +02:00
|
|
|
for i in range(0, len(self.placeimages.all())):
|
|
|
|
image = self.placeimages.all()[i]
|
|
|
|
if image == self.hero:
|
2021-12-30 23:20:05 +01:00
|
|
|
return i
|
2021-04-10 08:23:36 +02:00
|
|
|
return None
|
2020-09-22 21:56:51 +02:00
|
|
|
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
# Get center position of LP-geocoordinates.
|
|
|
|
def average_latlon(cls, place_list):
|
|
|
|
amount = len(place_list)
|
|
|
|
# Init fill values to prevent None
|
2021-10-01 09:29:09 +02:00
|
|
|
# China Corner in Münster
|
|
|
|
# Where I almost always eat lunch
|
|
|
|
# (Does'nt help losing wheight, tho)
|
|
|
|
longitude = 7.6295628132604385
|
|
|
|
latitude = 51.961922091398904
|
2020-09-22 21:56:51 +02:00
|
|
|
|
|
|
|
if amount > 0:
|
|
|
|
for place in place_list:
|
|
|
|
longitude += place.longitude
|
|
|
|
latitude += place.latitude
|
|
|
|
return {'latitude':latitude / amount, 'longitude': longitude / amount}
|
|
|
|
|
|
|
|
return {'latitude': latitude, 'longitude': longitude}
|
|
|
|
|
2021-12-30 23:20:05 +01:00
|
|
|
def calculate_place_level(self):
|
|
|
|
self.remove_expired_votes()
|
|
|
|
|
|
|
|
if self.placevotings.count() == 0:
|
2021-12-31 15:56:31 +01:00
|
|
|
self.level = 5
|
|
|
|
self.save()
|
|
|
|
return
|
2021-12-30 23:20:05 +01:00
|
|
|
|
|
|
|
level = 0
|
|
|
|
|
|
|
|
for vote in self.placevotings.all():
|
|
|
|
level += vote.vote
|
|
|
|
|
|
|
|
self.level = floor(level / self.placevotings.count())
|
2021-12-31 15:56:31 +01:00
|
|
|
self.save()
|
2021-12-30 23:20:05 +01:00
|
|
|
|
|
|
|
def remove_expired_votes(self):
|
|
|
|
for vote in self.placevotings.all():
|
|
|
|
if vote.is_expired:
|
|
|
|
vote.delete()
|
|
|
|
|
2020-09-22 21:56:51 +02:00
|
|
|
def __str__(self):
|
|
|
|
return self.name
|
|
|
|
|
2020-12-24 17:23:20 +01:00
|
|
|
def generate_place_image_filename(instance, filename):
|
2020-09-22 21:56:51 +02:00
|
|
|
"""
|
2020-12-24 17:23:20 +01:00
|
|
|
Callback for generating filename for uploaded place images.
|
|
|
|
Returns filename as: place_pk-placename{-number}.jpg
|
2020-09-22 21:56:51 +02:00
|
|
|
"""
|
|
|
|
|
|
|
|
return 'places/' + str(instance.place.pk) + '-' + str(instance.place.name) + '.' + filename.split('.')[-1]
|
|
|
|
|
2021-04-04 16:55:56 +02:00
|
|
|
def generate_image_upload_path(instance, filename):
|
|
|
|
return generate_place_image_filename(instance, filename)
|
|
|
|
|
2020-09-22 21:56:51 +02:00
|
|
|
class PlaceAsset(Submittable):
|
|
|
|
"""
|
|
|
|
Assets to a place, i.e. images
|
|
|
|
"""
|
|
|
|
|
|
|
|
class Meta:
|
2021-12-30 23:20:05 +01:00
|
|
|
abstract = True
|
2020-09-22 21:56:51 +02:00
|
|
|
|
|
|
|
place = models.ForeignKey(
|
|
|
|
Place,
|
|
|
|
on_delete=models.CASCADE,
|
|
|
|
related_name='%(class)ss',
|
|
|
|
null=True
|
|
|
|
)
|
|
|
|
|
2021-05-16 13:03:41 +02:00
|
|
|
class DummyAsset(PlaceAsset):
|
|
|
|
name = models.CharField(max_length=50)
|
|
|
|
|
2020-12-25 15:42:02 +01:00
|
|
|
class PlaceImage(PlaceAsset):
|
2020-09-22 21:56:51 +02:00
|
|
|
"""
|
|
|
|
PlaceImage defines an image file object that points to a file in uploads/.
|
|
|
|
Intermediate image sizes are generated as defined in THUMBNAIL_ALIASES.
|
|
|
|
PlaceImage references a Place to which it belongs.
|
|
|
|
"""
|
|
|
|
|
2020-10-11 02:07:14 +02:00
|
|
|
description = models.TextField(
|
|
|
|
blank=True,
|
|
|
|
verbose_name=_('Description'),
|
|
|
|
)
|
2020-09-22 21:56:51 +02:00
|
|
|
filename = ThumbnailerImageField(
|
2020-12-24 17:23:20 +01:00
|
|
|
upload_to=generate_place_image_filename,
|
2020-09-22 21:56:51 +02:00
|
|
|
resize_source=dict(size=(2560, 2560),
|
2020-10-11 01:39:38 +02:00
|
|
|
sharpen=True),
|
2021-04-10 08:23:36 +02:00
|
|
|
verbose_name=_('Images'),
|
2020-10-11 01:39:38 +02:00
|
|
|
help_text=_('Optional: One or more images to upload')
|
2020-09-22 21:56:51 +02:00
|
|
|
)
|
|
|
|
place = models.ForeignKey(
|
|
|
|
Place,
|
|
|
|
on_delete=models.CASCADE,
|
|
|
|
related_name='placeimages'
|
|
|
|
)
|
|
|
|
|
|
|
|
def __str__(self):
|
|
|
|
"""
|
|
|
|
Returning the name of the corresponding place + id
|
|
|
|
of this image as textual representation of this instance
|
|
|
|
"""
|
|
|
|
|
2021-04-10 08:23:36 +02:00
|
|
|
return 'Image ' + str(self.place.name)
|
2020-09-22 21:56:51 +02:00
|
|
|
|
|
|
|
# These two auto-delete files from filesystem when they are unneeded:
|
|
|
|
|
|
|
|
@receiver(post_delete, sender=PlaceImage)
|
|
|
|
def auto_delete_file_on_delete(sender, instance, **kwargs):
|
|
|
|
"""
|
|
|
|
Deletes file (including thumbnails) from filesystem
|
|
|
|
when corresponding `PlaceImage` object is deleted.
|
|
|
|
"""
|
|
|
|
if instance.filename:
|
|
|
|
# Get and delete all files and thumbnails from instance
|
|
|
|
thumbmanager = get_thumbnailer(instance.filename)
|
|
|
|
thumbmanager.delete(save=False)
|
|
|
|
|
|
|
|
@receiver(pre_save, sender=PlaceImage)
|
|
|
|
def auto_delete_file_on_change(sender, instance, **kwargs):
|
|
|
|
"""
|
|
|
|
Deletes old file from filesystem
|
|
|
|
when corresponding `PlaceImage` object is updated
|
|
|
|
with new file.
|
|
|
|
"""
|
|
|
|
if not instance.pk:
|
|
|
|
return False
|
|
|
|
|
|
|
|
try:
|
|
|
|
old_file = PlaceImage.objects.get(pk=instance.pk).filename
|
|
|
|
except PlaceImage.DoesNotExist:
|
|
|
|
return False
|
|
|
|
|
|
|
|
# No need to delete thumbnails, as they will be overwritten on regeneration.
|
|
|
|
new_file = instance.filename
|
|
|
|
if not old_file == new_file:
|
2020-12-25 15:42:02 +01:00
|
|
|
old_file.delete(save=False)
|
2021-12-30 23:20:05 +01:00
|
|
|
|
|
|
|
|
|
|
|
class PlaceVoting(PlaceAsset, Expireable):
|
|
|
|
vote = models.IntegerField(choices=PLACE_LEVELS)
|
|
|
|
|
|
|
|
def get_human_readable_level(self):
|
|
|
|
return PLACE_LEVELS[self.vote - 1][1]
|
|
|
|
|
|
|
|
def get_all_choices(self):
|
|
|
|
return reversed(PLACE_LEVELS)
|