~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-04-28 07:08:56 UTC
  • Revision ID: grantw@unimelb.edu.au-20090428070856-75yc00g6ea24qfqz
Drop ivle.conf.{subjects,exercises}_base - they were unused.

Show diffs side-by-side

added added

removed removed

Lines of Context:
26
26
import urllib
27
27
import cgi
28
28
 
29
 
from common import util
30
 
import common.db
31
 
 
32
 
def handle(req):
33
 
    """Handler for the Subjects application. Links to subject home pages."""
34
 
 
35
 
    req.styles = ["media/subjects/subjects.css"]
36
 
    if req.path == "":
37
 
        handle_toplevel_menu(req)
38
 
    else:
39
 
        handle_subject_page(req, req.path)
40
 
 
41
 
def handle_toplevel_menu(req):
42
 
    # This is represented as a directory. Redirect and add a slash if it is
43
 
    # missing.
44
 
    if req.uri[-1] != '/':
45
 
        req.throw_redirect(req.uri + '/')
46
 
 
47
 
    (enrolled_subjects, unenrolled_subjects) = \
48
 
              common.db.DB().get_subjects_status(req.user.login)
49
 
 
50
 
    def print_subject(subject):
51
 
        if subject['url'] is None:
52
 
            req.write('  <li>%s (no home page)</li>\n'
53
 
                % cgi.escape(subject['subj_name']))
54
 
        else:
55
 
            req.write('  <li><a href="%s">%s</a></li>\n'
56
 
                % (cgi.escape(subject['url']),
57
 
                   cgi.escape(subject['subj_name'])))
58
 
 
59
 
    req.content_type = "text/html"
60
 
    req.write_html_head_foot = True
61
 
    req.write('<div id="ivle_padding">\n')
62
 
    req.write("<h2>IVLE Subject Homepages</h2>\n")
63
 
    req.write("<h2>Subjects</h2>\n<ul>\n")
64
 
    for subject in enrolled_subjects:
65
 
        print_subject(subject)
66
 
    req.write("</ul>\n")
67
 
    if len(unenrolled_subjects) > 0:
68
 
        req.write("<h3>Other Subjects</h3>\n")
69
 
        req.write("<p>You are not currently enrolled in these subjects</p>\n")
70
 
        req.write("<ul>\n")
71
 
        for subject in unenrolled_subjects:
72
 
            print_subject(subject)
73
 
        req.write("</ul>\n")
74
 
    req.write("</div>\n")
75
 
 
76
 
def handle_subject_page(req, path):
77
 
    req.content_type = "text/html"
78
 
    req.write_html_head_foot = True     # Have dispatch print head and foot
79
 
 
80
 
    # Just make the iframe pointing to media/subjects
81
 
    serve_loc = util.make_path(os.path.join('media', 'subjects', path))
82
 
    req.write('<object class="fullscreen" type="text/html" \
83
 
data="%s"></iframe>'% urllib.quote(serve_loc))
 
29
from storm.locals import Desc
 
30
from genshi.filters import HTMLFormFiller
 
31
import formencode
 
32
 
 
33
from ivle.webapp.base.xhtml import XHTMLView
 
34
from ivle.webapp.base.plugins import ViewPlugin, MediaPlugin
 
35
from ivle.webapp.errors import NotFound
 
36
from ivle.database import Subject, Semester, Offering, Enrolment, User
 
37
 
 
38
 
 
39
class SubjectsView(XHTMLView):
 
40
    '''The view of the list of subjects.'''
 
41
    template = 'subjects.html'
 
42
    tab = 'subjects'
 
43
 
 
44
    def authorize(self, req):
 
45
        return req.user is not None
 
46
 
 
47
    def populate(self, req, ctx):
 
48
        ctx['user'] = req.user
 
49
        ctx['semesters'] = []
 
50
        for semester in req.store.find(Semester).order_by(Desc(Semester.year),
 
51
                                                     Desc(Semester.semester)):
 
52
            enrolments = semester.enrolments.find(user=req.user)
 
53
            if enrolments.count():
 
54
                ctx['semesters'].append((semester, enrolments))
 
55
 
 
56
 
 
57
class UserValidator(formencode.FancyValidator):
 
58
    """A FormEncode validator that turns a username into a user.
 
59
 
 
60
    The state must have a 'store' attribute, which is the Storm store
 
61
    to use."""
 
62
    def _to_python(self, value, state):
 
63
        user = User.get_by_login(state.store, value)
 
64
        if user:
 
65
            return user
 
66
        else:
 
67
            raise formencode.Invalid('User does not exist', value, state)
 
68
 
 
69
 
 
70
class NoEnrolmentValidator(formencode.FancyValidator):
 
71
    """A FormEncode validator that ensures absence of an enrolment.
 
72
 
 
73
    The state must have an 'offering' attribute.
 
74
    """
 
75
    def _to_python(self, value, state):
 
76
        if state.offering.get_enrolment(value):
 
77
            raise formencode.Invalid('User already enrolled', value, state)
 
78
        return value
 
79
 
 
80
 
 
81
class EnrolSchema(formencode.Schema):
 
82
    user = formencode.All(NoEnrolmentValidator(), UserValidator())
 
83
 
 
84
 
 
85
class EnrolView(XHTMLView):
 
86
    """A form to enrol a user in an offering."""
 
87
    template = 'enrol.html'
 
88
    tab = 'subjects'
 
89
    permission = 'edit'
 
90
 
 
91
    def __init__(self, req, subject, year, semester):
 
92
        """Find the given offering by subject, year and semester."""
 
93
        self.context = req.store.find(Offering,
 
94
            Offering.subject_id == Subject.id,
 
95
            Subject.short_name == subject,
 
96
            Offering.semester_id == Semester.id,
 
97
            Semester.year == year,
 
98
            Semester.semester == semester).one()
 
99
 
 
100
        if not self.context:
 
101
            raise NotFound()
 
102
 
 
103
    def filter(self, stream, ctx):
 
104
        return stream | HTMLFormFiller(data=ctx['data'])
 
105
 
 
106
    def populate(self, req, ctx):
 
107
        if req.method == 'POST':
 
108
            data = dict(req.get_fieldstorage())
 
109
            try:
 
110
                validator = EnrolSchema()
 
111
                req.offering = self.context # XXX: Getting into state.
 
112
                data = validator.to_python(data, state=req)
 
113
                self.context.enrol(data['user'])
 
114
                req.store.commit()
 
115
                req.throw_redirect(req.uri)
 
116
            except formencode.Invalid, e:
 
117
                errors = e.unpack_errors()
 
118
        else:
 
119
            data = {}
 
120
            errors = {}
 
121
 
 
122
        ctx['data'] = data or {}
 
123
        ctx['offering'] = self.context
 
124
        ctx['errors'] = errors
 
125
 
 
126
 
 
127
class Plugin(ViewPlugin, MediaPlugin):
 
128
    urls = [
 
129
        ('subjects/', SubjectsView),
 
130
        ('subjects/:subject/:year/:semester/+enrolments/+new', EnrolView),
 
131
    ]
 
132
 
 
133
    tabs = [
 
134
        ('subjects', 'Subjects',
 
135
         'View subject content and complete worksheets',
 
136
         'subjects.png', 'subjects', 5)
 
137
    ]
 
138
 
 
139
    media = 'subject-media'