source: main/waeup.sirp/trunk/src/waeup/sirp/applicants/browser.py @ 6647

Last change on this file since 6647 was 6644, checked in by Henrik Bettermann, 13 years ago

Don't add log message when no attribute has been changed, but log all transitions.

File size: 30.7 KB
RevLine 
[5273]1##
2## browser.py
3## Login : <uli@pu.smp.net>
[6153]4## Started on  Sun Jun 27 11:03:10 2010 Uli Fouquet & Henrik Bettermann
[5273]5## $Id$
[6078]6##
[6063]7## Copyright (C) 2010 Uli Fouquet & Henrik Bettermann
[5273]8## This program is free software; you can redistribute it and/or modify
9## it under the terms of the GNU General Public License as published by
10## the Free Software Foundation; either version 2 of the License, or
11## (at your option) any later version.
[6078]12##
[5273]13## This program is distributed in the hope that it will be useful,
14## but WITHOUT ANY WARRANTY; without even the implied warranty of
15## MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16## GNU General Public License for more details.
[6078]17##
[5273]18## You should have received a copy of the GNU General Public License
19## along with this program; if not, write to the Free Software
20## Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
21##
[5824]22"""UI components for basic applicants and related components.
[5273]23"""
[6082]24import sys
[5273]25import grok
26
[6305]27from datetime import datetime
[6081]28from zope.formlib.widget import CustomWidgetFactory
[6358]29from zope.formlib.form import setUpEditWidgets
[5937]30from zope.securitypolicy.interfaces import IPrincipalRoleManager
[6153]31from zope.traversing.browser import absoluteURL
[6081]32
[6303]33from hurry.workflow.interfaces import IWorkflowInfo, IWorkflowState
[6357]34from reportlab.pdfgen import canvas
[6364]35from reportlab.lib.units import cm
[6390]36from reportlab.lib.pagesizes import A4
[6364]37from reportlab.lib.styles import getSampleStyleSheet
38from reportlab.platypus import (Frame, Paragraph, Image,
39    Table, Spacer)
40from reportlab.platypus.tables import TableStyle
[6357]41
[6469]42from waeup.sirp.accesscodes import invalidate_accesscode, get_access_code
43from waeup.sirp.accesscodes.workflow import USED
[5273]44from waeup.sirp.browser import (
[6321]45    WAeUPPage, WAeUPEditFormPage, WAeUPAddFormPage, WAeUPDisplayFormPage)
[6081]46from waeup.sirp.browser.breadcrumbs import Breadcrumb
[6103]47from waeup.sirp.browser.layout import NullValidator
[6321]48from waeup.sirp.browser.pages import add_local_role, del_local_roles
[6013]49from waeup.sirp.browser.resources import datepicker, tabs, datatable
[6153]50from waeup.sirp.browser.viewlets import (
51    ManageActionButton, PrimaryNavTab, LeftSidebarLink
52    )
[6081]53from waeup.sirp.image.browser.widget import (
54    ThumbnailWidget, EncodingImageFileWidget,
[5822]55    )
[6598]56from waeup.sirp.image.image import createWAeUPImageFile
[6184]57from waeup.sirp.interfaces import IWAeUPObject, ILocalRolesAssignable
[6321]58from waeup.sirp.permissions import get_users_with_local_roles
[6254]59from waeup.sirp.university.interfaces import ICertificate
[6054]60from waeup.sirp.widgets.datewidget import (
61    FriendlyDateWidget, FriendlyDateDisplayWidget)
[6084]62from waeup.sirp.widgets.restwidget import ReSTDisplayWidget
[5303]63from waeup.sirp.widgets.objectwidget import (
[5301]64    WAeUPObjectWidget, WAeUPObjectDisplayWidget)
[5303]65from waeup.sirp.widgets.multilistwidget import (
[5273]66    MultiListWidget, MultiListDisplayWidget)
[6153]67from waeup.sirp.applicants import ResultEntry, Applicant, get_applicant_data
[6081]68from waeup.sirp.applicants.interfaces import (
[6321]69    IApplicant, IApplicantPrincipal,IApplicantEdit, IApplicantsRoot,
70    IApplicantsContainer, IApplicantsContainerAdd, application_types_vocab
[5686]71    )
[6353]72from waeup.sirp.applicants.workflow import INITIALIZED, STARTED
[5320]73
[5273]74results_widget = CustomWidgetFactory(
[5301]75    WAeUPObjectWidget, ResultEntry)
[5273]76
77results_display_widget = CustomWidgetFactory(
[5301]78    WAeUPObjectDisplayWidget, ResultEntry)
[5273]79
80list_results_widget = CustomWidgetFactory(
81    MultiListWidget, subwidget=results_widget)
82
83list_results_display_widget = CustomWidgetFactory(
84    MultiListDisplayWidget, subwidget=results_display_widget)
85
[6353]86#TRANSITION_OBJECTS = create_workflow()
[6305]87
[6322]88#TRANSITION_DICT = dict([
89#    (transition_object.transition_id,transition_object.title)
90#    for transition_object in TRANSITION_OBJECTS])
[6305]91
[6067]92class ApplicantsRootPage(WAeUPPage):
[5822]93    grok.context(IApplicantsRoot)
94    grok.name('index')
[6153]95    grok.require('waeup.Public')
[5822]96    title = 'Applicants'
[6069]97    label = 'Application Section'
[5843]98    pnav = 3
[6012]99
100    def update(self):
[6067]101        super(ApplicantsRootPage, self).update()
[6012]102        datatable.need()
103        return
104
[5828]105class ManageApplicantsRootActionButton(ManageActionButton):
106    grok.context(IApplicantsRoot)
[6067]107    grok.view(ApplicantsRootPage)
[6198]108    grok.require('waeup.manageApplications')
[6069]109    text = 'Manage application section'
[5828]110
[6069]111class ApplicantsRootManageFormPage(WAeUPEditFormPage):
[5828]112    grok.context(IApplicantsRoot)
113    grok.name('manage')
[6107]114    grok.template('applicantsrootmanagepage')
[6069]115    title = 'Applicants'
116    label = 'Manage application section'
[5843]117    pnav = 3
[6198]118    grok.require('waeup.manageApplications')
[6069]119    taboneactions = ['Add applicants container', 'Remove selected','Cancel']
[6184]120    tabtwoactions1 = ['Remove selected local roles']
121    tabtwoactions2 = ['Add local role']
[6069]122    subunits = 'Applicants Containers'
[6078]123
[6069]124    def update(self):
125        tabs.need()
[6108]126        datatable.need()
[6069]127        return super(ApplicantsRootManageFormPage, self).update()
[5828]128
[6184]129    def getLocalRoles(self):
130        roles = ILocalRolesAssignable(self.context)
131        return roles()
132
133    def getUsers(self):
134        """Get a list of all users.
135        """
136        for key, val in grok.getSite()['users'].items():
137            url = self.url(val)
138            yield(dict(url=url, name=key, val=val))
139
140    def getUsersWithLocalRoles(self):
141        return get_users_with_local_roles(self.context)
142
[6069]143    # ToDo: Show warning message before deletion
144    @grok.action('Remove selected')
145    def delApplicantsContainers(self, **data):
146        form = self.request.form
147        child_id = form['val_id']
148        if not isinstance(child_id, list):
149            child_id = [child_id]
150        deleted = []
151        for id in child_id:
152            try:
153                del self.context[id]
154                deleted.append(id)
155            except:
156                self.flash('Could not delete %s: %s: %s' % (
157                        id, sys.exc_info()[0], sys.exc_info()[1]))
158        if len(deleted):
159            self.flash('Successfully removed: %s' % ', '.join(deleted))
[6078]160        self.redirect(self.url(self.context, '@@manage')+'#tab-1')
161        return
[5828]162
[6069]163    @grok.action('Add applicants container', validator=NullValidator)
164    def addApplicantsContainer(self, **data):
165        self.redirect(self.url(self.context, '@@add'))
[6078]166        return
167
[6069]168    @grok.action('Cancel', validator=NullValidator)
169    def cancel(self, **data):
170        self.redirect(self.url(self.context))
[6078]171        return
172
[6184]173    @grok.action('Add local role', validator=NullValidator)
174    def addLocalRole(self, **data):
175        return add_local_role(self,2, **data)
176
177    @grok.action('Remove selected local roles')
178    def delLocalRoles(self, **data):
179        return del_local_roles(self,2,**data)
180
[6069]181class ApplicantsContainerAddFormPage(WAeUPAddFormPage):
[5822]182    grok.context(IApplicantsRoot)
[6198]183    grok.require('waeup.manageApplications')
[5822]184    grok.name('add')
[6107]185    grok.template('applicantscontaineraddpage')
[6069]186    title = 'Applicants'
187    label = 'Add applicants container'
[5843]188    pnav = 3
[6078]189
[6103]190    form_fields = grok.AutoFields(
191        IApplicantsContainerAdd).omit('code').omit('title')
[6083]192    form_fields['startdate'].custom_widget = FriendlyDateWidget('le')
193    form_fields['enddate'].custom_widget = FriendlyDateWidget('le')
[6078]194
[6083]195    def update(self):
196        datepicker.need() # Enable jQuery datepicker in date fields.
197        return super(ApplicantsContainerAddFormPage, self).update()
198
[6069]199    @grok.action('Add applicants container')
200    def addApplicantsContainer(self, **data):
[6103]201        year = data['year']
202        code = u'%s%s' % (data['prefix'], year)
203        prefix = application_types_vocab.getTerm(data['prefix'])
204        title = u'%s %s/%s' % (prefix.title, year, year + 1)
[6087]205        if code in self.context.keys():
[6105]206            self.flash(
207                'An applicants container for the same application '
208                'type and entrance year exists already in the database.')
[5822]209            return
210        # Add new applicants container...
[6083]211        provider = data['provider'][1]
[5822]212        container = provider.factory()
[6069]213        self.applyData(container, **data)
[6087]214        container.code = code
215        container.title = title
216        self.context[code] = container
[6105]217        self.flash('Added: "%s".' % code)
[6069]218        self.redirect(self.url(self.context, u'@@manage')+'#tab-1')
[5822]219        return
[6078]220
[6103]221    @grok.action('Cancel', validator=NullValidator)
[6069]222    def cancel(self, **data):
[6103]223        self.redirect(self.url(self.context, '@@manage') + '#tab-1')
[6078]224
[5845]225class ApplicantsRootBreadcrumb(Breadcrumb):
226    """A breadcrumb for applicantsroot.
227    """
228    grok.context(IApplicantsRoot)
[6153]229    title = u'Application Section'
[6078]230
[5845]231class ApplicantsContainerBreadcrumb(Breadcrumb):
232    """A breadcrumb for applicantscontainers.
233    """
234    grok.context(IApplicantsContainer)
[6319]235
[6153]236class ApplicantBreadcrumb(Breadcrumb):
237    """A breadcrumb for applicants.
238    """
239    grok.context(IApplicant)
[6319]240
[6153]241    @property
242    def title(self):
243        """Get a title for a context.
244        """
245        return self.context.access_code
[5828]246
247class ApplicantsTab(PrimaryNavTab):
[6153]248    """Applicants tab in primary navigation.
[5828]249    """
[6078]250
[5828]251    grok.context(IWAeUPObject)
252    grok.order(3)
[6336]253    grok.require('waeup.Public')
[5828]254    grok.template('primarynavtab')
255
[5843]256    pnav = 3
[5828]257    tab_title = u'Applicants'
258
259    @property
260    def link_target(self):
261        return self.view.application_url('applicants')
262
[6029]263class ApplicantsContainerPage(WAeUPDisplayFormPage):
[5830]264    """The standard view for regular applicant containers.
265    """
266    grok.context(IApplicantsContainer)
267    grok.name('index')
[6153]268    grok.require('waeup.Public')
[6029]269    grok.template('applicantscontainerpage')
[5850]270    pnav = 3
[6053]271
[6105]272    form_fields = grok.AutoFields(IApplicantsContainer).omit('title')
[6054]273    form_fields['startdate'].custom_widget = FriendlyDateDisplayWidget('le')
274    form_fields['enddate'].custom_widget = FriendlyDateDisplayWidget('le')
[6084]275    form_fields['description'].custom_widget = ReSTDisplayWidget
[6053]276
[5837]277    @property
278    def title(self):
[6087]279        return "Applicants Container: %s" % self.context.title
[5837]280
281    @property
282    def label(self):
[6087]283        return self.context.title
[5830]284
[6107]285class ApplicantsContainerManageActionButton(ManageActionButton):
[6336]286    grok.order(1)
[5832]287    grok.context(IApplicantsContainer)
288    grok.view(ApplicantsContainerPage)
[6198]289    grok.require('waeup.manageApplications')
[6070]290    text = 'Manage applicants container'
[5832]291
[6336]292class LoginApplicantActionButton(ManageActionButton):
293    grok.order(2)
294    grok.context(IApplicantsContainer)
295    grok.view(ApplicantsContainerPage)
296    grok.require('waeup.Anonymous')
297    icon = 'login.png'
298    text = 'Login for applicants'
299    target = 'login'
[5832]300
[6107]301class ApplicantsContainerManageFormPage(WAeUPEditFormPage):
[5837]302    grok.context(IApplicantsContainer)
[5850]303    grok.name('manage')
[6107]304    grok.template('applicantscontainermanagepage')
[6105]305    form_fields = grok.AutoFields(IApplicantsContainer).omit('title')
306    taboneactions = ['Save','Cancel']
307    tabtwoactions = ['Add applicant', 'Remove selected','Cancel']
[6184]308    tabthreeactions1 = ['Remove selected local roles']
309    tabthreeactions2 = ['Add local role']
[5844]310    # Use friendlier date widget...
[6054]311    form_fields['startdate'].custom_widget = FriendlyDateWidget('le')
312    form_fields['enddate'].custom_widget = FriendlyDateWidget('le')
[6198]313    grok.require('waeup.manageApplications')
[5850]314
315    @property
316    def title(self):
[6087]317        return "Applicants Container: %s" % self.context.title
[6078]318
[5850]319    @property
320    def label(self):
[6087]321        return 'Manage applicants container'
[5850]322
[5845]323    pnav = 3
[5837]324
325    def update(self):
[5850]326        datepicker.need() # Enable jQuery datepicker in date fields.
[5982]327        tabs.need()
[6015]328        datatable.need()  # Enable jQurey datatables for contents listing
[6107]329        return super(ApplicantsContainerManageFormPage, self).update()
[5837]330
[6184]331    def getLocalRoles(self):
332        roles = ILocalRolesAssignable(self.context)
333        return roles()
334
335    def getUsers(self):
336        """Get a list of all users.
337        """
338        for key, val in grok.getSite()['users'].items():
339            url = self.url(val)
340            yield(dict(url=url, name=key, val=val))
341
342    def getUsersWithLocalRoles(self):
343        return get_users_with_local_roles(self.context)
344
[5850]345    @grok.action('Save')
[5837]346    def apply(self, **data):
347        self.applyData(self.context, **data)
348        self.flash('Data saved.')
349        return
[6078]350
[6105]351    # ToDo: Show warning message before deletion
352    @grok.action('Remove selected')
353    def delApplicant(self, **data):
[6189]354        form = self.request.form
355        if form.has_key('val_id'):
356            child_id = form['val_id']
357        else:
358            self.flash('No applicant selected!')
359            self.redirect(self.url(self.context, '@@manage')+'#tab-2')
360            return
361        if not isinstance(child_id, list):
362            child_id = [child_id]
363        deleted = []
364        for id in child_id:
365            try:
366                del self.context[id]
367                deleted.append(id)
368            except:
369                self.flash('Could not delete %s: %s: %s' % (
370                        id, sys.exc_info()[0], sys.exc_info()[1]))
371        if len(deleted):
372            self.flash('Successfully removed: %s' % ', '.join(deleted))
373        self.redirect(self.url(self.context, u'@@manage')+'#tab-2')
374        return
[6105]375
376    @grok.action('Add applicant', validator=NullValidator)
377    def addApplicant(self, **data):
[6327]378        self.redirect(self.url(self.context, 'addapplicant'))
379        return
[6105]380
381    @grok.action('Cancel', validator=NullValidator)
[5837]382    def cancel(self, **data):
383        self.redirect(self.url(self.context))
384        return
[5886]385
[6184]386    @grok.action('Add local role', validator=NullValidator)
387    def addLocalRole(self, **data):
388        return add_local_role(self,3, **data)
[6105]389
[6184]390    @grok.action('Remove selected local roles')
391    def delLocalRoles(self, **data):
392        return del_local_roles(self,3,**data)
393
[5886]394class LoginApplicant(WAeUPPage):
395    grok.context(IApplicantsContainer)
396    grok.name('login')
[6153]397    grok.require('waeup.Public')
[5886]398
[6110]399    @property
400    def title(self):
401        return u"Applicant Login: %s" % self.context.title
[6078]402
[5886]403    @property
404    def label(self):
[6110]405        return u'Login for applicants only'
[5886]406
407    pnav = 3
[6319]408
[6110]409    @property
410    def ac_prefix(self):
411        return self.context.ac_prefix
[6078]412
[5896]413    def update(self, SUBMIT=None):
414        self.ac_series = self.request.form.get('form.ac_series', None)
415        self.ac_number = self.request.form.get('form.ac_number', None)
[5886]416        if SUBMIT is None:
417            return
[5894]418        if self.request.principal.id == 'zope.anybody':
[6105]419            self.flash('Entered credentials are invalid.')
[5886]420            return
[5894]421        if not IApplicantPrincipal.providedBy(self.request.principal):
422            # Don't care if user is already authenticated as non-applicant
423            return
[6377]424
425        # From here we handle an applicant (not an officer browsing)
[5905]426        pin = self.request.principal.access_code
[6375]427
428        # Mark pin as used (this also fires a pin related transition)
[6470]429        if get_access_code(pin).state == USED:
[6469]430            pass
431        else:
[6471]432            comment = u"AC invalidated"
[6469]433            # Here we know that the ac is in state initialized so we do not
434            # expect an exception
[6471]435            invalidate_accesscode(pin,comment)
[6375]436
[6377]437        if not pin in self.context.keys():
438            # Create applicant record
439            applicant = Applicant()
440            applicant.access_code = pin
441            self.context[pin] = applicant
[6359]442
[6377]443        role_manager = IPrincipalRoleManager(self.context[pin])
[6394]444        role_manager.assignRoleToPrincipal(
445            'waeup.local.ApplicationOwner', self.request.principal.id)
[6359]446
[6394]447        # Assign current principal the PortalUser role
[6397]448        role_manager = IPrincipalRoleManager(grok.getSite())
[6394]449        role_manager.assignRoleToPrincipal(
450            'waeup.PortalUser', self.request.principal.id)
[6377]451
452        # Mark application as started
453        if IWorkflowState(self.context[pin]).getState() is INITIALIZED:
454            IWorkflowInfo(self.context[pin]).fireTransition('start')
455
[5937]456        self.redirect(self.url(self.context[pin], 'edit'))
[5886]457        return
[6319]458
[6327]459class ApplicantAddFormPage(WAeUPAddFormPage):
[6622]460    """Add-form to add an applicant.
[6327]461    """
462    grok.context(IApplicantsContainer)
463    grok.require('waeup.manageApplications')
464    grok.name('addapplicant')
465    grok.template('applicantaddpage')
466    title = 'Applicants'
467    label = 'Add applicant'
468    pnav = 3
469
470    @property
471    def title(self):
472        return "Applicants Container: %s" % self.context.title
473
474    @property
475    def ac_prefix(self):
476        return self.context.ac_prefix
477
478    @grok.action('Create application record')
479    def addApplicant(self, **data):
480        ac_series = self.request.form.get('form.ac_series', None)
481        ac_number = self.request.form.get('form.ac_number', None)
482        pin = '%s-%s-%s' % (self.ac_prefix,ac_series,ac_number)
[6459]483        if not invalidate_accesscode(pin, comment=u"Invalidated by system"):
[6383]484            self.flash('%s is not a valid access code.' % pin)
485            self.redirect(self.url(self.context, '@@manage')+'#tab-2')
486            return
487        else:
[6327]488            # Create applicant record
489            applicant = Applicant()
490            applicant.access_code = pin
491            self.context[pin] = applicant
492        self.redirect(self.url(self.context[pin], 'edit'))
493        return
494
[6396]495class AccessCodeViewLink(LeftSidebarLink):
[6153]496    grok.order(1)
497    grok.require('waeup.Public')
[6396]498    icon = 'actionicon_view.png'
499    title = 'View Record'
500    target = '/@@index'
[5886]501
[6396]502    @property
503    def url(self):
[6153]504        if not IApplicantPrincipal.providedBy(self.request.principal):
505            return ''
506        access_code = getattr(self.request.principal,'access_code',None)
507        if access_code:
508            applicant_object = get_applicant_data(access_code)
[6396]509            return absoluteURL(applicant_object, self.request) + self.target
[6595]510        #else:
511        #    return ''
[6153]512
[6396]513class AccessCodeEditLink(AccessCodeViewLink):
514    grok.order(2)
515    grok.require('waeup.Public')
516    icon = 'actionicon_modify.png'
517    title = 'Edit Record'
518    target = '/@@edit'
519
520    @property
521    def url(self):
522        if not IApplicantPrincipal.providedBy(self.request.principal):
523            return ''
524        access_code = getattr(self.request.principal,'access_code',None)
525        if access_code:
526            applicant_object = get_applicant_data(access_code)
527            if applicant_object.locked:
528                return ''
529            return absoluteURL(applicant_object, self.request) + self.target
[6595]530        #else:
531        #    return ''
[6396]532
533class AccessCodeSlipLink(AccessCodeViewLink):
534    grok.order(3)
535    grok.require('waeup.Public')
536    icon = 'actionicon_pdf.png'
537    title = 'Download Slip'
538    target = '/application_slip.pdf'
539
[5273]540class DisplayApplicant(WAeUPDisplayFormPage):
541    grok.context(IApplicant)
542    grok.name('index')
[6198]543    grok.require('waeup.handleApplication')
[6320]544    form_fields = grok.AutoFields(IApplicant).omit(
545        'locked').omit('course_admitted')
[6196]546    #form_fields['fst_sit_results'].custom_widget = list_results_display_widget
[5919]547    form_fields['passport'].custom_widget = ThumbnailWidget
[6054]548    form_fields['date_of_birth'].custom_widget = FriendlyDateDisplayWidget('le')
[5273]549    label = 'Applicant'
[6254]550    grok.template('form_display')
[5843]551    pnav = 3
[5273]552
[6196]553    @property
554    def title(self):
[6465]555        if self.request.principal.title == 'Applicant':
556            return u'Your Application Record'
[6196]557        return '%s' % self.context.access_code
558
559    @property
560    def label(self):
561        container_title = self.context.__parent__.title
562        return '%s Application Record' % container_title
563
[6254]564    def getCourseAdmitted(self):
[6355]565        """Return link, title and code in html format to the certificate
566           admitted.
[6351]567        """
[6254]568        course_admitted = self.context.course_admitted
569        if ICertificate.providedBy(course_admitted):
570            url = self.url(course_admitted)
571            title = course_admitted.title
572            code = course_admitted.code
[6366]573            return '<a href="%s">%s - %s</a>' %(url,code,title)
[6457]574        return ''
[6254]575
[6358]576class PDFActionButton(ManageActionButton):
577    grok.context(IApplicant)
[6396]578    #grok.view(DisplayApplicant)
579    grok.require('waeup.manageApplications')
[6358]580    icon = 'actionicon_pdf.png'
[6367]581    text = 'Download application slip'
[6358]582    target = 'application_slip.pdf'
583
584class ExportPDFPage(grok.View):
585    """Deliver a PDF slip of the context.
586    """
587    grok.context(IApplicant)
588    grok.name('application_slip.pdf')
589    grok.require('waeup.handleApplication')
590    form_fields = grok.AutoFields(IApplicant).omit(
591        'locked').omit('course_admitted')
592    form_fields['date_of_birth'].custom_widget = FriendlyDateDisplayWidget('le')
593    prefix = 'form'
594
[6363]595    @property
596    def label(self):
597        container_title = self.context.__parent__.title
598        return '%s Application Record' % container_title
599
[6358]600    def getCourseAdmitted(self):
601        """Return title and code in html format to the certificate
602           admitted.
603        """
604        course_admitted = self.context.course_admitted
605        if ICertificate.providedBy(course_admitted):
606            title = course_admitted.title
607            code = course_admitted.code
[6365]608            return '%s - %s' %(code,title)
[6462]609        return ''
[6358]610
611    def setUpWidgets(self, ignore_request=False):
612        self.adapters = {}
613        self.widgets = setUpEditWidgets(
614            self.form_fields, self.prefix, self.context, self.request,
615            adapters=self.adapters, for_display=True,
616            ignore_request=ignore_request
617            )
618
619    def render(self):
[6364]620        # (0,0),(-1,-1) = whole table
621        # (0,0),(0,-1) = first column
622        # (-1,0),(-1,-1) = last column
623        # (0,0),(-1,0) = first row
624        # (0,-1),(-1,-1) = last row
625        SLIP_STYLE = TableStyle(
626            [('VALIGN',(0,0),(-1,-1),'TOP')]
627            )
[6358]628
629        pdf = canvas.Canvas('application_slip.pdf',pagesize=A4)
[6364]630        pdf.setTitle(self.label)
631        pdf.setSubject('Application')
632        pdf.setAuthor('%s (%s)' % (self.request.principal.title,
633            self.request.principal.id))
634        pdf.setCreator('WAeUP SIRP')
[6358]635        width, height = A4
636        style = getSampleStyleSheet()
[6365]637        pdf.line(1*cm,height-(1.8*cm),width-(1*cm),height-(1.8*cm))
[6363]638
[6358]639        story = []
[6365]640        frame_header = Frame(1*cm,1*cm,width-(1.7*cm),height-(1.7*cm))
[6363]641        header_title = getattr(grok.getSite(), 'name', u'Sample University')
642        story.append(Paragraph(header_title, style["Heading1"]))
643        frame_header.addFromList(story,pdf)
644
645        story = []
[6365]646        frame_body = Frame(1*cm,1*cm,width-(2*cm),height-(3.5*cm))
[6364]647        story.append(Paragraph(self.label, style["Heading2"]))
648        #story.append(HRFlowable())
649        story.append(Spacer(1, 18))
650        for msg in self.context.history.messages:
651            f_msg = '<font face="Courier" size=10>%s</font>' % msg
652            story.append(Paragraph(f_msg, style["Normal"]))
[6363]653        story.append(Spacer(1, 24))
[6358]654        self.setUpWidgets()
[6363]655        data = []
[6358]656        for widget in self.widgets:
[6462]657            f_label = '<font size=12>%s</font>:' % widget.label.strip()
[6363]658            f_label = Paragraph(f_label, style["Normal"])
[6358]659            if widget.name != 'form.passport':
[6363]660                f_text = '<font size=12>%s</font>' % widget()
661                f_text = Paragraph(f_text, style["Normal"])
662                data.append([f_label,f_text])
[6360]663            else:
664                filename = widget._data.file.name
[6364]665                im = Image(filename,width=4*cm, height=3*cm,kind='bound')
[6363]666                data.append([f_label,im])
[6462]667        f_label = '<font size=12>Admitted Course of Study:</font>'
[6363]668        f_text = '<font size=12>%s</font>' % self.getCourseAdmitted()
669        f_label = Paragraph(f_label, style["Normal"])
670        f_text = Paragraph(f_text, style["Normal"])
671        data.append([f_label,f_text])
[6364]672        table = Table(data,style=SLIP_STYLE)
[6363]673        story.append(table)
674        frame_body.addFromList(story,pdf)
675
[6364]676        story = []
[6365]677        frame_footer = Frame(1*cm,0,width-(2*cm),1*cm)
[6364]678        timestamp = datetime.now().strftime("%d/%m/%Y %H:%M:%S")
679        f_text = '<font size=10>%s</font>' % timestamp
680        story.append(Paragraph(f_text, style["Normal"]))
681        frame_footer.addFromList(story,pdf)
682
[6358]683        self.response.setHeader(
684            'Content-Type', 'application/pdf')
685        return pdf.getpdfdata()
686
[6383]687class ApplicantManageActionButton(ManageActionButton):
[6198]688    grok.context(IApplicant)
689    grok.view(DisplayApplicant)
690    grok.require('waeup.manageApplications')
[6383]691    text = 'Manage application record'
[6198]692    target = 'edit_full'
693
[6196]694class EditApplicantFull(WAeUPEditFormPage):
695    """A full edit view for applicant data.
696    """
697    grok.context(IApplicant)
698    grok.name('edit_full')
[6198]699    grok.require('waeup.manageApplications')
[6476]700    form_fields = grok.AutoFields(IApplicant)
[6196]701    form_fields['passport'].custom_widget = EncodingImageFileWidget
702    form_fields['date_of_birth'].custom_widget = FriendlyDateWidget('le-year')
703    grok.template('form_edit')
[6322]704    manage_applications = True
[6196]705    pnav = 3
706
707    def update(self):
708        datepicker.need() # Enable jQuery datepicker in date fields.
709        super(EditApplicantFull, self).update()
[6353]710        self.wf_info = IWorkflowInfo(self.context)
[6598]711        upload = self.request.form.get('form.passport', None)
712        if upload:
713            # We got a fresh upload
714            image = createWAeUPImageFile(upload.filename, upload)
715            # This would normally be totally illegal. We set context
716            # data without the complete form data being
717            # verified. Normally data is set in methods like `save`
718            # only, which is only called when all form data is correct
719            # (and if not, the context is not touched). With images
720            # and their uploads the problem then is, that the uploaded
721            # image won't be shown in a partially erranous form
722            # because the new uploaded image is still not part of the
723            # context obeject and the image widget cannot work around
724            # that (it has no access to the new data). For this reason
725            # we set the image here and not in 'save()' or other
726            # methods.
727            self.context.passport = image
728            upload.seek(0)
729            self.passport_changed = True
[6196]730        return
731
732    @property
733    def title(self):
734        return self.context.access_code
735
736    @property
737    def label(self):
738        container_title = self.context.__parent__.title
739        return '%s Application Form' % container_title
740
[6303]741    def getTransitions(self):
[6351]742        """Return a list of dicts of allowed transition ids and titles.
[6353]743
744        Each list entry provides keys ``name`` and ``title`` for
745        internal name and (human readable) title of a single
746        transition.
[6349]747        """
[6353]748        allowed_transitions = self.wf_info.getManualTransitions()
[6355]749        return [dict(name='', title='No transition')] +[
750            dict(name=x, title=y) for x, y in allowed_transitions]
[6303]751
[6196]752    @grok.action('Save')
753    def save(self, **data):
[6475]754        changed_fields = self.applyData(self.context, **data)
755        changed_fields = changed_fields.values()
[6582]756        fields_string = '+'.join(' + '.join(str(i) for i in b) for b in changed_fields)
[6196]757        self.context._p_changed = True
[6303]758        form = self.request.form
759        if form.has_key('transition') and form['transition']:
[6305]760            transition_id = form['transition']
[6353]761            self.wf_info.fireTransition(transition_id)
[6196]762        self.flash('Form has been saved.')
[6475]763        ob_class = self.__implemented__.__name__.replace('waeup.sirp.','')
[6644]764        if fields_string:
765            self.context.loggerInfo(ob_class, 'saved: % s' % fields_string)
[6196]766        return
767
768class EditApplicantStudent(EditApplicantFull):
[5982]769    """An applicant-centered edit view for applicant data.
770    """
[6196]771    grok.context(IApplicantEdit)
[5273]772    grok.name('edit')
[6198]773    grok.require('waeup.handleApplication')
[6459]774    form_fields = grok.AutoFields(IApplicantEdit).omit(
[6476]775        'locked', 'course_admitted', 'student_id',
[6459]776        'screening_score',
777        )
[5686]778    form_fields['passport'].custom_widget = EncodingImageFileWidget
[6054]779    form_fields['date_of_birth'].custom_widget = FriendlyDateWidget('le-year')
[6196]780    grok.template('form_edit')
[6322]781    manage_applications = False
[6465]782    title = u'Your Application Form'
[5484]783
[6322]784
[5941]785    def emitLockMessage(self):
[6105]786        self.flash('The requested form is locked (read-only).')
[5941]787        self.redirect(self.url(self.context))
788        return
[6078]789
[5686]790    def update(self):
[5941]791        if self.context.locked:
[6367]792            self.emitLockMessage()
[5941]793            return
[6040]794        datepicker.need() # Enable jQuery datepicker in date fields.
[6598]795        upload = self.request.form.get('form.passport', None)
796        if upload:
797            # We got a fresh upload
798            image = createWAeUPImageFile(upload.filename, upload)
799            # This would normally be totally illegal. We set context
800            # data without the complete form data being
801            # verified. Normally data is set in methods like `save`
802            # only, which is only called when all form data is correct
803            # (and if not, the context is not touched). With images
804            # and their uploads the problem then is, that the uploaded
805            # image won't be shown in a partially erranous form
806            # because the new uploaded image is still not part of the
807            # context obeject and the image widget cannot work around
808            # that (it has no access to the new data). For this reason
809            # we set the image here and not in 'save()' or other
810            # methods.
811            self.context.passport = image
812            upload.seek(0)
813            self.passport_changed = True
[5982]814        super(EditApplicantStudent, self).update()
[5686]815        return
[5952]816
[6196]817    def dataNotComplete(self):
[6322]818        if not self.request.form.get('confirm_passport', False):
[6196]819            return 'Passport confirmation box not ticked.'
[6595]820        #if len(self.errors) > 0:
821        #    return 'Form has errors.'
[6196]822        return False
[5952]823
[5273]824    @grok.action('Save')
825    def save(self, **data):
[6595]826        #if self.context.locked:
827        #    self.emitLockMessage()
828        #    return
[5273]829        self.applyData(self.context, **data)
830        self.context._p_changed = True
[6196]831        self.flash('Form has been saved.')
[5273]832        return
833
[5484]834    @grok.action('Final Submit')
835    def finalsubmit(self, **data):
[6595]836        #if self.context.locked:
837        #    self.emitLockMessage()
838        #    return
[5273]839        self.applyData(self.context, **data)
[5484]840        self.context._p_changed = True
[6196]841        if self.dataNotComplete():
842            self.flash(self.dataNotComplete())
[5941]843            return
[6303]844        state = IWorkflowState(self.context).getState()
[6322]845        # This shouldn't happen, but the application officer
846        # might have forgotten to lock the form after changing the state
[6303]847        if state != STARTED:
[6322]848            self.flash('This form cannot be submitted. Wrong state!')
[6303]849            return
850        IWorkflowInfo(self.context).fireTransition('submit')
[6476]851        self.context.application_date = datetime.now()
[5941]852        self.context.locked = True
[6196]853        self.flash('Form has been submitted.')
854        self.redirect(self.url(self.context))
[5273]855        return
[5941]856
[6367]857class ApplicantViewActionButton(ManageActionButton):
858    grok.context(IApplicant)
[6383]859    grok.view(EditApplicantFull)
[6396]860    grok.require('waeup.manageApplications')
[6383]861    icon = 'actionicon_view.png'
[6367]862    text = 'View application record'
[6598]863    target = 'index'
Note: See TracBrowser for help on using the repository browser.