source: main/waeup.kofa/trunk/src/waeup/kofa/applicants/browser.py @ 9004

Last change on this file since 9004 was 8983, checked in by Henrik Bettermann, 12 years ago

Add boolean field 'suspended' to IStudent and IApplicant and extend authentication (checkPassword) slightly. Test will follow

  • Property svn:keywords set to Id
File size: 39.7 KB
RevLine 
[5273]1## $Id: browser.py 8983 2012-07-12 11:43:12Z henrik $
[6078]2##
[7192]3## Copyright (C) 2011 Uli Fouquet & Henrik Bettermann
[5273]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.
[6078]8##
[5273]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.
[6078]13##
[5273]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##
[5824]18"""UI components for basic applicants and related components.
[5273]19"""
[7063]20import os
[6082]21import sys
[5273]22import grok
[7370]23from datetime import datetime, date
[8042]24from zope.event import notify
[7392]25from zope.component import getUtility, createObject, getAdapter
[8033]26from zope.catalog.interfaces import ICatalog
[7714]27from zope.i18n import translate
[7322]28from hurry.workflow.interfaces import (
29    IWorkflowInfo, IWorkflowState, InvalidTransitionError)
[7811]30from waeup.kofa.applicants.interfaces import (
[7363]31    IApplicant, IApplicantEdit, IApplicantsRoot,
[7683]32    IApplicantsContainer, IApplicantsContainerAdd,
[8033]33    MAX_UPLOAD_SIZE, IApplicantOnlinePayment, IApplicantsUtils,
[8037]34    IApplicantRegisterUpdate
[7363]35    )
[8404]36from waeup.kofa.applicants.applicant import search
[8636]37from waeup.kofa.applicants.workflow import (
38    INITIALIZED, STARTED, PAID, SUBMITTED, ADMITTED)
[7811]39from waeup.kofa.browser import (
[7819]40    KofaPage, KofaEditFormPage, KofaAddFormPage, KofaDisplayFormPage,
[7363]41    DEFAULT_PASSPORT_IMAGE_PATH)
[7811]42from waeup.kofa.browser.interfaces import ICaptchaManager
43from waeup.kofa.browser.breadcrumbs import Breadcrumb
[8314]44from waeup.kofa.browser.resources import toggleall
[7811]45from waeup.kofa.browser.layout import (
[8550]46    NullValidator, jsaction, action, UtilityView, JSAction)
[7811]47from waeup.kofa.browser.pages import add_local_role, del_local_roles
48from waeup.kofa.browser.resources import datepicker, tabs, datatable, warning
49from waeup.kofa.interfaces import (
[7819]50    IKofaObject, ILocalRolesAssignable, IExtFileStore, IPDF,
51    IFileStoreNameChooser, IPasswordValidator, IUserAccount, IKofaUtils)
[7811]52from waeup.kofa.interfaces import MessageFactory as _
53from waeup.kofa.permissions import get_users_with_local_roles
54from waeup.kofa.students.interfaces import IStudentsUtils
[8186]55from waeup.kofa.utils.helpers import string_from_bytes, file_size, now
[8170]56from waeup.kofa.widgets.datewidget import (
57    FriendlyDateDisplayWidget, FriendlyDateDisplayWidget,
58    FriendlyDatetimeDisplayWidget)
[8365]59from waeup.kofa.widgets.htmlwidget import HTMLDisplayWidget
[5320]60
[7819]61grok.context(IKofaObject) # Make IKofaObject the default context
[5273]62
[8550]63class SubmitJSAction(JSAction):
64
65    msg = _('\'You can not edit your application records after final submission.'
66            ' You really want to submit?\'')
67
68class submitaction(grok.action):
69
70    def __call__(self, success):
71        action = SubmitJSAction(self.label, success=success, **self.options)
72        self.actions.append(action)
73        return action
74
[8388]75class ApplicantsRootPage(KofaDisplayFormPage):
[5822]76    grok.context(IApplicantsRoot)
77    grok.name('index')
[6153]78    grok.require('waeup.Public')
[8388]79    form_fields = grok.AutoFields(IApplicantsRoot)
80    form_fields['description'].custom_widget = HTMLDisplayWidget
[7710]81    label = _('Application Section')
[8404]82    search_button = _('Search')
[5843]83    pnav = 3
[6012]84
85    def update(self):
[6067]86        super(ApplicantsRootPage, self).update()
[6012]87        return
88
[8388]89    @property
90    def introduction(self):
91        # Here we know that the cookie has been set
92        lang = self.request.cookies.get('kofa.language')
93        html = self.context.description_dict.get(lang,'')
94        if html == '':
95            portal_language = getUtility(IKofaUtils).PORTAL_LANGUAGE
96            html = self.context.description_dict.get(portal_language,'')
97        return html
98
[8404]99class ApplicantsSearchPage(KofaPage):
100    grok.context(IApplicantsRoot)
101    grok.name('search')
102    grok.require('waeup.viewApplication')
103    label = _('Search applicants')
104    search_button = _('Search')
105    pnav = 3
106
107    def update(self, *args, **kw):
108        datatable.need()
109        form = self.request.form
110        self.results = []
111        if 'searchterm' in form and form['searchterm']:
112            self.searchterm = form['searchterm']
113            self.searchtype = form['searchtype']
114        elif 'old_searchterm' in form:
115            self.searchterm = form['old_searchterm']
116            self.searchtype = form['old_searchtype']
117        else:
118            if 'search' in form:
119                self.flash(_('Empty search string'))
120            return
121        self.results = search(query=self.searchterm,
122            searchtype=self.searchtype, view=self)
123        if not self.results:
124            self.flash(_('No applicant found.'))
125        return
126
[7819]127class ApplicantsRootManageFormPage(KofaEditFormPage):
[5828]128    grok.context(IApplicantsRoot)
129    grok.name('manage')
[6107]130    grok.template('applicantsrootmanagepage')
[8388]131    form_fields = grok.AutoFields(IApplicantsRoot)
[7710]132    label = _('Manage application section')
[5843]133    pnav = 3
[7136]134    grok.require('waeup.manageApplication')
[8388]135    taboneactions = [_('Save')]
136    tabtwoactions = [_('Add applicants container'), _('Remove selected')]
137    tabthreeactions1 = [_('Remove selected local roles')]
138    tabthreeactions2 = [_('Add local role')]
[7710]139    subunits = _('Applicants Containers')
[6078]140
[6069]141    def update(self):
142        tabs.need()
[6108]143        datatable.need()
[7330]144        warning.need()
[8388]145        self.tab1 = self.tab2 = self.tab3 = ''
146        qs = self.request.get('QUERY_STRING', '')
147        if not qs:
148            qs = 'tab1'
149        setattr(self, qs, 'active')
[6069]150        return super(ApplicantsRootManageFormPage, self).update()
[5828]151
[6184]152    def getLocalRoles(self):
153        roles = ILocalRolesAssignable(self.context)
154        return roles()
155
156    def getUsers(self):
157        """Get a list of all users.
158        """
159        for key, val in grok.getSite()['users'].items():
160            url = self.url(val)
161            yield(dict(url=url, name=key, val=val))
162
163    def getUsersWithLocalRoles(self):
164        return get_users_with_local_roles(self.context)
165
[7710]166    @jsaction(_('Remove selected'))
[6069]167    def delApplicantsContainers(self, **data):
168        form = self.request.form
[8388]169        if form.has_key('val_id'):
170            child_id = form['val_id']
171        else:
172            self.flash(_('No container selected!'))
173            self.redirect(self.url(self.context, '@@manage')+'?tab2')
174            return
[6069]175        if not isinstance(child_id, list):
176            child_id = [child_id]
177        deleted = []
178        for id in child_id:
179            try:
180                del self.context[id]
181                deleted.append(id)
182            except:
[7710]183                self.flash(_('Could not delete:') + ' %s: %s: %s' % (
[6069]184                        id, sys.exc_info()[0], sys.exc_info()[1]))
185        if len(deleted):
[7738]186            self.flash(_('Successfully removed: ${a}',
187                mapping = {'a':', '.join(deleted)}))
[8388]188        self.redirect(self.url(self.context, '@@manage')+'?tab2')
[6078]189        return
[5828]190
[7710]191    @action(_('Add applicants container'), validator=NullValidator)
[6069]192    def addApplicantsContainer(self, **data):
193        self.redirect(self.url(self.context, '@@add'))
[6078]194        return
195
[7710]196    @action(_('Add local role'), validator=NullValidator)
[6184]197    def addLocalRole(self, **data):
[7484]198        return add_local_role(self,3, **data)
[6184]199
[7710]200    @action(_('Remove selected local roles'))
[6184]201    def delLocalRoles(self, **data):
[7484]202        return del_local_roles(self,3,**data)
[6184]203
[8388]204    def _description(self):
205        view = ApplicantsRootPage(
206            self.context,self.request)
207        view.setUpWidgets()
208        return view.widgets['description']()
209
210    @action(_('Save'), style='primary')
211    def save(self, **data):
212        self.applyData(self.context, **data)
213        self.context.description_dict = self._description()
[8390]214        self.flash(_('Form has been saved.'))
[8388]215        return
216
[7819]217class ApplicantsContainerAddFormPage(KofaAddFormPage):
[5822]218    grok.context(IApplicantsRoot)
[7136]219    grok.require('waeup.manageApplication')
[5822]220    grok.name('add')
[6107]221    grok.template('applicantscontaineraddpage')
[7710]222    label = _('Add applicants container')
[5843]223    pnav = 3
[6078]224
[6103]225    form_fields = grok.AutoFields(
[7903]226        IApplicantsContainerAdd).omit('code').omit('title')
[6078]227
[6083]228    def update(self):
229        datepicker.need() # Enable jQuery datepicker in date fields.
230        return super(ApplicantsContainerAddFormPage, self).update()
231
[7710]232    @action(_('Add applicants container'))
[6069]233    def addApplicantsContainer(self, **data):
[6103]234        year = data['year']
235        code = u'%s%s' % (data['prefix'], year)
[7844]236        appcats_dict = getUtility(IApplicantsUtils).APP_TYPES_DICT
[7685]237        title = appcats_dict[data['prefix']][0]
238        title = u'%s %s/%s' % (title, year, year + 1)
[6087]239        if code in self.context.keys():
[6105]240            self.flash(
[7710]241                _('An applicants container for the same application type and entrance year exists already in the database.'))
[5822]242            return
243        # Add new applicants container...
[8009]244        container = createObject(u'waeup.ApplicantsContainer')
[6069]245        self.applyData(container, **data)
[6087]246        container.code = code
247        container.title = title
248        self.context[code] = container
[7710]249        self.flash(_('Added:') + ' "%s".' % code)
[7484]250        self.redirect(self.url(self.context, u'@@manage'))
[5822]251        return
[6078]252
[7710]253    @action(_('Cancel'), validator=NullValidator)
[6069]254    def cancel(self, **data):
[7484]255        self.redirect(self.url(self.context, '@@manage'))
[6078]256
[5845]257class ApplicantsRootBreadcrumb(Breadcrumb):
258    """A breadcrumb for applicantsroot.
259    """
260    grok.context(IApplicantsRoot)
[7710]261    title = _(u'Applicants')
[6078]262
[5845]263class ApplicantsContainerBreadcrumb(Breadcrumb):
264    """A breadcrumb for applicantscontainers.
265    """
266    grok.context(IApplicantsContainer)
[6319]267
[6153]268class ApplicantBreadcrumb(Breadcrumb):
269    """A breadcrumb for applicants.
270    """
271    grok.context(IApplicant)
[6319]272
[6153]273    @property
274    def title(self):
275        """Get a title for a context.
276        """
[7240]277        return self.context.application_number
[5828]278
[7250]279class OnlinePaymentBreadcrumb(Breadcrumb):
280    """A breadcrumb for payments.
281    """
282    grok.context(IApplicantOnlinePayment)
283
284    @property
285    def title(self):
286        return self.context.p_id
287
[8563]288class ApplicantsStatisticsPage(KofaDisplayFormPage):
289    """Some statistics about applicants in a container.
290    """
291    grok.context(IApplicantsContainer)
292    grok.name('statistics')
[8565]293    grok.require('waeup.viewApplicationStatistics')
[8563]294    grok.template('applicantcontainerstatistics')
295
296    @property
297    def label(self):
298        return "%s" % self.context.title
299
[7819]300class ApplicantsContainerPage(KofaDisplayFormPage):
[5830]301    """The standard view for regular applicant containers.
302    """
303    grok.context(IApplicantsContainer)
304    grok.name('index')
[6153]305    grok.require('waeup.Public')
[6029]306    grok.template('applicantscontainerpage')
[5850]307    pnav = 3
[6053]308
[8128]309    form_fields = grok.AutoFields(IApplicantsContainer).omit('title')
[8365]310    form_fields['description'].custom_widget = HTMLDisplayWidget
[8203]311    form_fields[
312        'startdate'].custom_widget = FriendlyDatetimeDisplayWidget('le')
313    form_fields[
314        'enddate'].custom_widget = FriendlyDatetimeDisplayWidget('le')
[6053]315
[5837]316    @property
[7708]317    def introduction(self):
[7833]318        # Here we know that the cookie has been set
319        lang = self.request.cookies.get('kofa.language')
[7708]320        html = self.context.description_dict.get(lang,'')
[8388]321        if html == '':
[7833]322            portal_language = getUtility(IKofaUtils).PORTAL_LANGUAGE
[7708]323            html = self.context.description_dict.get(portal_language,'')
[8388]324        return html
[7708]325
326    @property
[7467]327    def label(self):
[7493]328        return "%s" % self.context.title
[5837]329
[7819]330class ApplicantsContainerManageFormPage(KofaEditFormPage):
[5837]331    grok.context(IApplicantsContainer)
[5850]332    grok.name('manage')
[6107]333    grok.template('applicantscontainermanagepage')
[7903]334    form_fields = grok.AutoFields(IApplicantsContainer).omit('title')
[7710]335    taboneactions = [_('Save'),_('Cancel')]
[8684]336    tabtwoactions = [_('Remove selected'),_('Cancel'),
[8314]337        _('Create students from selected')]
[7710]338    tabthreeactions1 = [_('Remove selected local roles')]
339    tabthreeactions2 = [_('Add local role')]
[5844]340    # Use friendlier date widget...
[7136]341    grok.require('waeup.manageApplication')
[5850]342
343    @property
344    def label(self):
[7710]345        return _('Manage applicants container')
[5850]346
[5845]347    pnav = 3
[5837]348
[8547]349    @property
350    def showApplicants(self):
351        if len(self.context) < 5000:
352            return True
353        return False
354
[5837]355    def update(self):
[5850]356        datepicker.need() # Enable jQuery datepicker in date fields.
[5982]357        tabs.need()
[8314]358        toggleall.need()
[7484]359        self.tab1 = self.tab2 = self.tab3 = ''
360        qs = self.request.get('QUERY_STRING', '')
361        if not qs:
362            qs = 'tab1'
363        setattr(self, qs, 'active')
[7330]364        warning.need()
[6015]365        datatable.need()  # Enable jQurey datatables for contents listing
[6107]366        return super(ApplicantsContainerManageFormPage, self).update()
[5837]367
[6184]368    def getLocalRoles(self):
369        roles = ILocalRolesAssignable(self.context)
370        return roles()
371
372    def getUsers(self):
373        """Get a list of all users.
374        """
375        for key, val in grok.getSite()['users'].items():
376            url = self.url(val)
377            yield(dict(url=url, name=key, val=val))
378
379    def getUsersWithLocalRoles(self):
380        return get_users_with_local_roles(self.context)
381
[7708]382    def _description(self):
383        view = ApplicantsContainerPage(
384            self.context,self.request)
385        view.setUpWidgets()
386        return view.widgets['description']()
387
[7714]388    @action(_('Save'), style='primary')
[7489]389    def save(self, **data):
[5837]390        self.applyData(self.context, **data)
[7708]391        self.context.description_dict = self._description()
[8562]392        # Always refresh title. So we can change titles
393        # if APP_TYPES_DICT has been edited.
394        appcats_dict = getUtility(IApplicantsUtils).APP_TYPES_DICT
395        title = appcats_dict[self.context.prefix][0]
396        self.context.title = u'%s %s/%s' % (
397            title, self.context.year, self.context.year + 1)
[7710]398        self.flash(_('Form has been saved.'))
[5837]399        return
[6078]400
[7710]401    @jsaction(_('Remove selected'))
[6105]402    def delApplicant(self, **data):
[6189]403        form = self.request.form
404        if form.has_key('val_id'):
405            child_id = form['val_id']
406        else:
[7710]407            self.flash(_('No applicant selected!'))
[7484]408            self.redirect(self.url(self.context, '@@manage')+'?tab2')
[6189]409            return
410        if not isinstance(child_id, list):
411            child_id = [child_id]
412        deleted = []
413        for id in child_id:
414            try:
415                del self.context[id]
416                deleted.append(id)
417            except:
[7710]418                self.flash(_('Could not delete:') + ' %s: %s: %s' % (
[6189]419                        id, sys.exc_info()[0], sys.exc_info()[1]))
420        if len(deleted):
[7741]421            self.flash(_('Successfully removed: ${a}',
[7738]422                mapping = {'a':', '.join(deleted)}))
[7484]423        self.redirect(self.url(self.context, u'@@manage')+'?tab2')
[6189]424        return
[6105]425
[8314]426    @action(_('Create students from selected'))
427    def createStudents(self, **data):
428        form = self.request.form
429        if form.has_key('val_id'):
430            child_id = form['val_id']
431        else:
432            self.flash(_('No applicant selected!'))
433            self.redirect(self.url(self.context, '@@manage')+'?tab2')
434            return
435        if not isinstance(child_id, list):
436            child_id = [child_id]
437        created = []
438        for id in child_id:
439            success, msg = self.context[id].createStudent(view=self)
440            if success:
441                created.append(id)
442        if len(created):
443            self.flash(_('${a} students successfully created.',
444                mapping = {'a': len(created)}))
445        else:
446            self.flash(_('No student could be created.'))
447        self.redirect(self.url(self.context, u'@@manage')+'?tab2')
448        return
449
[7710]450    @action(_('Cancel'), validator=NullValidator)
[5837]451    def cancel(self, **data):
452        self.redirect(self.url(self.context))
453        return
[5886]454
[7710]455    @action(_('Add local role'), validator=NullValidator)
[6184]456    def addLocalRole(self, **data):
457        return add_local_role(self,3, **data)
[6105]458
[7710]459    @action(_('Remove selected local roles'))
[6184]460    def delLocalRoles(self, **data):
461        return del_local_roles(self,3,**data)
462
[7819]463class ApplicantAddFormPage(KofaAddFormPage):
[6622]464    """Add-form to add an applicant.
[6327]465    """
466    grok.context(IApplicantsContainer)
[7136]467    grok.require('waeup.manageApplication')
[6327]468    grok.name('addapplicant')
[7240]469    #grok.template('applicantaddpage')
470    form_fields = grok.AutoFields(IApplicant).select(
[7356]471        'firstname', 'middlename', 'lastname',
[7240]472        'email', 'phone')
[7714]473    label = _('Add applicant')
[6327]474    pnav = 3
475
[7714]476    @action(_('Create application record'))
[6327]477    def addApplicant(self, **data):
[8008]478        applicant = createObject(u'waeup.Applicant')
[7240]479        self.applyData(applicant, **data)
480        self.context.addApplicant(applicant)
[7714]481        self.flash(_('Applicant record created.'))
[7363]482        self.redirect(
483            self.url(self.context[applicant.application_number], 'index'))
[6327]484        return
485
[7819]486class ApplicantDisplayFormPage(KofaDisplayFormPage):
[8014]487    """A display view for applicant data.
488    """
[5273]489    grok.context(IApplicant)
490    grok.name('index')
[7113]491    grok.require('waeup.viewApplication')
[7200]492    grok.template('applicantdisplaypage')
[6320]493    form_fields = grok.AutoFields(IApplicant).omit(
[8983]494        'locked', 'course_admitted', 'password', 'suspended')
[7714]495    label = _('Applicant')
[5843]496    pnav = 3
[8922]497    hide_hint = False
[5273]498
[8046]499    @property
500    def separators(self):
501        return getUtility(IApplicantsUtils).SEPARATORS_DICT
502
[7063]503    def update(self):
504        self.passport_url = self.url(self.context, 'passport.jpg')
[7240]505        # Mark application as started if applicant logs in for the first time
[7272]506        usertype = getattr(self.request.principal, 'user_type', None)
507        if usertype == 'applicant' and \
508            IWorkflowState(self.context).getState() == INITIALIZED:
[7240]509            IWorkflowInfo(self.context).fireTransition('start')
[7063]510        return
511
[6196]512    @property
[7240]513    def hasPassword(self):
514        if self.context.password:
[7714]515            return _('set')
516        return _('unset')
[7240]517
518    @property
[6196]519    def label(self):
520        container_title = self.context.__parent__.title
[8096]521        return _('${a} <br /> Application Record ${b}', mapping = {
[7714]522            'a':container_title, 'b':self.context.application_number})
[6196]523
[7347]524    def getCourseAdmitted(self):
525        """Return link, title and code in html format to the certificate
526           admitted.
527        """
528        course_admitted = self.context.course_admitted
[7351]529        if getattr(course_admitted, '__parent__',None):
[7347]530            url = self.url(course_admitted)
531            title = course_admitted.title
532            code = course_admitted.code
533            return '<a href="%s">%s - %s</a>' %(url,code,title)
534        return ''
[6254]535
[7259]536class ApplicantBaseDisplayFormPage(ApplicantDisplayFormPage):
537    grok.context(IApplicant)
538    grok.name('base')
539    form_fields = grok.AutoFields(IApplicant).select(
540        'applicant_id', 'firstname', 'lastname','email', 'course1')
541
[7459]542class CreateStudentPage(UtilityView, grok.View):
[8636]543    """Create a student object from applicant data.
[7341]544    """
545    grok.context(IApplicant)
546    grok.name('createstudent')
547    grok.require('waeup.manageStudent')
548
549    def update(self):
[8314]550        msg = self.context.createStudent(view=self)[1]
[7341]551        self.flash(msg)
552        self.redirect(self.url(self.context))
553        return
554
555    def render(self):
556        return
557
[8636]558class CreateAllStudentsPage(UtilityView, grok.View):
559    """Create all student objects from applicant data
560    in a container.
561
562    This is a hidden page, no link or button will
563    be provided and only PortalManagers can do this.
564    """
565    grok.context(IApplicantsContainer)
566    grok.name('createallstudents')
567    grok.require('waeup.managePortal')
568
569    def update(self):
570        cat = getUtility(ICatalog, name='applicants_catalog')
571        results = list(cat.searchResults(state=(ADMITTED, ADMITTED)))
572        created = []
573        for result in results:
574            if not self.context.has_key(result.application_number):
575                continue
576            success, msg = result.createStudent(view=self)
577            if success:
578                created.append(result.applicant_id)
579            else:
580                ob_class = self.__implemented__.__name__.replace('waeup.kofa.','')
[8742]581                self.context.__parent__.logger.info(
582                    '%s - %s - %s' % (ob_class, result.applicant_id, msg))
[8636]583        if len(created):
584            self.flash(_('${a} students successfully created.',
585                mapping = {'a': len(created)}))
586        else:
587            self.flash(_('No student could be created.'))
588        self.redirect(self.url(self.context, u'@@manage')+'?tab2')
589        return
590
591    def render(self):
592        return
593
[8260]594class ApplicationFeePaymentAddPage(UtilityView, grok.View):
[7250]595    """ Page to add an online payment ticket
596    """
597    grok.context(IApplicant)
598    grok.name('addafp')
599    grok.require('waeup.payApplicant')
[8243]600    factory = u'waeup.ApplicantOnlinePayment'
[7250]601
602    def update(self):
603        for key in self.context.keys():
604            ticket = self.context[key]
605            if ticket.p_state == 'paid':
606                  self.flash(
[7714]607                      _('This type of payment has already been made.'))
[7250]608                  self.redirect(self.url(self.context))
609                  return
[8524]610        applicants_utils = getUtility(IApplicantsUtils)
611        container = self.context.__parent__
[8243]612        payment = createObject(self.factory)
[8524]613        error = applicants_utils.setPaymentDetails(container, payment)
614        if error is not None:
615            self.flash(error)
616            self.redirect(self.url(self.context))
617            return
[7250]618        self.context[payment.p_id] = payment
[7714]619        self.flash(_('Payment ticket created.'))
[8280]620        self.redirect(self.url(payment))
[7250]621        return
622
623    def render(self):
624        return
625
626
[7819]627class OnlinePaymentDisplayFormPage(KofaDisplayFormPage):
[7250]628    """ Page to view an online payment ticket
629    """
630    grok.context(IApplicantOnlinePayment)
631    grok.name('index')
632    grok.require('waeup.viewApplication')
633    form_fields = grok.AutoFields(IApplicantOnlinePayment)
[8170]634    form_fields[
635        'creation_date'].custom_widget = FriendlyDatetimeDisplayWidget('le')
636    form_fields[
637        'payment_date'].custom_widget = FriendlyDatetimeDisplayWidget('le')
[7250]638    pnav = 3
639
640    @property
641    def label(self):
[7714]642        return _('${a}: Online Payment Ticket ${b}', mapping = {
[8170]643            'a':self.context.__parent__.display_fullname,
644            'b':self.context.p_id})
[7250]645
[8420]646class OnlinePaymentApprovePage(UtilityView, grok.View):
647    """ Approval view
[7250]648    """
649    grok.context(IApplicantOnlinePayment)
[8420]650    grok.name('approve')
651    grok.require('waeup.managePortal')
[7250]652
653    def update(self):
[8428]654        success, msg, log = self.context.approveApplicantPayment()
655        if log is not None:
[8742]656            self.context.__parent__.writeLogMessage(self, log)
[8422]657        self.flash(msg)
[7250]658        return
659
660    def render(self):
661        self.redirect(self.url(self.context, '@@index'))
662        return
663
[7459]664class ExportPDFPaymentSlipPage(UtilityView, grok.View):
[7250]665    """Deliver a PDF slip of the context.
666    """
667    grok.context(IApplicantOnlinePayment)
[8262]668    grok.name('payment_slip.pdf')
[7250]669    grok.require('waeup.viewApplication')
670    form_fields = grok.AutoFields(IApplicantOnlinePayment)
[8173]671    form_fields['creation_date'].custom_widget = FriendlyDatetimeDisplayWidget('le')
672    form_fields['payment_date'].custom_widget = FriendlyDatetimeDisplayWidget('le')
[7250]673    prefix = 'form'
[8258]674    note = None
[7250]675
676    @property
[7714]677    def title(self):
[7819]678        portal_language = getUtility(IKofaUtils).PORTAL_LANGUAGE
[7811]679        return translate(_('Payment Data'), 'waeup.kofa',
[7714]680            target_language=portal_language)
681
682    @property
[7250]683    def label(self):
[7819]684        portal_language = getUtility(IKofaUtils).PORTAL_LANGUAGE
[8262]685        return translate(_('Online Payment Slip'),
[7811]686            'waeup.kofa', target_language=portal_language) \
[7714]687            + ' %s' % self.context.p_id
[7250]688
689    def render(self):
[8262]690        #if self.context.p_state != 'paid':
691        #    self.flash(_('Ticket not yet paid.'))
692        #    self.redirect(self.url(self.context))
693        #    return
[7259]694        applicantview = ApplicantBaseDisplayFormPage(self.context.__parent__,
[7250]695            self.request)
696        students_utils = getUtility(IStudentsUtils)
[8262]697        return students_utils.renderPDF(self,'payment_slip.pdf',
[8258]698            self.context.__parent__, applicantview, note=self.note)
[7250]699
[7459]700class ExportPDFPage(UtilityView, grok.View):
[6358]701    """Deliver a PDF slip of the context.
702    """
703    grok.context(IApplicant)
704    grok.name('application_slip.pdf')
[7136]705    grok.require('waeup.viewApplication')
[6358]706    prefix = 'form'
707
[8666]708    def update(self):
709        if self.context.state in ('initialized', 'started'):
710            self.flash(
711                _('Please pay before trying to download the application slip.'))
712            return self.redirect(self.url(self.context))
713        return
714
[6358]715    def render(self):
[7392]716        pdfstream = getAdapter(self.context, IPDF, name='application_slip')(
717            view=self)
[6358]718        self.response.setHeader(
719            'Content-Type', 'application/pdf')
[7392]720        return pdfstream
[6358]721
[7081]722def handle_img_upload(upload, context, view):
[7063]723    """Handle upload of applicant image.
[7081]724
725    Returns `True` in case of success or `False`.
726
727    Please note that file pointer passed in (`upload`) most probably
728    points to end of file when leaving this function.
[7063]729    """
[7081]730    size = file_size(upload)
731    if size > MAX_UPLOAD_SIZE:
[7714]732        view.flash(_('Uploaded image is too big!'))
[7081]733        return False
[7247]734    dummy, ext = os.path.splitext(upload.filename)
735    ext.lower()
736    if ext != '.jpg':
[7714]737        view.flash(_('jpg file extension expected.'))
[7247]738        return False
[7081]739    upload.seek(0) # file pointer moved when determining size
[7063]740    store = getUtility(IExtFileStore)
741    file_id = IFileStoreNameChooser(context).chooseName()
742    store.createFile(file_id, upload)
[7081]743    return True
[7063]744
[7819]745class ApplicantManageFormPage(KofaEditFormPage):
[6196]746    """A full edit view for applicant data.
747    """
748    grok.context(IApplicant)
[7200]749    grok.name('manage')
[7136]750    grok.require('waeup.manageApplication')
[6476]751    form_fields = grok.AutoFields(IApplicant)
[7351]752    form_fields['student_id'].for_display = True
[7378]753    form_fields['applicant_id'].for_display = True
[7200]754    grok.template('applicanteditpage')
[6322]755    manage_applications = True
[6196]756    pnav = 3
[7714]757    display_actions = [[_('Save'), _('Final Submit')],
758        [_('Add online payment ticket'),_('Remove selected tickets')]]
[6196]759
[8046]760    @property
761    def separators(self):
762        return getUtility(IApplicantsUtils).SEPARATORS_DICT
763
[6196]764    def update(self):
765        datepicker.need() # Enable jQuery datepicker in date fields.
[7330]766        warning.need()
[7200]767        super(ApplicantManageFormPage, self).update()
[6353]768        self.wf_info = IWorkflowInfo(self.context)
[7081]769        self.max_upload_size = string_from_bytes(MAX_UPLOAD_SIZE)
[7084]770        self.passport_changed = None
[6598]771        upload = self.request.form.get('form.passport', None)
772        if upload:
773            # We got a fresh upload
[7084]774            self.passport_changed = handle_img_upload(
775                upload, self.context, self)
[6196]776        return
777
778    @property
779    def label(self):
780        container_title = self.context.__parent__.title
[8096]781        return _('${a} <br /> Application Form ${b}', mapping = {
[7714]782            'a':container_title, 'b':self.context.application_number})
[6196]783
[6303]784    def getTransitions(self):
[6351]785        """Return a list of dicts of allowed transition ids and titles.
[6353]786
787        Each list entry provides keys ``name`` and ``title`` for
788        internal name and (human readable) title of a single
789        transition.
[6349]790        """
[8434]791        allowed_transitions = [t for t in self.wf_info.getManualTransitions()
792            if not t[0] == 'pay']
[7687]793        return [dict(name='', title=_('No transition'))] +[
[6355]794            dict(name=x, title=y) for x, y in allowed_transitions]
[6303]795
[7714]796    @action(_('Save'), style='primary')
[6196]797    def save(self, **data):
[7240]798        form = self.request.form
799        password = form.get('password', None)
800        password_ctl = form.get('control_password', None)
801        if password:
802            validator = getUtility(IPasswordValidator)
803            errors = validator.validate_password(password, password_ctl)
804            if errors:
805                self.flash( ' '.join(errors))
806                return
[7084]807        if self.passport_changed is False:  # False is not None!
808            return # error during image upload. Ignore other values
[6475]809        changed_fields = self.applyData(self.context, **data)
[7199]810        # Turn list of lists into single list
811        if changed_fields:
812            changed_fields = reduce(lambda x,y: x+y, changed_fields.values())
[7240]813        else:
814            changed_fields = []
815        if self.passport_changed:
816            changed_fields.append('passport')
817        if password:
818            # Now we know that the form has no errors and can set password ...
819            IUserAccount(self.context).setPassword(password)
820            changed_fields.append('password')
[7199]821        fields_string = ' + '.join(changed_fields)
[7085]822        trans_id = form.get('transition', None)
823        if trans_id:
824            self.wf_info.fireTransition(trans_id)
[7714]825        self.flash(_('Form has been saved.'))
[6644]826        if fields_string:
[8742]827            self.context.writeLogMessage(self, 'saved: % s' % fields_string)
[6196]828        return
829
[7250]830    def unremovable(self, ticket):
[7330]831        return False
[7250]832
833    # This method is also used by the ApplicantEditFormPage
834    def delPaymentTickets(self, **data):
835        form = self.request.form
836        if form.has_key('val_id'):
837            child_id = form['val_id']
838        else:
[7714]839            self.flash(_('No payment selected.'))
[7250]840            self.redirect(self.url(self.context))
841            return
842        if not isinstance(child_id, list):
843            child_id = [child_id]
844        deleted = []
845        for id in child_id:
846            # Applicants are not allowed to remove used payment tickets
847            if not self.unremovable(self.context[id]):
848                try:
849                    del self.context[id]
850                    deleted.append(id)
851                except:
[7714]852                    self.flash(_('Could not delete:') + ' %s: %s: %s' % (
[7250]853                            id, sys.exc_info()[0], sys.exc_info()[1]))
854        if len(deleted):
[7741]855            self.flash(_('Successfully removed: ${a}',
[7738]856                mapping = {'a':', '.join(deleted)}))
[8742]857            self.context.writeLogMessage(
858                self, 'removed: % s' % ', '.join(deleted))
[7250]859        return
860
[7252]861    # We explicitely want the forms to be validated before payment tickets
862    # can be created. If no validation is requested, use
[7459]863    # 'validator=NullValidator' in the action directive
[7714]864    @action(_('Add online payment ticket'))
[7250]865    def addPaymentTicket(self, **data):
866        self.redirect(self.url(self.context, '@@addafp'))
[7252]867        return
[7250]868
[7714]869    @jsaction(_('Remove selected tickets'))
[7250]870    def removePaymentTickets(self, **data):
871        self.delPaymentTickets(**data)
872        self.redirect(self.url(self.context) + '/@@manage')
873        return
874
[7200]875class ApplicantEditFormPage(ApplicantManageFormPage):
[5982]876    """An applicant-centered edit view for applicant data.
877    """
[6196]878    grok.context(IApplicantEdit)
[5273]879    grok.name('edit')
[6198]880    grok.require('waeup.handleApplication')
[6459]881    form_fields = grok.AutoFields(IApplicantEdit).omit(
[6476]882        'locked', 'course_admitted', 'student_id',
[8983]883        'screening_score', 'suspended'
[6459]884        )
[7459]885    form_fields['applicant_id'].for_display = True
[8039]886    form_fields['reg_number'].for_display = True
[7200]887    grok.template('applicanteditpage')
[6322]888    manage_applications = False
[5484]889
[7250]890    @property
891    def display_actions(self):
[8286]892        state = IWorkflowState(self.context).getState()
893        if state == INITIALIZED:
[7250]894            actions = [[],[]]
[8286]895        elif state == STARTED:
[7714]896            actions = [[_('Save')],
897                [_('Add online payment ticket'),_('Remove selected tickets')]]
[8286]898        elif state == PAID:
[7714]899            actions = [[_('Save'), _('Final Submit')],
900                [_('Remove selected tickets')]]
[7351]901        else:
[7250]902            actions = [[],[]]
903        return actions
904
[7330]905    def unremovable(self, ticket):
[8286]906        state = IWorkflowState(self.context).getState()
907        return ticket.r_code or state in (INITIALIZED, SUBMITTED)
[7330]908
[7145]909    def emit_lock_message(self):
[7714]910        self.flash(_('The requested form is locked (read-only).'))
[5941]911        self.redirect(self.url(self.context))
912        return
[6078]913
[5686]914    def update(self):
[8665]915        if self.context.locked or (
916            self.context.__parent__.expired and
917            self.context.__parent__.strict_deadline):
[7145]918            self.emit_lock_message()
[5941]919            return
[7200]920        super(ApplicantEditFormPage, self).update()
[5686]921        return
[5952]922
[6196]923    def dataNotComplete(self):
[7252]924        store = getUtility(IExtFileStore)
925        if not store.getFileByContext(self.context, attr=u'passport.jpg'):
[7714]926            return _('No passport picture uploaded.')
[6322]927        if not self.request.form.get('confirm_passport', False):
[7714]928            return _('Passport picture confirmation box not ticked.')
[6196]929        return False
[5952]930
[7252]931    # We explicitely want the forms to be validated before payment tickets
932    # can be created. If no validation is requested, use
[7459]933    # 'validator=NullValidator' in the action directive
[7714]934    @action(_('Add online payment ticket'))
[7250]935    def addPaymentTicket(self, **data):
936        self.redirect(self.url(self.context, '@@addafp'))
[7252]937        return
[7250]938
[7714]939    @jsaction(_('Remove selected tickets'))
[7250]940    def removePaymentTickets(self, **data):
941        self.delPaymentTickets(**data)
942        self.redirect(self.url(self.context) + '/@@edit')
943        return
944
[7996]945    @action(_('Save'), style='primary')
[5273]946    def save(self, **data):
[7084]947        if self.passport_changed is False:  # False is not None!
948            return # error during image upload. Ignore other values
[5273]949        self.applyData(self.context, **data)
[6196]950        self.flash('Form has been saved.')
[5273]951        return
952
[8550]953    @submitaction(_('Final Submit'))
[5484]954    def finalsubmit(self, **data):
[7084]955        if self.passport_changed is False:  # False is not None!
956            return # error during image upload. Ignore other values
[6196]957        if self.dataNotComplete():
958            self.flash(self.dataNotComplete())
[5941]959            return
[7252]960        self.applyData(self.context, **data)
[8286]961        state = IWorkflowState(self.context).getState()
[6322]962        # This shouldn't happen, but the application officer
963        # might have forgotten to lock the form after changing the state
[8286]964        if state != PAID:
[8589]965            self.flash(_('The form cannot be submitted. Wrong state!'))
[6303]966            return
967        IWorkflowInfo(self.context).fireTransition('submit')
[8589]968        # application_date is used in export files for sorting.
969        # We can thus store utc.
[8194]970        self.context.application_date = datetime.utcnow()
[5941]971        self.context.locked = True
[7714]972        self.flash(_('Form has been submitted.'))
[6196]973        self.redirect(self.url(self.context))
[5273]974        return
[5941]975
[7063]976class PassportImage(grok.View):
977    """Renders the passport image for applicants.
978    """
979    grok.name('passport.jpg')
980    grok.context(IApplicant)
[7113]981    grok.require('waeup.viewApplication')
[7063]982
983    def render(self):
984        # A filename chooser turns a context into a filename suitable
985        # for file storage.
986        image = getUtility(IExtFileStore).getFileByContext(self.context)
987        self.response.setHeader(
988            'Content-Type', 'image/jpeg')
989        if image is None:
990            # show placeholder image
[7089]991            return open(DEFAULT_PASSPORT_IMAGE_PATH, 'rb').read()
[7063]992        return image
[7363]993
[7819]994class ApplicantRegistrationPage(KofaAddFormPage):
[7363]995    """Captcha'd registration page for applicants.
996    """
997    grok.context(IApplicantsContainer)
998    grok.name('register')
[7373]999    grok.require('waeup.Anonymous')
[7363]1000    grok.template('applicantregister')
1001
[7368]1002    @property
[8033]1003    def form_fields(self):
1004        form_fields = None
[8128]1005        if self.context.mode == 'update':
1006            form_fields = grok.AutoFields(IApplicantRegisterUpdate).select(
1007                'firstname','reg_number','email')
1008        else: #if self.context.mode == 'create':
[8033]1009            form_fields = grok.AutoFields(IApplicantEdit).select(
1010                'firstname', 'middlename', 'lastname', 'email', 'phone')
1011        return form_fields
1012
1013    @property
[7368]1014    def label(self):
[8078]1015        return _('Apply for ${a}',
[7714]1016            mapping = {'a':self.context.title})
[7368]1017
[7363]1018    def update(self):
[8665]1019        if self.context.expired:
1020            self.flash(_('Outside application period.'))
[7368]1021            self.redirect(self.url(self.context))
1022            return
1023        # Handle captcha
[7363]1024        self.captcha = getUtility(ICaptchaManager).getCaptcha()
1025        self.captcha_result = self.captcha.verify(self.request)
1026        self.captcha_code = self.captcha.display(self.captcha_result.error_code)
1027        return
1028
[8629]1029    def _redirect(self, email, password, applicant_id):
1030        # Forward only email to landing page in base package.
1031        self.redirect(self.url(self.context, 'registration_complete',
1032            data = dict(email=email)))
1033        return
1034
[7714]1035    @action(_('Get login credentials'), style='primary')
[7363]1036    def register(self, **data):
1037        if not self.captcha_result.is_valid:
[8037]1038            # Captcha will display error messages automatically.
[7363]1039            # No need to flash something.
1040            return
[8033]1041        if self.context.mode == 'create':
1042            # Add applicant
1043            applicant = createObject(u'waeup.Applicant')
1044            self.applyData(applicant, **data)
1045            self.context.addApplicant(applicant)
[8042]1046            applicant.reg_number = applicant.applicant_id
1047            notify(grok.ObjectModifiedEvent(applicant))
[8033]1048        elif self.context.mode == 'update':
1049            # Update applicant
[8037]1050            reg_number = data.get('reg_number','')
1051            firstname = data.get('firstname','')
[8033]1052            cat = getUtility(ICatalog, name='applicants_catalog')
1053            results = list(
1054                cat.searchResults(reg_number=(reg_number, reg_number)))
1055            if results:
1056                applicant = results[0]
[8042]1057                if getattr(applicant,'firstname',None) is None:
[8037]1058                    self.flash(_('An error occurred.'))
1059                    return
1060                elif applicant.firstname.lower() != firstname.lower():
[8042]1061                    # Don't tell the truth here. Anonymous must not
1062                    # know that a record was found and only the firstname
1063                    # verification failed.
[8037]1064                    self.flash(_('No application record found.'))
1065                    return
[8627]1066                elif applicant.password is not None and \
1067                    applicant.state != INITIALIZED:
1068                    self.flash(_('Your password has already been set and used. '
[8042]1069                                 'Please proceed to the login page.'))
1070                    return
1071                # Store email address but nothing else.
[8033]1072                applicant.email = data['email']
[8042]1073                notify(grok.ObjectModifiedEvent(applicant))
[8033]1074            else:
[8042]1075                # No record found, this is the truth.
[8033]1076                self.flash(_('No application record found.'))
1077                return
1078        else:
[8042]1079            # Does not happen but anyway ...
[8033]1080            return
[7819]1081        kofa_utils = getUtility(IKofaUtils)
[7811]1082        password = kofa_utils.genPassword()
[7380]1083        IUserAccount(applicant).setPassword(password)
[7365]1084        # Send email with credentials
[7399]1085        login_url = self.url(grok.getSite(), 'login')
[8853]1086        url_info = u'Login: %s' % login_url
[7714]1087        msg = _('You have successfully been registered for the')
[7811]1088        if kofa_utils.sendCredentials(IUserAccount(applicant),
[8853]1089            password, url_info, msg):
[8629]1090            email_sent = applicant.email
[7380]1091        else:
[8629]1092            email_sent = None
1093        self._redirect(email=email_sent, password=password,
1094            applicant_id=applicant.applicant_id)
[7380]1095        return
1096
[7819]1097class ApplicantRegistrationEmailSent(KofaPage):
[7380]1098    """Landing page after successful registration.
[8629]1099
[7380]1100    """
1101    grok.name('registration_complete')
1102    grok.require('waeup.Public')
1103    grok.template('applicantregemailsent')
[7714]1104    label = _('Your registration was successful.')
[7380]1105
[8629]1106    def update(self, email=None, applicant_id=None, password=None):
[7380]1107        self.email = email
[8629]1108        self.password = password
1109        self.applicant_id = applicant_id
[7380]1110        return
Note: See TracBrowser for help on using the repository browser.