source: main/waeup.uniben/trunk/src/waeup/uniben/students/browser.py @ 13580

Last change on this file since 13580 was 13335, checked in by Henrik Bettermann, 9 years ago

Line break must be at the end of line0.

  • Property svn:keywords set to Id
File size: 20.8 KB
Line 
1## $Id: browser.py 13335 2015-10-16 17:49:33Z henrik $
2##
3## Copyright (C) 2012 Uli Fouquet & Henrik Bettermann
4## This program is free software; you can redistribute it and/or modify
5## it under the terms of the GNU General Public License as published by
6## the Free Software Foundation; either version 2 of the License, or
7## (at your option) any later version.
8##
9## This program is distributed in the hope that it will be useful,
10## but WITHOUT ANY WARRANTY; without even the implied warranty of
11## MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12## GNU General Public License for more details.
13##
14## You should have received a copy of the GNU General Public License
15## along with this program; if not, write to the Free Software
16## Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
17##
18import grok
19from zope.i18n import translate
20from zope.schema.interfaces import ConstraintNotSatisfied
21from zope.formlib.textwidgets import BytesDisplayWidget
22from zope.component import getUtility
23from hurry.workflow.interfaces import IWorkflowInfo
24from waeup.kofa.interfaces import (
25    REQUESTED, IExtFileStore, IKofaUtils, IObjectHistory)
26from waeup.kofa.widgets.datewidget import FriendlyDatetimeDisplayWidget
27from waeup.kofa.browser.layout import action, KofaEditFormPage, UtilityView
28from waeup.kofa.students.browser import (
29    StudyLevelEditFormPage, StudyLevelDisplayFormPage,
30    StudentBasePDFFormPage, ExportPDFCourseRegistrationSlip,
31    CourseTicketDisplayFormPage, StudentTriggerTransitionFormPage,
32    msave, emit_lock_message,
33    StudentActivateView, StudentDeactivateView,
34    ExportPDFTranscriptSlip)
35from waeup.kofa.students.workflow import (CREATED, ADMITTED, PAID,
36    CLEARANCE, REQUESTED, RETURNING, CLEARED, REGISTERED, VALIDATED,
37    GRADUATED, TRANSCRIPT, FORBIDDEN_POSTGRAD_TRANS)
38from waeup.kofa.students.interfaces import IStudentsUtils, ICourseTicket
39from waeup.kofa.students.workflow import FORBIDDEN_POSTGRAD_TRANS
40from kofacustom.nigeria.students.browser import (
41    NigeriaOnlinePaymentDisplayFormPage,
42    NigeriaStudentBaseManageFormPage,
43    NigeriaStudentClearanceEditFormPage,
44    NigeriaOnlinePaymentAddFormPage,
45    NigeriaExportPDFPaymentSlip,
46    NigeriaExportPDFClearanceSlip,
47    NigeriaExportPDFBedTicketSlip,
48    NigeriaStudentPersonalDisplayFormPage,
49    NigeriaStudentPersonalManageFormPage,
50    NigeriaStudentPersonalEditFormPage
51    )
52
53from waeup.uniben.students.interfaces import (
54    ICustomStudent,
55    ICustomStudentOnlinePayment,
56    ICustomStudentStudyCourse,
57    ICustomStudentStudyLevel,
58    ICustomUGStudentClearance,
59    ICustomPGStudentClearance,
60    ICustomStudentPersonal,
61    ICustomStudentPersonalEdit)
62from waeup.uniben.interfaces import MessageFactory as _
63
64class CustomOnlinePaymentDisplayFormPage(NigeriaOnlinePaymentDisplayFormPage):
65    """ Page to view an online payment ticket
66    """
67    grok.context(ICustomStudentOnlinePayment)
68    form_fields = grok.AutoFields(ICustomStudentOnlinePayment).omit(
69        'provider_amt', 'gateway_amt', 'thirdparty_amt', 'p_item')
70    form_fields[
71        'creation_date'].custom_widget = FriendlyDatetimeDisplayWidget('le')
72    form_fields[
73        'payment_date'].custom_widget = FriendlyDatetimeDisplayWidget('le')
74
75class CustomStudentClearanceEditFormPage(NigeriaStudentClearanceEditFormPage):
76    """ View to edit student clearance data by student
77    """
78
79    @property
80    def form_fields(self):
81        if self.context.is_postgrad:
82            form_fields = grok.AutoFields(ICustomPGStudentClearance).omit(
83            'clearance_locked', 'nysc_location', 'clr_code', 'officer_comment',
84            'physical_clearance_date')
85        else:
86            form_fields = grok.AutoFields(ICustomUGStudentClearance).omit(
87            'clearance_locked', 'clr_code', 'officer_comment',
88            'physical_clearance_date')
89            form_fields['date_of_birth'].for_display = True
90            form_fields['nationality'].for_display = True
91            form_fields['lga'].for_display = True
92        return form_fields
93
94    def dataNotComplete(self):
95        store = getUtility(IExtFileStore)
96        if not store.getFileByContext(self.context, attr=u'birth_certificate.jpg'):
97            return _('No birth certificate uploaded.')
98        if not store.getFileByContext(self.context, attr=u'ref_let.jpg'):
99            return _('No guarantor/referee letter uploaded.')
100        if not store.getFileByContext(self.context, attr=u'acc_let.jpg'):
101            return _('No acceptance letter uploaded.')
102        if not store.getFileByContext(self.context, attr=u'fst_sit_scan.jpg'):
103            return _('No first sitting result uploaded.')
104        #if not store.getFileByContext(self.context, attr=u'scd_sit_scan.jpg'):
105        #    return _('No second sitting result uploaded.')
106        if not store.getFileByContext(self.context, attr=u'secr_cults.jpg'):
107            return _('No affidavit of non-membership of secret cults uploaded.')
108        return False
109
110class CustomOnlinePaymentAddFormPage(NigeriaOnlinePaymentAddFormPage):
111    """ Page to add an online payment ticket
112    """
113    form_fields = grok.AutoFields(ICustomStudentOnlinePayment).select(
114        'p_category')
115
116class CustomExportPDFPaymentSlip(NigeriaExportPDFPaymentSlip):
117    """Deliver a PDF slip of the context.
118    """
119    grok.context(ICustomStudentOnlinePayment)
120    form_fields = grok.AutoFields(ICustomStudentOnlinePayment).omit(
121        'provider_amt', 'gateway_amt', 'thirdparty_amt', 'p_item')
122    form_fields['creation_date'].custom_widget = FriendlyDatetimeDisplayWidget('le')
123    form_fields['payment_date'].custom_widget = FriendlyDatetimeDisplayWidget('le')
124
125
126class CustomExportPDFClearanceSlip(NigeriaExportPDFClearanceSlip):
127    """Deliver a PDF slip of the context.
128    """
129
130    @property
131    def omit_fields(self):
132        omit_fields = ('password', 'suspended', 'suspended_comment',
133                       'phone', 'adm_code', 'email', 'date_of_birth')
134        if self.context.faccode == 'JUPEB':
135            omit_fields += ('faculty', 'department')
136        return omit_fields
137
138    @property
139    def label(self):
140        portal_language = getUtility(IKofaUtils).PORTAL_LANGUAGE
141        line0 = ''
142        if self.context.faccode == 'JUPEB':
143            line0 = 'Joint Universities Preliminary Examinations Board (JUPEB)\n'
144        line1 = translate(_('Clearance Slip of'),
145            'waeup.kofa', target_language=portal_language) \
146            + ' %s' % self.context.display_fullname
147        return '%s%s' % (line0, line1)
148
149    def _sigsInFooter(self):
150        isStudent = getattr(
151            self.request.principal, 'user_type', None) == 'student'
152        if not isStudent and self.context.state in (CLEARED, RETURNING):
153            return (_('Date, Student Signature'),
154                    _('Date, Clearance Officer Signature'),
155                    )
156        return ()
157
158    def render(self):
159        studentview = StudentBasePDFFormPage(self.context.student,
160            self.request, self.omit_fields)
161        students_utils = getUtility(IStudentsUtils)
162        return students_utils.renderPDF(
163            self, 'clearance_slip.pdf',
164            self.context.student, studentview, signatures=self._signatures(),
165            sigs_in_footer=self._sigsInFooter(), show_scans=False,
166            omit_fields=self.omit_fields)
167
168
169class ExportClearanceInvitationSlip(UtilityView, grok.View):
170    """Deliver an invitation letter to physical clearance.
171
172    This form page is available only in Uniben.
173    """
174    grok.context(ICustomStudent)
175    grok.name('clearance_invitation_slip.pdf')
176    grok.require('waeup.viewStudent')
177    prefix = 'form'
178
179    label = u'Invitation Letter for Physical Clearance'
180
181    omit_fields = (
182        'suspended', 'phone', 'email',
183        'adm_code', 'suspended_comment',
184        'date_of_birth', 'current_level',
185        'department', 'current_mode',
186        'entry_session', 'matric_number', 'sex')
187
188    form_fields = []
189
190    @property
191    def note(self):
192        if self.context.physical_clearance_date:
193            return """
194<br /><br /><br /><br /><font size='12'>
195Dear %s,
196<br /><br /><br />
197You are invited for your physical clearance on:
198<br /><br />
199<strong>%s</strong>.
200<br /><br />
201Please bring along this letter of invitation to the University Main Auditorium
202<br /><br />
203on your clearance date.
204<br /><br /><br />
205Signed,
206<br /><br />
207The Registrar<br />
208</font>
209
210""" % (self.context.display_fullname, self.context.physical_clearance_date)
211        return
212
213
214    def update(self):
215        if self.context.student.state != REQUESTED \
216            or not  self.context.student.physical_clearance_date:
217            self.flash(_('Forbidden'), type="warning")
218            self.redirect(self.url(self.context))
219
220    def render(self):
221        studentview = StudentBasePDFFormPage(self.context.student,
222            self.request, self.omit_fields)
223        students_utils = getUtility(IStudentsUtils)
224        return students_utils.renderPDF(
225            self, 'clearance_invitation_slip',
226            self.context.student, studentview,
227            omit_fields=self.omit_fields,
228            note=self.note)
229
230class CustomStudentPersonalDisplayFormPage(
231    NigeriaStudentPersonalDisplayFormPage):
232    """ Page to display student personal data
233    """
234
235    form_fields = grok.AutoFields(ICustomStudentPersonal)
236    form_fields['perm_address'].custom_widget = BytesDisplayWidget
237    form_fields['next_kin_address'].custom_widget = BytesDisplayWidget
238    form_fields[
239        'personal_updated'].custom_widget = FriendlyDatetimeDisplayWidget('le')
240
241class CustomStudentPersonalManageFormPage(
242    NigeriaStudentPersonalManageFormPage):
243    """ Page to manage personal data
244    """
245
246    form_fields = grok.AutoFields(ICustomStudentPersonal)
247    form_fields['personal_updated'].for_display = True
248    form_fields[
249        'personal_updated'].custom_widget = FriendlyDatetimeDisplayWidget('le')
250
251class CstomStudentPersonalEditFormPage(NigeriaStudentPersonalEditFormPage):
252    """ Page to edit personal data
253    """
254    form_fields = grok.AutoFields(
255        ICustomStudentPersonalEdit).omit('personal_updated')
256
257class StudyCourseCOEditFormPage(KofaEditFormPage):
258    """ Page to edit the student study course data by clearance officers.
259
260    This form page is available only in Uniben.
261    """
262    grok.context(ICustomStudentStudyCourse)
263    grok.name('edit_level')
264    grok.require('waeup.clearStudent')
265    label = _('Edit current level')
266    pnav = 4
267    form_fields = grok.AutoFields(
268        ICustomStudentStudyCourse).select('current_level')
269
270    def update(self):
271        if not (self.context.is_current and
272            self.context.student.state == REQUESTED):
273            emit_lock_message(self)
274            return
275        super(StudyCourseCOEditFormPage, self).update()
276        return
277
278    @action(_('Save'), style='primary')
279    def save(self, **data):
280        try:
281            msave(self, **data)
282        except ConstraintNotSatisfied:
283            # The selected level might not exist in certificate
284            self.flash(_('Current level not available for certificate.'))
285            return
286        #notify(grok.ObjectModifiedEvent(self.context.__parent__))
287        return
288
289class CustomStudyLevelEditFormPage(StudyLevelEditFormPage):
290    """ Page to edit the student study level data by students.
291
292    """
293    grok.template('studyleveleditpage')
294
295class CustomStudyLevelDisplayFormPage(StudyLevelDisplayFormPage):
296    """ Page to display student study levels
297    """
298    grok.template('studylevelpage')
299
300class CustomExportPDFCourseRegistrationSlip(
301    ExportPDFCourseRegistrationSlip):
302    """Deliver a PDF slip of the context.
303    """
304
305    form_fields = grok.AutoFields(ICustomStudentStudyLevel).omit(
306        'level_verdict', 'gpa', 'level')
307
308    def update(self):
309        if self.context.student.state != REGISTERED \
310            or self.context.student.current_level != self.context.level:
311            self.flash(_('Forbidden'), type="warning")
312            self.redirect(self.url(self.context))
313
314    @property
315    def tabletitle(self):
316        portal_language = getUtility(IKofaUtils).PORTAL_LANGUAGE
317        tabletitle = []
318        tabletitle.append(translate(_('1st Semester Courses'), 'waeup.kofa',
319            target_language=portal_language))
320        tabletitle.append(translate(_('2nd Semester Courses'), 'waeup.kofa',
321            target_language=portal_language))
322        tabletitle.append(translate(_('Level Courses'), 'waeup.kofa',
323            target_language=portal_language))
324        tabletitle.append(translate(_('1st Trimester Courses'), 'waeup.kofa',
325            target_language=portal_language))
326        tabletitle.append(translate(_('2nd Trimester Courses'), 'waeup.kofa',
327            target_language=portal_language))
328        tabletitle.append(translate(_('3rd Trimester Courses'), 'waeup.kofa',
329            target_language=portal_language))
330        return tabletitle
331
332    def render(self):
333        portal_language = getUtility(IKofaUtils).PORTAL_LANGUAGE
334        Code = translate('Code', 'waeup.kofa', target_language=portal_language)
335        Title = translate('Title', 'waeup.kofa', target_language=portal_language)
336        Dept = translate('Dept.', 'waeup.kofa', target_language=portal_language)
337        Faculty = translate('Faculty', 'waeup.kofa', target_language=portal_language)
338        Cred = translate(_('Credits'), 'waeup.uniben', target_language=portal_language)
339        studentview = StudentBasePDFFormPage(self.context.student,
340            self.request, self.omit_fields)
341        students_utils = getUtility(IStudentsUtils)
342
343        tabledata = []
344        tableheader = []
345        contenttitle = []
346        for i in range(1,7):
347            tabledata.append(sorted(
348                [value for value in self.context.values() if value.semester == i],
349                key=lambda value: str(value.semester) + value.code))
350            tableheader.append([(Code,'code', 2.5),
351                             (Title,'title', 5),
352                             (Dept,'dcode', 1.5), (Faculty,'fcode', 1.5),
353                             (Cred, 'credits', 1.5),
354                             ])
355        return students_utils.renderPDF(
356            self, 'course_registration_slip.pdf',
357            self.context.student, studentview,
358            tableheader=tableheader,
359            tabledata=tabledata,
360            omit_fields=self.omit_fields
361            )
362
363class UnibenExportPDFCourseResultSlip(ExportPDFCourseRegistrationSlip):
364    """Deliver a PDF slip of the context.
365    """
366
367    grok.name('course_result_slip.pdf')
368
369    @property
370    def tabletitle(self):
371        portal_language = getUtility(IKofaUtils).PORTAL_LANGUAGE
372        tabletitle = []
373        tabletitle.append(translate(_('1st Semester Courses'), 'waeup.kofa',
374            target_language=portal_language))
375        tabletitle.append(translate(_('2nd Semester Courses'), 'waeup.kofa',
376            target_language=portal_language))
377        tabletitle.append(translate(_('Level Courses'), 'waeup.kofa',
378            target_language=portal_language))
379        tabletitle.append(translate(_('1st Trimester Courses'), 'waeup.kofa',
380            target_language=portal_language))
381        tabletitle.append(translate(_('2nd Trimester Courses'), 'waeup.kofa',
382            target_language=portal_language))
383        tabletitle.append(translate(_('3rd Trimester Courses'), 'waeup.kofa',
384            target_language=portal_language))
385        return tabletitle
386
387    @property
388    def label(self):
389        portal_language = getUtility(IKofaUtils).PORTAL_LANGUAGE
390        lang = self.request.cookies.get('kofa.language', portal_language)
391        level_title = translate(self.context.level_title, 'waeup.kofa',
392            target_language=lang)
393        return translate(_('Course Result Slip'),
394            'waeup.uniben', target_language=portal_language) \
395            + ' %s' % level_title
396
397    def render(self):
398        portal_language = getUtility(IKofaUtils).PORTAL_LANGUAGE
399        Code = translate('Code', 'waeup.kofa', target_language=portal_language)
400        Title = translate('Title', 'waeup.kofa', target_language=portal_language)
401        Dept = translate('Dept.', 'waeup.kofa', target_language=portal_language)
402        Faculty = translate('Faculty', 'waeup.kofa', target_language=portal_language)
403        Cred = translate(_('Credits'), 'waeup.uniben', target_language=portal_language)
404        Grade = translate('Grade', 'waeup.kofa', target_language=portal_language)
405        studentview = StudentBasePDFFormPage(self.context.student,
406            self.request, self.omit_fields)
407        students_utils = getUtility(IStudentsUtils)
408
409        tabledata = []
410        tableheader = []
411        contenttitle = []
412        for i in range(1,7):
413            tabledata.append(sorted(
414                [value for value in self.context.values() if value.semester == i],
415                key=lambda value: str(value.semester) + value.code))
416            tableheader.append([(Code,'code', 2.5),
417                             (Title,'title', 5),
418                             (Dept,'dcode', 1.5), (Faculty,'fcode', 1.5),
419                             (Cred, 'credits', 1.5),
420                             (Grade, 'grade', 1.5),
421                             ])
422        return students_utils.renderPDF(
423            self, 'course_registration_slip.pdf',
424            self.context.student, studentview,
425            tableheader=tableheader,
426            tabledata=tabledata,
427            omit_fields=self.omit_fields
428            )
429
430class CustomCourseTicketDisplayFormPage(CourseTicketDisplayFormPage):
431    """ Page to display course tickets
432    """
433    form_fields = grok.AutoFields(ICourseTicket).omit('score')
434
435class CustomStudentActivateView(StudentActivateView):
436    """ Activate student account
437    """
438
439    def update(self):
440        self.context.suspended = False
441        self.context.writeLogMessage(self, 'account activated')
442        history = IObjectHistory(self.context)
443        history.addMessage('Student account activated', user='undisclosed')
444        self.flash(_('Student account has been activated.'))
445        self.redirect(self.url(self.context))
446        return
447
448class CustomStudentDeactivateView(StudentDeactivateView):
449    """ Deactivate student account
450    """
451    def update(self):
452        self.context.suspended = True
453        self.context.writeLogMessage(self, 'account deactivated')
454        history = IObjectHistory(self.context)
455        history.addMessage('Student account deactivated', user='undisclosed')
456        self.flash(_('Student account has been deactivated.'))
457        self.redirect(self.url(self.context))
458        return
459
460class CustomExportPDFTranscriptSlip(ExportPDFTranscriptSlip):
461    """Deliver a PDF slip of the context.
462    """
463
464    def _sigsInFooter(self):
465        return []
466        #return (_('CERTIFIED TRUE COPY'),)
467
468    def _signatures(self):
469        return ([(
470            'Current HD<br /> D. R. (Exams & Records)<br /> '
471            'For: Registrar')],)
472
473    def render(self):
474        portal_language = getUtility(IKofaUtils).PORTAL_LANGUAGE
475        Term = translate(_('Term'), 'waeup.kofa', target_language=portal_language)
476        Code = translate(_('Code'), 'waeup.kofa', target_language=portal_language)
477        Title = translate(_('Title'), 'waeup.kofa', target_language=portal_language)
478        Cred = translate(_('Credits'), 'waeup.kofa', target_language=portal_language)
479        #Score = translate(_('Score'), 'waeup.kofa', target_language=portal_language)
480        Grade = translate(_('Grade'), 'waeup.kofa', target_language=portal_language)
481        studentview = StudentBasePDFFormPage(self.context.student,
482            self.request, self.omit_fields)
483        students_utils = getUtility(IStudentsUtils)
484
485        tableheader = [(Code,'code', 2.5),
486                         (Title,'title', 8.5),
487                         (Term, 'semester', 1.5),
488                         (Cred, 'credits', 1.5),
489                         #(Score, 'score', 1.5),
490                         (Grade, 'grade', 1.5),
491                         ]
492
493        return students_utils.renderPDFTranscript(
494            self, 'transcript.pdf',
495            self.context.student, studentview,
496            omit_fields=self.omit_fields,
497            tableheader=tableheader,
498            signatures=self._signatures(),
499            sigs_in_footer=self._sigsInFooter(),
500            )
501
502class CustomExportPDFBedTicketSlip(NigeriaExportPDFBedTicketSlip):
503    """Deliver a PDF slip of the context.
504    """
505
506    omit_fields = ('password', 'suspended', 'suspended_comment',
507        'phone', 'adm_code', 'email', 'date_of_birth')
508
509    def render(self):
510        studentview = StudentBasePDFFormPage(self.context.student,
511            self.request, self.omit_fields)
512        students_utils = getUtility(IStudentsUtils)
513
514        note = """
515<br /><br /><br /><br /><br /><font size="14">
516Please endeavour to pay your hostel maintenance charge within <br /><br />
5174 days of being allocated a space or else you are deemed to have <br /><br />
518voluntarily forfeited it and it goes back into circulation to be <br /><br />
519available for booking afresh!</font>
520"""
521
522        return students_utils.renderPDF(
523            self, 'bed_allocation_slip.pdf',
524            self.context.student, studentview,
525            omit_fields=self.omit_fields,
526            note=note)
Note: See TracBrowser for help on using the repository browser.