~azzar1/unity/add-show-desktop-key

« back to all changes in this revision

Viewing changes to ivle/webapp/admin/subject.py

  • Committer: William Grant
  • Date: 2009-02-17 00:01:10 UTC
  • mto: (1099.1.143 new-dispatch)
  • mto: This revision was merged to the branch mainline in revision 1100.
  • Revision ID: grantw@unimelb.edu.au-20090217000110-mysp1vl5n29uwm2v
Change serveservice's interface - it now uses a JSON-based non-CGI one.
This breaks serving things.

Show diffs side-by-side

added added

removed removed

Lines of Context:
26
26
import urllib
27
27
import cgi
28
28
 
29
 
from storm.locals import Desc
30
 
from genshi.filters import HTMLFormFiller
31
 
import formencode
32
 
 
33
29
from ivle.webapp.base.xhtml import XHTMLView
34
30
from ivle.webapp.base.plugins import ViewPlugin, MediaPlugin
35
31
from ivle.webapp.errors import NotFound
36
 
from ivle.database import Subject, Semester, Offering, Enrolment, User
 
32
from ivle.database import Subject
37
33
from ivle import util
38
34
 
39
35
 
40
36
class SubjectsView(XHTMLView):
41
37
    '''The view of the list of subjects.'''
42
38
    template = 'subjects.html'
43
 
    tab = 'subjects'
 
39
    appname = 'subjects' # XXX
44
40
 
45
41
    def authorize(self, req):
46
42
        return req.user is not None
47
43
 
48
44
    def populate(self, req, ctx):
49
 
        ctx['user'] = req.user
50
 
        ctx['semesters'] = []
51
 
        for semester in req.store.find(Semester).order_by(Desc(Semester.year),
52
 
                                                     Desc(Semester.semester)):
53
 
            enrolments = semester.enrolments.find(user=req.user)
54
 
            if enrolments.count():
55
 
                ctx['semesters'].append((semester, enrolments))
56
 
 
57
 
 
58
 
class UserValidator(formencode.FancyValidator):
59
 
    """A FormEncode validator that turns a username into a user.
60
 
 
61
 
    The state must have a 'store' attribute, which is the Storm store
62
 
    to use."""
63
 
    def _to_python(self, value, state):
64
 
        user = User.get_by_login(state.store, value)
65
 
        if user:
66
 
            return user
67
 
        else:
68
 
            raise formencode.Invalid('User does not exist', value, state)
69
 
 
70
 
 
71
 
class NoEnrolmentValidator(formencode.FancyValidator):
72
 
    """A FormEncode validator that ensures absence of an enrolment.
73
 
 
74
 
    The state must have an 'offering' attribute.
75
 
    """
76
 
    def _to_python(self, value, state):
77
 
        if state.offering.get_enrolment(value):
78
 
            raise formencode.Invalid('User already enrolled', value, state)
79
 
        return value
80
 
 
81
 
 
82
 
class EnrolSchema(formencode.Schema):
83
 
    user = formencode.All(NoEnrolmentValidator(), UserValidator())
84
 
 
85
 
 
86
 
class EnrolView(XHTMLView):
87
 
    """A form to enrol a user in an offering."""
88
 
    template = 'enrol.html'
89
 
    tab = 'subjects'
90
 
    permission = 'edit'
91
 
 
92
 
    def __init__(self, req, subject, year, semester):
93
 
        """Find the given offering by subject, year and semester."""
94
 
        self.context = req.store.find(Offering,
95
 
            Offering.subject_id == Subject.id,
96
 
            Subject.short_name == subject,
97
 
            Offering.semester_id == Semester.id,
98
 
            Semester.year == year,
99
 
            Semester.semester == semester).one()
100
 
 
101
 
        if not self.context:
102
 
            raise NotFound()
103
 
 
104
 
    def filter(self, stream, ctx):
105
 
        return stream | HTMLFormFiller(data=ctx['data'])
106
 
 
107
 
    def populate(self, req, ctx):
108
 
        if req.method == 'POST':
109
 
            data = dict(req.get_fieldstorage())
110
 
            try:
111
 
                validator = EnrolSchema()
112
 
                req.offering = self.context # XXX: Getting into state.
113
 
                data = validator.to_python(data, state=req)
114
 
                self.context.enrol(data['user'])
115
 
                req.store.commit()
116
 
                req.throw_redirect(req.uri)
117
 
            except formencode.Invalid, e:
118
 
                errors = e.unpack_errors()
119
 
        else:
120
 
            data = {}
121
 
            errors = {}
122
 
 
123
 
        ctx['data'] = data or {}
124
 
        ctx['offering'] = self.context
125
 
        ctx['errors'] = errors
 
45
        enrolled_subjects = req.user.subjects
 
46
        unenrolled_subjects = [subject for subject in
 
47
                               req.store.find(Subject)
 
48
                               if subject not in enrolled_subjects]
 
49
 
 
50
        ctx['enrolled_subjects'] = []
 
51
        ctx['other_subjects'] = []
 
52
 
 
53
        req.content_type = "text/html"
 
54
        req.write_html_head_foot = True
 
55
 
 
56
        for subject in enrolled_subjects:
 
57
            new_subj = {}
 
58
            new_subj['name'] = subject.name
 
59
            new_subj['url'] = subject.url
 
60
            ctx['enrolled_subjects'].append(new_subj)
 
61
 
 
62
        if len(unenrolled_subjects) > 0:
 
63
            for subject in unenrolled_subjects:
 
64
                new_subj = {}
 
65
                new_subj['name'] = subject.name
 
66
                new_subj['url'] = subject.url
 
67
                ctx['other_subjects'].append(new_subj)
126
68
 
127
69
 
128
70
class Plugin(ViewPlugin, MediaPlugin):
129
71
    urls = [
130
72
        ('subjects/', SubjectsView),
131
 
        ('subjects/:subject/:year/:semester/+enrolments/+new', EnrolView),
132
73
    ]
133
74
 
134
75
    tabs = [
135
 
        ('subjects', 'Subjects',
136
 
         'View subject content and complete worksheets',
137
 
         'subjects.png', 'subjects', 5)
 
76
        ('subjects', 'Subjects', 'Announcements and information about the '
 
77
         'subjects you are enrolled in.', 'subjects.png', 'subjects', 5)
138
78
    ]
139
79
 
140
80
    media = 'subject-media'