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

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

Adjust to changes in base package.

  • Property svn:keywords set to Id
File size: 20.8 KB
RevLine 
[8911]1## $Id: browser.py 13624 2016-01-17 07:28:57Z 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
[9210]20from zope.schema.interfaces import ConstraintNotSatisfied
[12856]21from zope.formlib.textwidgets import BytesDisplayWidget
[9380]22from zope.component import getUtility
[9281]23from hurry.workflow.interfaces import IWorkflowInfo
[11069]24from waeup.kofa.interfaces import (
25    REQUESTED, IExtFileStore, IKofaUtils, IObjectHistory)
[8911]26from waeup.kofa.widgets.datewidget import FriendlyDatetimeDisplayWidget
[12121]27from waeup.kofa.browser.layout import action, KofaEditFormPage, UtilityView
[9205]28from waeup.kofa.students.browser import (
[9686]29    StudyLevelEditFormPage, StudyLevelDisplayFormPage,
[13063]30    StudentBasePDFFormPage, ExportPDFCourseRegistrationSlip,
[10025]31    CourseTicketDisplayFormPage, StudentTriggerTransitionFormPage,
[11069]32    msave, emit_lock_message,
[13063]33    StudentActivateView, StudentDeactivateView,
34    ExportPDFTranscriptSlip)
[12845]35from waeup.kofa.students.workflow import (CREATED, ADMITTED, PAID,
36    CLEARANCE, REQUESTED, RETURNING, CLEARED, REGISTERED, VALIDATED,
37    GRADUATED, TRANSCRIPT, FORBIDDEN_POSTGRAD_TRANS)
[9686]38from waeup.kofa.students.interfaces import IStudentsUtils, ICourseTicket
[9251]39from waeup.kofa.students.workflow import FORBIDDEN_POSTGRAD_TRANS
[8911]40from kofacustom.nigeria.students.browser import (
41    NigeriaOnlinePaymentDisplayFormPage,
[9251]42    NigeriaStudentBaseManageFormPage,
[9380]43    NigeriaStudentClearanceEditFormPage,
[8911]44    NigeriaOnlinePaymentAddFormPage,
[13063]45    NigeriaExportPDFPaymentSlip,
46    NigeriaExportPDFClearanceSlip,
[13297]47    NigeriaExportPDFBedTicketSlip,
[12856]48    NigeriaStudentPersonalDisplayFormPage,
[13085]49    NigeriaStudentPersonalManageFormPage,
50    NigeriaStudentPersonalEditFormPage
[9281]51    )
[8911]52
[9210]53from waeup.uniben.students.interfaces import (
[12121]54    ICustomStudent,
[12856]55    ICustomStudentOnlinePayment,
56    ICustomStudentStudyCourse,
[11552]57    ICustomStudentStudyLevel,
58    ICustomUGStudentClearance,
[12856]59    ICustomPGStudentClearance,
[13085]60    ICustomStudentPersonal,
61    ICustomStudentPersonalEdit)
[9210]62from waeup.uniben.interfaces import MessageFactory as _
[8911]63
64class CustomOnlinePaymentDisplayFormPage(NigeriaOnlinePaymentDisplayFormPage):
65    """ Page to view an online payment ticket
66    """
67    grok.context(ICustomStudentOnlinePayment)
[9775]68    form_fields = grok.AutoFields(ICustomStudentOnlinePayment).omit(
[9993]69        'provider_amt', 'gateway_amt', 'thirdparty_amt', 'p_item')
[8911]70    form_fields[
71        'creation_date'].custom_widget = FriendlyDatetimeDisplayWidget('le')
72    form_fields[
73        'payment_date'].custom_widget = FriendlyDatetimeDisplayWidget('le')
74
[9380]75class CustomStudentClearanceEditFormPage(NigeriaStudentClearanceEditFormPage):
76    """ View to edit student clearance data by student
77    """
78
[11552]79    @property
80    def form_fields(self):
81        if self.context.is_postgrad:
82            form_fields = grok.AutoFields(ICustomPGStudentClearance).omit(
[12391]83            'clearance_locked', 'nysc_location', 'clr_code', 'officer_comment',
84            'physical_clearance_date')
[11552]85        else:
86            form_fields = grok.AutoFields(ICustomUGStudentClearance).omit(
[12391]87            'clearance_locked', 'clr_code', 'officer_comment',
88            'physical_clearance_date')
[11552]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
[9380]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.')
[9395]104        #if not store.getFileByContext(self.context, attr=u'scd_sit_scan.jpg'):
105        #    return _('No second sitting result uploaded.')
[9380]106        if not store.getFileByContext(self.context, attr=u'secr_cults.jpg'):
[9848]107            return _('No affidavit of non-membership of secret cults uploaded.')
[9380]108        return False
109
[8911]110class CustomOnlinePaymentAddFormPage(NigeriaOnlinePaymentAddFormPage):
111    """ Page to add an online payment ticket
112    """
113    form_fields = grok.AutoFields(ICustomStudentOnlinePayment).select(
114        'p_category')
115
[13063]116class CustomExportPDFPaymentSlip(NigeriaExportPDFPaymentSlip):
[8911]117    """Deliver a PDF slip of the context.
118    """
119    grok.context(ICustomStudentOnlinePayment)
[9775]120    form_fields = grok.AutoFields(ICustomStudentOnlinePayment).omit(
[9993]121        'provider_amt', 'gateway_amt', 'thirdparty_amt', 'p_item')
[8911]122    form_fields['creation_date'].custom_widget = FriendlyDatetimeDisplayWidget('le')
[9205]123    form_fields['payment_date'].custom_widget = FriendlyDatetimeDisplayWidget('le')
124
125
[13063]126class CustomExportPDFClearanceSlip(NigeriaExportPDFClearanceSlip):
[9551]127    """Deliver a PDF slip of the context.
128    """
129
[13303]130    @property
131    def omit_fields(self):
132        omit_fields = ('password', 'suspended', 'suspended_comment',
[13309]133                       'phone', 'adm_code', 'email', 'date_of_birth')
[13303]134        if self.context.faccode == 'JUPEB':
135            omit_fields += ('faculty', 'department')
136        return omit_fields
[11537]137
[13299]138    @property
139    def label(self):
140        portal_language = getUtility(IKofaUtils).PORTAL_LANGUAGE
[13300]141        line0 = ''
142        if self.context.faccode == 'JUPEB':
[13335]143            line0 = 'Joint Universities Preliminary Examinations Board (JUPEB)\n'
[13300]144        line1 = translate(_('Clearance Slip of'),
[13299]145            'waeup.kofa', target_language=portal_language) \
146            + ' %s' % self.context.display_fullname
[13335]147        return '%s%s' % (line0, line1)
[13299]148
[12846]149    def _sigsInFooter(self):
[12845]150        isStudent = getattr(
151            self.request.principal, 'user_type', None) == 'student'
[12852]152        if not isStudent and self.context.state in (CLEARED, RETURNING):
[12846]153            return (_('Date, Student Signature'),
154                    _('Date, Clearance Officer Signature'),
155                    )
156        return ()
[12845]157
[9551]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(),
[10271]165            sigs_in_footer=self._sigsInFooter(), show_scans=False,
166            omit_fields=self.omit_fields)
[9551]167
[12121]168
[13063]169class ExportClearanceInvitationSlip(UtilityView, grok.View):
[12121]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
[12263]179    label = u'Invitation Letter for Physical Clearance'
[12121]180
181    omit_fields = (
182        'suspended', 'phone', 'email',
183        'adm_code', 'suspended_comment',
184        'date_of_birth', 'current_level',
[12150]185        'department', 'current_mode',
[12121]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 """
[12147]194<br /><br /><br /><br /><font size='12'>
195Dear %s,
196<br /><br /><br />
[12264]197You are invited for your physical clearance on:
[12121]198<br /><br />
[12264]199<strong>%s</strong>.
200<br /><br />
[12265]201Please bring along this letter of invitation to the University Main Auditorium
[12147]202<br /><br />
[12263]203on your clearance date.
[12147]204<br /><br /><br />
[12263]205Signed,
[12147]206<br /><br />
207The Registrar<br />
[12121]208</font>
209
[12147]210""" % (self.context.display_fullname, self.context.physical_clearance_date)
[12121]211        return
212
[12122]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
[12121]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
[12856]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
[13085]251class CstomStudentPersonalEditFormPage(NigeriaStudentPersonalEditFormPage):
252    """ Page to edit personal data
253    """
254    form_fields = grok.AutoFields(
255        ICustomStudentPersonalEdit).omit('personal_updated')
256
[9210]257class StudyCourseCOEditFormPage(KofaEditFormPage):
[9281]258    """ Page to edit the student study course data by clearance officers.
[9210]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__))
[9281]287        return
288
289class CustomStudyLevelEditFormPage(StudyLevelEditFormPage):
290    """ Page to edit the student study level data by students.
291
292    """
[9686]293    grok.template('studyleveleditpage')
[9281]294
[9686]295class CustomStudyLevelDisplayFormPage(StudyLevelDisplayFormPage):
296    """ Page to display student study levels
297    """
298    grok.template('studylevelpage')
299
[13063]300class CustomExportPDFCourseRegistrationSlip(
301    ExportPDFCourseRegistrationSlip):
[9686]302    """Deliver a PDF slip of the context.
303    """
304
[9921]305    form_fields = grok.AutoFields(ICustomStudentStudyLevel).omit(
[12879]306        'level_verdict', 'gpa', 'level')
[9921]307
[12855]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
[10441]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
[9686]332    def render(self):
333        portal_language = getUtility(IKofaUtils).PORTAL_LANGUAGE
[9909]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)
[9686]339        studentview = StudentBasePDFFormPage(self.context.student,
340            self.request, self.omit_fields)
341        students_utils = getUtility(IStudentsUtils)
[10441]342
343        tabledata = []
344        tableheader = []
345        for i in range(1,7):
346            tabledata.append(sorted(
347                [value for value in self.context.values() if value.semester == i],
348                key=lambda value: str(value.semester) + value.code))
349            tableheader.append([(Code,'code', 2.5),
350                             (Title,'title', 5),
351                             (Dept,'dcode', 1.5), (Faculty,'fcode', 1.5),
352                             (Cred, 'credits', 1.5),
353                             ])
[9686]354        return students_utils.renderPDF(
355            self, 'course_registration_slip.pdf',
356            self.context.student, studentview,
[10441]357            tableheader=tableheader,
358            tabledata=tabledata,
[10271]359            omit_fields=self.omit_fields
[9908]360            )
[9848]361
[13063]362class UnibenExportPDFCourseResultSlip(ExportPDFCourseRegistrationSlip):
[9848]363    """Deliver a PDF slip of the context.
364    """
365
366    grok.name('course_result_slip.pdf')
367
368    @property
[10441]369    def tabletitle(self):
370        portal_language = getUtility(IKofaUtils).PORTAL_LANGUAGE
371        tabletitle = []
372        tabletitle.append(translate(_('1st Semester Courses'), 'waeup.kofa',
373            target_language=portal_language))
374        tabletitle.append(translate(_('2nd Semester Courses'), 'waeup.kofa',
375            target_language=portal_language))
376        tabletitle.append(translate(_('Level Courses'), 'waeup.kofa',
377            target_language=portal_language))
378        tabletitle.append(translate(_('1st Trimester Courses'), 'waeup.kofa',
379            target_language=portal_language))
380        tabletitle.append(translate(_('2nd Trimester Courses'), 'waeup.kofa',
381            target_language=portal_language))
382        tabletitle.append(translate(_('3rd Trimester Courses'), 'waeup.kofa',
383            target_language=portal_language))
384        return tabletitle
385
386    @property
[9848]387    def label(self):
388        portal_language = getUtility(IKofaUtils).PORTAL_LANGUAGE
389        lang = self.request.cookies.get('kofa.language', portal_language)
390        level_title = translate(self.context.level_title, 'waeup.kofa',
391            target_language=lang)
392        return translate(_('Course Result Slip'),
393            'waeup.uniben', target_language=portal_language) \
394            + ' %s' % level_title
395
396    def render(self):
397        portal_language = getUtility(IKofaUtils).PORTAL_LANGUAGE
[9909]398        Code = translate('Code', 'waeup.kofa', target_language=portal_language)
399        Title = translate('Title', 'waeup.kofa', target_language=portal_language)
400        Dept = translate('Dept.', 'waeup.kofa', target_language=portal_language)
401        Faculty = translate('Faculty', 'waeup.kofa', target_language=portal_language)
402        Cred = translate(_('Credits'), 'waeup.uniben', target_language=portal_language)
403        Grade = translate('Grade', 'waeup.kofa', target_language=portal_language)
[9848]404        studentview = StudentBasePDFFormPage(self.context.student,
405            self.request, self.omit_fields)
406        students_utils = getUtility(IStudentsUtils)
[10441]407
408        tabledata = []
409        tableheader = []
410        for i in range(1,7):
411            tabledata.append(sorted(
412                [value for value in self.context.values() if value.semester == i],
413                key=lambda value: str(value.semester) + value.code))
414            tableheader.append([(Code,'code', 2.5),
415                             (Title,'title', 5),
416                             (Dept,'dcode', 1.5), (Faculty,'fcode', 1.5),
417                             (Cred, 'credits', 1.5),
418                             (Grade, 'grade', 1.5),
419                             ])
[9908]420        return students_utils.renderPDF(
421            self, 'course_registration_slip.pdf',
422            self.context.student, studentview,
[10441]423            tableheader=tableheader,
424            tabledata=tabledata,
[10271]425            omit_fields=self.omit_fields
[9908]426            )
[9686]427
428class CustomCourseTicketDisplayFormPage(CourseTicketDisplayFormPage):
429    """ Page to display course tickets
430    """
431    form_fields = grok.AutoFields(ICourseTicket).omit('score')
[11069]432
[13063]433class CustomStudentActivateView(StudentActivateView):
[11069]434    """ Activate student account
435    """
436
437    def update(self):
438        self.context.suspended = False
439        self.context.writeLogMessage(self, 'account activated')
440        history = IObjectHistory(self.context)
441        history.addMessage('Student account activated', user='undisclosed')
442        self.flash(_('Student account has been activated.'))
443        self.redirect(self.url(self.context))
444        return
445
[13063]446class CustomStudentDeactivateView(StudentDeactivateView):
[11069]447    """ Deactivate student account
448    """
449    def update(self):
450        self.context.suspended = True
451        self.context.writeLogMessage(self, 'account deactivated')
452        history = IObjectHistory(self.context)
453        history.addMessage('Student account deactivated', user='undisclosed')
454        self.flash(_('Student account has been deactivated.'))
455        self.redirect(self.url(self.context))
456        return
[12845]457
[13063]458class CustomExportPDFTranscriptSlip(ExportPDFTranscriptSlip):
[12845]459    """Deliver a PDF slip of the context.
460    """
461
462    def _sigsInFooter(self):
463        return []
464        #return (_('CERTIFIED TRUE COPY'),)
465
466    def _signatures(self):
467        return ([(
468            'Current HD<br /> D. R. (Exams & Records)<br /> '
469            'For: Registrar')],)
470
471    def render(self):
472        portal_language = getUtility(IKofaUtils).PORTAL_LANGUAGE
473        Term = translate(_('Term'), 'waeup.kofa', target_language=portal_language)
474        Code = translate(_('Code'), 'waeup.kofa', target_language=portal_language)
475        Title = translate(_('Title'), 'waeup.kofa', target_language=portal_language)
476        Cred = translate(_('Credits'), 'waeup.kofa', target_language=portal_language)
477        #Score = translate(_('Score'), 'waeup.kofa', target_language=portal_language)
478        Grade = translate(_('Grade'), 'waeup.kofa', target_language=portal_language)
479        studentview = StudentBasePDFFormPage(self.context.student,
480            self.request, self.omit_fields)
481        students_utils = getUtility(IStudentsUtils)
482
483        tableheader = [(Code,'code', 2.5),
484                         (Title,'title', 8.5),
485                         (Term, 'semester', 1.5),
486                         (Cred, 'credits', 1.5),
487                         #(Score, 'score', 1.5),
488                         (Grade, 'grade', 1.5),
489                         ]
490
491        return students_utils.renderPDFTranscript(
492            self, 'transcript.pdf',
493            self.context.student, studentview,
494            omit_fields=self.omit_fields,
495            tableheader=tableheader,
496            signatures=self._signatures(),
497            sigs_in_footer=self._sigsInFooter(),
[13297]498            )
499
500class CustomExportPDFBedTicketSlip(NigeriaExportPDFBedTicketSlip):
501    """Deliver a PDF slip of the context.
502    """
503
[13317]504    omit_fields = ('password', 'suspended', 'suspended_comment',
505        'phone', 'adm_code', 'email', 'date_of_birth')
[13297]506
507    def render(self):
508        studentview = StudentBasePDFFormPage(self.context.student,
509            self.request, self.omit_fields)
510        students_utils = getUtility(IStudentsUtils)
511
512        note = """
[13328]513<br /><br /><br /><br /><br /><font size="14">
514Please endeavour to pay your hostel maintenance charge within <br /><br />
5154 days of being allocated a space or else you are deemed to have <br /><br />
516voluntarily forfeited it and it goes back into circulation to be <br /><br />
517available for booking afresh!</font>
518"""
[13297]519
520        return students_utils.renderPDF(
521            self, 'bed_allocation_slip.pdf',
522            self.context.student, studentview,
523            omit_fields=self.omit_fields,
524            note=note)
Note: See TracBrowser for help on using the repository browser.