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.

265 lines
14 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. class Volunteer(models.Model):
  6. realname = models.CharField(max_length=200, null=True, verbose_name="Realname",
  7. help_text="Bitte gib deinen Vornamen und deinen Nachnamen ein.")
  8. email = models.EmailField(max_length=200, null=True, verbose_name='E-Mail-Adresse',
  9. 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.'))
  10. # the following Fields are not supposed to be edited by users
  11. granted = models.BooleanField(null=True)
  12. granted_date = models.DateField(null=True)
  13. survey_mail_send = models.BooleanField(default=False, verbose_name='Keine Umfragemail schicken')
  14. survey_mail_date = models.DateField(verbose_name='Umfragemail wurde verschickt am', null=True, blank=True)
  15. @classmethod
  16. def set_granted(cl, key, b):
  17. obj = cl.objects.get(pk=key)
  18. obj.granted = b
  19. obj.granted_date = date.today()
  20. obj.save()
  21. class Meta:
  22. abstract = True
  23. class Extern(Volunteer):
  24. ''' abstract basis class for all data entered by extern volunteers '''
  25. username = models.CharField(max_length=200, null=True, verbose_name='Benutzer_innenname',
  26. help_text=format_html("Bitte gib den Namen ein, mit dem du dich<br>in den Wikimedia-Projekten registriert hast."))
  27. # the following Fields are not supposed to be edited by users
  28. service_id = models.CharField(max_length=15, null=True, blank=True)
  29. def save(self,*args,**kwargs):
  30. # is there a way to call super().save() only once?
  31. super().save(*args,**kwargs)
  32. self.service_id = type(self).__name__ + str(self.pk)
  33. super().save(*args,**kwargs)
  34. class Meta:
  35. abstract = True
  36. class ConcreteExtern(Extern):
  37. ''' needed because we can't initiate abstract base classes in the view'''
  38. pass
  39. class Account(models.Model):
  40. code = models.CharField('Kostenstelle', max_length=5, default="DEF",
  41. null=False, primary_key = True)
  42. description = models.CharField('Beschreibung', max_length=30, default='NO DESCRIPTION')
  43. def __str__(self):
  44. return f"{self.code} {self.description}"
  45. class Project(Volunteer):
  46. name = models.CharField(max_length=200, verbose_name='Name des Projekts')
  47. description = models.CharField(max_length=500, verbose_name="Kurzbeschreibung", null=True)
  48. start = models.DateField('Startdatum', null=True, blank= True)
  49. end = models.DateField('Erwartetes Projektende', null=True)
  50. otrs = models.URLField(max_length=300, null=True, verbose_name='OTRS-Link')
  51. plan = models.URLField(max_length=2000, null=True, blank=True, verbose_name="Link zum Förderplan")
  52. page = models.URLField(max_length=2000, null=True, blank=True, verbose_name="Link zur Projektseite")
  53. urls = models.CharField(max_length=2000, null=True, blank=True, verbose_name="Weitere Links")
  54. group = models.CharField(max_length=2000, null=True, blank=True, verbose_name="Mitorganisierende")
  55. location = models.CharField(max_length=2000, null=True, blank=True, verbose_name="Ort/Adresse/Location")
  56. participants_estimated = models.IntegerField(blank=True, null=True, verbose_name='Teilnehmende angefragt')
  57. participants_real = models.IntegerField(blank=True, null=True, verbose_name='Teilnehmende ausgezählt')
  58. insurance = models.BooleanField(default=False, verbose_name='Haftpflichtversicherung')
  59. insurance_technic = models.BooleanField(default=False, verbose_name='Technikversicherung Ausland')
  60. support = models.CharField(max_length=300, blank=True, null=True, verbose_name='Betreuungsperson und Vertretung')
  61. cost = models.IntegerField(blank=True, null=True)
  62. account = models.ForeignKey('Account', on_delete=models.CASCADE, null=True, to_field='code')
  63. granted_from = models.CharField(max_length=100,null=True,verbose_name='Bewilligt von')
  64. notes = models.CharField(max_length=1000,null=True,blank=True,verbose_name='Anmerkungen')
  65. # the following Fields are not supposed to be edited by users
  66. pid = models.CharField(max_length=15, null=True, blank=True)
  67. end_mail_send = models.BooleanField(null=True)
  68. status = models.CharField(max_length=3,choices=(('RUN', 'läuft'),('END','beendet')),default='RUN')
  69. persons = models.IntegerField(default=1)
  70. finance_id = models.CharField(max_length=15, null= True, blank=True)
  71. project_of_year = models.IntegerField(default=0)
  72. def save(self,*args,**kwargs):
  73. # is there a way to call super().save() only once?
  74. #print(f'1: {self.id}')
  75. # super().save(*args,**kwargs)
  76. #print(f'2: {self.id}')
  77. self.pid = str(self.account.code) + str(self.pk).zfill(3)
  78. #print(f'3: {self.id}')
  79. if not self.project_of_year:
  80. # self.finance_id = "1234000"
  81. year = self.start.year
  82. print(f"startyear: {year}")
  83. projects = Project.objects.filter(start__year=year)
  84. print(projects)
  85. if not projects:
  86. self.project_of_year = 1
  87. else:
  88. pass
  89. else:
  90. print(f"project_of_year is {self.project_of_year}")
  91. super().save(*args,**kwargs)
  92. def __str__(self):
  93. return f"{self.pid} {self.name}"
  94. class Intern(Volunteer):
  95. '''abstrat base class for data entry from /intern (except Project)'''
  96. request_url = models.URLField(max_length=2000, verbose_name='Antrag (URL)')
  97. class Meta:
  98. abstract = True
  99. class ConcreteVolunteer(Volunteer):
  100. ''' needed because we can't initiate abstract base classes in the view'''
  101. pass
  102. class HonoraryCertificate(Intern):
  103. ''' this class is also used for accreditations '''
  104. project = models.ForeignKey(Project, null=True, blank=True, on_delete=models.SET_NULL)
  105. def __str__(self):
  106. return "Certificate for " + self.realname
  107. TRANSPORT_CHOICES = {'BAHN': 'Bahn',
  108. 'NONE': 'Keine Fahrtkosten',
  109. 'OTHER': 'Sonstiges (mit Begründung)'}
  110. PAYEDBY_CHOICES = {'WMDE': 'WMDE',
  111. 'REQU': 'Antragstellender Mensch'}
  112. class Travel(Intern):
  113. project = models.ForeignKey(Project, on_delete=models.CASCADE)
  114. transport = models.CharField(max_length=5, choices=TRANSPORT_CHOICES.items(), default='BAHN')
  115. other_transport = models.CharField(max_length=200, null=True, blank=True, verbose_name='Sonstige Transportmittel (mit Begründung)')
  116. travelcost = models.CharField(max_length=10, default="0", verbose_name="Fahrtkosten")
  117. checkin = models.DateField(blank=True, null=True, verbose_name='Check In')
  118. checkout = models.DateField(blank=True, null=True, verbose_name='Check Out')
  119. payed_for_hotel_by = models.CharField(max_length=4, choices=PAYEDBY_CHOICES.items(), blank=True, null=True, verbose_name='Kostenauslage Hotel durch')
  120. payed_for_travel_by = models.CharField(max_length=4, choices=PAYEDBY_CHOICES.items(), blank=True, null=True, verbose_name='Kostenauslage Fahrt durch')
  121. notes = models.CharField(max_length=500, blank=True)
  122. #abstract base class for Library and IFG
  123. class Grant(Extern):
  124. cost = models.CharField(max_length=10, verbose_name='Kosten',
  125. help_text="Bitte gib die ungefähr zu erwartenden Kosten in Euro an.")
  126. notes = models.TextField(max_length=500, blank=True, verbose_name='Anmerkungen',
  127. help_text="Bitte gib an wofür Du das Stipendium verwenden willst.")
  128. class Meta:
  129. abstract = True
  130. TYPE_CHOICES = {'BIB': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/Zugang_zu_Fachliteratur#Bibliotheksstipendium">Bibliotheksstipendium</a>'),
  131. 'ELIT': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/Zugang_zu_Fachliteratur#eLiteraturstipendium">eLiteraturstipendium</a>'),
  132. '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>'),
  133. 'IFG': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/Gebührenerstattungen_für_Behördenanfragen">Kostenübernahme IFG-Anfrage</a>'),
  134. 'LIT': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/Zugang_zu_Fachliteratur#Literaturstipendium">Literaturstipendium</a>'),
  135. 'LIST': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/E-Mail-Adressen_und_Visitenkarten#Mailinglisten">Mailingliste</a>'),
  136. 'SOFT': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/Software-Stipendien">Softwarestipendium</a>'),
  137. 'VIS': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/E-Mail-Adressen_und_Visitenkarten#Visitenkarten">Visitenkarten</a>'),
  138. }
  139. # same model is used for Library, ELitStip and Software!
  140. class Library(Grant):
  141. type = models.CharField(
  142. max_length=4,
  143. choices=TYPE_CHOICES.items(), #attention: actually only BIB, ELIT, SOFT should be used here
  144. default='LIB',
  145. )
  146. library = models.CharField(max_length=200)
  147. duration = models.CharField(max_length=100, verbose_name="Dauer")
  148. def __str__(self):
  149. return self.library
  150. class Literature(Grant):
  151. info = models.CharField(max_length=500, verbose_name='Informationen zum Werk',
  152. help_text=format_html("Bitte gib alle Informationen zum benötigten Werk an,<br>\
  153. die eine eindeutige Identifizierung ermöglichen (Autor, Titel, Verlag, ISBN, ...)"))
  154. source = models.CharField(max_length=200, verbose_name='Bezugsquelle',
  155. help_text="Bitte gib an, wo du das Werk kaufen möchtest.")
  156. class IFG(Grant):
  157. url = models.URLField(max_length=2000, verbose_name="URL",
  158. help_text="Bitte gib den Link zu deiner Anfrage bei Frag den Staat an.")
  159. def __str__(self):
  160. return "IFG-Anfrage von " + self.realname
  161. DOMAIN_CHOICES = {'PEDIA': '@wikipedia.de',
  162. 'BOOKS': '@wikibooks.de',
  163. 'QUOTE': '@wikiquote.de',
  164. 'SOURCE': '@wikisource.de',
  165. 'VERSITY': '@wikiversity.de',}
  166. class Domain(Extern):
  167. domain = models.CharField(max_length=10,
  168. choices=DOMAIN_CHOICES.items(),
  169. default='PEDIA')
  170. class Meta:
  171. abstract = True
  172. MAIL_CHOICES = {'REALNAME': 'Vorname.Nachname',
  173. 'USERNAME': 'Username',
  174. 'OTHER': 'Sonstiges:'}
  175. class Email(Domain):
  176. address = models.CharField(max_length=50,
  177. choices=MAIL_CHOICES.items(),
  178. default='USERNAME', verbose_name='Adressbestandteil',
  179. help_text=format_html("Bitte gib hier den gewünschten Adressbestandteil an,<br>der sich vor der Domain befinden soll."))
  180. other = models.CharField(max_length=50,blank=True,null=True, verbose_name="Sonstiges")
  181. class List(Domain):
  182. address = models.CharField(max_length=50, default='NO_ADDRESS',
  183. verbose_name="Adressbestandteil für Projektmailingliste",
  184. help_text=format_html("Bitte gib hier den gewünschten Adressbestandteil an,<br>der sich vor der Domain befinden soll."))
  185. PROJECT_CHOICE = {'PEDIA': 'Wikipedia',
  186. 'SOURCE': 'Wikisource',
  187. 'BOOKS': 'Wikibooks',
  188. 'QUOTE': 'Wikiquote',
  189. 'VERSITY': 'Wikiversity',
  190. 'VOYAGE': 'Wikivoyage',
  191. 'DATA': 'Wikidata',
  192. 'NEWS': 'Wikinews',
  193. 'COMMONS': 'Wikimedia Commons'}
  194. BC_VARIANT = {'PIC': 'mit Bild',
  195. 'NOPIC': 'ohne Bild'}
  196. class BusinessCard(Extern):
  197. project = models.CharField(max_length=20, choices=PROJECT_CHOICE.items(),
  198. default='PEDIA', verbose_name='Wikimedia-Projekt',
  199. help_text='Für welches Wikimedia-Projekt möchtest Du Visitenkarten?')
  200. data = models.TextField(max_length=1000, verbose_name='Persönliche Daten für die Visitenkarten', default='',
  201. help_text=format_html("Bitte gib hier alle persönlichen Daten an, und zwar genau so,<br>\
  202. wie sie (auch in der entsprechenden Reihenfolge) auf den Visitenkarten stehen sollen<br>\
  203. (z.B. Vorname Nachname, Benutzer:/Benutzerin:, Benutzer-/-innenname, Anschrift,<br>\
  204. Telefonnummer, E-Mail-Adresse usw.). Trenne die einzelnen Angaben durch Zeilenumbrüche.<br>\
  205. Hinweis: Telefonnummern bilden wir üblicherweise im internationalen Format gemäß<br>\
  206. DIN 5008 ab. Als anzugebende E-Mail-Adresse empfehlen wir dir eine Wikimedia-Projekt-<br>\
  207. Adresse, die du ebenfalls beantragen kannst, sofern du nicht bereits eine besitzt."))
  208. variant = models.CharField(max_length=5, choices=BC_VARIANT.items(),
  209. default='NOPIC', verbose_name='Variante',
  210. help_text=format_html('so sehen die Varianten aus: <a href="https://upload.wikimedia.org/wikipedia/commons/c/cd/Muster_Visitenkarten_WMDE_2018.jpg">\
  211. mit Bild</a> <a href="https://upload.wikimedia.org/wikipedia/commons/d/d3/Muster_Visitenkarte_WMDE.png">ohne Bild</a>' ))
  212. sent_to = models.TextField(max_length=1000, verbose_name='Versandadresse',
  213. default='', help_text="Bitte gib den Namen und die vollständige Adresse ein, an welche die Visitenkarten geschickt werden sollen.")