mirror of
				https://gitlab.crans.org/bde/nk20
				synced 2025-10-26 05:23:18 +01:00 
			
		
		
		
	| @@ -4347,7 +4347,23 @@ | |||||||
|             "mask": 3, |             "mask": 3, | ||||||
|             "field": "", |             "field": "", | ||||||
|             "permanent": false, |             "permanent": false, | ||||||
|             "description": "Ajouter un membre au BDE ou à la Kfet" |             "description": "Faire adhérer BDE ou Kfet" | ||||||
|  |         } | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         "model": "permission.permission", | ||||||
|  |         "pk": 293, | ||||||
|  |         "fields": { | ||||||
|  |             "model": [ | ||||||
|  |                 "wei", | ||||||
|  |                 "weimembership" | ||||||
|  |             ], | ||||||
|  |             "query": "[\"AND\", {\"bus\": [\"membership\", \"weimembership\", \"bus\"]}, {\"club\": [\"club\"], \"club__weiclub__membership_end__gte\": [\"today\"]}]", | ||||||
|  |             "type": "change", | ||||||
|  |             "mask": 2, | ||||||
|  |             "field": "team", | ||||||
|  |             "permanent": false, | ||||||
|  |             "description": "Modifier l'équipe d'une adhésion WEI à son bus" | ||||||
|         } |         } | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
| @@ -4764,7 +4780,6 @@ | |||||||
|             "name": "Chef\u22c5fe de bus", |             "name": "Chef\u22c5fe de bus", | ||||||
|             "permissions": [ |             "permissions": [ | ||||||
|                 22, |                 22, | ||||||
|                 84, |  | ||||||
|                 115, |                 115, | ||||||
|                 117, |                 117, | ||||||
|                 118, |                 118, | ||||||
| @@ -4778,7 +4793,8 @@ | |||||||
|                 287, |                 287, | ||||||
|                 289, |                 289, | ||||||
|                 290, |                 290, | ||||||
|                 291 |                 291, | ||||||
|  |                 293 | ||||||
|             ] |             ] | ||||||
|         } |         } | ||||||
|     }, |     }, | ||||||
| @@ -4790,7 +4806,6 @@ | |||||||
|             "name": "Chef\u22c5fe d'\u00e9quipe", |             "name": "Chef\u22c5fe d'\u00e9quipe", | ||||||
|             "permissions": [ |             "permissions": [ | ||||||
|                 22, |                 22, | ||||||
|                 84, |  | ||||||
|                 116, |                 116, | ||||||
|                 123, |                 123, | ||||||
|                 124, |                 124, | ||||||
| @@ -4805,8 +4820,7 @@ | |||||||
|             "for_club": null, |             "for_club": null, | ||||||
|             "name": "\u00c9lectron libre", |             "name": "\u00c9lectron libre", | ||||||
|             "permissions": [ |             "permissions": [ | ||||||
|                 22, |                 22 | ||||||
|                 84 |  | ||||||
|             ] |             ] | ||||||
|         } |         } | ||||||
|     }, |     }, | ||||||
| @@ -4957,7 +4971,6 @@ | |||||||
|             "name": "Référent⋅e Bus", |             "name": "Référent⋅e Bus", | ||||||
|             "permissions": [ |             "permissions": [ | ||||||
|                 22, |                 22, | ||||||
|                 84, |  | ||||||
|                 115, |                 115, | ||||||
|                 117, |                 117, | ||||||
|                 118, |                 118, | ||||||
| @@ -4971,7 +4984,8 @@ | |||||||
|                 287, |                 287, | ||||||
|                 289, |                 289, | ||||||
|                 290, |                 290, | ||||||
|                 291 |                 291, | ||||||
|  |                 293 | ||||||
|             ] |             ] | ||||||
|         } |         } | ||||||
|     },  |     },  | ||||||
|   | |||||||
| @@ -14,7 +14,8 @@ from django.utils.translation import gettext_lazy as _ | |||||||
| from .base import WEISurvey, WEISurveyInformation, WEISurveyAlgorithm, WEIBusInformation | from .base import WEISurvey, WEISurveyInformation, WEISurveyAlgorithm, WEIBusInformation | ||||||
| from ...models import WEIMembership, Bus | from ...models import WEIMembership, Bus | ||||||
|  |  | ||||||
| WORDS = [ | WORDS = { | ||||||
|  |     'list': [ | ||||||
|         '13 organisé', '3ième mi temps', 'Années 2000', 'Apéro', 'BBQ', 'BP', 'Beauf', 'Binge drinking', 'Bon enfant', |         '13 organisé', '3ième mi temps', 'Années 2000', 'Apéro', 'BBQ', 'BP', 'Beauf', 'Binge drinking', 'Bon enfant', | ||||||
|         'Cartouche', 'Catacombes', 'Chansons paillardes', 'Chansons populaires', 'Chanteur', 'Chartreuse', 'Chill', |         'Cartouche', 'Catacombes', 'Chansons paillardes', 'Chansons populaires', 'Chanteur', 'Chartreuse', 'Chill', | ||||||
|         'Core', 'DJ', 'Dancefloor', 'Danse', 'David Guetta', 'Disco', 'Eau de vie', 'Électro', 'Escalade', 'Familial', |         'Core', 'DJ', 'Dancefloor', 'Danse', 'David Guetta', 'Disco', 'Eau de vie', 'Électro', 'Escalade', 'Familial', | ||||||
| @@ -23,7 +24,129 @@ WORDS = [ | |||||||
|         'Karaoké', 'LGBTQI+', 'Lady Gaga', 'Loup garou', 'Morning beer', 'Métal', 'Nuit blanche', 'Ovalie', 'Psychedelic', |         'Karaoké', 'LGBTQI+', 'Lady Gaga', 'Loup garou', 'Morning beer', 'Métal', 'Nuit blanche', 'Ovalie', 'Psychedelic', | ||||||
|         'Pétanque', 'Rave', 'Reggae', 'Rhum', 'Ricard', 'Rock', 'Rosé', 'Rétro', 'Séducteur', 'Techno', 'Thérapie taxi', |         'Pétanque', 'Rave', 'Reggae', 'Rhum', 'Ricard', 'Rock', 'Rosé', 'Rétro', 'Séducteur', 'Techno', 'Thérapie taxi', | ||||||
|         'Théâtre', 'Trap', 'Turn up', 'Underground', 'Volley', 'Wati B', 'Zinédine Zidane', |         'Théâtre', 'Trap', 'Turn up', 'Underground', 'Volley', 'Wati B', 'Zinédine Zidane', | ||||||
| ] |     ], | ||||||
|  |     'questions': { | ||||||
|  |         'Question 1': [ | ||||||
|  |             'Description 1', | ||||||
|  |             { | ||||||
|  |                 3: 'Réponse 1 Madagas[car]', | ||||||
|  |                 43: 'Réponse 1 Y2[KAR]', | ||||||
|  |                 2: 'Réponse 1 Tcherno[bus]', | ||||||
|  |                 45: 'Réponse 1 [Kar]tier', | ||||||
|  |                 1: 'Réponse 1 [Car]cassonne', | ||||||
|  |                 47: 'Réponse 1 O[car]ina', | ||||||
|  |                 48: 'Réponse 1 Show[bus]', | ||||||
|  |                 49: 'Réponse 1 [Car]ioca' | ||||||
|  |             } | ||||||
|  |         ], | ||||||
|  |         'Question 2': [ | ||||||
|  |             'Description 2', | ||||||
|  |             { | ||||||
|  |                 3: 'Réponse 2 Madagas[car]', | ||||||
|  |                 43: 'Réponse 2 Y2[KAR]', | ||||||
|  |                 2: 'Réponse 2 Tcherno[bus]', | ||||||
|  |                 45: 'Réponse 2 [Kar]tier', | ||||||
|  |                 1: 'Réponse 2 [Car]cassonne', | ||||||
|  |                 47: 'Réponse 2 O[car]ina', | ||||||
|  |                 48: 'Réponse 2 Show[bus]', | ||||||
|  |                 49: 'Réponse 2 [Car]ioca' | ||||||
|  |             } | ||||||
|  |         ], | ||||||
|  |         'Question 3': [ | ||||||
|  |             'Description 3', | ||||||
|  |             { | ||||||
|  |                 3: 'Réponse 3 Madagas[car]', | ||||||
|  |                 43: 'Réponse 3 Y2[KAR]', | ||||||
|  |                 2: 'Réponse 3 Tcherno[bus]', | ||||||
|  |                 45: 'Réponse 3 [Kar]tier', | ||||||
|  |                 1: 'Réponse 3 [Car]cassonne', | ||||||
|  |                 47: 'Réponse 3 O[car]ina', | ||||||
|  |                 48: 'Réponse 3 Show[bus]', | ||||||
|  |                 49: 'Réponse 3 [Car]ioca' | ||||||
|  |             } | ||||||
|  |         ], | ||||||
|  |         'Question 4': [ | ||||||
|  |             'Description 4', | ||||||
|  |             { | ||||||
|  |                 3: 'Réponse 4 Madagas[car]', | ||||||
|  |                 43: 'Réponse 4 Y2[KAR]', | ||||||
|  |                 2: 'Réponse 4 Tcherno[bus]', | ||||||
|  |                 45: 'Réponse 4 [Kar]tier', | ||||||
|  |                 1: 'Réponse 4 [Car]cassonne', | ||||||
|  |                 47: 'Réponse 4 O[car]ina', | ||||||
|  |                 48: 'Réponse 4 Show[bus]', | ||||||
|  |                 49: 'Réponse 4 [Car]ioca' | ||||||
|  |             } | ||||||
|  |         ], | ||||||
|  |         'Question 5': [ | ||||||
|  |             'Description 5', | ||||||
|  |             { | ||||||
|  |                 3: 'Réponse 5 Madagas[car]', | ||||||
|  |                 43: 'Réponse 5 Y2[KAR]', | ||||||
|  |                 2: 'Réponse 5 Tcherno[bus]', | ||||||
|  |                 45: 'Réponse 5 [Kar]tier', | ||||||
|  |                 1: 'Réponse 5 [Car]cassonne', | ||||||
|  |                 47: 'Réponse 5 O[car]ina', | ||||||
|  |                 48: 'Réponse 5 Show[bus]', | ||||||
|  |                 49: 'Réponse 5 [Car]ioca' | ||||||
|  |             } | ||||||
|  |         ], | ||||||
|  |         'Question 6': [ | ||||||
|  |             'Description 6', | ||||||
|  |             { | ||||||
|  |                 3: 'Réponse 6 Madagas[car]', | ||||||
|  |                 43: 'Réponse 6 Y2[KAR]', | ||||||
|  |                 2: 'Réponse 6 Tcherno[bus]', | ||||||
|  |                 45: 'Réponse 6 [Kar]tier', | ||||||
|  |                 1: 'Réponse 6 [Car]cassonne', | ||||||
|  |                 47: 'Réponse 6 O[car]ina', | ||||||
|  |                 48: 'Réponse 6 Show[bus]', | ||||||
|  |                 49: 'Réponse 6 [Car]ioca' | ||||||
|  |             } | ||||||
|  |         ], | ||||||
|  |         'Question 7': [ | ||||||
|  |             'Description 7', | ||||||
|  |             { | ||||||
|  |                 3: 'Réponse 7 Madagas[car]', | ||||||
|  |                 43: 'Réponse 7 Y2[KAR]', | ||||||
|  |                 2: 'Réponse 7 Tcherno[bus]', | ||||||
|  |                 45: 'Réponse 7 [Kar]tier', | ||||||
|  |                 1: 'Réponse 7 [Car]cassonne', | ||||||
|  |                 47: 'Réponse 7 O[car]ina', | ||||||
|  |                 48: 'Réponse 7 Show[bus]', | ||||||
|  |                 49: 'Réponse 7 [Car]ioca' | ||||||
|  |             } | ||||||
|  |         ], | ||||||
|  |         'Question 8': [ | ||||||
|  |             'Description 8', | ||||||
|  |             { | ||||||
|  |                 3: 'Réponse 8 Madagas[car]', | ||||||
|  |                 43: 'Réponse 8 Y2[KAR]', | ||||||
|  |                 2: 'Réponse 8 Tcherno[bus]', | ||||||
|  |                 45: 'Réponse 8 [Kar]tier', | ||||||
|  |                 1: 'Réponse 8 [Car]cassonne', | ||||||
|  |                 47: 'Réponse 8 O[car]ina', | ||||||
|  |                 48: 'Réponse 8 Show[bus]', | ||||||
|  |                 49: 'Réponse 8 [Car]ioca' | ||||||
|  |             } | ||||||
|  |         ], | ||||||
|  |         'Question 9': [ | ||||||
|  |             'Description 9', | ||||||
|  |             { | ||||||
|  |                 3: 'Réponse 9 Madagas[car]', | ||||||
|  |                 43: 'Réponse 9 Y2[KAR]', | ||||||
|  |                 2: 'Réponse 9 Tcherno[bus]', | ||||||
|  |                 45: 'Réponse 9 [Kar]tier', | ||||||
|  |                 1: 'Réponse 9 [Car]cassonne', | ||||||
|  |                 47: 'Réponse 9 O[car]ina', | ||||||
|  |                 48: 'Réponse 9 Show[bus]', | ||||||
|  |                 49: 'Réponse 9 [Car]ioca' | ||||||
|  |             } | ||||||
|  |         ] | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | NB_WORDS = 5 | ||||||
|  |  | ||||||
|  |  | ||||||
| class WEISurveyForm2025(forms.Form): | class WEISurveyForm2025(forms.Form): | ||||||
| @@ -32,11 +155,6 @@ class WEISurveyForm2025(forms.Form): | |||||||
|     Members choose 20 words, from which we calculate the best associated bus. |     Members choose 20 words, from which we calculate the best associated bus. | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     word = forms.ChoiceField( |  | ||||||
|         label=_("Choose a word:"), |  | ||||||
|         widget=forms.RadioSelect(), |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     def set_registration(self, registration): |     def set_registration(self, registration): | ||||||
|         """ |         """ | ||||||
|         Filter the bus selector with the buses of the current WEI. |         Filter the bus selector with the buses of the current WEI. | ||||||
| @@ -48,13 +166,18 @@ class WEISurveyForm2025(forms.Form): | |||||||
|             registration._force_save = True |             registration._force_save = True | ||||||
|             registration.save() |             registration.save() | ||||||
|  |  | ||||||
|         if self.data: |         rng = Random((information.step + 1) * information.seed) | ||||||
|             self.fields["word"].choices = [(w, w) for w in WORDS] |  | ||||||
|  |         if information.step == 0: | ||||||
|  |             self.fields["words"] = forms.MultipleChoiceField( | ||||||
|  |                 label=_(f"Choose {NB_WORDS} words:"), | ||||||
|  |                 choices=[(w, w) for w in WORDS['list']], | ||||||
|  |                 widget=forms.CheckboxSelectMultiple(), | ||||||
|  |                 required=True, | ||||||
|  |             ) | ||||||
|             if self.is_valid(): |             if self.is_valid(): | ||||||
|                 return |                 return | ||||||
|  |  | ||||||
|         rng = Random((information.step + 1) * information.seed) |  | ||||||
|  |  | ||||||
|             buses = WEISurveyAlgorithm2025.get_buses() |             buses = WEISurveyAlgorithm2025.get_buses() | ||||||
|             informations = {bus: WEIBusInformation2025(bus) for bus in buses} |             informations = {bus: WEIBusInformation2025(bus) for bus in buses} | ||||||
|             scores = sum((list(informations[bus].scores.values()) for bus in buses), []) |             scores = sum((list(informations[bus].scores.values()) for bus in buses), []) | ||||||
| @@ -63,19 +186,36 @@ class WEISurveyForm2025(forms.Form): | |||||||
|             else: |             else: | ||||||
|                 average_score = 0 |                 average_score = 0 | ||||||
|  |  | ||||||
|         preferred_words = {bus: [word for word in WORDS |             preferred_words = { | ||||||
|                                  if informations[bus].scores[word] >= average_score] |                 bus: [word for word in WORDS['list'] if informations[bus].scores[word] >= average_score] | ||||||
|                            for bus in buses} |                 for bus in buses | ||||||
|  |             } | ||||||
|  |  | ||||||
|         # Correction : proposer plusieurs mots différents à chaque étape |  | ||||||
|         n_choices = 4  # Nombre de mots à proposer à chaque étape |  | ||||||
|             all_preferred_words = set() |             all_preferred_words = set() | ||||||
|             for bus_words in preferred_words.values(): |             for bus_words in preferred_words.values(): | ||||||
|                 all_preferred_words.update(bus_words) |                 all_preferred_words.update(bus_words) | ||||||
|             all_preferred_words = list(all_preferred_words) |             all_preferred_words = list(all_preferred_words) | ||||||
|             rng.shuffle(all_preferred_words) |             rng.shuffle(all_preferred_words) | ||||||
|         words = all_preferred_words[:n_choices] |             self.fields["words"].choices = [(w, w) for w in all_preferred_words] | ||||||
|         self.fields["word"].choices = [(w, w) for w in words] |         else: | ||||||
|  |             questions = list(WORDS['questions'].items()) | ||||||
|  |             idx = information.step - 1 | ||||||
|  |             if idx < len(questions): | ||||||
|  |                 q, (desc, answers) = questions[idx] | ||||||
|  |                 choices = [(k, v) for k, v in answers.items()] | ||||||
|  |                 rng.shuffle(choices) | ||||||
|  |                 self.fields[q] = forms.ChoiceField( | ||||||
|  |                     label=desc, | ||||||
|  |                     choices=choices, | ||||||
|  |                     widget=forms.RadioSelect, | ||||||
|  |                     required=True, | ||||||
|  |                 ) | ||||||
|  |  | ||||||
|  |     def clean_words(self): | ||||||
|  |         data = self.cleaned_data['words'] | ||||||
|  |         if len(data) != NB_WORDS: | ||||||
|  |             raise forms.ValidationError(_(f"Please choose exactly {NB_WORDS} words")) | ||||||
|  |         return data | ||||||
|  |  | ||||||
|  |  | ||||||
| class WEIBusInformation2025(WEIBusInformation): | class WEIBusInformation2025(WEIBusInformation): | ||||||
| @@ -86,7 +226,7 @@ class WEIBusInformation2025(WEIBusInformation): | |||||||
|  |  | ||||||
|     def __init__(self, bus): |     def __init__(self, bus): | ||||||
|         self.scores = {} |         self.scores = {} | ||||||
|         for word in WORDS: |         for word in WORDS['list']: | ||||||
|             self.scores[word] = 0 |             self.scores[word] = 0 | ||||||
|         super().__init__(bus) |         super().__init__(bus) | ||||||
|  |  | ||||||
| @@ -108,7 +248,7 @@ class BusInformationForm2025(forms.ModelForm): | |||||||
|             except (json.JSONDecodeError, TypeError, AttributeError): |             except (json.JSONDecodeError, TypeError, AttributeError): | ||||||
|                 initial_scores = {} |                 initial_scores = {} | ||||||
|         if words is None: |         if words is None: | ||||||
|             words = WORDS |             words = WORDS['list'] | ||||||
|         self.words = words |         self.words = words | ||||||
|  |  | ||||||
|         choices = [(i, str(i)) for i in range(6)]  # [(0, '0'), (1, '1'), ..., (5, '5')] |         choices = [(i, str(i)) for i in range(6)]  # [(0, '0'), (1, '1'), ..., (5, '5')] | ||||||
| @@ -145,10 +285,26 @@ class WEISurveyInformation2025(WEISurveyInformation): | |||||||
|     step = 0 |     step = 0 | ||||||
|  |  | ||||||
|     def __init__(self, registration): |     def __init__(self, registration): | ||||||
|         for i in range(1, 21): |         for i in range(1, 5): | ||||||
|             setattr(self, "word" + str(i), None) |             setattr(self, "word" + str(i), None) | ||||||
|  |         for q in WORDS['questions']: | ||||||
|  |             setattr(self, q, None) | ||||||
|         super().__init__(registration) |         super().__init__(registration) | ||||||
|  |  | ||||||
|  |     def reset(self, registration): | ||||||
|  |         """ | ||||||
|  |         Réinitialise complètement le questionnaire : step, seed, mots choisis et réponses aux questions. | ||||||
|  |         """ | ||||||
|  |         self.step = 0 | ||||||
|  |         self.seed = 0 | ||||||
|  |         for i in range(1, 5): | ||||||
|  |             setattr(self, f"word{i}", None) | ||||||
|  |         for q in WORDS['questions']: | ||||||
|  |             setattr(self, q, None) | ||||||
|  |         self.save(registration) | ||||||
|  |         registration._force_save = True | ||||||
|  |         registration.save() | ||||||
|  |  | ||||||
|  |  | ||||||
| class WEISurvey2025(WEISurvey): | class WEISurvey2025(WEISurvey): | ||||||
|     """ |     """ | ||||||
| @@ -174,9 +330,19 @@ class WEISurvey2025(WEISurvey): | |||||||
|  |  | ||||||
|     @transaction.atomic |     @transaction.atomic | ||||||
|     def form_valid(self, form): |     def form_valid(self, form): | ||||||
|         word = form.cleaned_data["word"] |         if self.information.step == 0: | ||||||
|  |             words = form.cleaned_data['words'] | ||||||
|  |             for i, word in enumerate(words, 1): | ||||||
|  |                 setattr(self.information, "word" + str(i), word) | ||||||
|  |             self.information.step += 1 | ||||||
|  |             self.save() | ||||||
|  |         else: | ||||||
|  |             questions = list(WORDS['questions'].keys()) | ||||||
|  |             idx = self.information.step - 1 | ||||||
|  |             if idx < len(questions): | ||||||
|  |                 q = questions[idx] | ||||||
|  |                 setattr(self.information, q, form.cleaned_data[q]) | ||||||
|                 self.information.step += 1 |                 self.information.step += 1 | ||||||
|         setattr(self.information, "word" + str(self.information.step), word) |  | ||||||
|                 self.save() |                 self.save() | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
| @@ -187,7 +353,7 @@ class WEISurvey2025(WEISurvey): | |||||||
|         """ |         """ | ||||||
|         The survey is complete once the bus is chosen. |         The survey is complete once the bus is chosen. | ||||||
|         """ |         """ | ||||||
|         return self.information.step == 20 |         return self.information.step > len(WORDS['questions']) | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     @lru_cache() |     @lru_cache() | ||||||
| @@ -206,7 +372,8 @@ class WEISurvey2025(WEISurvey): | |||||||
|         bus_info = self.get_algorithm_class().get_bus_information(bus) |         bus_info = self.get_algorithm_class().get_bus_information(bus) | ||||||
|         # Score is the given score by the bus subtracted to the mid-score of the buses. |         # Score is the given score by the bus subtracted to the mid-score of the buses. | ||||||
|         s = sum(bus_info.scores[getattr(self.information, 'word' + str(i))] |         s = sum(bus_info.scores[getattr(self.information, 'word' + str(i))] | ||||||
|                 - self.word_mean(getattr(self.information, 'word' + str(i))) for i in range(1, 21)) / 20 |                 - self.word_mean(getattr(self.information, 'word' + str(i))) for i in range(1, 1 + NB_WORDS)) / NB_WORDS | ||||||
|  |         s += sum(1 for q in WORDS['questions'] if getattr(self.information, q) == str(bus.pk)) | ||||||
|         return s |         return s | ||||||
|  |  | ||||||
|     @lru_cache() |     @lru_cache() | ||||||
| @@ -243,6 +410,13 @@ class WEISurveyAlgorithm2025(WEISurveyAlgorithm): | |||||||
|     def get_bus_information_form(cls): |     def get_bus_information_form(cls): | ||||||
|         return BusInformationForm2025 |         return BusInformationForm2025 | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def get_buses(cls): | ||||||
|  |  | ||||||
|  |         if not hasattr(cls, '_buses'): | ||||||
|  |             cls._buses = Bus.objects.filter(wei__year=cls.get_survey_class().get_year(), size__gt=0).all().exclude(name='Staff') | ||||||
|  |         return cls._buses | ||||||
|  |  | ||||||
|     def run_algorithm(self, display_tqdm=False): |     def run_algorithm(self, display_tqdm=False): | ||||||
|         """ |         """ | ||||||
|         Gale-Shapley algorithm implementation. |         Gale-Shapley algorithm implementation. | ||||||
|   | |||||||
| @@ -71,7 +71,7 @@ class WEIRegistrationTable(tables.Table): | |||||||
|         'wei:wei_delete_registration', |         'wei:wei_delete_registration', | ||||||
|         args=[A('pk')], |         args=[A('pk')], | ||||||
|         orderable=False, |         orderable=False, | ||||||
|         verbose_name=_("delete"), |         verbose_name=_("Delete"), | ||||||
|         text=_("Delete"), |         text=_("Delete"), | ||||||
|         attrs={ |         attrs={ | ||||||
|             'th': { |             'th': { | ||||||
| @@ -136,8 +136,8 @@ class WEIRegistrationTable(tables.Table): | |||||||
|  |  | ||||||
| class WEIMembershipTable(tables.Table): | class WEIMembershipTable(tables.Table): | ||||||
|     user = tables.LinkColumn( |     user = tables.LinkColumn( | ||||||
|         'wei:wei_update_registration', |         'wei:wei_update_membership', | ||||||
|         args=[A('registration__pk')], |         args=[A('pk')], | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     year = tables.Column( |     year = tables.Column( | ||||||
|   | |||||||
| @@ -50,7 +50,7 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
|                     {% endif %} |                     {% endif %} | ||||||
|  |  | ||||||
|                     {% if club.deposit_amount > 0 %} |                     {% if club.deposit_amount > 0 %} | ||||||
|                     <dt class="col-xl-6">{% trans 'Deposit amount'|capfirst %}</dt> |                     <dt class="col-xl-6">{% trans 'deposit amount'|capfirst %}</dt> | ||||||
|                     <dd class="col-xl-6">{{ club.deposit_amount|pretty_money }}</dd> |                     <dd class="col-xl-6">{{ club.deposit_amount|pretty_money }}</dd> | ||||||
|                     {% endif %} |                     {% endif %} | ||||||
|  |  | ||||||
|   | |||||||
| @@ -39,6 +39,11 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
|             data-turbolinks="false"> |             data-turbolinks="false"> | ||||||
|             {% trans "Update my registration" %} |             {% trans "Update my registration" %} | ||||||
|         </a> |         </a> | ||||||
|  |         {% if not not_first_year %} | ||||||
|  |         <a class="btn btn-warning" href="{% url "wei:wei_survey" pk=my_registration.pk %}?reset=true" data-turbolinks="false"> | ||||||
|  |             {% trans "Restart survey" %} | ||||||
|  |         </a> | ||||||
|  |         {% endif %} | ||||||
|         {% endif %} |         {% endif %} | ||||||
|     </div> |     </div> | ||||||
|     {% endif %} |     {% endif %} | ||||||
|   | |||||||
| @@ -149,6 +149,7 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
|                         <li>{% blocktrans trimmed with amount=fee|pretty_money %} |                         <li>{% blocktrans trimmed with amount=fee|pretty_money %} | ||||||
|                             Membership fees: {{ amount }} |                             Membership fees: {{ amount }} | ||||||
|                         {% endblocktrans %}</li> |                         {% endblocktrans %}</li> | ||||||
|  |                         {% if not registration.first_year %} | ||||||
|                         {% if registration.deposit_type == 'note' %} |                         {% if registration.deposit_type == 'note' %} | ||||||
|                             <li>{% blocktrans trimmed with amount=club.deposit_amount|pretty_money %} |                             <li>{% blocktrans trimmed with amount=club.deposit_amount|pretty_money %} | ||||||
|                                 Deposit (by Note transaction): {{ amount }} |                                 Deposit (by Note transaction): {{ amount }} | ||||||
| @@ -158,6 +159,7 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
|                                 Deposit (by check): {{ amount }} |                                 Deposit (by check): {{ amount }} | ||||||
|                             {% endblocktrans %}</li> |                             {% endblocktrans %}</li> | ||||||
|                         {% endif %} |                         {% endif %} | ||||||
|  |                         {% endif %} | ||||||
|                         <li><strong>{% blocktrans trimmed with total=total_needed|pretty_money %} |                         <li><strong>{% blocktrans trimmed with total=total_needed|pretty_money %} | ||||||
|                             Total needed: {{ total }} |                             Total needed: {{ total }} | ||||||
|                         {% endblocktrans %}</strong></li> |                         {% endblocktrans %}</strong></li> | ||||||
| @@ -213,7 +215,6 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
|                 $("input[name='bus']:checked").each(function (ignored) { |                 $("input[name='bus']:checked").each(function (ignored) { | ||||||
|                     buses.push($(this).parent().text().trim()); |                     buses.push($(this).parent().text().trim()); | ||||||
|                 }); |                 }); | ||||||
|                 console.log(buses); |  | ||||||
|                 $("input[name='team']").each(function () { |                 $("input[name='team']").each(function () { | ||||||
|                     let label = $(this).parent(); |                     let label = $(this).parent(); | ||||||
|                     $(this).parent().addClass('d-none'); |                     $(this).parent().addClass('d-none'); | ||||||
|   | |||||||
							
								
								
									
										46
									
								
								apps/wei/templates/wei/weimembership_update.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								apps/wei/templates/wei/weimembership_update.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,46 @@ | |||||||
|  | {% extends "base.html" %} | ||||||
|  | {% comment %} | ||||||
|  | Copyright (C) 2018-2025 by BDE ENS Paris-Saclay | ||||||
|  | SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  | {% endcomment %} | ||||||
|  | {% load i18n crispy_forms_tags %} | ||||||
|  |  | ||||||
|  | {% block content %} | ||||||
|  | <div class="card bg-white mb-3"> | ||||||
|  |   <h3 class="card-header text-center"> | ||||||
|  |     {{ title }} | ||||||
|  |   </h3> | ||||||
|  |   <div class="card-body" id="form"> | ||||||
|  |     <form method="post"> | ||||||
|  |       {% csrf_token %} | ||||||
|  |       {{ form | crispy }} | ||||||
|  |       <button class="btn btn-primary" type="submit">{% trans "Submit"%}</button> | ||||||
|  |     </form> | ||||||
|  |   </div> | ||||||
|  | </div> | ||||||
|  | {% endblock %} | ||||||
|  |  | ||||||
|  | {% block extrajavascript %} | ||||||
|  |     <script> | ||||||
|  |         $(document).ready(function () { | ||||||
|  |             function refreshTeams() { | ||||||
|  |                 let buses = []; | ||||||
|  |                 $("input[name='bus']:checked").each(function (ignored) { | ||||||
|  |                     buses.push($(this).parent().text().trim()); | ||||||
|  |                 }); | ||||||
|  |                 $("input[name='team']").each(function () { | ||||||
|  |                     let label = $(this).parent(); | ||||||
|  |                     $(this).parent().addClass('d-none'); | ||||||
|  |                     buses.forEach(function (bus) { | ||||||
|  |                         if (label.text().includes(bus)) | ||||||
|  |                             label.removeClass('d-none'); | ||||||
|  |                     }); | ||||||
|  |                 }); | ||||||
|  |             } | ||||||
|  |      | ||||||
|  |             $("input[name='bus']").change(refreshTeams); | ||||||
|  |      | ||||||
|  |             refreshTeams(); | ||||||
|  |         }); | ||||||
|  |     </script> | ||||||
|  | {% endblock %} | ||||||
| @@ -6,7 +6,7 @@ import random | |||||||
| from django.contrib.auth.models import User | from django.contrib.auth.models import User | ||||||
| from django.test import TestCase | from django.test import TestCase | ||||||
|  |  | ||||||
| from ..forms.surveys.wei2025 import WEIBusInformation2025, WEISurvey2025, WORDS, WEISurveyInformation2025 | from ..forms.surveys.wei2025 import WEIBusInformation2025, WEISurvey2025, WORDS, NB_WORDS, WEISurveyInformation2025 | ||||||
| from ..models import Bus, WEIClub, WEIRegistration | from ..models import Bus, WEIClub, WEIRegistration | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -34,8 +34,8 @@ class TestWEIAlgorithm(TestCase): | |||||||
|             bus = Bus.objects.create(wei=self.wei, name=f"Bus {i}", size=10) |             bus = Bus.objects.create(wei=self.wei, name=f"Bus {i}", size=10) | ||||||
|             self.buses.append(bus) |             self.buses.append(bus) | ||||||
|             information = WEIBusInformation2025(bus) |             information = WEIBusInformation2025(bus) | ||||||
|             for word in WORDS: |             for word in WORDS['list']: | ||||||
|                 information.scores[word] = random.randint(0, 101) |                 information.scores[word] = random.randint(0, 6) | ||||||
|             information.save() |             information.save() | ||||||
|             bus.save() |             bus.save() | ||||||
|  |  | ||||||
| @@ -54,7 +54,7 @@ class TestWEIAlgorithm(TestCase): | |||||||
|             ) |             ) | ||||||
|             information = WEISurveyInformation2025(registration) |             information = WEISurveyInformation2025(registration) | ||||||
|             for j in range(1, 21): |             for j in range(1, 21): | ||||||
|                 setattr(information, f'word{j}', random.choice(WORDS)) |                 setattr(information, f'word{j}', random.choice(WORDS['list'])) | ||||||
|             information.step = 20 |             information.step = 20 | ||||||
|             information.save(registration) |             information.save(registration) | ||||||
|             registration.save() |             registration.save() | ||||||
| @@ -83,9 +83,11 @@ class TestWEIAlgorithm(TestCase): | |||||||
|                 birth_date='2000-01-01', |                 birth_date='2000-01-01', | ||||||
|             ) |             ) | ||||||
|             information = WEISurveyInformation2025(registration) |             information = WEISurveyInformation2025(registration) | ||||||
|             for j in range(1, 21): |             for j in range(1, 1 + NB_WORDS): | ||||||
|                 setattr(information, f'word{j}', random.choice(WORDS)) |                 setattr(information, f'word{j}', random.choice(WORDS['list'])) | ||||||
|             information.step = 20 |             for q in WORDS['questions']: | ||||||
|  |                 setattr(information, q, random.choice(list(WORDS['questions'][q][1].keys()))) | ||||||
|  |             information.step = len(WORDS['questions']) + 1 | ||||||
|             information.save(registration) |             information.save(registration) | ||||||
|             registration.save() |             registration.save() | ||||||
|  |  | ||||||
| @@ -106,6 +108,6 @@ class TestWEIAlgorithm(TestCase): | |||||||
|             max_score = buses[0][1] |             max_score = buses[0][1] | ||||||
|             penalty += (max_score - score) ** 2 |             penalty += (max_score - score) ** 2 | ||||||
|  |  | ||||||
|             self.assertLessEqual(max_score - score, 25)  # Always less than 25 % of tolerance |             self.assertLessEqual(max_score - score, 1)  # Always less than 25 % of tolerance | ||||||
|  |  | ||||||
|         self.assertLessEqual(penalty / 100, 25)  # Tolerance of 5 % |         self.assertLessEqual(penalty / 100, 25)  # Tolerance of 5 % | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ from .views import CurrentWEIDetailView, WEI1AListView, WEIListView, WEICreateVi | |||||||
|     WEIRegistrationsView, WEIMembershipsView, MemberListRenderView, BusInformationUpdateView, \ |     WEIRegistrationsView, WEIMembershipsView, MemberListRenderView, BusInformationUpdateView, \ | ||||||
|     BusCreateView, BusManageView, BusUpdateView, BusTeamCreateView, BusTeamManageView, BusTeamUpdateView, \ |     BusCreateView, BusManageView, BusUpdateView, BusTeamCreateView, BusTeamManageView, BusTeamUpdateView, \ | ||||||
|     WEIAttributeBus1AView, WEIAttributeBus1ANextView, WEIRegister1AView, WEIRegister2AView, WEIUpdateRegistrationView, \ |     WEIAttributeBus1AView, WEIAttributeBus1ANextView, WEIRegister1AView, WEIRegister2AView, WEIUpdateRegistrationView, \ | ||||||
|     WEIDeleteRegistrationView, WEIValidateRegistrationView, WEISurveyView, WEISurveyEndView, WEIClosedView |     WEIDeleteRegistrationView, WEIValidateRegistrationView, WEISurveyView, WEISurveyEndView, WEIClosedView, WEIUpdateMembershipView | ||||||
|  |  | ||||||
| app_name = 'wei' | app_name = 'wei' | ||||||
| urlpatterns = [ | urlpatterns = [ | ||||||
| @@ -43,4 +43,6 @@ urlpatterns = [ | |||||||
|     path('bus-1A/<int:pk>/', WEIAttributeBus1AView.as_view(), name="wei_bus_1A"), |     path('bus-1A/<int:pk>/', WEIAttributeBus1AView.as_view(), name="wei_bus_1A"), | ||||||
|     path('bus-1A/next/<int:pk>/', WEIAttributeBus1ANextView.as_view(), name="wei_bus_1A_next"), |     path('bus-1A/next/<int:pk>/', WEIAttributeBus1ANextView.as_view(), name="wei_bus_1A_next"), | ||||||
|     path('update-bus-info/<int:pk>/', BusInformationUpdateView.as_view(), name="update_bus_info"), |     path('update-bus-info/<int:pk>/', BusInformationUpdateView.as_view(), name="update_bus_info"), | ||||||
|  |  | ||||||
|  |     path('edit_membership/<int:pk>/', WEIUpdateMembershipView.as_view(), name="wei_update_membership"), | ||||||
| ] | ] | ||||||
|   | |||||||
| @@ -816,7 +816,10 @@ class WEIUpdateRegistrationView(ProtectQuerysetMixin, LoginRequiredMixin, Update | |||||||
|             del form.fields["deposit_check"] |             del form.fields["deposit_check"] | ||||||
|  |  | ||||||
|         # S'assurer que le champ deposit_type est obligatoire pour les 2A+ |         # S'assurer que le champ deposit_type est obligatoire pour les 2A+ | ||||||
|         if not self.object.first_year and "deposit_type" in form.fields: |         if "deposit_type" in form.fields: | ||||||
|  |             if self.object.first_year: | ||||||
|  |                 del form.fields["deposit_type"] | ||||||
|  |             else: | ||||||
|                 form.fields["deposit_type"].required = True |                 form.fields["deposit_type"].required = True | ||||||
|                 form.fields["deposit_type"].help_text = _("Choose how you want to pay the deposit") |                 form.fields["deposit_type"].help_text = _("Choose how you want to pay the deposit") | ||||||
|  |  | ||||||
| @@ -879,7 +882,6 @@ class WEIUpdateRegistrationView(ProtectQuerysetMixin, LoginRequiredMixin, Update | |||||||
|             information["preferred_roles_name"] = [role.name for role in choose_bus_form.cleaned_data["roles"]] |             information["preferred_roles_name"] = [role.name for role in choose_bus_form.cleaned_data["roles"]] | ||||||
|             form.instance.information = information |             form.instance.information = information | ||||||
|  |  | ||||||
|             # Sauvegarder le type de caution pour les 2A+ |  | ||||||
|             if "deposit_type" in form.cleaned_data: |             if "deposit_type" in form.cleaned_data: | ||||||
|                 form.instance.deposit_type = form.cleaned_data["deposit_type"] |                 form.instance.deposit_type = form.cleaned_data["deposit_type"] | ||||||
|             form.instance.save() |             form.instance.save() | ||||||
| @@ -1181,6 +1183,49 @@ class WEIValidateRegistrationView(ProtectQuerysetMixin, ProtectedCreateView): | |||||||
|         return reverse_lazy("wei:wei_registrations", kwargs={"pk": self.object.club.pk}) |         return reverse_lazy("wei:wei_registrations", kwargs={"pk": self.object.club.pk}) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class WEIUpdateMembershipView(ProtectQuerysetMixin, LoginRequiredMixin, UpdateView): | ||||||
|  |     """ | ||||||
|  |     Update a membership for the WEI | ||||||
|  |     """ | ||||||
|  |     model = WEIMembership | ||||||
|  |     context_object_name = "membership" | ||||||
|  |     template_name = "wei/weimembership_update.html" | ||||||
|  |     extra_context = {"title": _("Update WEI Membership")} | ||||||
|  |  | ||||||
|  |     def dispatch(self, request, *args, **kwargs): | ||||||
|  |         wei = self.get_object().registration.wei | ||||||
|  |         today = date.today() | ||||||
|  |         # We can't update a registration once the WEI is started and before the membership start date | ||||||
|  |         if today >= wei.date_start or today < wei.membership_start: | ||||||
|  |             return redirect(reverse_lazy('wei:wei_closed', args=(wei.pk,))) | ||||||
|  |         # Store the validate parameter in the view's state | ||||||
|  |         return super().dispatch(request, *args, **kwargs) | ||||||
|  |  | ||||||
|  |     def get_form(self): | ||||||
|  |         form = WEIMembershipForm( | ||||||
|  |             self.request.POST or None, | ||||||
|  |             self.request.FILES or None, | ||||||
|  |             instance=self.object, | ||||||
|  |             wei=self.object.registration.wei, | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         form.fields["roles"].initial = self.object.roles.all() | ||||||
|  |         form.fields["bus"].initial = self.object.bus | ||||||
|  |         form.fields["team"].initial = self.object.team | ||||||
|  |  | ||||||
|  |         del form.fields["credit_type"] | ||||||
|  |         del form.fields["credit_amount"] | ||||||
|  |         del form.fields["first_name"] | ||||||
|  |         del form.fields["last_name"] | ||||||
|  |         del form.fields["bank"] | ||||||
|  |  | ||||||
|  |         return form | ||||||
|  |  | ||||||
|  |     def get_success_url(self): | ||||||
|  |         print("get_success_url") | ||||||
|  |         return reverse_lazy("wei:wei_detail", kwargs={"pk": self.object.registration.wei.pk}) | ||||||
|  |  | ||||||
|  |  | ||||||
| class WEISurveyView(LoginRequiredMixin, BaseFormView, DetailView): | class WEISurveyView(LoginRequiredMixin, BaseFormView, DetailView): | ||||||
|     """ |     """ | ||||||
|     Display the survey for the WEI for first year members. |     Display the survey for the WEI for first year members. | ||||||
| @@ -1203,6 +1248,10 @@ class WEISurveyView(LoginRequiredMixin, BaseFormView, DetailView): | |||||||
|  |  | ||||||
|         if not self.survey: |         if not self.survey: | ||||||
|             self.survey = CurrentSurvey(obj) |             self.survey = CurrentSurvey(obj) | ||||||
|  |  | ||||||
|  |         if request.GET.get("reset") == "true": | ||||||
|  |             info = self.survey.information | ||||||
|  |             info.reset(obj) | ||||||
|         # If the survey is complete, then display the end page. |         # If the survey is complete, then display the end page. | ||||||
|         if self.survey.is_complete(): |         if self.survey.is_complete(): | ||||||
|             return redirect(reverse_lazy('wei:wei_survey_end', args=(self.survey.registration.pk,))) |             return redirect(reverse_lazy('wei:wei_survey_end', args=(self.survey.registration.pk,))) | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user