X-Git-Url: https://git.mdrn.pl/wolnelektury.git/blobdiff_plain/6d42bc478e3d1bd90eb294464748c21e4de0fc63..29b00497f9103bb31f9e236bdf5844b9a6fa79ea:/src/social/models.py
diff --git a/src/social/models.py b/src/social/models.py
index 29029ee6b..862db4cdd 100644
--- a/src/social/models.py
+++ b/src/social/models.py
@@ -1,24 +1,31 @@
-# This file is part of Wolnelektury, licensed under GNU Affero GPLv3 or later.
-# Copyright © Fundacja Nowoczesna Polska. See NOTICE for more information.
+# This file is part of Wolne Lektury, licensed under GNU Affero GPLv3 or later.
+# Copyright © Fundacja Wolne Lektury. See NOTICE for more information.
#
+from datetime import datetime
+import uuid
+from oauthlib.common import urlencode, generate_token
from random import randint
from django.db import models
from django.conf import settings
+from django.contrib.auth.models import User
from django.core.exceptions import ValidationError
+from django.core.mail import send_mail
from django.urls import reverse
-from django.utils.translation import ugettext_lazy as _
+from django.utils.timezone import now, utc
from catalogue.models import Book
+from catalogue.utils import get_random_hash
from wolnelektury.utils import cached_render, clear_cached_renders
+from .syncable import Syncable
class BannerGroup(models.Model):
- name = models.CharField(_('name'), max_length=255, unique=True)
- created_at = models.DateTimeField(_('created at'), auto_now_add=True)
+ name = models.CharField('nazwa', max_length=255, unique=True)
+ created_at = models.DateTimeField('utworzona', auto_now_add=True)
class Meta:
ordering = ('name',)
- verbose_name = _('banner group')
- verbose_name_plural = _('banner groups')
+ verbose_name = 'grupa bannerów'
+ verbose_name_plural = 'grupy bannerów'
def __str__(self):
return self.name
@@ -36,49 +43,42 @@ class BannerGroup(models.Model):
class Cite(models.Model):
- book = models.ForeignKey(Book, models.CASCADE, verbose_name=_('book'), null=True, blank=True)
- text = models.TextField(_('text'), blank=True)
- small = models.BooleanField(_('small'), default=False, help_text=_('Make this cite display smaller.'))
- vip = models.CharField(_('VIP'), max_length=128, null=True, blank=True)
- link = models.URLField(_('link'))
- video = models.URLField(_('video'), blank=True)
- picture = models.ImageField(_('picture'), blank=True,
+ book = models.ForeignKey(Book, models.CASCADE, verbose_name='ksiÄ
żka', null=True, blank=True)
+ text = models.TextField('tekst', blank=True)
+ small = models.BooleanField(
+ 'maÅy', default=False, help_text='Sprawia, że cytat wyÅwietla siÄ mniejszym fontem.')
+ vip = models.CharField('VIP', max_length=128, null=True, blank=True)
+ link = models.URLField('odnoÅnik')
+ video = models.URLField('wideo', blank=True)
+ picture = models.ImageField('ilustracja', blank=True,
help_text='Najlepsze wymiary: 975 x 315 z tekstem, 487 x 315 bez tekstu.')
- picture_alt = models.CharField(_('picture alternative text'), max_length=255, blank=True)
- picture_title = models.CharField(_('picture title'), max_length=255, null=True, blank=True)
- picture_author = models.CharField(_('picture author'), max_length=255, blank=True, null=True)
- picture_link = models.URLField(_('picture link'), blank=True, null=True)
- picture_license = models.CharField(_('picture license name'), max_length=255, blank=True, null=True)
- picture_license_link = models.URLField(_('picture license link'), blank=True, null=True)
-
- sticky = models.BooleanField(_('sticky'), default=False, db_index=True,
- help_text=_('Sticky cites will take precedense.'))
- banner = models.BooleanField(_('banner'), default=False, help_text=
- 'Dostosuj wielkoÅÄ do obrazu tÅa, zignoruj tekst.'
- '
(PrzestarzaÅe; użyj funkcji "Obraz" w sekcji "Media box")')
-
- background_plain = models.BooleanField(_('plain background'), default=False)
- background_color = models.CharField(_('background color'), max_length=32, blank=True)
+ picture_alt = models.CharField('alternatywny tekst ilustracji', max_length=255, blank=True)
+ picture_title = models.CharField('tytuÅ ilustracji', max_length=255, null=True, blank=True)
+ picture_author = models.CharField('autor ilustracji', max_length=255, blank=True, null=True)
+ picture_link = models.URLField('link ilustracji', blank=True, null=True)
+ picture_license = models.CharField('nazwa licencji ilustracji', max_length=255, blank=True, null=True)
+ picture_license_link = models.URLField('adres licencji ilustracji', blank=True, null=True)
+
+ sticky = models.BooleanField('przyklejony', default=False, db_index=True,
+ help_text='Przyklejone cytaty majÄ
pierwszeÅstwo.')
+ background_plain = models.BooleanField('jednobarwne tÅo', default=False)
+ background_color = models.CharField('kolor tÅa', max_length=32, blank=True)
image = models.ImageField(
- _('image'), upload_to='social/cite', null=True, blank=True,
- help_text=_('Best image is exactly 975px wide and weights under 100kB.'))
- image_shift = models.IntegerField(
- _('shift'), null=True, blank=True,
- help_text='PrzesuniÄcie w pionie, w procentach. 0 to wyrównanie do górnej krawÄdzi, 100 do dolnej. DomyÅlne jest 50%.'
- '
(PrzestarzaÅe; użyj obrazka o wÅaÅciwych proporcjach;)')
- image_title = models.CharField(_('title'), max_length=255, null=True, blank=True)
- image_author = models.CharField(_('author'), max_length=255, blank=True, null=True)
- image_link = models.URLField(_('link'), blank=True, null=True)
- image_license = models.CharField(_('license name'), max_length=255, blank=True, null=True)
- image_license_link = models.URLField(_('license link'), blank=True, null=True)
-
- created_at = models.DateTimeField(_('created at'), auto_now_add=True)
- group = models.ForeignKey(BannerGroup, verbose_name=_('group'), null=True, blank=True, on_delete=models.SET_NULL)
+ 'obraz tÅa', upload_to='social/cite', null=True, blank=True,
+ help_text='Najlepsze tÅo ma wielkoÅÄ 975 x 315 px i waży poniżej 100kB.')
+ image_title = models.CharField('tytuÅ obrazu tÅa', max_length=255, null=True, blank=True)
+ image_author = models.CharField('autor obrazu tÅa', max_length=255, blank=True, null=True)
+ image_link = models.URLField('link obrazu tÅa', blank=True, null=True)
+ image_license = models.CharField('nazwa licencji obrazu tÅa', max_length=255, blank=True, null=True)
+ image_license_link = models.URLField('adres licencji obrazu tÅa', blank=True, null=True)
+
+ created_at = models.DateTimeField('utworzony', auto_now_add=True)
+ group = models.ForeignKey(BannerGroup, verbose_name='grupa', null=True, blank=True, on_delete=models.SET_NULL)
class Meta:
ordering = ('vip', 'text')
- verbose_name = _('banner')
- verbose_name_plural = _('banners')
+ verbose_name = 'banner'
+ verbose_name_plural = 'bannery'
def __str__(self):
t = []
@@ -106,9 +106,6 @@ class Cite(models.Model):
return self.vip or self.text or self.book
def layout(self):
- if self.banner:
- # TODO: move all banners to pictures.
- return 'banner'
pieces = []
if self.has_box():
pieces.append('box')
@@ -134,36 +131,312 @@ class Cite(models.Model):
class Carousel(models.Model):
- slug = models.SlugField(_('slug'), unique=True)
+ placement = models.SlugField('miejsce', choices=[
+ ('main', 'main'),
+ ('main_2022', 'main 2022'),
+ ])
+ priority = models.SmallIntegerField('priorytet', default=0)
+ language = models.CharField('jÄzyk', max_length=2, blank=True, default='', choices=settings.LANGUAGES)
class Meta:
- ordering = ('slug',)
- verbose_name = _('carousel')
- verbose_name_plural = _('carousels')
+# ordering = ('placement', '-priority')
+ verbose_name = 'karuzela'
+ verbose_name_plural = 'karuzele'
def __str__(self):
- return self.slug
+ return self.placement
+
+ @classmethod
+ def get(cls, placement):
+ carousel = cls.objects.filter(placement=placement).order_by('-priority', '?').first()
+ if carousel is None:
+ carousel = cls.objects.create(placement=placement)
+ return carousel
+
class CarouselItem(models.Model):
- order = models.PositiveSmallIntegerField(_('order'), unique=True)
- carousel = models.ForeignKey(Carousel, models.CASCADE, verbose_name=_('carousel'))
- banner = models.ForeignKey(Cite, models.CASCADE, null=True, blank=True, verbose_name=_('banner'))
- banner_group = models.ForeignKey(BannerGroup, models.CASCADE, null=True, blank=True, verbose_name=_('banner group'))
+ order = models.PositiveSmallIntegerField('kolejnoÅÄ')
+ carousel = models.ForeignKey(Carousel, models.CASCADE, verbose_name='karuzela')
+ banner = models.ForeignKey(
+ Cite, models.CASCADE, null=True, blank=True, verbose_name='banner')
+ banner_group = models.ForeignKey(
+ BannerGroup, models.CASCADE, null=True, blank=True, verbose_name='grupa bannerów')
class Meta:
ordering = ('order',)
- unique_together = [('carousel', 'order')]
- verbose_name = _('carousel item')
- verbose_name_plural = _('carousel items')
+ verbose_name = 'element karuzeli'
+ verbose_name_plural = 'elementy karuzeli'
def __str__(self):
return str(self.banner or self.banner_group)
def clean(self):
if not self.banner and not self.banner_group:
- raise ValidationError(_('Either banner or banner group is required.'))
+ raise ValidationError('ProszÄ wskazaÄ banner albo grupÄ bannerów.')
elif self.banner and self.banner_group:
- raise ValidationError(_('Either banner or banner group is required.'))
+ raise ValidationError('ProszÄ wskazaÄ banner albo grupÄ bannerów.')
def get_banner(self):
return self.banner or self.banner_group.get_banner()
+
+
+class UserProfile(models.Model):
+ user = models.OneToOneField(User, models.CASCADE)
+ notifications = models.BooleanField(default=False)
+
+ @classmethod
+ def get_for(cls, user):
+ obj, created = cls.objects.get_or_create(user=user)
+ return obj
+
+
+class UserConfirmation(models.Model):
+ user = models.ForeignKey(User, models.CASCADE)
+ created_at = models.DateTimeField(auto_now_add=True)
+ key = models.CharField(max_length=128, unique=True)
+
+ def send(self):
+ send_mail(
+ 'Potwierdź konto w bibliotece Wolne Lektury',
+ f'https://beta.wolnelektury.pl/ludzie/potwierdz/{self.key}/',
+ settings.CONTACT_EMAIL,
+ [self.user.email]
+ )
+
+ def use(self):
+ user = self.user
+ user.is_active = True
+ user.save()
+ self.delete()
+
+ @classmethod
+ def request(cls, user):
+ cls.objects.create(
+ user=user,
+ key=generate_token()
+ ).send()
+
+
+class Progress(Syncable, models.Model):
+ user = models.ForeignKey(User, models.CASCADE)
+ book = models.ForeignKey('catalogue.Book', models.CASCADE)
+ created_at = models.DateTimeField(auto_now_add=True)
+ updated_at = models.DateTimeField(auto_now=True)
+ reported_timestamp = models.DateTimeField()
+ deleted = models.BooleanField(default=False)
+ last_mode = models.CharField(max_length=64, choices=[
+ ('text', 'text'),
+ ('audio', 'audio'),
+ ])
+ text_percent = models.FloatField(null=True, blank=True)
+ text_anchor = models.CharField(max_length=64, blank=True)
+ audio_percent = models.FloatField(null=True, blank=True)
+ audio_timestamp = models.FloatField(null=True, blank=True)
+ implicit_text_percent = models.FloatField(null=True, blank=True)
+ implicit_text_anchor = models.CharField(max_length=64, blank=True)
+ implicit_audio_percent = models.FloatField(null=True, blank=True)
+ implicit_audio_timestamp = models.FloatField(null=True, blank=True)
+
+ syncable_fields = [
+ 'deleted',
+ 'last_mode', 'text_anchor', 'audio_timestamp'
+ ]
+
+ class Meta:
+ unique_together = [('user', 'book')]
+
+ @property
+ def timestamp(self):
+ return self.updated_at.timestamp()
+
+ @classmethod
+ def create_from_data(cls, user, data):
+ return cls.objects.create(
+ user=user,
+ book=data['book'],
+ reported_timestamp=now(),
+ )
+
+ def save(self, *args, **kwargs):
+ try:
+ audio_l = self.book.get_audio_length()
+ except:
+ audio_l = 60
+ if self.text_anchor:
+ self.text_percent = 33
+ if audio_l:
+ self.implicit_audio_percent = 40
+ self.implicit_audio_timestamp = audio_l * .4
+ if self.audio_timestamp:
+ if self.audio_timestamp > audio_l:
+ self.audio_timestamp = audio_l
+ if audio_l:
+ self.audio_percent = 100 * self.audio_timestamp / audio_l
+ self.implicit_text_percent = 60
+ self.implicit_text_anchor = 'f20'
+ return super().save(*args, **kwargs)
+
+
+class UserList(Syncable, models.Model):
+ slug = models.SlugField(unique=True)
+ user = models.ForeignKey(User, models.CASCADE)
+ name = models.CharField(max_length=1024)
+ favorites = models.BooleanField(default=False)
+ public = models.BooleanField(default=False)
+ deleted = models.BooleanField(default=False)
+ created_at = models.DateTimeField(auto_now_add=True)
+ updated_at = models.DateTimeField(auto_now=True)
+ reported_timestamp = models.DateTimeField()
+
+ syncable_fields = ['name', 'public', 'deleted']
+
+ def get_absolute_url(self):
+ return reverse(
+ 'tagged_object_list',
+ args=[f'polka/{self.slug}']
+ )
+
+ def __str__(self):
+ return self.name
+
+ @property
+ def url_chunk(self):
+ return f'polka/{self.slug}'
+
+ @classmethod
+ def create_from_data(cls, user, data):
+ return cls.create(user, data['name'])
+
+ @classmethod
+ def create(cls, user, name):
+ n = now()
+ return cls.objects.create(
+ user=user,
+ name=name,
+ slug=get_random_hash(name),
+ updated_at=n,
+ reported_timestamp=n,
+ )
+
+ @classmethod
+ def get_by_name(cls, user, name, create=False):
+ l = cls.objects.filter(
+ user=user,
+ name=name
+ ).first()
+ if l is None and create:
+ l = cls.create(user, name)
+ return l
+
+ @classmethod
+ def get_favorites_list(cls, user, create=False):
+ try:
+ return cls.objects.get(
+ user=user,
+ favorites=True
+ )
+ except cls.DoesNotExist:
+ n = now()
+ if create:
+ return cls.objects.create(
+ user=user,
+ favorites=True,
+ slug=get_random_hash('favorites'),
+ updated_at=n,
+ reported_timestamp=n,
+ )
+ else:
+ return None
+ except cls.MultipleObjectsReturned:
+ # merge?
+ lists = list(cls.objects.filter(user=user, favorites=True))
+ for l in lists[1:]:
+ t.userlistitem_set.all().update(
+ list=lists[0]
+ )
+ l.delete()
+ return lists[0]
+
+ @classmethod
+ def likes(cls, user, book):
+ ls = cls.get_favorites_list(user)
+ if ls is None:
+ return False
+ return ls.userlistitem_set.filter(deleted=False, book=book).exists()
+
+ def append(self, book):
+ n = now()
+ items = self.userlistitem_set.filter(
+ book=book,
+ )
+ if items.exists():
+ items.update(
+ deleted=False,
+ reported_timestamp=n,
+ )
+ item = items.first()
+ else:
+ item = self.userlistitem_set.create(
+ book=book,
+ order=(self.userlistitem_set.aggregate(m=models.Max('order'))['m'] or 0) + 1,
+ updated_at=n,
+ reported_timestamp=n,
+ )
+ book.update_popularity()
+ return item
+
+ def remove(self, book):
+ self.userlistitem_set.filter(book=book).update(
+ deleted=True,
+ updated_at=now()
+ )
+ book.update_popularity()
+
+ @classmethod
+ def like(cls, user, book):
+ ul = cls.get_favorites_list(user, create=True)
+ ul.append(book)
+
+ @classmethod
+ def unlike(cls, user, book):
+ ul = cls.get_favorites_list(user)
+ if ul is not None:
+ ul.remove(book)
+
+ def get_books(self):
+ return [item.book for item in self.userlistitem_set.exclude(deleted=True).exclude(book=None)]
+
+
+class UserListItem(Syncable, models.Model):
+ list = models.ForeignKey(UserList, models.CASCADE)
+ uuid = models.UUIDField(unique=True, default=uuid.uuid4, editable=False, blank=True)
+ order = models.IntegerField()
+ deleted = models.BooleanField(default=False)
+ created_at = models.DateTimeField(auto_now_add=True)
+ updated_at = models.DateTimeField(auto_now=True)
+ reported_timestamp = models.DateTimeField()
+
+ book = models.ForeignKey('catalogue.Book', models.SET_NULL, null=True, blank=True)
+ fragment = models.ForeignKey('catalogue.Fragment', models.SET_NULL, null=True, blank=True)
+ quote = models.ForeignKey('bookmarks.Quote', models.SET_NULL, null=True, blank=True)
+ bookmark = models.ForeignKey('bookmarks.Bookmark', models.SET_NULL, null=True, blank=True)
+
+ note = models.TextField(blank=True)
+
+ syncable_fields = ['order', 'deleted', 'book', 'fragment', 'quote', 'bookmark', 'note']
+
+ @classmethod
+ def create_from_data(cls, user, data):
+ if data.get('favorites'):
+ l = UserList.get_favorites_list(user, create=True)
+ else:
+ l = data['list']
+ try:
+ assert l.user == user
+ except AssertionError:
+ return
+ return l.append(book=data['book'])
+
+ @property
+ def favorites(self):
+ return self.list.favorites