registration
[wolnelektury.git] / src / social / models.py
1 # This file is part of Wolne Lektury, licensed under GNU Affero GPLv3 or later.
2 # Copyright © Fundacja Wolne Lektury. See NOTICE for more information.
3 #
4 from oauthlib.common import urlencode, generate_token
5 from random import randint
6 from django.db import models
7 from django.conf import settings
8 from django.contrib.auth.models import User
9 from django.core.exceptions import ValidationError
10 from django.core.mail import send_mail
11 from django.urls import reverse
12 from catalogue.models import Book
13 from wolnelektury.utils import cached_render, clear_cached_renders
14
15
16 class BannerGroup(models.Model):
17     name = models.CharField('nazwa', max_length=255, unique=True)
18     created_at = models.DateTimeField('utworzona', auto_now_add=True)
19
20     class Meta:
21         ordering = ('name',)
22         verbose_name = 'grupa bannerów'
23         verbose_name_plural = 'grupy bannerów'
24
25     def __str__(self):
26         return self.name
27
28     def get_absolute_url(self):
29         """This is used for testing."""
30         return "%s?banner_group=%d" % (reverse('main_page'), self.id)
31
32     def get_banner(self):
33         banners = self.cite_set.all()
34         count = banners.count()
35         if not count:
36             return None
37         return banners[randint(0, count-1)]
38
39
40 class Cite(models.Model):
41     book = models.ForeignKey(Book, models.CASCADE, verbose_name='książka', null=True, blank=True)
42     text = models.TextField('tekst', blank=True)
43     small = models.BooleanField(
44         'mały', default=False, help_text='Sprawia, że cytat wyświetla się mniejszym fontem.')
45     vip = models.CharField('VIP', max_length=128, null=True, blank=True)
46     link = models.URLField('odnośnik')
47     video = models.URLField('wideo', blank=True)
48     picture = models.ImageField('ilustracja', blank=True,
49             help_text='Najlepsze wymiary: 975 x 315 z tekstem, 487 x 315 bez tekstu.')
50     picture_alt = models.CharField('alternatywny tekst ilustracji', max_length=255, blank=True)
51     picture_title = models.CharField('tytuł ilustracji', max_length=255, null=True, blank=True)
52     picture_author = models.CharField('autor ilustracji', max_length=255, blank=True, null=True)
53     picture_link = models.URLField('link ilustracji', blank=True, null=True)
54     picture_license = models.CharField('nazwa licencji ilustracji', max_length=255, blank=True, null=True)
55     picture_license_link = models.URLField('adres licencji ilustracji', blank=True, null=True)
56
57     sticky = models.BooleanField('przyklejony', default=False, db_index=True,
58                                  help_text='Przyklejone cytaty mają pierwszeństwo.')
59     background_plain = models.BooleanField('jednobarwne tło', default=False)
60     background_color = models.CharField('kolor tła', max_length=32, blank=True)
61     image = models.ImageField(
62         'obraz tła', upload_to='social/cite', null=True, blank=True,
63         help_text='Najlepsze tło ma wielkość 975 x 315 px i waży poniżej 100kB.')
64     image_title = models.CharField('tytuł obrazu tła', max_length=255, null=True, blank=True)
65     image_author = models.CharField('autor obrazu tła', max_length=255, blank=True, null=True)
66     image_link = models.URLField('link obrazu tła', blank=True, null=True)
67     image_license = models.CharField('nazwa licencji obrazu tła', max_length=255, blank=True, null=True)
68     image_license_link = models.URLField('adres licencji obrazu tła', blank=True, null=True)
69
70     created_at = models.DateTimeField('utworzony', auto_now_add=True)
71     group = models.ForeignKey(BannerGroup, verbose_name='grupa', null=True, blank=True, on_delete=models.SET_NULL)
72
73     class Meta:
74         ordering = ('vip', 'text')
75         verbose_name = 'banner'
76         verbose_name_plural = 'bannery'
77
78     def __str__(self):
79         t = []
80         if self.text:
81             t.append(self.text[:60])
82         if self.book_id:
83             t.append('[ks.]'[:60])
84         t.append(self.link[:60])
85         if self.vip:
86             t.append('vip: ' + self.vip)
87         if self.picture:
88             t.append('[obr.]')
89         if self.video:
90             t.append('[vid.]')
91         return ', '.join(t)
92
93     def get_absolute_url(self):
94         """This is used for testing."""
95         return "%s?banner=%d" % (reverse('main_page'), self.id)
96
97     def has_box(self):
98         return self.video or self.picture
99
100     def has_body(self):
101         return self.vip or self.text or self.book
102
103     def layout(self):
104         pieces = []
105         if self.has_box():
106             pieces.append('box')
107         if self.has_body():
108             pieces.append('text')
109         return '-'.join(pieces)
110
111
112     def save(self, *args, **kwargs):
113         ret = super(Cite, self).save(*args, **kwargs)
114         self.clear_cache()
115         return ret
116
117     @cached_render('social/cite_promo.html')
118     def main_box(self):
119         return {
120             'cite': self,
121             'main': True,
122         }
123
124     def clear_cache(self):
125         clear_cached_renders(self.main_box)
126
127
128 class Carousel(models.Model):
129     placement = models.SlugField('miejsce', choices=[
130         ('main', 'main'),
131         ('main_2022', 'main 2022'),
132     ])
133     priority = models.SmallIntegerField('priorytet', default=0)
134     language = models.CharField('język', max_length=2, blank=True, default='', choices=settings.LANGUAGES)
135
136     class Meta:
137 #        ordering = ('placement', '-priority')
138         verbose_name = 'karuzela'
139         verbose_name_plural = 'karuzele'
140
141     def __str__(self):
142         return self.placement
143
144     @classmethod
145     def get(cls, placement):
146         carousel = cls.objects.filter(placement=placement).order_by('-priority', '?').first()
147         if carousel is None:
148             carousel = cls.objects.create(placement=placement)
149         return carousel
150
151
152 class CarouselItem(models.Model):
153     order = models.PositiveSmallIntegerField('kolejność')
154     carousel = models.ForeignKey(Carousel, models.CASCADE, verbose_name='karuzela')
155     banner = models.ForeignKey(
156         Cite, models.CASCADE, null=True, blank=True, verbose_name='banner')
157     banner_group = models.ForeignKey(
158         BannerGroup, models.CASCADE, null=True, blank=True, verbose_name='grupa bannerów')
159
160     class Meta:
161         ordering = ('order',)
162         verbose_name = 'element karuzeli'
163         verbose_name_plural = 'elementy karuzeli'
164
165     def __str__(self):
166         return str(self.banner or self.banner_group)
167
168     def clean(self):
169         if not self.banner and not self.banner_group:
170             raise ValidationError('Proszę wskazać banner albo grupę bannerów.')
171         elif self.banner and self.banner_group:
172             raise ValidationError('Proszę wskazać banner albo grupę bannerów.')
173
174     def get_banner(self):
175         return self.banner or self.banner_group.get_banner()
176
177
178 class UserConfirmation(models.Model):
179     user = models.ForeignKey(User, models.CASCADE)
180     created_at = models.DateTimeField(auto_now_add=True)
181     key = models.CharField(max_length=128, unique=True)
182
183     def send(self):
184         send_mail(
185             'Potwierdź konto w bibliotece Wolne Lektury',
186             f'https://beta.wolnelektury.pl/ludzie/potwierdz/{self.key}/',
187             settings.CONTACT_EMAIL,
188             [self.user.email]
189         )
190
191     def use(self):
192         user = self.user
193         user.is_active = True
194         user.save()
195         self.delete()
196     
197     @classmethod
198     def request(cls, user):
199         cls.objects.create(
200             user=user,
201             key=generate_token()
202         ).send()