| [aba463b] | 1 | import forms.models |
|---|
| [90afb00] | 2 | import groups.models |
|---|
| 3 | import settings |
|---|
| 4 | |
|---|
| [00e16ed] | 5 | from django.contrib.auth.decorators import user_passes_test, login_required |
|---|
| [ea42397] | 6 | from django.views.generic import list_detail, ListView, DetailView |
|---|
| [aba463b] | 7 | from django.shortcuts import render_to_response, get_object_or_404 |
|---|
| [90afb00] | 8 | from django.template import RequestContext |
|---|
| [f921734] | 9 | from django.template import Context, Template |
|---|
| 10 | from django.template.loader import get_template |
|---|
| [90afb00] | 11 | from django.http import Http404, HttpResponseRedirect |
|---|
| 12 | from django.core.urlresolvers import reverse |
|---|
| [d199ba4] | 13 | from django.core.mail import EmailMessage, mail_admins |
|---|
| [90afb00] | 14 | from django.forms import Form |
|---|
| 15 | from django.forms import ModelForm |
|---|
| [b1a8ea7] | 16 | from django.forms import ModelChoiceField, ModelMultipleChoiceField |
|---|
| [ea42397] | 17 | from django.db.models import Q, Count |
|---|
| [aba463b] | 18 | |
|---|
| 19 | import datetime |
|---|
| 20 | |
|---|
| [90afb00] | 21 | ################# |
|---|
| 22 | # GENERIC VIEWS # |
|---|
| 23 | ################# |
|---|
| 24 | |
|---|
| 25 | class SelectGroupForm(Form): |
|---|
| 26 | group = ModelChoiceField(queryset=groups.models.Group.objects.all()) |
|---|
| [2f7114b] | 27 | def __init__(self, *args, **kwargs): |
|---|
| 28 | queryset = None |
|---|
| 29 | if 'queryset' in kwargs: |
|---|
| 30 | queryset = kwargs['queryset'] |
|---|
| 31 | del kwargs['queryset'] |
|---|
| [f921734] | 32 | super(SelectGroupForm, self).__init__(*args, **kwargs) |
|---|
| 33 | if queryset is not None: |
|---|
| 34 | self.fields["group"].queryset = queryset |
|---|
| [90afb00] | 35 | |
|---|
| [f921734] | 36 | def select_group(request, url_name_after, pagename='homepage', queryset=None, ): |
|---|
| [90afb00] | 37 | if request.method == 'POST': # If the form has been submitted... |
|---|
| [f921734] | 38 | # A form bound to the POST data |
|---|
| 39 | form = SelectGroupForm(request.POST, queryset=queryset, ) |
|---|
| [90afb00] | 40 | if form.is_valid(): # All validation rules pass |
|---|
| 41 | group = form.cleaned_data['group'].id |
|---|
| 42 | return HttpResponseRedirect(reverse(url_name_after, args=[group],)) # Redirect after POST |
|---|
| 43 | else: |
|---|
| [f921734] | 44 | form = SelectGroupForm(queryset=queryset, ) # An unbound form |
|---|
| [90afb00] | 45 | |
|---|
| 46 | context = { |
|---|
| 47 | 'form':form, |
|---|
| [0b488d4] | 48 | 'pagename':pagename, |
|---|
| [90afb00] | 49 | } |
|---|
| 50 | return render_to_response('forms/select.html', context, context_instance=RequestContext(request), ) |
|---|
| 51 | |
|---|
| 52 | ############################# |
|---|
| 53 | # FIRST-YEAR SUMMER MAILING # |
|---|
| 54 | ############################# |
|---|
| 55 | |
|---|
| [00e16ed] | 56 | @login_required |
|---|
| [97399af] | 57 | def fysm_by_years(request, year, category, ): |
|---|
| [aba463b] | 58 | if year is None: year = datetime.date.today().year |
|---|
| 59 | queryset = forms.models.FYSM.objects.filter(year=year).order_by('group__name') |
|---|
| [dafa3c8] | 60 | category_obj = None |
|---|
| [a10dd4b] | 61 | category_name = 'main' |
|---|
| [dafa3c8] | 62 | if category != None: |
|---|
| [c27da9e] | 63 | category_obj = get_object_or_404(forms.models.FYSMCategory, slug=category) |
|---|
| [a10dd4b] | 64 | category_name = category_obj.name |
|---|
| [c27da9e] | 65 | queryset = queryset.filter(categories=category_obj) |
|---|
| [a10dd4b] | 66 | forms.models.FYSMView.record_metric(request=request, fysm=None, year=year, page=category_name, ) |
|---|
| [c27da9e] | 67 | categories = forms.models.FYSMCategory.objects.all() |
|---|
| [aba463b] | 68 | return list_detail.object_list( |
|---|
| 69 | request, |
|---|
| 70 | queryset=queryset, |
|---|
| [90afb00] | 71 | template_name="fysm/fysm_listing.html", |
|---|
| [aba463b] | 72 | template_object_name="fysm", |
|---|
| 73 | extra_context={ |
|---|
| 74 | "year": year, |
|---|
| 75 | "pagename": "fysm", |
|---|
| [dafa3c8] | 76 | "category": category_obj, |
|---|
| [189506e] | 77 | "categories": categories, |
|---|
| [aba463b] | 78 | } |
|---|
| 79 | ) |
|---|
| [90afb00] | 80 | |
|---|
| [00e16ed] | 81 | @login_required |
|---|
| [aaa8e04] | 82 | def fysm_view(request, year, submission, ): |
|---|
| [ae881e5] | 83 | submit_obj = get_object_or_404(forms.models.FYSM, pk=submission,) |
|---|
| 84 | all = forms.models.FYSM.objects.only("id", "display_name", ) |
|---|
| 85 | try: |
|---|
| 86 | prev = all.filter(display_name__lt=submit_obj.display_name).order_by("-display_name")[0] |
|---|
| 87 | except IndexError: |
|---|
| 88 | prev = None |
|---|
| 89 | try: |
|---|
| 90 | next = all.filter(display_name__gt=submit_obj.display_name).order_by("display_name")[0] |
|---|
| 91 | except IndexError: |
|---|
| 92 | next = None |
|---|
| [a10dd4b] | 93 | forms.models.FYSMView.record_metric(request=request, fysm=submit_obj, year=year, page="detail", ) |
|---|
| [aaa8e04] | 94 | return list_detail.object_detail( |
|---|
| 95 | request, |
|---|
| 96 | forms.models.FYSM.objects, |
|---|
| 97 | object_id=submission, |
|---|
| 98 | template_name="fysm/fysm_detail.html", |
|---|
| 99 | template_object_name="fysm", |
|---|
| 100 | extra_context={ |
|---|
| 101 | "year": year, |
|---|
| 102 | "pagename": "fysm", |
|---|
| [ae881e5] | 103 | "prev": prev, |
|---|
| 104 | "next": next, |
|---|
| [aaa8e04] | 105 | }, |
|---|
| 106 | ) |
|---|
| 107 | |
|---|
| [f462be6] | 108 | def fysm_link(request, year, link_type, submission, ): |
|---|
| 109 | submit_obj = get_object_or_404(forms.models.FYSM, pk=submission,) |
|---|
| 110 | if submit_obj.year != int(year): |
|---|
| 111 | raise Http404("Year mismatch: fysm.year='%s', request's year='%s'" % (submit_obj.year, year, )) |
|---|
| 112 | if link_type == 'join': |
|---|
| 113 | url = submit_obj.join_url |
|---|
| 114 | elif link_type == 'website': |
|---|
| 115 | url = submit_obj.website |
|---|
| 116 | else: |
|---|
| 117 | raise Http404("Unknown link type") |
|---|
| [a10dd4b] | 118 | forms.models.FYSMView.record_metric(request=request, fysm=submit_obj, year=year, page=link_type, ) |
|---|
| [f462be6] | 119 | return HttpResponseRedirect(url) |
|---|
| 120 | |
|---|
| [f921734] | 121 | def select_group_fysm(request, ): |
|---|
| [2c0eaa1] | 122 | qobj = Q(activity_category__isnull = True) | ~(Q(activity_category__name='Dorm') | Q(activity_category__name='FSILG')) |
|---|
| [d3167b9] | 123 | queryset = groups.models.Group.active_groups.filter(qobj) |
|---|
| [f921734] | 124 | return select_group( |
|---|
| 125 | request, |
|---|
| 126 | url_name_after='fysm-manage', |
|---|
| 127 | pagename='fysm', |
|---|
| 128 | queryset=queryset, |
|---|
| 129 | ) |
|---|
| 130 | |
|---|
| [90afb00] | 131 | class FYSMRequestForm(ModelForm): |
|---|
| 132 | class Meta: |
|---|
| 133 | model = forms.models.FYSM |
|---|
| 134 | fields = ( |
|---|
| 135 | 'display_name', |
|---|
| 136 | 'website', |
|---|
| 137 | 'join_url', |
|---|
| 138 | 'contact_email', |
|---|
| 139 | 'description', |
|---|
| 140 | 'logo', |
|---|
| [3400018] | 141 | 'slide', |
|---|
| [57f8ffa] | 142 | 'tags', |
|---|
| [c27da9e] | 143 | 'categories', |
|---|
| [90afb00] | 144 | ) |
|---|
| 145 | |
|---|
| [00e16ed] | 146 | @login_required |
|---|
| [90afb00] | 147 | def fysm_manage(request, group, ): |
|---|
| 148 | year = datetime.date.today().year |
|---|
| 149 | group_obj = get_object_or_404(groups.models.Group, pk=group) |
|---|
| 150 | |
|---|
| 151 | initial = {} |
|---|
| 152 | try: |
|---|
| 153 | fysm_obj = forms.models.FYSM.objects.get(group=group_obj, year=year, ) |
|---|
| 154 | print "Successfully found", fysm_obj.__dict__ |
|---|
| 155 | except forms.models.FYSM.DoesNotExist: |
|---|
| 156 | fysm_obj = forms.models.FYSM() |
|---|
| 157 | fysm_obj.group = group_obj |
|---|
| 158 | fysm_obj.year = year |
|---|
| 159 | initial['display_name'] = group_obj.name |
|---|
| 160 | initial['year'] = year |
|---|
| 161 | initial['website'] = group_obj.website_url |
|---|
| 162 | initial['join_url'] = group_obj.website_url |
|---|
| 163 | initial['contact_email'] = group_obj.officer_email |
|---|
| 164 | |
|---|
| 165 | if request.method == 'POST': # If the form has been submitted... |
|---|
| 166 | form = FYSMRequestForm(request.POST, request.FILES, instance=fysm_obj, ) # A form bound to the POST data |
|---|
| 167 | |
|---|
| 168 | if form.is_valid(): # All validation rules pass |
|---|
| 169 | request_obj = form.save() |
|---|
| 170 | |
|---|
| [adfe18c] | 171 | view_path = reverse('fysm-view', args=[year, request_obj.pk, ]) |
|---|
| 172 | view_uri = '%s://%s%s' % (request.is_secure() and 'https' or 'http', |
|---|
| 173 | request.get_host(), view_path) |
|---|
| 174 | |
|---|
| [90afb00] | 175 | # Send email |
|---|
| 176 | tmpl = get_template('fysm/update_email.txt') |
|---|
| 177 | ctx = Context({ |
|---|
| 178 | 'group': group_obj, |
|---|
| 179 | 'fysm': fysm_obj, |
|---|
| [adfe18c] | 180 | 'view_uri': view_uri, |
|---|
| [90afb00] | 181 | 'submitter': request.user, |
|---|
| 182 | 'request': request, |
|---|
| 183 | 'sender': "ASA FYSM team", |
|---|
| 184 | }) |
|---|
| 185 | body = tmpl.render(ctx) |
|---|
| [d199ba4] | 186 | email = EmailMessage( |
|---|
| 187 | subject='FYSM entry for "%s" updated by "%s"' % ( |
|---|
| [90afb00] | 188 | group_obj.name, |
|---|
| 189 | request.user, |
|---|
| 190 | ), |
|---|
| [d199ba4] | 191 | body=body, |
|---|
| 192 | from_email='asa-fysm@mit.edu', |
|---|
| [c49fbe2] | 193 | to=[group_obj.officer_email, request.user.email, ], |
|---|
| [dbaeedf] | 194 | bcc=['asa-fysm-submissions@mit.edu', ] |
|---|
| [90afb00] | 195 | ) |
|---|
| [d199ba4] | 196 | email.send() |
|---|
| [90afb00] | 197 | return HttpResponseRedirect(reverse('fysm-thanks', args=[fysm_obj.pk],)) # Redirect after POST |
|---|
| 198 | |
|---|
| 199 | else: |
|---|
| 200 | form = FYSMRequestForm(instance=fysm_obj, initial=initial, ) # An unbound form |
|---|
| 201 | |
|---|
| 202 | context = { |
|---|
| 203 | 'group':group_obj, |
|---|
| 204 | 'fysm':fysm_obj, |
|---|
| 205 | 'form':form, |
|---|
| [07caee0] | 206 | 'categories':forms.models.FYSMCategory.objects.all(), |
|---|
| [90afb00] | 207 | 'pagename':'fysm', |
|---|
| 208 | } |
|---|
| 209 | return render_to_response('fysm/submit.html', context, context_instance=RequestContext(request), ) |
|---|
| 210 | |
|---|
| 211 | |
|---|
| 212 | def fysm_thanks(request, fysm, ): |
|---|
| 213 | year = datetime.date.today().year |
|---|
| 214 | fysm_obj = get_object_or_404(forms.models.FYSM, pk=fysm) |
|---|
| 215 | |
|---|
| 216 | context = { |
|---|
| 217 | 'group':fysm_obj.group, |
|---|
| 218 | 'fysm':fysm_obj, |
|---|
| 219 | 'pagename':'fysm', |
|---|
| 220 | } |
|---|
| 221 | return render_to_response('fysm/thanks.html', context, context_instance=RequestContext(request), ) |
|---|
| [5b834ab] | 222 | |
|---|
| 223 | ##################### |
|---|
| 224 | # Membership update # |
|---|
| 225 | ##################### |
|---|
| 226 | |
|---|
| 227 | class Form_GroupMembershipUpdate(ModelForm): |
|---|
| [b1a8ea7] | 228 | group = ModelChoiceField(queryset=groups.models.Group.active_groups.all()) |
|---|
| 229 | |
|---|
| [5b834ab] | 230 | def __init__(self, *args, **kwargs): |
|---|
| 231 | super(Form_GroupMembershipUpdate, self).__init__(*args, **kwargs) |
|---|
| 232 | self.fields['no_hazing'].required = True |
|---|
| 233 | |
|---|
| 234 | class Meta: |
|---|
| 235 | model = forms.models.GroupMembershipUpdate |
|---|
| 236 | fields = [ |
|---|
| 237 | 'group', |
|---|
| 238 | 'updater_title', |
|---|
| 239 | 'group_email', |
|---|
| 240 | 'officer_email', |
|---|
| [185fde1] | 241 | 'email_preface', |
|---|
| [5b834ab] | 242 | 'no_hazing', |
|---|
| [185fde1] | 243 | 'no_discrimination', |
|---|
| [5b834ab] | 244 | 'membership_definition', |
|---|
| 245 | 'num_undergrads', |
|---|
| 246 | 'num_grads', |
|---|
| [ceaf3bd] | 247 | 'num_alum', |
|---|
| 248 | 'num_other_affiliate', |
|---|
| [5b834ab] | 249 | 'num_other', |
|---|
| 250 | 'membership_list', |
|---|
| 251 | ] |
|---|
| 252 | |
|---|
| 253 | @login_required |
|---|
| 254 | def group_membership_update(request, ): |
|---|
| 255 | year = datetime.date.today().year |
|---|
| 256 | |
|---|
| 257 | initial = { |
|---|
| 258 | } |
|---|
| 259 | update_obj = forms.models.GroupMembershipUpdate() |
|---|
| 260 | update_obj.update_time = datetime.datetime.now() |
|---|
| 261 | update_obj.updater_name = request.user.username |
|---|
| 262 | |
|---|
| 263 | confirm_path = reverse('membership-confirm', ) |
|---|
| 264 | confirm_uri = '%s://%s%s' % (request.is_secure() and 'https' or 'http', |
|---|
| 265 | request.get_host(), confirm_path) |
|---|
| 266 | |
|---|
| 267 | if request.method == 'POST': # If the form has been submitted... |
|---|
| 268 | form = Form_GroupMembershipUpdate(request.POST, request.FILES, instance=update_obj) # A form bound to the POST data |
|---|
| 269 | |
|---|
| 270 | if form.is_valid(): # All validation rules pass |
|---|
| 271 | request_obj = form.save() |
|---|
| 272 | group_obj = request_obj.group |
|---|
| 273 | |
|---|
| 274 | |
|---|
| 275 | # Send email |
|---|
| 276 | tmpl = get_template('membership/anti-hazing.txt') |
|---|
| 277 | ctx = Context({ |
|---|
| 278 | 'update': request_obj, |
|---|
| 279 | 'group': group_obj, |
|---|
| 280 | 'submitter': request.user, |
|---|
| 281 | }) |
|---|
| 282 | body = tmpl.render(ctx) |
|---|
| 283 | email = EmailMessage( |
|---|
| [185fde1] | 284 | subject='Anti-Hazing and Non-Discrimination Acknowledgement for %s' % ( |
|---|
| [5b834ab] | 285 | group_obj.name, |
|---|
| 286 | ), |
|---|
| 287 | body=body, |
|---|
| 288 | from_email=request.user.email, |
|---|
| 289 | to=[request_obj.group_email, ], |
|---|
| 290 | cc=[request_obj.officer_email, ], |
|---|
| [5c68184] | 291 | bcc=['asa-db-outgoing@mit.edu', ], |
|---|
| [5b834ab] | 292 | ) |
|---|
| 293 | email.send() |
|---|
| 294 | |
|---|
| 295 | # Send email |
|---|
| 296 | tmpl = get_template('membership/submit-confirm-email.txt') |
|---|
| 297 | ctx = Context({ |
|---|
| 298 | 'update': request_obj, |
|---|
| 299 | 'group': group_obj, |
|---|
| 300 | 'submitter': request.user, |
|---|
| 301 | 'confirm_uri': confirm_uri, |
|---|
| 302 | }) |
|---|
| 303 | body = tmpl.render(ctx) |
|---|
| 304 | email = EmailMessage( |
|---|
| 305 | subject='ASA Membership Information for %s' % ( |
|---|
| 306 | group_obj.name, |
|---|
| 307 | ), |
|---|
| 308 | body=body, |
|---|
| 309 | from_email=request.user.email, |
|---|
| 310 | to=[request_obj.officer_email, ], |
|---|
| [5c68184] | 311 | bcc=['asa-db-outgoing@mit.edu', ], |
|---|
| [5b834ab] | 312 | ) |
|---|
| 313 | email.send() |
|---|
| 314 | |
|---|
| 315 | return HttpResponseRedirect(reverse('membership-thanks', )) # Redirect after POST |
|---|
| 316 | |
|---|
| 317 | else: |
|---|
| 318 | form = Form_GroupMembershipUpdate(initial=initial, ) # An unbound form |
|---|
| 319 | |
|---|
| 320 | context = { |
|---|
| 321 | 'form':form, |
|---|
| 322 | 'confirm_uri': confirm_uri, |
|---|
| 323 | 'pagename':'groups', |
|---|
| 324 | } |
|---|
| 325 | return render_to_response('membership/update.html', context, context_instance=RequestContext(request), ) |
|---|
| 326 | |
|---|
| 327 | class Form_PersonMembershipUpdate(ModelForm): |
|---|
| [b1a8ea7] | 328 | groups = ModelMultipleChoiceField(queryset=groups.models.Group.active_groups.all()) |
|---|
| [5b834ab] | 329 | class Meta: |
|---|
| 330 | model = forms.models.PersonMembershipUpdate |
|---|
| 331 | fields = [ |
|---|
| 332 | 'groups', |
|---|
| 333 | ] |
|---|
| 334 | |
|---|
| 335 | @login_required |
|---|
| 336 | def person_membership_update(request, ): |
|---|
| 337 | year = datetime.date.today().year |
|---|
| 338 | |
|---|
| 339 | initial = { |
|---|
| 340 | } |
|---|
| 341 | update_obj = forms.models.PersonMembershipUpdate() |
|---|
| 342 | update_obj.update_time = datetime.datetime.now() |
|---|
| 343 | update_obj.username = request.user.username |
|---|
| 344 | |
|---|
| 345 | if request.method == 'POST': # If the form has been submitted... |
|---|
| 346 | form = Form_PersonMembershipUpdate(request.POST, request.FILES, instance=update_obj) # A form bound to the POST data |
|---|
| 347 | |
|---|
| 348 | if form.is_valid(): # All validation rules pass |
|---|
| 349 | request_obj = form.save() |
|---|
| 350 | return HttpResponseRedirect(reverse('membership-thanks', )) # Redirect after POST |
|---|
| 351 | |
|---|
| 352 | else: |
|---|
| 353 | form = Form_PersonMembershipUpdate(initial=initial, ) # An unbound form |
|---|
| 354 | |
|---|
| 355 | context = { |
|---|
| 356 | 'form':form, |
|---|
| 357 | 'pagename':'groups', |
|---|
| 358 | } |
|---|
| 359 | return render_to_response('membership/confirm.html', context, context_instance=RequestContext(request), ) |
|---|
| [ea42397] | 360 | |
|---|
| 361 | |
|---|
| 362 | class View_GroupMembershipList(ListView): |
|---|
| 363 | context_object_name = "group_list" |
|---|
| 364 | template_name = "membership/submitted.html" |
|---|
| 365 | |
|---|
| 366 | def get_queryset(self): |
|---|
| 367 | group_updates = forms.models.GroupMembershipUpdate.objects.all() |
|---|
| 368 | group_updates = group_updates.annotate(num_confirms=Count('group__personmembershipupdate')) |
|---|
| 369 | return group_updates |
|---|
| 370 | |
|---|
| 371 | #def get_context_data(self, **kwargs): |
|---|
| 372 | # context = super(GroupHistoryView, self).get_context_data(**kwargs) |
|---|
| 373 | # if 'pk' in self.kwargs: |
|---|
| 374 | # group = get_object_or_404(groups.models.Group, pk=self.kwargs['pk']) |
|---|
| 375 | # context['title'] = "History for %s" % (group.name, ) |
|---|
| 376 | # else: |
|---|
| 377 | # context['title'] = "Recent Changes" |
|---|
| 378 | # return context |
|---|