source: main/waeup.kofa/trunk/src/waeup/kofa/students/interfaces.py @ 10036

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

FCEOkene wants to hide the bed coordinates if maintenance fee is not yet paid. Thus we need additional property attributes which return the p_item (payment tickets) and the bed_coordinates attributes (bed tickets) by default and can be easily customized to hide this information in certain cases. bed_coordinates and p_item must be omitted on forms. The new display_ attributes are displayed instead.

All packages must now be adjusted.

The 'cost-benefit ratio' of these kinds of customizations is quite bad and we should think about declining such customization requests. However, I started customization and these are the changed made in the base package.

  • Property svn:keywords set to Id
File size: 21.9 KB
Line 
1## $Id: interfaces.py 9984 2013-02-24 08:29:24Z henrik $
2##
3## Copyright (C) 2011 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##
18#from datetime import datetime
19from zope.component import getUtility
20from zope.interface import Attribute, Interface
21from zope import schema
22from zc.sourcefactory.contextual import BasicContextualSourceFactory
23from waeup.kofa.browser.interfaces import IStudentNavigationBase
24from waeup.kofa.interfaces import (
25    IKofaObject, academic_sessions_vocab, validate_email, ICSVExporter)
26from waeup.kofa.interfaces import MessageFactory as _
27from waeup.kofa.schema import TextLineChoice, FormattedDate, PhoneNumber
28from waeup.kofa.students.vocabularies import (
29    StudyLevelSource, contextual_reg_num_source, contextual_mat_num_source,
30    GenderSource, nats_vocab
31    )
32from waeup.kofa.payments.interfaces import (
33    IPaymentsContainer, IOnlinePayment)
34from waeup.kofa.university.vocabularies import (
35    CourseSource, StudyModeSource, CertificateSource, SemesterSource,
36    ContextualDictSourceFactoryBase)
37
38class PreviousPaymentCategorySource(ContextualDictSourceFactoryBase):
39    """A source that delivers all selectable categories of previous session
40    payments.
41    """
42    #: name of dict to deliver from kofa utils.
43    DICT_NAME = 'PREVIOUS_PAYMENT_CATEGORIES'
44
45class BalancePaymentCategorySource(ContextualDictSourceFactoryBase):
46    """A source that delivers all selectable items of balance payments.
47    """
48    #: name of dict to deliver from kofa utils.
49    DICT_NAME = 'BALANCE_PAYMENT_CATEGORIES'
50
51# VerdictSource can't be placed into the vocabularies module because it
52# requires importing IStudentsUtils which then leads to circular imports.
53class VerdictSource(BasicContextualSourceFactory):
54    """A verdicts source delivers all verdicts provided
55    in the portal.
56    """
57    def getValues(self, context):
58        verdicts_dict = getUtility(IStudentsUtils).VERDICTS_DICT
59        return sorted(verdicts_dict.keys())
60
61    def getToken(self, context, value):
62        return value
63
64    def getTitle(self, context, value):
65        verdicts_dict = getUtility(IStudentsUtils).VERDICTS_DICT
66        if value != '0':
67            return verdicts_dict[value] + ' (%s)' % value
68        return verdicts_dict[value]
69
70
71class IStudentsUtils(Interface):
72    """A collection of methods which are subject to customization.
73
74    """
75    def setReturningData(student):
76        """ This method defines what happens after school fee payment
77        depending on the student's senate verdict.
78
79        In the base configuration current level is always increased
80        by 100 no matter which verdict has been assigned.
81        """
82
83    def setPaymentDetails(category, student, previous_session=None,
84            previous_level=None,):
85        """Create Payment object and set the payment data of a student for
86        the payment category specified.
87
88        """
89
90    def getAccommodation_details(student):
91        """Determine the accommodation dates of a student.
92
93        """
94
95    def selectBed(available_beds):
96        """Select a bed from a list of available beds.
97
98        In the standard configuration we select the first bed found,
99        but can also randomize the selection if we like.
100        """
101
102    def getPDFCreator(context):
103        """Get some IPDFCreator instance suitable for use with `context`.
104        """
105
106    def renderPDF(view, subject='', filename='slip.pdf',):
107        """Render pdf slips for various pages.
108
109        """
110
111class IStudentsContainer(IKofaObject):
112    """A students container contains university students.
113
114    """
115    def addStudent(student):
116        """Add an IStudent object and subcontainers.
117
118        """
119
120    def archive(id=None):
121        """Create on-dist archive of students.
122
123        If id is `None`, all students are archived.
124
125        If id contains a single id string, only the respective
126        students are archived.
127
128        If id contains a list of id strings all of the respective
129        students types are saved to disk.
130        """
131
132    def clear(id=None, archive=True):
133        """Remove students of type given by 'id'.
134
135        Optionally archive the students.
136
137        If id is `None`, all students are archived.
138
139        If id contains a single id string, only the respective
140        students are archived.
141
142        If id contains a list of id strings all of the respective
143        student types are saved to disk.
144
145        If `archive` is ``False`` none of the archive-handling is done
146        and respective students are simply removed from the
147        database.
148        """
149
150    unique_student_id = Attribute("""A unique student id.""")
151
152class IStudentNavigation(IStudentNavigationBase):
153    """Interface needed for student navigation, logging, etc.
154
155    """
156    student = Attribute('Student object of context.')
157
158    def writeLogMessage(view, message):
159        """Write a view specific log message into students.log.
160
161        """
162
163class IStudentBase(IKofaObject):
164    """Representation of student base data.
165
166    """
167    history = Attribute('Object history, a list of messages')
168    state = Attribute('Returns the registration state of a student')
169    password = Attribute('Encrypted password of a student')
170    temp_password = Attribute(
171        'Dictionary with user name, timestamp and encrypted password')
172    certcode = Attribute('The certificate code of any chosen study course')
173    depcode = Attribute('The department code of any chosen study course')
174    faccode = Attribute('The faculty code of any chosen study course')
175    entry_session = Attribute('The entry session of the student')
176    current_session = Attribute('The current session of the student')
177    current_level = Attribute('The current level of the student')
178    current_mode = Attribute('The current mode of the student')
179    current_verdict = Attribute('The current verdict of the student')
180    fullname = Attribute('All name parts separated by hyphens')
181    display_fullname = Attribute('The fullname of an applicant')
182    is_postgrad = Attribute('True if postgraduate student')
183
184    suspended = schema.Bool(
185        title = _(u'Account suspended'),
186        default = False,
187        required = False,
188        )
189
190    suspended_comment = schema.Text(
191        title = _(u"Reasons for Deactivation"),
192        required = False,
193        description = _(
194            u'This message will be shown if and only if deactivated '
195            'students try to login.'),
196        )
197
198    student_id = schema.TextLine(
199        title = _(u'Student Id'),
200        required = False,
201        )
202
203    firstname = schema.TextLine(
204        title = _(u'First Name'),
205        required = True,
206        )
207
208    middlename = schema.TextLine(
209        title = _(u'Middle Name'),
210        required = False,
211        )
212
213    lastname = schema.TextLine(
214        title = _(u'Last Name (Surname)'),
215        required = True,
216        )
217
218    sex = schema.Choice(
219        title = _(u'Sex'),
220        source = GenderSource(),
221        required = True,
222        )
223
224    reg_number = TextLineChoice(
225        title = _(u'Registration Number'),
226        required = True,
227        readonly = False,
228        source = contextual_reg_num_source,
229        )
230
231    matric_number = TextLineChoice(
232        title = _(u'Matriculation Number'),
233        required = False,
234        readonly = False,
235        source = contextual_mat_num_source,
236        )
237
238    adm_code = schema.TextLine(
239        title = _(u'PWD Activation Code'),
240        required = False,
241        readonly = False,
242        )
243
244    email = schema.ASCIILine(
245        title = _(u'Email'),
246        required = False,
247        constraint=validate_email,
248        )
249    phone = PhoneNumber(
250        title = _(u'Phone'),
251        description = u'',
252        required = False,
253        )
254
255    def setTempPassword(user, password):
256        """Set a temporary password (LDAP-compatible) SSHA encoded for
257        officers.
258
259        """
260
261    def getTempPassword():
262        """Check if a temporary password has been set and if it
263        is not expired.
264
265        Return the temporary password if valid,
266        None otherwise. Unset the temporary password if expired.
267        """
268
269    def transfer(certificate, current_session,
270        current_level, current_verdict):
271        """ Creates a new studycourse and backups the old one.
272
273        """
274
275class IUGStudentClearance(IKofaObject):
276    """Representation of undergraduate student clearance data.
277
278    """
279    officer_comment = schema.Text(
280        title = _(u"Officer's Comment"),
281        required = False,
282        )
283
284    clearance_locked = schema.Bool(
285        title = _(u'Clearance form locked'),
286        default = False,
287        required = False,
288        )
289
290    clr_code = schema.TextLine(
291        title = _(u'CLR Activation Code'),
292        required = False,
293        readonly = False,
294        )
295
296    date_of_birth = FormattedDate(
297        title = _(u'Date of Birth'),
298        required = True,
299        show_year = True,
300        )
301
302    nationality = schema.Choice(
303        vocabulary = nats_vocab,
304        title = _(u'Nationality'),
305        required = False,
306        )
307
308class IPGStudentClearance(IUGStudentClearance):
309    """Representation of postgraduate student clearance data.
310
311    """
312    employer = schema.TextLine(
313        title = _(u'Employer'),
314        required = False,
315        readonly = False,
316        )
317
318class IStudentPersonal(IKofaObject):
319    """Representation of student personal data.
320
321    """
322    personal_updated = schema.Datetime(
323        title = _(u'Updated'),
324        required = False,
325        readonly = False,
326        )
327
328    perm_address = schema.Text(
329        title = _(u'Permanent Address'),
330        required = False,
331        )
332
333class IStudent(IStudentBase,IUGStudentClearance,IPGStudentClearance,
334    IStudentPersonal):
335    """Representation of a student.
336
337    """
338
339class IStudentPersonalEdit(IStudentPersonal):
340    """Interface for editing personal data by students.
341
342    Here we can repeat the fields from IStudentPersonal and set the
343    `required` if necessary.
344    """
345
346    perm_address = schema.Text(
347        title = _(u'Permanent Address'),
348        required = True,
349        )
350
351class IStudentUpdateByRegNo(IStudent):
352    """Representation of a student. Skip regular reg_number validation.
353
354    """
355    reg_number = schema.TextLine(
356        title = _(u'Registration Number'),
357        required = False,
358        )
359
360class IStudentUpdateByMatricNo(IStudent):
361    """Representation of a student. Skip regular matric_number validation.
362
363    """
364    matric_number = schema.TextLine(
365        title = _(u'Matriculation Number'),
366        required = False,
367        )
368
369class IStudentRequestPW(IStudent):
370    """Representation of an student for first-time password request.
371
372    This interface is used when students use the requestpw page to
373    login for the the first time.
374    """
375    number = schema.TextLine(
376        title = _(u'Registr. or Matric. Number'),
377        required = True,
378        )
379
380    firstname = schema.TextLine(
381        title = _(u'First Name'),
382        required = True,
383        )
384
385    email = schema.ASCIILine(
386        title = _(u'Email Address'),
387        required = True,
388        constraint=validate_email,
389        )
390
391class IStudentStudyCourse(IKofaObject):
392    """A container for student study levels.
393
394    """
395    certificate = schema.Choice(
396        title = _(u'Certificate'),
397        source = CertificateSource(),
398        required = False,
399        )
400
401    entry_mode = schema.Choice(
402        title = _(u'Entry Mode'),
403        source = StudyModeSource(),
404        required = True,
405        readonly = False,
406        )
407
408    entry_session = schema.Choice(
409        title = _(u'Entry Session'),
410        source = academic_sessions_vocab,
411        #default = datetime.now().year,
412        required = True,
413        readonly = False,
414        )
415
416    current_session = schema.Choice(
417        title = _(u'Current Session'),
418        source = academic_sessions_vocab,
419        required = True,
420        readonly = False,
421        )
422
423    current_level = schema.Choice(
424        title = _(u'Current Level'),
425        source = StudyLevelSource(),
426        required = False,
427        readonly = False,
428        )
429
430    current_verdict = schema.Choice(
431        title = _(u'Current Verdict'),
432        source = VerdictSource(),
433        default = '0',
434        required = False,
435        )
436
437    previous_verdict = schema.Choice(
438        title = _(u'Previous Verdict'),
439        source = VerdictSource(),
440        default = '0',
441        required = False,
442        )
443
444class IStudentStudyCourseTransfer(IStudentStudyCourse):
445    """An student transfers.
446
447    """
448
449    certificate = schema.Choice(
450        title = _(u'Certificate'),
451        source = CertificateSource(),
452        required = True,
453        )
454
455    current_level = schema.Choice(
456        title = _(u'Current Level'),
457        source = StudyLevelSource(),
458        required = True,
459        readonly = False,
460        )
461
462    entry_session = schema.Choice(
463        title = _(u'Entry Session'),
464        source = academic_sessions_vocab,
465        #default = datetime.now().year,
466        required = False,
467        readonly = False,
468        )
469
470
471IStudentStudyCourseTransfer['certificate'].order = IStudentStudyCourse[
472    'certificate'].order
473IStudentStudyCourseTransfer['current_level'].order = IStudentStudyCourse[
474    'current_level'].order
475
476class IStudentVerdictUpdate(IKofaObject):
477    """A interface for verdict imports.
478
479    """
480
481    current_verdict = schema.Choice(
482        title = _(u'Current Verdict'),
483        source = VerdictSource(),
484        required = True,
485        )
486
487    current_session = schema.Choice(
488        title = _(u'Current Session'),
489        source = academic_sessions_vocab,
490        required = True,
491        )
492
493    current_level = schema.Choice(
494        title = _(u'Current Level'),
495        source = StudyLevelSource(),
496        required = True,
497        )
498
499    bypass_validation = schema.Bool(
500        title = _(u'Bypass validation'),
501        required = False,
502        )
503
504    validated_by = schema.TextLine(
505        title = _(u'Validated by'),
506        required = False,
507        )
508
509class IStudentStudyLevel(IKofaObject):
510    """A container for course tickets.
511
512    """
513    level = Attribute('The level code')
514    number_of_tickets = Attribute('Number of tickets contained in this level')
515    certcode = Attribute('The certificate code of the study course')
516    is_current_level = Attribute('Is this level the current level of the student?')
517
518    level_session = schema.Choice(
519        title = _(u'Session'),
520        source = academic_sessions_vocab,
521        required = True,
522        )
523
524    level_verdict = schema.Choice(
525        title = _(u'Verdict'),
526        source = VerdictSource(),
527        default = '0',
528        required = False,
529        )
530
531    validated_by = schema.TextLine(
532        title = _(u'Validated by'),
533        default = None,
534        required = False,
535        )
536
537    validation_date = schema.Datetime(
538        title = _(u'Validation Date'),
539        required = False,
540        readonly = False,
541        )
542
543    total_credits = schema.Int(
544        title = _(u'Total Credits'),
545        required = False,
546        readonly = True,
547        )
548
549    gpa = schema.Int(
550        title = _(u'GPA'),
551        required = False,
552        readonly = True,
553        )
554
555    def addCourseTicket(ticket, course):
556        """Add a course ticket object.
557        """
558
559    def addCertCourseTickets(cert):
560        """Collect all certificate courses and create course
561        tickets automatically.
562        """
563
564class ICourseTicket(IKofaObject):
565    """An interface for course tickets.
566
567    """
568    code = Attribute('code of the original course')
569    certcode = Attribute('certificate code of the study course')
570    grade = Attribute('grade calculated from score')
571    weight = Attribute('weight calculated from score')
572    removable_by_student = Attribute('Is student allowed to remove the ticket?')
573    level_session = Attribute('session of the level the ticket has been added to')
574    level = Attribute('id of the level the ticket has been added to')
575
576    title = schema.TextLine(
577        title = _(u'Title'),
578        required = False,
579        )
580
581    fcode = schema.TextLine(
582        title = _(u'Faculty Code'),
583        required = False,
584        )
585
586    dcode = schema.TextLine(
587        title = _(u'Department Code'),
588        required = False,
589        )
590
591    semester = schema.Choice(
592        title = _(u'Semester/Term'),
593        source = SemesterSource(),
594        required = False,
595        )
596
597    passmark = schema.Int(
598        title = _(u'Passmark'),
599        required = False,
600        )
601
602    credits = schema.Int(
603        title = _(u'Credits'),
604        required = False,
605        )
606
607    mandatory = schema.Bool(
608        title = _(u'Required'),
609        default = False,
610        required = False,
611        )
612
613    score = schema.Int(
614        title = _(u'Score'),
615        default = None,
616        required = False,
617        )
618
619    carry_over = schema.Bool(
620        title = _(u'Carry-over Course'),
621        default = False,
622        required = False,
623        )
624
625    automatic = schema.Bool(
626        title = _(u'Automatical Creation'),
627        default = False,
628        required = False,
629        )
630
631
632class ICourseTicketAdd(IKofaObject):
633    """An interface for adding course tickets.
634
635    """
636    course = schema.Choice(
637        title = _(u'Course'),
638        source = CourseSource(),
639        readonly = False,
640        )
641
642class ICourseTicketImport(ICourseTicket):
643    """An interface for importing course results and nothing more.
644
645    """
646    score = schema.Int(
647        title = _(u'Score'),
648        required = False,
649        readonly = False,
650        )
651
652    level_session = schema.Choice(
653        title = _(u'Level Session'),
654        source = academic_sessions_vocab,
655        required = False,
656        readonly = False,
657        )
658
659class IStudentAccommodation(IKofaObject):
660    """A container for student accommodation objects.
661
662    """
663
664    def addBedTicket(bedticket):
665        """Add a bed ticket object.
666        """
667
668
669class IBedTicket(IKofaObject):
670    """A ticket for accommodation booking.
671
672    """
673    bed = Attribute('The bed object.')
674
675    bed_coordinates = schema.TextLine(
676        title = u'',
677        required = True,
678        readonly = False,
679        )
680
681    display_coordinates = schema.TextLine(
682        title = _(u'Allocated Bed'),
683        required = False,
684        readonly = True,
685        )
686
687    bed_type = schema.TextLine(
688        title = _(u'Requested Bed Type'),
689        required = True,
690        readonly = False,
691        )
692
693    booking_session = schema.Choice(
694        title = _(u'Session'),
695        source = academic_sessions_vocab,
696        required = True,
697        readonly = True,
698        )
699
700    booking_date = schema.Datetime(
701        title = _(u'Booking Date'),
702        required = False,
703        readonly = True,
704        )
705
706    booking_code = schema.TextLine(
707        title = _(u'Booking Activation Code'),
708        required = False,
709        readonly = True,
710        )
711
712    def getSessionString():
713        """Returns the title of academic_sessions_vocab term.
714
715        """
716
717class IStudentPaymentsContainer(IPaymentsContainer):
718    """A container for student payment objects.
719
720    """
721
722class IStudentOnlinePayment(IOnlinePayment):
723    """A student payment via payment gateways.
724
725    """
726
727    p_current = schema.Bool(
728        title = _(u'Current Session Payment'),
729        default = True,
730        required = False,
731        )
732
733    p_level = schema.Int(
734        title = _(u'Payment Level'),
735        required = False,
736        )
737
738    def doAfterStudentPayment():
739        """Process student after payment was made.
740
741        """
742
743    def doAfterStudentPaymentApproval():
744        """Process student after payment was approved.
745
746        """
747
748    def approveStudentPayment():
749        """Approve payment and process student.
750
751        """
752
753IStudentOnlinePayment['p_level'].order = IStudentOnlinePayment[
754    'p_session'].order
755
756class IStudentPreviousPayment(IKofaObject):
757    """An interface for adding previous session payments.
758
759    """
760
761    p_category = schema.Choice(
762        title = _(u'Payment Category'),
763        default = u'schoolfee',
764        source = PreviousPaymentCategorySource(),
765        required = True,
766        )
767
768    p_session = schema.Choice(
769        title = _(u'Payment Session'),
770        source = academic_sessions_vocab,
771        required = True,
772        )
773
774    p_level = schema.Choice(
775        title = _(u'Payment Level'),
776        source = StudyLevelSource(),
777        required = True,
778        )
779
780class IStudentBalancePayment(IKofaObject):
781    """An interface for adding balances.
782
783    """
784
785    p_category = schema.Choice(
786        title = _(u'Payment Category'),
787        default = u'schoolfee',
788        required = True,
789        source = BalancePaymentCategorySource(),
790        )
791
792    balance_session = schema.Choice(
793        title = _(u'Payment Session'),
794        source = academic_sessions_vocab,
795        required = True,
796        )
797
798    balance_level = schema.Choice(
799        title = _(u'Payment Level'),
800        source = StudyLevelSource(),
801        required = True,
802        )
803
804    balance_amount = schema.Float(
805        title = _(u'Balance Amount'),
806        default = None,
807        required = True,
808        readonly = False,
809        description = _(
810            u'Balance in Naira '),
811        )
812
813class ICSVStudentExporter(ICSVExporter):
814    """A regular ICSVExporter that additionally supports exporting
815      data from a given student object.
816    """
817    def get_filtered(site, **kw):
818        """Get a filtered set of students.
819        """
820
821    def export_student(student, filepath=None):
822        """Export data for a given student.
823        """
824
825    def export_filtered(site, filepath=None, **kw):
826        """Export filtered set of students.
827        """
Note: See TracBrowser for help on using the repository browser.