You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

311 lines
18 KiB

4 years ago
  1. from datetime import date
  2. from django.db import models
  3. from django.utils.html import format_html
  4. from .settings import ACCOUNTS
  5. EMAIL_STATES = {'NONE': 'noch keine Mail versendet',
  6. 'INF': 'die Benachrichtigung End-Mail wurde bereits versendet',
  7. 'END': 'alle automatischen Mails, auch surveyMail, wurden bereits versendet'}
  8. class Volunteer(models.Model):
  9. realname = models.CharField(max_length=200, null=True, verbose_name="Realname",
  10. help_text="Bitte gib deinen Vornamen und deinen Nachnamen ein.")
  11. email = models.EmailField(max_length=200, null=True, verbose_name='E-Mail-Adresse',
  12. 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.'))
  13. # the following Fields are not supposed to be edited by users
  14. granted = models.BooleanField(null=True)
  15. granted_date = models.DateField(null=True)
  16. survey_mail_send = models.BooleanField(default=False, verbose_name='Keine Umfragemail schicken')
  17. survey_mail_date = models.DateField(verbose_name='Umfragemail wurde verschickt am', null=True, blank=True)
  18. mail_state = models.CharField(max_length=6, choices=EMAIL_STATES.items(), default='NONE')
  19. @classmethod
  20. def set_granted(cl, key, b):
  21. obj = cl.objects.get(pk=key)
  22. obj.granted = b
  23. obj.granted_date = date.today()
  24. obj.save()
  25. class Meta:
  26. abstract = True
  27. class Extern(Volunteer):
  28. ''' abstract basis class for all data entered by extern volunteers '''
  29. username = models.CharField(max_length=200, null=True, verbose_name='Benutzer_innenname',
  30. help_text=format_html("Bitte gib den Namen ein, mit dem du dich<br>in den Wikimedia-Projekten registriert hast."))
  31. # the following Fields are not supposed to be edited by users
  32. service_id = models.CharField(max_length=15, null=True, blank=True)
  33. def save(self,*args,**kwargs):
  34. # we don't call save with args/kwargs to avoid UNIQUE CONSTRAINT errors
  35. # but maybe there is a better solution?
  36. super().save()
  37. self.service_id = type(self).__name__ + str(self.pk)
  38. super().save()
  39. class Meta:
  40. abstract = True
  41. class ConcreteExtern(Extern):
  42. ''' needed because we can't initiate abstract base classes in the view'''
  43. pass
  44. class Account(models.Model):
  45. code = models.CharField('Kostenstelle', max_length=5, default="DEF",
  46. null=False, primary_key = True)
  47. description = models.CharField('Beschreibung', max_length=60, default='NO DESCRIPTION')
  48. intern_notes = models.TextField(max_length=1000, blank=True)
  49. def __str__(self):
  50. return f"{self.code} {self.description}"
  51. class Project(Volunteer):
  52. name = models.CharField(max_length=200, verbose_name='Name des Projekts')
  53. description = models.CharField(max_length=500, verbose_name="Kurzbeschreibung", null=True)
  54. start = models.DateField('Startdatum', null=True)
  55. end = models.DateField('Erwartetes Projektende', null=True)
  56. otrs = models.URLField(max_length=300, null=True, verbose_name='OTRS-Link')
  57. plan = models.URLField(max_length=2000, null=True, blank=True, verbose_name="Link zum Förderplan")
  58. page = models.URLField(max_length=2000, null=True, blank=True, verbose_name="Link zur Projektseite")
  59. urls = models.CharField(max_length=2000, null=True, blank=True, verbose_name="Weitere Links")
  60. group = models.CharField(max_length=2000, null=True, blank=True, verbose_name="Mitorganisierende")
  61. location = models.CharField(max_length=2000, null=True, blank=True, verbose_name="Ort/Adresse/Location")
  62. participants_estimated = models.IntegerField(blank=True, null=True, verbose_name='Teilnehmende angefragt')
  63. participants_real = models.IntegerField(blank=True, null=True, verbose_name='Teilnehmende ausgezählt')
  64. insurance = models.BooleanField(default=False, verbose_name='Haftpflichtversicherung')
  65. insurance_technic = models.BooleanField(default=False, verbose_name='Technikversicherung Ausland')
  66. support = models.CharField(max_length=300, blank=True, null=True, verbose_name='Betreuungsperson und Vertretung')
  67. cost = models.IntegerField(blank=True, null=True)
  68. account = models.ForeignKey('Account', on_delete=models.CASCADE, null=True, to_field='code', db_constraint = False)
  69. granted_from = models.CharField(max_length=100,null=True,verbose_name='Bewilligt von')
  70. notes = models.TextField(max_length=1000,null=True,blank=True,verbose_name='Anmerkungen')
  71. intern_notes = models.TextField(max_length=1000, blank=True)
  72. # the following Fields are not supposed to be edited by users
  73. pid = models.CharField(max_length=15, null=True, blank=True)
  74. end_mail_send = models.BooleanField(null=True)
  75. status = models.CharField(max_length=3,choices=(('RUN', 'läuft'),('END','beendet'),('NOT','nicht stattgefunden')),default='RUN')
  76. persons = models.IntegerField(default=1)
  77. finance_id = models.CharField(max_length=15, null= True, blank=True)
  78. project_of_year = models.IntegerField(default=0)
  79. def save(self,*args,**kwargs):
  80. '''we generate the autogenerated fields here'''
  81. # we don't call save with args/kwargs to avoid UNIQUE CONSTRAINT errors
  82. # but maybe there is a better solution?
  83. super().save()
  84. self.pid = str(self.start.year) + '-' + str(self.account.code) + str(self.pk).zfill(3)
  85. # self.pid = str(self.account.code) + str(self.pk).zfill(3)
  86. # generation of finance_id
  87. if not self.project_of_year:
  88. print('AAA')
  89. # we need to determine if this is a new year with its first new project...
  90. year = self.start.year
  91. print(year)
  92. projects = Project.objects.filter(start__year=year)
  93. if not projects:
  94. #print('BBB')
  95. self.project_of_year = 1
  96. else:
  97. #print('CCC')
  98. projects = projects.order_by("-project_of_year")[0]
  99. self.project_of_year = int(projects.project_of_year) + 1
  100. self.finance_id = str(self.account.code) + str(self.project_of_year).zfill(3)
  101. super().save()
  102. def __str__(self):
  103. return f"{self.pid} {self.name}"
  104. class Intern(Volunteer):
  105. '''abstract base class for data entry from /intern (except Project)'''
  106. request_url = models.URLField(max_length=2000, verbose_name='Antrag (URL)')
  107. intern_notes = models.TextField(max_length=1000, blank=True)
  108. class Meta:
  109. abstract = True
  110. class ConcreteVolunteer(Volunteer):
  111. ''' needed because we can't initiate abstract base classes in the view'''
  112. pass
  113. class HonoraryCertificate(Intern):
  114. ''' this class is also used for accreditations '''
  115. project = models.ForeignKey(Project, null=True, blank=True, on_delete=models.SET_NULL)
  116. def __str__(self):
  117. return "Certificate for " + self.realname
  118. TRANSPORT_CHOICES = {'BAHN': 'Bahn',
  119. 'NONE': 'Keine Fahrtkosten',
  120. 'OTHER': 'Sonstiges (mit Begründung)'}
  121. PAYEDBY_CHOICES = {'WMDE': 'WMDE',
  122. 'REQU': 'Antragstellender Mensch'}
  123. class Travel(Intern):
  124. # project variable is now null true and blank true, which means it can be saved without project id to be later on filled out by admins
  125. project = models.ForeignKey(Project, on_delete=models.CASCADE, null=True, blank=True)
  126. project_name = models.CharField(max_length=50, null=True, blank=True, verbose_name='Projektname:')
  127. transport = models.CharField(max_length=5, choices=TRANSPORT_CHOICES.items(), default='BAHN', verbose_name='Transportmittel:')
  128. other_transport = models.CharField(max_length=200, null=True, blank=True, verbose_name='Sonstige Transportmittel (mit Begründung)')
  129. travelcost = models.CharField(max_length=10, default="0", verbose_name="Fahrtkosten")
  130. checkin = models.DateField(blank=True, null=True, verbose_name='Check In')
  131. checkout = models.DateField(blank=True, null=True, verbose_name='Check Out')
  132. payed_for_hotel_by = models.CharField(max_length=4, choices=PAYEDBY_CHOICES.items(), blank=True, null=True, verbose_name='Kostenauslage Hotel durch')
  133. payed_for_travel_by = models.CharField(max_length=4, choices=PAYEDBY_CHOICES.items(), blank=True, null=True, verbose_name='Kostenauslage Fahrt durch')
  134. hotel = models.BooleanField(default=False, verbose_name='Hotelzimmer benötigt:')
  135. notes = models.TextField(max_length=1000, blank=True)
  136. #abstract base class for Library and IFG
  137. class Grant(Extern):
  138. cost = models.CharField(max_length=10, verbose_name='Kosten',
  139. help_text="Bitte gib die ungefähr zu erwartenden Kosten in Euro an.")
  140. notes = models.TextField(max_length=1000, blank=True, verbose_name='Anmerkungen',
  141. help_text="Bitte gib an wofür Du das Stipendium verwenden willst.")
  142. class Meta:
  143. abstract = True
  144. TYPE_CHOICES = {'BIB': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/Zugang_zu_Fachliteratur#Bibliotheksstipendium" target="_blank" rel="noopener">Bibliotheksstipendium</a>'),
  145. 'ELIT': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/Zugang_zu_Fachliteratur#eLiteraturstipendium" target="_blank" rel="noopener">eLiteraturstipendium</a>'),
  146. 'MAIL': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/E-Mail-Adressen_und_Visitenkarten#E-Mail-Adressen" target="_blank" rel="noopener">E-Mail-Adresse</a>'),
  147. 'IFG': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/Gebührenerstattungen_für_Behördenanfragen" target="_blank" rel="noopener">Kostenübernahme IFG-Anfrage</a>'),
  148. 'LIT': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/Zugang_zu_Fachliteratur#Literaturstipendium" target="_blank" rel="noopener">Literaturstipendium</a>'),
  149. 'LIST': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/E-Mail-Adressen_und_Visitenkarten#Mailinglisten" target="_blank" rel="noopener">Mailingliste</a>'),
  150. 'TRAV': format_html('Reisekosten'),
  151. 'SOFT': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/Software-Stipendien" target="_blank" rel="noopener">Softwarestipendium</a>'),
  152. 'VIS': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/E-Mail-Adressen_und_Visitenkarten#Visitenkarten" target="_blank" rel="noopener">Visitenkarten</a>'),
  153. }
  154. # same model is used for Library, ELitStip and Software!
  155. class Library(Grant):
  156. type = models.CharField(
  157. max_length=4,
  158. choices=TYPE_CHOICES.items(), #attention: actually only BIB, ELIT, SOFT should be used here
  159. default='BIB',
  160. )
  161. library = models.CharField(max_length=200)
  162. duration = models.CharField(max_length=100, verbose_name="Dauer")
  163. intern_notes = models.TextField(max_length=1000, blank=True)
  164. def __str__(self):
  165. return self.library
  166. SELFBUY_CHOICES = {'TRUE': format_html('Ich möchte das Werk selbst kaufen und per Kostenerstattung bei Wikimedia Deutschland abrechnen.'),
  167. 'FALSE': format_html('Ich möchte, dass Wikimedia Deutschland das Werk für mich kauft'),
  168. 'NONE': format_html('Nichts ausgewählt')
  169. }
  170. class Literature(Grant):
  171. info = models.CharField(max_length=500, verbose_name='Informationen zum Werk',
  172. help_text=format_html("Bitte gib alle Informationen zum benötigten Werk an,<br>\
  173. die eine eindeutige Identifizierung ermöglichen (Autor, Titel, Verlag, ISBN, ...)"))
  174. source = models.CharField(max_length=200, verbose_name='Bezugsquelle',
  175. help_text="Bitte gib an, wo du das Werk kaufen möchtest.")
  176. selfbuy = models.CharField( max_length=10, verbose_name='Selbstkauf?', choices=SELFBUY_CHOICES.items(), default='TRUE')
  177. selfbuy_give_data = models.BooleanField(verbose_name=format_html('Datenweitergabe erlauben'), help_text=format_html('Weitergabe meiner Daten (Name, Postadresse) an den von mir angegebenen Anbieter/Dienstleister.'))
  178. selfbuy_data = models.TextField(max_length=1000, verbose_name='Persönliche Daten sowie Adresse', default='',
  179. help_text=format_html("Bitte gib hier alle persönlichen Daten an, die wir benötigen, um das Werk<br>\
  180. für dich zu kaufen und es dir anschließend zu schicken (z.B. Vorname Nachname, Anschrift, <br>\
  181. Telefonnummer, E-Mail-Adresse usw.). Trenne die einzelnen Angaben durch Zeilenumbrüche. <br>\
  182. Hinweis: Telefonnummern bilden wir üblicherweise im internationalen Format gemäß<br>\
  183. DIN 5008 ab. Als anzugebende E-Mail-Adresse empfehlen wir dir eine Wikimedia-Projekt-<br>\
  184. Adresse, die du ebenfalls beantragen kannst, sofern du nicht bereits eine besitzt."))
  185. intern_notes = models.TextField(max_length=1000, blank=True)
  186. class IFG(Grant):
  187. url = models.URLField(max_length=2000, verbose_name="URL",
  188. help_text="Bitte gib den Link zu deiner Anfrage bei Frag den Staat an.")
  189. intern_notes = models.TextField(max_length=1000, blank=True)
  190. def __str__(self):
  191. return "IFG-Anfrage von " + self.realname
  192. DOMAIN_CHOICES = {'PEDIA': '@wikipedia.de',
  193. 'BOOKS': '@wikibooks.de',
  194. 'QUOTE': '@wikiquote.de',
  195. 'SOURCE': '@wikisource.de',
  196. 'VERSITY': '@wikiversity.de',}
  197. class Domain(Extern):
  198. domain = models.CharField(max_length=10,
  199. choices=DOMAIN_CHOICES.items(),
  200. default='PEDIA')
  201. class Meta:
  202. abstract = True
  203. MAIL_CHOICES = {'REALNAME': 'Vorname.Nachname',
  204. 'USERNAME': 'Username',
  205. 'OTHER': 'Sonstiges:'}
  206. ADULT_CHOICES = {'TRUE': format_html('Ich bin volljährig.'),
  207. 'FALSE': format_html('Ich bin noch nicht volljährig.'),
  208. 'NONE': format_html('Nichts ausgewählt')
  209. }
  210. class Email(Domain):
  211. address = models.CharField(max_length=50,
  212. choices=MAIL_CHOICES.items(),
  213. default='USERNAME', verbose_name='Adressbestandteil',
  214. help_text=format_html("Bitte gib hier den gewünschten Adressbestandteil an,<br>der sich vor der Domain befinden soll."))
  215. other = models.CharField(max_length=50,blank=True,null=True, verbose_name="Sonstiges")
  216. adult = models.CharField( max_length=10, verbose_name='Volljährigkeit', choices=ADULT_CHOICES.items(), default='NONE')
  217. intern_notes = models.TextField(max_length=1000, blank=True)
  218. class List(Domain):
  219. address = models.CharField(max_length=50, default='NO_ADDRESS',
  220. verbose_name="Adressbestandteil für Projektmailingliste",
  221. help_text=format_html("Bitte gib hier den gewünschten Adressbestandteil an,<br>der sich vor der Domain befinden soll."))
  222. intern_notes = models.TextField(max_length=1000, blank=True)
  223. PROJECT_CHOICE = {'PEDIA': 'Wikipedia',
  224. 'SOURCE': 'Wikisource',
  225. 'BOOKS': 'Wikibooks',
  226. 'QUOTE': 'Wikiquote',
  227. 'VERSITY': 'Wikiversity',
  228. 'VOYAGE': 'Wikivoyage',
  229. 'DATA': 'Wikidata',
  230. 'NEWS': 'Wikinews',
  231. 'COMMONS': 'Wikimedia Commons'}
  232. BC_VARIANT = {'PIC': 'mit Bild',
  233. 'NOPIC': 'ohne Bild'}
  234. class BusinessCard(Extern):
  235. project = models.CharField(max_length=20, choices=PROJECT_CHOICE.items(),
  236. default='PEDIA', verbose_name='Wikimedia-Projekt',
  237. help_text='Für welches Wikimedia-Projekt möchtest Du Visitenkarten?')
  238. data = models.TextField(max_length=1000, verbose_name='Persönliche Daten für die Visitenkarten', default='',
  239. help_text=format_html("Bitte gib hier alle persönlichen Daten an, und zwar genau so,<br>\
  240. wie sie (auch in der entsprechenden Reihenfolge) auf den Visitenkarten stehen sollen<br>\
  241. (z.B. Vorname Nachname, Benutzer:/Benutzerin:, Benutzer-/-innenname, Anschrift,<br>\
  242. Telefonnummer, E-Mail-Adresse usw.). Trenne die einzelnen Angaben durch Zeilenumbrüche.<br>\
  243. Hinweis: Telefonnummern bilden wir üblicherweise im internationalen Format gemäß<br>\
  244. DIN 5008 ab. Als anzugebende E-Mail-Adresse empfehlen wir dir eine Wikimedia-Projekt-<br>\
  245. Adresse, die du ebenfalls beantragen kannst, sofern du nicht bereits eine besitzt."))
  246. variant = models.CharField(max_length=5, choices=BC_VARIANT.items(),
  247. default='NOPIC', verbose_name='Variante',
  248. help_text=format_html('so sehen die Varianten aus: <a href="https://upload.wikimedia.org/wikipedia/commons/c/cd/Muster_Visitenkarten_WMDE_2018.jpg">\
  249. mit Bild</a> <a href="https://upload.wikimedia.org/wikipedia/commons/d/d3/Muster_Visitenkarte_WMDE.png">ohne Bild</a>' ))
  250. url_of_pic = models.CharField(max_length=200, verbose_name='Url des Bildes', default='', help_text="Bitte gib die Wikimedia-Commons-URL des Bildes an.")
  251. sent_to = models.TextField(max_length=1000, verbose_name='Versandadresse',
  252. default='', help_text="Bitte gib den Namen und die vollständige Adresse ein, an welche die Visitenkarten geschickt werden sollen.")
  253. send_data_to_print = models.BooleanField(default=False, verbose_name=format_html('Datenweitergabe erlauben'), help_text=format_html('Hiermit erlaube ich die Weitergabe meiner Daten (Name, Postadresse) an den von Wikimedia Deutschland ausgewählten Dienstleister (z. B. <a href="wir-machen-druck.de">wir-machen-druck.de</a>) zum Zwecke des direkten Versands der Druckerzeugnisse an mich.'))
  254. intern_notes = models.TextField(max_length=1000, blank=True)