2022-05-20 17:14:48 +00:00
|
|
|
from django.contrib.postgres.fields import ArrayField
|
|
|
|
from django.db import models
|
|
|
|
from django.dispatch import receiver
|
2022-10-31 08:20:43 +00:00
|
|
|
from django.contrib.auth import get_user_model
|
2022-05-20 17:14:48 +00:00
|
|
|
|
2022-06-21 13:48:07 +00:00
|
|
|
from model_utils import FieldTracker
|
|
|
|
|
|
|
|
from . import tasks
|
|
|
|
|
2022-06-15 10:05:29 +00:00
|
|
|
from colorfield.fields import ColorField
|
|
|
|
|
2022-05-20 17:14:48 +00:00
|
|
|
from datetime import date
|
|
|
|
|
|
|
|
import os
|
|
|
|
import uuid
|
|
|
|
|
|
|
|
|
2022-10-31 08:20:43 +00:00
|
|
|
UserModel = get_user_model()
|
|
|
|
|
|
|
|
|
2022-06-15 10:05:29 +00:00
|
|
|
class PhotoTag(models.Model):
|
|
|
|
name = models.CharField(unique=True, null=False, blank=False, max_length=100)
|
|
|
|
color = ColorField(default='#FFE5B4')
|
|
|
|
|
|
|
|
def __str__(self):
|
|
|
|
return self.name
|
|
|
|
|
2022-10-31 08:20:43 +00:00
|
|
|
class Meta:
|
|
|
|
ordering = ('name',)
|
|
|
|
|
2022-06-15 10:05:29 +00:00
|
|
|
|
2022-05-20 17:14:48 +00:00
|
|
|
class PhotoGroup(models.Model):
|
2022-10-31 08:20:43 +00:00
|
|
|
name = models.CharField(unique=False, null=False, max_length=200)
|
2022-05-20 17:14:48 +00:00
|
|
|
date = models.DateField(null=True)
|
2022-10-31 08:20:43 +00:00
|
|
|
parent = models.ForeignKey('self', blank=True, null=True, on_delete=models.SET_NULL)
|
|
|
|
|
|
|
|
owner = models.ForeignKey(
|
|
|
|
UserModel,
|
|
|
|
on_delete=models.CASCADE,
|
|
|
|
related_name='photogroups',
|
|
|
|
)
|
2022-05-20 17:14:48 +00:00
|
|
|
|
|
|
|
def __str__(self):
|
|
|
|
return self.name
|
|
|
|
|
|
|
|
|
|
|
|
def get_default_photogroup():
|
|
|
|
return PhotoGroup.objects.get_or_create(
|
|
|
|
name="Unsortiert",
|
|
|
|
date=None,
|
|
|
|
)
|
|
|
|
|
|
|
|
def get_original_photo_path(instance, filename):
|
|
|
|
_, ext = os.path.splitext(filename)
|
2022-06-21 13:48:07 +00:00
|
|
|
return 'original_images/%s%s' % (str(uuid.uuid4()), ext)
|
2022-05-20 17:14:48 +00:00
|
|
|
|
|
|
|
def get_cropped_photo_path(instance, filename):
|
|
|
|
_, ext = os.path.splitext(filename)
|
2022-06-21 13:48:07 +00:00
|
|
|
return 'cropped_images/%s%s' % (str(uuid.uuid4()), ext)
|
2022-05-20 17:14:48 +00:00
|
|
|
|
|
|
|
|
|
|
|
class Photo(models.Model):
|
|
|
|
legacy_id = models.IntegerField(unique=True, blank=True, null=True)
|
|
|
|
original_image = models.ImageField(upload_to=get_original_photo_path, null=False, blank=False)
|
|
|
|
cropped_image = models.ImageField(upload_to=get_cropped_photo_path, null=True, blank=True)
|
|
|
|
bbox_coords = ArrayField(
|
|
|
|
ArrayField(
|
|
|
|
models.IntegerField(unique=False, blank=False, null=True, default=None),
|
|
|
|
size=2
|
|
|
|
),
|
|
|
|
blank=True,
|
|
|
|
null=True,
|
|
|
|
default=None,
|
|
|
|
size=4,
|
|
|
|
)
|
|
|
|
rotate = models.FloatField(blank=True, null=True)
|
|
|
|
intersections = ArrayField(
|
|
|
|
ArrayField(
|
|
|
|
models.IntegerField(unique=False, blank=False, null=True, default=None),
|
|
|
|
size=2
|
|
|
|
),
|
|
|
|
blank=True,
|
|
|
|
null=True,
|
|
|
|
default=None
|
|
|
|
)
|
|
|
|
group = models.ForeignKey(
|
|
|
|
PhotoGroup,
|
|
|
|
null=False,
|
|
|
|
blank=True,
|
|
|
|
on_delete=models.SET_DEFAULT,
|
|
|
|
default=get_default_photogroup
|
|
|
|
)
|
|
|
|
ocr_text = models.CharField(max_length=200, null=True, blank=True)
|
2022-06-15 10:05:29 +00:00
|
|
|
tag = models.ForeignKey(
|
|
|
|
PhotoTag,
|
|
|
|
null=True,
|
|
|
|
blank=False,
|
|
|
|
on_delete=models.SET_NULL,
|
|
|
|
default=None,
|
|
|
|
)
|
2022-05-20 17:14:48 +00:00
|
|
|
|
2022-10-31 08:20:43 +00:00
|
|
|
owner = models.ForeignKey(
|
|
|
|
UserModel,
|
|
|
|
on_delete=models.CASCADE,
|
|
|
|
related_name='photos',
|
|
|
|
)
|
|
|
|
|
2022-06-21 13:48:07 +00:00
|
|
|
tracker = FieldTracker()
|
|
|
|
|
2022-05-20 17:14:48 +00:00
|
|
|
def __str__(self):
|
|
|
|
return "Photo #" + str(self.id)
|
|
|
|
|
|
|
|
|
|
|
|
@receiver(models.signals.post_delete, sender=Photo)
|
|
|
|
def auto_delete_file_on_delete(sender, instance, **kwargs):
|
|
|
|
if instance.original_image:
|
2022-06-21 13:48:07 +00:00
|
|
|
instance.original_image.delete(save=False)
|
2022-05-20 17:14:48 +00:00
|
|
|
|
|
|
|
if instance.cropped_image:
|
2022-06-21 13:48:07 +00:00
|
|
|
instance.cropped_image.delete(save=False)
|
2022-05-20 17:14:48 +00:00
|
|
|
|
|
|
|
|
|
|
|
@receiver(models.signals.pre_save, sender=Photo)
|
|
|
|
def auto_delete_file_on_change(sender, instance, **kwargs):
|
|
|
|
if not instance.pk:
|
|
|
|
return False
|
|
|
|
|
|
|
|
try:
|
2022-06-21 13:48:07 +00:00
|
|
|
photo = Photo.objects.get(pk=instance.pk)
|
2022-05-20 17:14:48 +00:00
|
|
|
except Photo.DoesNotExist:
|
|
|
|
return False
|
|
|
|
|
2022-06-21 13:48:07 +00:00
|
|
|
# check original image for change
|
2022-05-20 17:14:48 +00:00
|
|
|
new_file = instance.original_image
|
2022-06-21 13:48:07 +00:00
|
|
|
old_file = photo.original_image
|
2022-05-20 17:14:48 +00:00
|
|
|
if not old_file == new_file:
|
2022-06-21 13:48:07 +00:00
|
|
|
photo.original_image.delete(save=False)
|
|
|
|
|
|
|
|
# check cropped image for change
|
|
|
|
new_file = instance.cropped_image
|
|
|
|
old_file = photo.cropped_image
|
|
|
|
if not old_file == new_file:
|
|
|
|
photo.cropped_image.delete(save=False)
|
2022-05-20 17:14:48 +00:00
|
|
|
|
|
|
|
|
2022-06-21 13:48:07 +00:00
|
|
|
@receiver(models.signals.post_save, sender=Photo)
|
|
|
|
def max_resize_images(sender, instance, **kwargs):
|
|
|
|
MAX_IMAGE_WIDTH = 800
|
|
|
|
|
|
|
|
tracker = instance.tracker
|
|
|
|
|
|
|
|
original_image_s3_path = instance.original_image.name
|
|
|
|
if original_image_s3_path and tracker.has_changed('original_image'):
|
|
|
|
tasks.max_resize_image_chain(original_image_s3_path, MAX_IMAGE_WIDTH)
|
|
|
|
|
|
|
|
cropped_image_s3_path = instance.cropped_image.name
|
|
|
|
if cropped_image_s3_path and tracker.has_changed('cropped_image'):
|
|
|
|
tasks.max_resize_image_chain(cropped_image_s3_path, MAX_IMAGE_WIDTH)
|
2022-05-20 17:14:48 +00:00
|
|
|
|
|
|
|
|
|
|
|
def get_empty_photolog_default():
|
|
|
|
return []
|
|
|
|
|
|
|
|
def get_photolog_pdf_path(instance, filename):
|
2022-06-21 13:48:07 +00:00
|
|
|
return "photolog_pdf/%s.pdf" % str(uuid.uuid4())
|
2022-05-20 17:14:48 +00:00
|
|
|
|
|
|
|
class PhotoLog(models.Model):
|
|
|
|
title = models.CharField(null=False, blank=False, max_length=5000)
|
|
|
|
date = models.DateField(auto_now=False, default=date.today)
|
|
|
|
render_date = models.BooleanField(null=False, blank=True, default=True)
|
|
|
|
start_slide_image = models.IntegerField(null=True, blank=True, default=3)
|
|
|
|
slides = ArrayField(
|
|
|
|
ArrayField(
|
|
|
|
models.IntegerField(blank=False, null=True),
|
|
|
|
null=True,
|
|
|
|
blank=True,
|
|
|
|
size=3
|
|
|
|
),
|
|
|
|
null=False,
|
|
|
|
blank=True,
|
|
|
|
default=get_empty_photolog_default
|
|
|
|
)
|
|
|
|
pdf = models.FileField(upload_to=get_photolog_pdf_path, null=True, blank=True)
|
|
|
|
|
|
|
|
def __str__(self):
|
|
|
|
return self.title
|
|
|
|
|
|
|
|
|
2022-06-21 13:48:07 +00:00
|
|
|
class PhotoLogTemplate(models.Model):
|
|
|
|
title = models.CharField(null=False, blank=False, max_length=5000)
|
|
|
|
date = models.DateField(auto_now=False, default=date.today)
|
|
|
|
render_date = models.BooleanField(null=False, blank=True, default=True)
|
|
|
|
start_slide_image = models.IntegerField(null=True, blank=True, default=3)
|
|
|
|
slides = models.JSONField(
|
|
|
|
null=False,
|
|
|
|
blank=False,
|
|
|
|
default=list,
|
|
|
|
)
|
2022-05-20 17:14:48 +00:00
|
|
|
|
2022-06-21 13:48:07 +00:00
|
|
|
def __str__(self):
|
|
|
|
return self.title
|
2022-05-20 17:14:48 +00:00
|
|
|
|
|
|
|
|
2022-06-21 13:48:07 +00:00
|
|
|
@receiver(models.signals.post_delete, sender=PhotoLog)
|
|
|
|
def auto_delete_file_on_delete(sender, instance, **kwargs):
|
|
|
|
if instance.pdf:
|
|
|
|
instance.pdf.delete(save=False)
|