2008-08-10 22:55:25 +00:00
|
|
|
import datetime
|
2010-06-26 17:01:22 +00:00
|
|
|
import os
|
2013-03-22 20:55:28 +00:00
|
|
|
import hashlib
|
2013-08-01 12:48:43 +00:00
|
|
|
from PIL import Image
|
2008-08-01 09:27:59 +00:00
|
|
|
|
|
|
|
|
from django.db import models
|
2013-08-14 14:18:52 +00:00
|
|
|
from django.core.files import File
|
2008-10-28 06:39:21 +00:00
|
|
|
from django.core.files.base import ContentFile
|
2012-04-20 15:45:56 +00:00
|
|
|
from django.core.files.storage import get_storage_class
|
2008-08-10 22:55:25 +00:00
|
|
|
from django.utils.translation import ugettext as _
|
2013-08-01 11:19:56 +00:00
|
|
|
from django.utils import six
|
2010-06-26 17:00:31 +00:00
|
|
|
from django.db.models import signals
|
2008-10-28 06:39:21 +00:00
|
|
|
|
2013-09-13 15:59:52 +00:00
|
|
|
from avatar.conf import settings
|
|
|
|
|
from avatar.util import get_username, force_bytes, invalidate_cache
|
2010-02-22 23:52:37 +00:00
|
|
|
|
2012-01-24 15:44:25 +00:00
|
|
|
try:
|
|
|
|
|
from django.utils.timezone import now
|
|
|
|
|
except ImportError:
|
|
|
|
|
now = datetime.datetime.now
|
|
|
|
|
|
2012-04-20 15:45:56 +00:00
|
|
|
|
2013-09-13 15:59:52 +00:00
|
|
|
avatar_storage = get_storage_class(settings.AVATAR_STORAGE)()
|
2008-08-10 22:55:25 +00:00
|
|
|
|
2010-06-24 15:56:04 +00:00
|
|
|
|
2010-02-01 18:11:32 +00:00
|
|
|
def avatar_file_path(instance=None, filename=None, size=None, ext=None):
|
2013-09-13 15:59:52 +00:00
|
|
|
tmppath = [settings.AVATAR_STORAGE_DIR]
|
2014-08-19 10:34:43 +00:00
|
|
|
userdirname = get_username(instance.user)
|
|
|
|
|
if settings.AVATAR_USERID_AS_USERDIRNAME:
|
|
|
|
|
userdirname = str(instance.user_id)
|
2014-08-14 16:25:43 +00:00
|
|
|
if settings.AVATAR_HASH_USERDIRNAMES:
|
2014-08-19 10:34:43 +00:00
|
|
|
tmp = hashlib.md5(userdirname).hexdigest()
|
|
|
|
|
tmppath.extend([tmp[0], tmp[1], userdirname])
|
2010-01-25 16:59:23 +00:00
|
|
|
else:
|
2014-08-19 10:34:43 +00:00
|
|
|
tmppath.append(userdirname)
|
2010-01-25 16:59:23 +00:00
|
|
|
if not filename:
|
|
|
|
|
# Filename already stored in database
|
|
|
|
|
filename = instance.avatar.name
|
2013-09-13 15:59:52 +00:00
|
|
|
if ext and settings.AVATAR_HASH_FILENAMES:
|
2010-02-01 18:11:32 +00:00
|
|
|
# An extension was provided, probably because the thumbnail
|
|
|
|
|
# is in a different format than the file. Use it. Because it's
|
|
|
|
|
# only enabled if AVATAR_HASH_FILENAMES is true, we can trust
|
|
|
|
|
# it won't conflict with another filename
|
|
|
|
|
(root, oldext) = os.path.splitext(filename)
|
|
|
|
|
filename = root + "." + ext
|
2010-01-25 16:59:23 +00:00
|
|
|
else:
|
|
|
|
|
# File doesn't exist yet
|
2013-09-13 15:59:52 +00:00
|
|
|
if settings.AVATAR_HASH_FILENAMES:
|
2010-01-25 16:59:23 +00:00
|
|
|
(root, ext) = os.path.splitext(filename)
|
2013-08-01 12:32:03 +00:00
|
|
|
filename = hashlib.md5(force_bytes(filename)).hexdigest()
|
2010-01-25 16:59:23 +00:00
|
|
|
filename = filename + ext
|
2010-01-25 15:56:40 +00:00
|
|
|
if size:
|
2010-01-25 16:59:23 +00:00
|
|
|
tmppath.extend(['resized', str(size)])
|
|
|
|
|
tmppath.append(os.path.basename(filename))
|
|
|
|
|
return os.path.join(*tmppath)
|
2008-08-01 09:27:59 +00:00
|
|
|
|
2013-03-04 10:43:23 +00:00
|
|
|
|
2010-02-01 18:11:32 +00:00
|
|
|
def find_extension(format):
|
|
|
|
|
format = format.lower()
|
|
|
|
|
|
|
|
|
|
if format == 'jpeg':
|
|
|
|
|
format = 'jpg'
|
|
|
|
|
|
|
|
|
|
return format
|
|
|
|
|
|
2013-03-04 10:43:23 +00:00
|
|
|
|
2008-08-01 09:27:59 +00:00
|
|
|
class Avatar(models.Model):
|
2013-08-01 11:12:49 +00:00
|
|
|
user = models.ForeignKey(getattr(settings, 'AUTH_USER_MODEL', 'auth.User'))
|
2008-08-10 22:55:25 +00:00
|
|
|
primary = models.BooleanField(default=False)
|
2012-04-20 15:45:56 +00:00
|
|
|
avatar = models.ImageField(max_length=1024,
|
2013-03-04 10:43:23 +00:00
|
|
|
upload_to=avatar_file_path,
|
|
|
|
|
storage=avatar_storage,
|
|
|
|
|
blank=True)
|
2012-01-24 15:44:25 +00:00
|
|
|
date_uploaded = models.DateTimeField(default=now)
|
2012-10-05 10:07:47 +00:00
|
|
|
|
2008-08-01 09:27:59 +00:00
|
|
|
def __unicode__(self):
|
2013-08-01 11:19:56 +00:00
|
|
|
return _(six.u('Avatar for %s')) % self.user
|
2012-10-05 10:07:47 +00:00
|
|
|
|
2010-06-24 15:56:04 +00:00
|
|
|
def save(self, *args, **kwargs):
|
|
|
|
|
avatars = Avatar.objects.filter(user=self.user)
|
|
|
|
|
if self.pk:
|
|
|
|
|
avatars = avatars.exclude(pk=self.pk)
|
2013-09-13 15:59:52 +00:00
|
|
|
if settings.AVATAR_MAX_AVATARS_PER_USER > 1:
|
2010-01-25 17:16:33 +00:00
|
|
|
if self.primary:
|
2010-01-22 18:26:36 +00:00
|
|
|
avatars = avatars.filter(primary=True)
|
|
|
|
|
avatars.update(primary=False)
|
|
|
|
|
else:
|
|
|
|
|
avatars.delete()
|
2010-06-24 15:56:04 +00:00
|
|
|
super(Avatar, self).save(*args, **kwargs)
|
2012-10-05 10:07:47 +00:00
|
|
|
|
2008-08-10 22:55:25 +00:00
|
|
|
def thumbnail_exists(self, size):
|
2008-10-28 06:39:21 +00:00
|
|
|
return self.avatar.storage.exists(self.avatar_name(size))
|
2012-10-05 10:07:47 +00:00
|
|
|
|
2014-03-22 12:35:13 +00:00
|
|
|
def transpose_image(self, image):
|
|
|
|
|
"""
|
|
|
|
|
Transpose based on EXIF information.
|
|
|
|
|
Borrowed from django-imagekit:
|
|
|
|
|
imagekit.processors.Transpose
|
|
|
|
|
"""
|
|
|
|
|
EXIF_ORIENTATION_STEPS = {
|
|
|
|
|
1: [],
|
|
|
|
|
2: ['FLIP_LEFT_RIGHT'],
|
|
|
|
|
3: ['ROTATE_180'],
|
|
|
|
|
4: ['FLIP_TOP_BOTTOM'],
|
|
|
|
|
5: ['ROTATE_270', 'FLIP_LEFT_RIGHT'],
|
|
|
|
|
6: ['ROTATE_270'],
|
|
|
|
|
7: ['ROTATE_90', 'FLIP_LEFT_RIGHT'],
|
|
|
|
|
8: ['ROTATE_90'],
|
|
|
|
|
}
|
|
|
|
|
try:
|
|
|
|
|
orientation = image._getexif()[0x0112]
|
|
|
|
|
ops = EXIF_ORIENTATION_STEPS[orientation]
|
|
|
|
|
except:
|
|
|
|
|
ops = []
|
|
|
|
|
for method in ops:
|
|
|
|
|
image = image.transpose(getattr(Image, method))
|
|
|
|
|
return image
|
|
|
|
|
|
2010-03-16 13:56:15 +00:00
|
|
|
def create_thumbnail(self, size, quality=None):
|
2010-06-26 14:49:35 +00:00
|
|
|
# invalidate the cache of the thumbnail with the given size first
|
|
|
|
|
invalidate_cache(self.user, size)
|
2008-08-10 22:55:25 +00:00
|
|
|
try:
|
2013-08-01 12:09:04 +00:00
|
|
|
orig = self.avatar.storage.open(self.avatar.name, 'rb')
|
|
|
|
|
image = Image.open(orig)
|
2014-03-22 12:40:56 +00:00
|
|
|
image = self.transpose_image(image)
|
2013-09-13 15:59:52 +00:00
|
|
|
quality = quality or settings.AVATAR_THUMB_QUALITY
|
2013-08-01 12:32:03 +00:00
|
|
|
w, h = image.size
|
2012-10-05 10:07:13 +00:00
|
|
|
if w != size or h != size:
|
|
|
|
|
if w > h:
|
2013-08-01 12:32:03 +00:00
|
|
|
diff = int((w - h) / 2)
|
2012-10-05 10:07:13 +00:00
|
|
|
image = image.crop((diff, 0, w - diff, h))
|
|
|
|
|
else:
|
2013-08-01 12:32:03 +00:00
|
|
|
diff = int((h - w) / 2)
|
2012-10-05 10:07:13 +00:00
|
|
|
image = image.crop((0, diff, w, h - diff))
|
2014-01-10 09:35:07 +00:00
|
|
|
if image.mode not in ("RGB", "RGBA"):
|
2012-10-05 10:07:13 +00:00
|
|
|
image = image.convert("RGB")
|
2013-09-13 15:59:52 +00:00
|
|
|
image = image.resize((size, size), settings.AVATAR_RESIZE_METHOD)
|
2013-08-01 12:32:03 +00:00
|
|
|
thumb = six.BytesIO()
|
2013-09-13 15:59:52 +00:00
|
|
|
image.save(thumb, settings.AVATAR_THUMB_FORMAT, quality=quality)
|
2012-10-05 10:07:13 +00:00
|
|
|
thumb_file = ContentFile(thumb.getvalue())
|
|
|
|
|
else:
|
2013-08-14 14:18:52 +00:00
|
|
|
thumb_file = File(orig)
|
2012-10-05 10:07:13 +00:00
|
|
|
thumb = self.avatar.storage.save(self.avatar_name(size), thumb_file)
|
2008-08-10 22:55:25 +00:00
|
|
|
except IOError:
|
2012-10-05 10:07:47 +00:00
|
|
|
return # What should we do here? Render a "sorry, didn't work" img?
|
2010-06-26 14:49:35 +00:00
|
|
|
|
2008-08-10 22:55:25 +00:00
|
|
|
def avatar_url(self, size):
|
2008-10-28 06:39:21 +00:00
|
|
|
return self.avatar.storage.url(self.avatar_name(size))
|
2012-10-05 10:07:47 +00:00
|
|
|
|
2011-05-17 15:19:28 +00:00
|
|
|
def get_absolute_url(self):
|
2013-09-13 15:59:52 +00:00
|
|
|
return self.avatar_url(settings.AVATAR_DEFAULT_SIZE)
|
2011-05-17 15:19:28 +00:00
|
|
|
|
2008-10-28 06:39:21 +00:00
|
|
|
def avatar_name(self, size):
|
2013-09-13 15:59:52 +00:00
|
|
|
ext = find_extension(settings.AVATAR_THUMB_FORMAT)
|
2010-01-25 15:56:40 +00:00
|
|
|
return avatar_file_path(
|
2010-01-25 16:59:23 +00:00
|
|
|
instance=self,
|
2010-02-01 18:11:32 +00:00
|
|
|
size=size,
|
|
|
|
|
ext=ext
|
2010-01-25 15:56:40 +00:00
|
|
|
)
|
2010-06-26 17:00:31 +00:00
|
|
|
|
|
|
|
|
|
2012-10-05 13:52:02 +00:00
|
|
|
def invalidate_avatar_cache(sender, instance, **kwargs):
|
2013-09-09 23:19:29 +00:00
|
|
|
if hasattr(instance, 'user'):
|
|
|
|
|
invalidate_cache(instance.user)
|
2012-10-05 13:52:02 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
def create_default_thumbnails(sender, instance, created=False, **kwargs):
|
|
|
|
|
invalidate_avatar_cache(sender, instance)
|
2010-06-26 17:00:31 +00:00
|
|
|
if created:
|
2013-09-13 15:59:52 +00:00
|
|
|
for size in settings.AVATAR_AUTO_GENERATE_SIZES:
|
2010-06-26 17:00:31 +00:00
|
|
|
instance.create_thumbnail(size)
|
|
|
|
|
|
2012-10-05 13:52:02 +00:00
|
|
|
|
2013-01-19 01:39:31 +00:00
|
|
|
def remove_avatar_images(instance=None, **kwargs):
|
2013-09-09 23:19:29 +00:00
|
|
|
if hasattr(instance, 'user'):
|
2014-07-19 15:10:04 +00:00
|
|
|
for size in settings.AVATAR_AUTO_GENERATE_SIZES:
|
2013-09-09 23:19:29 +00:00
|
|
|
if instance.thumbnail_exists(size):
|
|
|
|
|
instance.avatar.storage.delete(instance.avatar_name(size))
|
|
|
|
|
instance.avatar.storage.delete(instance.avatar.name)
|
2013-01-19 01:39:31 +00:00
|
|
|
|
|
|
|
|
|
2010-06-26 17:00:31 +00:00
|
|
|
signals.post_save.connect(create_default_thumbnails, sender=Avatar)
|
2012-10-05 13:52:02 +00:00
|
|
|
signals.post_delete.connect(invalidate_avatar_cache, sender=Avatar)
|
2013-01-19 01:39:31 +00:00
|
|
|
|
2013-09-13 15:59:52 +00:00
|
|
|
if settings.AVATAR_CLEANUP_DELETED:
|
2013-01-19 01:39:31 +00:00
|
|
|
signals.post_delete.connect(remove_avatar_images, sender=Avatar)
|