|
from datetime import date
|
|
|
|
from django.db import models
|
|
from django.utils.html import format_html
|
|
|
|
from .settings import ACCOUNTS
|
|
|
|
|
|
class Volunteer(models.Model):
|
|
realname = models.CharField(max_length=200, null=True, verbose_name="Realname",
|
|
help_text="Bitte gib deinen Vornamen und deinen Nachnamen ein.")
|
|
email = models.EmailField(max_length=200, null=True, verbose_name='E-Mail-Adresse',
|
|
help_text=format_html('Bitte gib deine E-Mail-Adresse ein, damit dich<br>Wikimedia Deutschland bei Rückfragen oder für<br>die Zusage kontaktieren kann.'))
|
|
|
|
# the following Fields are not supposed to be edited by users
|
|
granted = models.BooleanField(null=True)
|
|
granted_date = models.DateField(null=True)
|
|
survey_mail_send = models.BooleanField(default=False, verbose_name='Keine Umfragemail schicken')
|
|
survey_mail_date = models.DateField(verbose_name='Umfragemail wurde verschickt am', null=True, blank=True)
|
|
|
|
@classmethod
|
|
def set_granted(cl, key, b):
|
|
obj = cl.objects.get(pk=key)
|
|
obj.granted = b
|
|
obj.granted_date = date.today()
|
|
obj.save()
|
|
|
|
class Meta:
|
|
abstract = True
|
|
|
|
|
|
class Extern(Volunteer):
|
|
''' abstract basis class for all data entered by extern volunteers '''
|
|
username = models.CharField(max_length=200, null=True, verbose_name='Benutzer_innenname',
|
|
help_text=format_html("Bitte gib den Namen ein, mit dem du dich<br>in den Wikimedia-Projekten registriert hast."))
|
|
|
|
# the following Fields are not supposed to be edited by users
|
|
service_id = models.CharField(max_length=15, null=True, blank=True)
|
|
|
|
def save(self,*args,**kwargs):
|
|
# we don't call save with args/kwargs to avoid UNIQUE CONSTRAINT errors
|
|
# but maybe there is a better solution?
|
|
super().save()
|
|
self.service_id = type(self).__name__ + str(self.pk)
|
|
super().save()
|
|
|
|
class Meta:
|
|
abstract = True
|
|
|
|
class ConcreteExtern(Extern):
|
|
''' needed because we can't initiate abstract base classes in the view'''
|
|
pass
|
|
|
|
class Account(models.Model):
|
|
code = models.CharField('Kostenstelle', max_length=5, default="DEF",
|
|
null=False, primary_key = True)
|
|
description = models.CharField('Beschreibung', max_length=60, default='NO DESCRIPTION')
|
|
|
|
def __str__(self):
|
|
return f"{self.code} {self.description}"
|
|
|
|
class Project(Volunteer):
|
|
name = models.CharField(max_length=200, verbose_name='Name des Projekts')
|
|
description = models.CharField(max_length=500, verbose_name="Kurzbeschreibung", null=True)
|
|
start = models.DateField('Startdatum', null=True)
|
|
end = models.DateField('Erwartetes Projektende', null=True)
|
|
otrs = models.URLField(max_length=300, null=True, verbose_name='OTRS-Link')
|
|
plan = models.URLField(max_length=2000, null=True, blank=True, verbose_name="Link zum Förderplan")
|
|
page = models.URLField(max_length=2000, null=True, blank=True, verbose_name="Link zur Projektseite")
|
|
urls = models.CharField(max_length=2000, null=True, blank=True, verbose_name="Weitere Links")
|
|
group = models.CharField(max_length=2000, null=True, blank=True, verbose_name="Mitorganisierende")
|
|
location = models.CharField(max_length=2000, null=True, blank=True, verbose_name="Ort/Adresse/Location")
|
|
participants_estimated = models.IntegerField(blank=True, null=True, verbose_name='Teilnehmende angefragt')
|
|
participants_real = models.IntegerField(blank=True, null=True, verbose_name='Teilnehmende ausgezählt')
|
|
insurance = models.BooleanField(default=False, verbose_name='Haftpflichtversicherung')
|
|
insurance_technic = models.BooleanField(default=False, verbose_name='Technikversicherung Ausland')
|
|
support = models.CharField(max_length=300, blank=True, null=True, verbose_name='Betreuungsperson und Vertretung')
|
|
cost = models.IntegerField(blank=True, null=True)
|
|
account = models.ForeignKey('Account', on_delete=models.CASCADE, null=True, to_field='code', db_constraint = False)
|
|
granted_from = models.CharField(max_length=100,null=True,verbose_name='Bewilligt von')
|
|
notes = models.CharField(max_length=1000,null=True,blank=True,verbose_name='Anmerkungen')
|
|
|
|
# the following Fields are not supposed to be edited by users
|
|
pid = models.CharField(max_length=15, null=True, blank=True)
|
|
end_mail_send = models.BooleanField(null=True)
|
|
status = models.CharField(max_length=3,choices=(('RUN', 'läuft'),('END','beendet')),default='RUN')
|
|
persons = models.IntegerField(default=1)
|
|
finance_id = models.CharField(max_length=15, null= True, blank=True)
|
|
project_of_year = models.IntegerField(default=0)
|
|
|
|
def save(self,*args,**kwargs):
|
|
'''we generate the autogenerated fields here'''
|
|
# we don't call save with args/kwargs to avoid UNIQUE CONSTRAINT errors
|
|
# but maybe there is a better solution?
|
|
super().save()
|
|
self.pid = str(self.account.code) + str(self.pk).zfill(3)
|
|
# generation of finance_id
|
|
if not self.project_of_year:
|
|
print('AAA')
|
|
# we need to determine if this is a new year with its first new project...
|
|
year = self.start.year
|
|
projects = Project.objects.filter(start__year=year)
|
|
if not projects:
|
|
print('BBB')
|
|
self.project_of_year = 1
|
|
else:
|
|
print('CCC')
|
|
projects = projects.order_by("-project_of_year")[0]
|
|
self.project_of_year = int(projects.project_of_year) + 1
|
|
self.finance_id = str(self.account.code) + str(self.project_of_year).zfill(3)
|
|
super().save()
|
|
|
|
def __str__(self):
|
|
return f"{self.pid} {self.name}"
|
|
|
|
|
|
class Intern(Volunteer):
|
|
'''abstrat base class for data entry from /intern (except Project)'''
|
|
request_url = models.URLField(max_length=2000, verbose_name='Antrag (URL)')
|
|
|
|
class Meta:
|
|
abstract = True
|
|
|
|
class ConcreteVolunteer(Volunteer):
|
|
''' needed because we can't initiate abstract base classes in the view'''
|
|
pass
|
|
|
|
class HonoraryCertificate(Intern):
|
|
''' this class is also used for accreditations '''
|
|
|
|
project = models.ForeignKey(Project, null=True, blank=True, on_delete=models.SET_NULL)
|
|
|
|
def __str__(self):
|
|
return "Certificate for " + self.realname
|
|
|
|
|
|
TRANSPORT_CHOICES = {'BAHN': 'Bahn',
|
|
'NONE': 'Keine Fahrtkosten',
|
|
'OTHER': 'Sonstiges (mit Begründung)'}
|
|
|
|
PAYEDBY_CHOICES = {'WMDE': 'WMDE',
|
|
'REQU': 'Antragstellender Mensch'}
|
|
|
|
class Travel(Intern):
|
|
project = models.ForeignKey(Project, on_delete=models.CASCADE)
|
|
transport = models.CharField(max_length=5, choices=TRANSPORT_CHOICES.items(), default='BAHN')
|
|
other_transport = models.CharField(max_length=200, null=True, blank=True, verbose_name='Sonstige Transportmittel (mit Begründung)')
|
|
travelcost = models.CharField(max_length=10, default="0", verbose_name="Fahrtkosten")
|
|
checkin = models.DateField(blank=True, null=True, verbose_name='Check In')
|
|
checkout = models.DateField(blank=True, null=True, verbose_name='Check Out')
|
|
payed_for_hotel_by = models.CharField(max_length=4, choices=PAYEDBY_CHOICES.items(), blank=True, null=True, verbose_name='Kostenauslage Hotel durch')
|
|
payed_for_travel_by = models.CharField(max_length=4, choices=PAYEDBY_CHOICES.items(), blank=True, null=True, verbose_name='Kostenauslage Fahrt durch')
|
|
notes = models.CharField(max_length=500, blank=True)
|
|
|
|
|
|
#abstract base class for Library and IFG
|
|
class Grant(Extern):
|
|
cost = models.CharField(max_length=10, verbose_name='Kosten',
|
|
help_text="Bitte gib die ungefähr zu erwartenden Kosten in Euro an.")
|
|
notes = models.TextField(max_length=500, blank=True, verbose_name='Anmerkungen',
|
|
help_text="Bitte gib an wofür Du das Stipendium verwenden willst.")
|
|
|
|
class Meta:
|
|
abstract = True
|
|
|
|
|
|
TYPE_CHOICES = {'BIB': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/Zugang_zu_Fachliteratur#Bibliotheksstipendium">Bibliotheksstipendium</a>'),
|
|
'ELIT': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/Zugang_zu_Fachliteratur#eLiteraturstipendium">eLiteraturstipendium</a>'),
|
|
'MAIL': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/E-Mail-Adressen_und_Visitenkarten#E-Mail-Adressen">E-Mail-Adresse</a>'),
|
|
'IFG': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/Gebührenerstattungen_für_Behördenanfragen">Kostenübernahme IFG-Anfrage</a>'),
|
|
'LIT': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/Zugang_zu_Fachliteratur#Literaturstipendium">Literaturstipendium</a>'),
|
|
'LIST': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/E-Mail-Adressen_und_Visitenkarten#Mailinglisten">Mailingliste</a>'),
|
|
'SOFT': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/Software-Stipendien">Softwarestipendium</a>'),
|
|
'VIS': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/E-Mail-Adressen_und_Visitenkarten#Visitenkarten">Visitenkarten</a>'),
|
|
}
|
|
|
|
# same model is used for Library, ELitStip and Software!
|
|
class Library(Grant):
|
|
|
|
type = models.CharField(
|
|
max_length=4,
|
|
choices=TYPE_CHOICES.items(), #attention: actually only BIB, ELIT, SOFT should be used here
|
|
default='BIB',
|
|
)
|
|
library = models.CharField(max_length=200)
|
|
duration = models.CharField(max_length=100, verbose_name="Dauer")
|
|
|
|
def __str__(self):
|
|
return self.library
|
|
|
|
class Literature(Grant):
|
|
info = models.CharField(max_length=500, verbose_name='Informationen zum Werk',
|
|
help_text=format_html("Bitte gib alle Informationen zum benötigten Werk an,<br>\
|
|
die eine eindeutige Identifizierung ermöglichen (Autor, Titel, Verlag, ISBN, ...)"))
|
|
source = models.CharField(max_length=200, verbose_name='Bezugsquelle',
|
|
help_text="Bitte gib an, wo du das Werk kaufen möchtest.")
|
|
|
|
class IFG(Grant):
|
|
url = models.URLField(max_length=2000, verbose_name="URL",
|
|
help_text="Bitte gib den Link zu deiner Anfrage bei Frag den Staat an.")
|
|
|
|
def __str__(self):
|
|
return "IFG-Anfrage von " + self.realname
|
|
|
|
DOMAIN_CHOICES = {'PEDIA': '@wikipedia.de',
|
|
'BOOKS': '@wikibooks.de',
|
|
'QUOTE': '@wikiquote.de',
|
|
'SOURCE': '@wikisource.de',
|
|
'VERSITY': '@wikiversity.de',}
|
|
|
|
class Domain(Extern):
|
|
domain = models.CharField(max_length=10,
|
|
choices=DOMAIN_CHOICES.items(),
|
|
default='PEDIA')
|
|
|
|
class Meta:
|
|
abstract = True
|
|
|
|
MAIL_CHOICES = {'REALNAME': 'Vorname.Nachname',
|
|
'USERNAME': 'Username',
|
|
'OTHER': 'Sonstiges:'}
|
|
|
|
class Email(Domain):
|
|
address = models.CharField(max_length=50,
|
|
choices=MAIL_CHOICES.items(),
|
|
default='USERNAME', verbose_name='Adressbestandteil',
|
|
help_text=format_html("Bitte gib hier den gewünschten Adressbestandteil an,<br>der sich vor der Domain befinden soll."))
|
|
|
|
other = models.CharField(max_length=50,blank=True,null=True, verbose_name="Sonstiges")
|
|
|
|
class List(Domain):
|
|
address = models.CharField(max_length=50, default='NO_ADDRESS',
|
|
verbose_name="Adressbestandteil für Projektmailingliste",
|
|
help_text=format_html("Bitte gib hier den gewünschten Adressbestandteil an,<br>der sich vor der Domain befinden soll."))
|
|
|
|
PROJECT_CHOICE = {'PEDIA': 'Wikipedia',
|
|
'SOURCE': 'Wikisource',
|
|
'BOOKS': 'Wikibooks',
|
|
'QUOTE': 'Wikiquote',
|
|
'VERSITY': 'Wikiversity',
|
|
'VOYAGE': 'Wikivoyage',
|
|
'DATA': 'Wikidata',
|
|
'NEWS': 'Wikinews',
|
|
'COMMONS': 'Wikimedia Commons'}
|
|
|
|
BC_VARIANT = {'PIC': 'mit Bild',
|
|
'NOPIC': 'ohne Bild'}
|
|
|
|
class BusinessCard(Extern):
|
|
project = models.CharField(max_length=20, choices=PROJECT_CHOICE.items(),
|
|
default='PEDIA', verbose_name='Wikimedia-Projekt',
|
|
help_text='Für welches Wikimedia-Projekt möchtest Du Visitenkarten?')
|
|
|
|
data = models.TextField(max_length=1000, verbose_name='Persönliche Daten für die Visitenkarten', default='',
|
|
help_text=format_html("Bitte gib hier alle persönlichen Daten an, und zwar genau so,<br>\
|
|
wie sie (auch in der entsprechenden Reihenfolge) auf den Visitenkarten stehen sollen<br>\
|
|
(z.B. Vorname Nachname, Benutzer:/Benutzerin:, Benutzer-/-innenname, Anschrift,<br>\
|
|
Telefonnummer, E-Mail-Adresse usw.). Trenne die einzelnen Angaben durch Zeilenumbrüche.<br>\
|
|
Hinweis: Telefonnummern bilden wir üblicherweise im internationalen Format gemäß<br>\
|
|
DIN 5008 ab. Als anzugebende E-Mail-Adresse empfehlen wir dir eine Wikimedia-Projekt-<br>\
|
|
Adresse, die du ebenfalls beantragen kannst, sofern du nicht bereits eine besitzt."))
|
|
variant = models.CharField(max_length=5, choices=BC_VARIANT.items(),
|
|
default='NOPIC', verbose_name='Variante',
|
|
help_text=format_html('so sehen die Varianten aus: <a href="https://upload.wikimedia.org/wikipedia/commons/c/cd/Muster_Visitenkarten_WMDE_2018.jpg">\
|
|
mit Bild</a> <a href="https://upload.wikimedia.org/wikipedia/commons/d/d3/Muster_Visitenkarte_WMDE.png">ohne Bild</a>' ))
|
|
sent_to = models.TextField(max_length=1000, verbose_name='Versandadresse',
|
|
default='', help_text="Bitte gib den Namen und die vollständige Adresse ein, an welche die Visitenkarten geschickt werden sollen.")
|