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

Last change on this file since 10322 was 10276, checked in by Henrik Bettermann, 11 years ago

Remove gpa field. The GPA must not be an attribute of the level. It contains also information on carry over courses of following levels. Thus it can only be reliably calculated at the end of a study course, e.g. in transcripts.

  • Property svn:keywords set to Id
File size: 22.3 KB
RevLine 
[7191]1## $Id: interfaces.py 10276 2013-06-05 08:02:29Z henrik $
[6621]2##
[7191]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##
[8409]18#from datetime import datetime
[7620]19from zope.component import getUtility
[7256]20from zope.interface import Attribute, Interface
[6621]21from zope import schema
[7620]22from zc.sourcefactory.contextual import BasicContextualSourceFactory
[9217]23from waeup.kofa.browser.interfaces import IStudentNavigationBase
[7811]24from waeup.kofa.interfaces import (
[8409]25    IKofaObject, academic_sessions_vocab, validate_email, ICSVExporter)
[7811]26from waeup.kofa.interfaces import MessageFactory as _
[8176]27from waeup.kofa.schema import TextLineChoice, FormattedDate, PhoneNumber
[7811]28from waeup.kofa.students.vocabularies import (
[7915]29    StudyLevelSource, contextual_reg_num_source, contextual_mat_num_source,
[9420]30    GenderSource, nats_vocab
[7915]31    )
[8160]32from waeup.kofa.payments.interfaces import (
[8174]33    IPaymentsContainer, IOnlinePayment)
[7915]34from waeup.kofa.university.vocabularies import (
[9864]35    CourseSource, StudyModeSource, CertificateSource, SemesterSource,
36    ContextualDictSourceFactoryBase)
[6621]37
[9864]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
[9868]45class BalancePaymentCategorySource(ContextualDictSourceFactoryBase):
[9864]46    """A source that delivers all selectable items of balance payments.
47    """
48    #: name of dict to deliver from kofa utils.
[9868]49    DICT_NAME = 'BALANCE_PAYMENT_CATEGORIES'
[9864]50
[7620]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):
[7841]58        verdicts_dict = getUtility(IStudentsUtils).VERDICTS_DICT
[8820]59        return sorted(verdicts_dict.keys())
[7620]60
61    def getToken(self, context, value):
62        return value
63
64    def getTitle(self, context, value):
[7841]65        verdicts_dict = getUtility(IStudentsUtils).VERDICTS_DICT
[8820]66        if value != '0':
67            return verdicts_dict[value] + ' (%s)' % value
[7688]68        return verdicts_dict[value]
[7620]69
[7681]70
[7150]71class IStudentsUtils(Interface):
72    """A collection of methods which are subject to customization.
73
74    """
[7841]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
[9148]83    def setPaymentDetails(category, student, previous_session=None,
84            previous_level=None,):
[8595]85        """Create Payment object and set the payment data of a student for
86        the payment category specified.
[7150]87
88        """
89
[7186]90    def getAccommodation_details(student):
[7150]91        """Determine the accommodation dates of a student.
92
93        """
94
[7186]95    def selectBed(available_beds):
[7150]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
[9949]102    def getPDFCreator(context):
103        """Get some IPDFCreator instance suitable for use with `context`.
104        """
105
[7186]106    def renderPDF(view, subject='', filename='slip.pdf',):
[7150]107        """Render pdf slips for various pages.
108
109        """
110
[7819]111class IStudentsContainer(IKofaObject):
[7096]112    """A students container contains university students.
[6692]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
[8408]150    unique_student_id = Attribute("""A unique student id.""")
151
[9217]152class IStudentNavigation(IStudentNavigationBase):
[8735]153    """Interface needed for student navigation, logging, etc.
[7150]154
[6642]155    """
[8736]156    student = Attribute('Student object of context.')
[7150]157
[8735]158    def writeLogMessage(view, message):
159        """Write a view specific log message into students.log.
160
161        """
162
[7819]163class IStudentBase(IKofaObject):
[6631]164    """Representation of student base data.
[7150]165
[6621]166    """
[7062]167    history = Attribute('Object history, a list of messages')
[6637]168    state = Attribute('Returns the registration state of a student')
[6699]169    password = Attribute('Encrypted password of a student')
[9334]170    temp_password = Attribute(
171        'Dictionary with user name, timestamp and encrypted password')
[7203]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')
[9761]175    entry_session = Attribute('The entry session of the student')
[7062]176    current_session = Attribute('The current session of the student')
[9142]177    current_level = Attribute('The current level of the student')
[7641]178    current_mode = Attribute('The current mode of the student')
[9182]179    current_verdict = Attribute('The current verdict of the student')
[7364]180    fullname = Attribute('All name parts separated by hyphens')
181    display_fullname = Attribute('The fullname of an applicant')
[8969]182    is_postgrad = Attribute('True if postgraduate student')
[6637]183
[8983]184    suspended = schema.Bool(
185        title = _(u'Account suspended'),
186        default = False,
[9035]187        required = False,
[8983]188        )
189
[9702]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
[6665]198    student_id = schema.TextLine(
[7723]199        title = _(u'Student Id'),
[6849]200        required = False,
[6665]201        )
202
[7357]203    firstname = schema.TextLine(
[7723]204        title = _(u'First Name'),
[6621]205        required = True,
206        )
207
[7357]208    middlename = schema.TextLine(
[7723]209        title = _(u'Middle Name'),
[7357]210        required = False,
211        )
212
213    lastname = schema.TextLine(
[7723]214        title = _(u'Last Name (Surname)'),
[7357]215        required = True,
216        )
217
[6996]218    sex = schema.Choice(
[7723]219        title = _(u'Sex'),
[6996]220        source = GenderSource(),
221        required = True,
222        )
223
[6788]224    reg_number = TextLineChoice(
[7723]225        title = _(u'Registration Number'),
[6696]226        required = True,
227        readonly = False,
[6788]228        source = contextual_reg_num_source,
[6696]229        )
230
[6788]231    matric_number = TextLineChoice(
[7723]232        title = _(u'Matriculation Number'),
[6750]233        required = False,
234        readonly = False,
[6788]235        source = contextual_mat_num_source,
[6750]236        )
237
[6769]238    adm_code = schema.TextLine(
[7723]239        title = _(u'PWD Activation Code'),
[6769]240        required = False,
[8977]241        readonly = False,
[6769]242        )
243
[7133]244    email = schema.ASCIILine(
[7723]245        title = _(u'Email'),
[7133]246        required = False,
247        constraint=validate_email,
248        )
[8176]249    phone = PhoneNumber(
[7723]250        title = _(u'Phone'),
[7331]251        description = u'',
[7133]252        required = False,
253        )
254
[9334]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
[9131]269    def transfer(certificate, current_session,
270        current_level, current_verdict):
271        """ Creates a new studycourse and backups the old one.
272
273        """
274
[7993]275class IUGStudentClearance(IKofaObject):
276    """Representation of undergraduate student clearance data.
[7150]277
[6631]278    """
[9543]279    officer_comment = schema.Text(
280        title = _(u"Officer's Comment"),
281        required = False,
[6631]282        )
283
[6695]284    clearance_locked = schema.Bool(
[7723]285        title = _(u'Clearance form locked'),
[6695]286        default = False,
[9035]287        required = False,
[6695]288        )
289
[6769]290    clr_code = schema.TextLine(
[7723]291        title = _(u'CLR Activation Code'),
[6769]292        required = False,
[8977]293        readonly = False,
[6769]294        )
295
[9543]296    date_of_birth = FormattedDate(
297        title = _(u'Date of Birth'),
298        required = True,
299        show_year = True,
300        )
301
[7523]302    nationality = schema.Choice(
[8069]303        vocabulary = nats_vocab,
[7723]304        title = _(u'Nationality'),
[7983]305        required = False,
[7523]306        )
307
[7993]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
[7819]318class IStudentPersonal(IKofaObject):
[6631]319    """Representation of student personal data.
[7150]320
[6631]321    """
[9543]322    personal_updated = schema.Datetime(
323        title = _(u'Updated'),
324        required = False,
325        readonly = False,
326        )
327
[6651]328    perm_address = schema.Text(
[7723]329        title = _(u'Permanent Address'),
[6631]330        required = False,
331        )
332
[8008]333class IStudent(IStudentBase,IUGStudentClearance,IPGStudentClearance,
334    IStudentPersonal):
[6631]335    """Representation of a student.
[7150]336
[6631]337    """
338
[9563]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
[6849]351class IStudentUpdateByRegNo(IStudent):
352    """Representation of a student. Skip regular reg_number validation.
[7150]353
[6849]354    """
355    reg_number = schema.TextLine(
[7723]356        title = _(u'Registration Number'),
[6849]357        required = False,
358        )
359
360class IStudentUpdateByMatricNo(IStudent):
361    """Representation of a student. Skip regular matric_number validation.
[7150]362
[6849]363    """
364    matric_number = schema.TextLine(
[7723]365        title = _(u'Matriculation Number'),
[6849]366        required = False,
367        )
368
[8779]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    """
[8854]375    number = schema.TextLine(
376        title = _(u'Registr. or Matric. Number'),
[8779]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
[7819]391class IStudentStudyCourse(IKofaObject):
[6633]392    """A container for student study levels.
393
394    """
[6648]395    certificate = schema.Choice(
[7723]396        title = _(u'Certificate'),
[6648]397        source = CertificateSource(),
[7209]398        required = False,
[6633]399        )
[6635]400
[6996]401    entry_mode = schema.Choice(
[7723]402        title = _(u'Entry Mode'),
[7681]403        source = StudyModeSource(),
[6996]404        required = True,
405        readonly = False,
406        )
407
408    entry_session = schema.Choice(
[7723]409        title = _(u'Entry Session'),
[6996]410        source = academic_sessions_vocab,
[7425]411        #default = datetime.now().year,
[6996]412        required = True,
413        readonly = False,
414        )
415
[6724]416    current_session = schema.Choice(
[7723]417        title = _(u'Current Session'),
[6744]418        source = academic_sessions_vocab,
[6724]419        required = True,
[6996]420        readonly = False,
[6724]421        )
422
423    current_level = schema.Choice(
[7723]424        title = _(u'Current Level'),
[6725]425        source = StudyLevelSource(),
426        required = False,
[6996]427        readonly = False,
[6724]428        )
429
430    current_verdict = schema.Choice(
[7723]431        title = _(u'Current Verdict'),
[7619]432        source = VerdictSource(),
[8820]433        default = '0',
[6725]434        required = False,
[6724]435        )
436
437    previous_verdict = schema.Choice(
[7723]438        title = _(u'Previous Verdict'),
[7619]439        source = VerdictSource(),
[8820]440        default = '0',
[6725]441        required = False,
[6724]442        )
443
[9138]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
[9960]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
[9138]471IStudentStudyCourseTransfer['certificate'].order = IStudentStudyCourse[
472    'certificate'].order
473IStudentStudyCourseTransfer['current_level'].order = IStudentStudyCourse[
474    'current_level'].order
475
[10250]476class IStudentStudyCourseTranscript(IKofaObject):
477    """An interface for student transcripts.
478
479    """
480
481    entry_mode = schema.Choice(
482        title = _(u'Entry Mode'),
483        source = StudyModeSource(),
484        required = True,
485        readonly = False,
486        )
487
488    entry_session = schema.Choice(
489        title = _(u'Entry Session'),
490        source = academic_sessions_vocab,
491        #default = datetime.now().year,
492        required = True,
493        readonly = False,
494        )
495
[7951]496class IStudentVerdictUpdate(IKofaObject):
497    """A interface for verdict imports.
498
499    """
500
501    current_verdict = schema.Choice(
502        title = _(u'Current Verdict'),
503        source = VerdictSource(),
504        required = True,
505        )
506
507    current_session = schema.Choice(
508        title = _(u'Current Session'),
509        source = academic_sessions_vocab,
510        required = True,
511        )
512
513    current_level = schema.Choice(
514        title = _(u'Current Level'),
515        source = StudyLevelSource(),
516        required = True,
517        )
518
[9296]519    bypass_validation = schema.Bool(
520        title = _(u'Bypass validation'),
521        required = False,
522        )
523
524    validated_by = schema.TextLine(
525        title = _(u'Validated by'),
526        required = False,
527        )
528
[7819]529class IStudentStudyLevel(IKofaObject):
[6774]530    """A container for course tickets.
531
532    """
533    level = Attribute('The level code')
[9235]534    number_of_tickets = Attribute('Number of tickets contained in this level')
[9253]535    certcode = Attribute('The certificate code of the study course')
[9257]536    is_current_level = Attribute('Is this level the current level of the student?')
[6774]537
[6793]538    level_session = schema.Choice(
[7723]539        title = _(u'Session'),
[6793]540        source = academic_sessions_vocab,
[9437]541        required = True,
[6793]542        )
[6781]543
[6793]544    level_verdict = schema.Choice(
[7723]545        title = _(u'Verdict'),
[7619]546        source = VerdictSource(),
[8820]547        default = '0',
[6793]548        required = False,
549        )
550
[9161]551    validated_by = schema.TextLine(
552        title = _(u'Validated by'),
553        default = None,
554        required = False,
555        )
556
557    validation_date = schema.Datetime(
558        title = _(u'Validation Date'),
559        required = False,
560        readonly = False,
561        )
562
[9690]563    total_credits = schema.Int(
564        title = _(u'Total Credits'),
565        required = False,
566        readonly = True,
567        )
568
[10276]569    #gpa = schema.Int(
570    #    title = _(u'GPA'),
571    #    required = False,
572    #    readonly = True,
573    #    )
[9690]574
[8920]575    def addCourseTicket(ticket, course):
[8182]576        """Add a course ticket object.
577        """
578
[9501]579    def addCertCourseTickets(cert):
580        """Collect all certificate courses and create course
581        tickets automatically.
582        """
583
[7819]584class ICourseTicket(IKofaObject):
[9420]585    """An interface for course tickets.
[6781]586
587    """
[6783]588    code = Attribute('code of the original course')
[9253]589    certcode = Attribute('certificate code of the study course')
[9684]590    grade = Attribute('grade calculated from score')
591    weight = Attribute('weight calculated from score')
[9698]592    removable_by_student = Attribute('Is student allowed to remove the ticket?')
[9925]593    level_session = Attribute('session of the level the ticket has been added to')
594    level = Attribute('id of the level the ticket has been added to')
[6781]595
[9420]596    title = schema.TextLine(
597        title = _(u'Title'),
598        required = False,
599        )
600
601    fcode = schema.TextLine(
602        title = _(u'Faculty Code'),
603        required = False,
604        )
605
606    dcode = schema.TextLine(
607        title = _(u'Department Code'),
608        required = False,
609        )
610
611    semester = schema.Choice(
612        title = _(u'Semester/Term'),
613        source = SemesterSource(),
614        required = False,
615        )
616
617    passmark = schema.Int(
618        title = _(u'Passmark'),
619        required = False,
620        )
621
622    credits = schema.Int(
623        title = _(u'Credits'),
624        required = False,
625        )
626
[7665]627    mandatory = schema.Bool(
[9320]628        title = _(u'Required'),
[6795]629        default = False,
630        required = False,
631        )
632
[6781]633    score = schema.Int(
[7723]634        title = _(u'Score'),
[9684]635        default = None,
[6781]636        required = False,
637        )
638
[7661]639    carry_over = schema.Bool(
[7723]640        title = _(u'Carry-over Course'),
[7661]641        default = False,
642        required = False,
643        )
644
[9420]645    automatic = schema.Bool(
646        title = _(u'Automatical Creation'),
647        default = False,
[9316]648        required = False,
649        )
650
[7633]651
[9420]652class ICourseTicketAdd(IKofaObject):
[7150]653    """An interface for adding course tickets.
[6795]654
655    """
656    course = schema.Choice(
[7723]657        title = _(u'Course'),
[6795]658        source = CourseSource(),
659        readonly = False,
660        )
661
[9420]662class ICourseTicketImport(ICourseTicket):
663    """An interface for importing course results and nothing more.
664
665    """
666    score = schema.Int(
667        title = _(u'Score'),
668        required = False,
669        readonly = False,
670        )
671
672    level_session = schema.Choice(
673        title = _(u'Level Session'),
674        source = academic_sessions_vocab,
675        required = False,
676        readonly = False,
677        )
678
[7819]679class IStudentAccommodation(IKofaObject):
[6635]680    """A container for student accommodation objects.
681
682    """
683
[9423]684    def addBedTicket(bedticket):
685        """Add a bed ticket object.
686        """
687
688
[7819]689class IBedTicket(IKofaObject):
[6989]690    """A ticket for accommodation booking.
691
692    """
[6996]693    bed = Attribute('The bed object.')
694
695    bed_coordinates = schema.TextLine(
[9984]696        title = u'',
[9423]697        required = True,
[7014]698        readonly = False,
[6992]699        )
700
[9984]701    display_coordinates = schema.TextLine(
702        title = _(u'Allocated Bed'),
703        required = False,
704        readonly = True,
705        )
706
[6996]707    bed_type = schema.TextLine(
[9424]708        title = _(u'Requested Bed Type'),
[9423]709        required = True,
[7014]710        readonly = False,
[6996]711        )
712
[6992]713    booking_session = schema.Choice(
[7723]714        title = _(u'Session'),
[6992]715        source = academic_sessions_vocab,
716        required = True,
[7014]717        readonly = True,
[6992]718        )
719
[8170]720    booking_date = schema.Datetime(
[7723]721        title = _(u'Booking Date'),
[6992]722        required = False,
[7014]723        readonly = True,
[6992]724        )
725
726    booking_code = schema.TextLine(
[7723]727        title = _(u'Booking Activation Code'),
[6992]728        required = False,
[7014]729        readonly = True,
[6992]730        )
731
[6994]732    def getSessionString():
[7633]733        """Returns the title of academic_sessions_vocab term.
[7150]734
[6994]735        """
[6992]736
[6860]737class IStudentPaymentsContainer(IPaymentsContainer):
[6635]738    """A container for student payment objects.
739
740    """
741
[6877]742class IStudentOnlinePayment(IOnlinePayment):
743    """A student payment via payment gateways.
744
745    """
746
[9148]747    p_current = schema.Bool(
748        title = _(u'Current Session Payment'),
749        default = True,
750        required = False,
751        )
752
[8268]753    p_level = schema.Int(
754        title = _(u'Payment Level'),
755        required = False,
756        )
[6877]757
[8422]758    def doAfterStudentPayment():
759        """Process student after payment was made.
760
761        """
762
[8453]763    def doAfterStudentPaymentApproval():
764        """Process student after payment was approved.
765
766        """
767
[8420]768    def approveStudentPayment():
[8422]769        """Approve payment and process student.
[8420]770
771        """
772
[8268]773IStudentOnlinePayment['p_level'].order = IStudentOnlinePayment[
774    'p_session'].order
[8408]775
[9864]776class IStudentPreviousPayment(IKofaObject):
[9148]777    """An interface for adding previous session payments.
778
779    """
780
[9864]781    p_category = schema.Choice(
782        title = _(u'Payment Category'),
783        default = u'schoolfee',
784        source = PreviousPaymentCategorySource(),
785        required = True,
786        )
787
[9148]788    p_session = schema.Choice(
789        title = _(u'Payment Session'),
790        source = academic_sessions_vocab,
791        required = True,
792        )
793
794    p_level = schema.Choice(
795        title = _(u'Payment Level'),
796        source = StudyLevelSource(),
797        required = True,
798        )
799
[9864]800class IStudentBalancePayment(IKofaObject):
801    """An interface for adding balances.
802
803    """
804
[9868]805    p_category = schema.Choice(
[9864]806        title = _(u'Payment Category'),
807        default = u'schoolfee',
808        required = True,
[9868]809        source = BalancePaymentCategorySource(),
[9864]810        )
811
812    balance_session = schema.Choice(
813        title = _(u'Payment Session'),
814        source = academic_sessions_vocab,
815        required = True,
816        )
817
818    balance_level = schema.Choice(
819        title = _(u'Payment Level'),
820        source = StudyLevelSource(),
821        required = True,
822        )
823
824    balance_amount = schema.Float(
825        title = _(u'Balance Amount'),
[9874]826        default = None,
[9864]827        required = True,
828        readonly = False,
829        description = _(
830            u'Balance in Naira '),
831        )
832
[8408]833class ICSVStudentExporter(ICSVExporter):
834    """A regular ICSVExporter that additionally supports exporting
835      data from a given student object.
836    """
[9801]837    def get_filtered(site, **kw):
[9797]838        """Get a filtered set of students.
839        """
[8408]840
841    def export_student(student, filepath=None):
842        """Export data for a given student.
843        """
[9734]844
[9797]845    def export_filtered(site, filepath=None, **kw):
[9734]846        """Export filtered set of students.
847        """
Note: See TracBrowser for help on using the repository browser.