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

« back to all changes in this revision

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

  • Committer: Matt Giuca
  • Date: 2009-02-25 03:04:08 UTC
  • mto: This revision was merged to the branch mainline in revision 1119.
  • Revision ID: matt.giuca@gmail.com-20090225030408-5e13v4felqc4aybr
(almost) all HTML pages updated: Moved <h1>s outside of the ivle-padding div.
    Added new <h1>s to pages which didn't have any, and reduced <h1s> down to
    <h2s> if they didn't represent the page heading.

h1s now represent the top-level heading of a page. The new style sheet styles
them specially, designed to appear directly underneath the tab bar. Therefore,
they need to appear outside of any padding.

util.py: Removed the <h1> from the default-generated terms of service page.
    (All other TOS pages should follow). There is now a <h1> at the top of the
    TOS template page instead.

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
 
import genshi
31
 
from genshi.filters import HTMLFormFiller
32
 
from genshi.template import Context, TemplateLoader
33
 
import formencode
34
 
 
35
29
from ivle.webapp.base.xhtml import XHTMLView
36
30
from ivle.webapp.base.plugins import ViewPlugin, MediaPlugin
37
31
from ivle.webapp.errors import NotFound
38
 
 
39
 
from ivle.database import Subject, Semester, Offering, Enrolment, User,\
40
 
                          ProjectSet, Project, ProjectSubmission
 
32
from ivle.database import Subject
41
33
from ivle import util
42
 
import ivle.date
43
 
 
44
 
from ivle.webapp.admin.projectservice import ProjectSetRESTView,\
45
 
                                             ProjectRESTView
46
 
from ivle.webapp.admin.offeringservice import OfferingRESTView
47
34
 
48
35
 
49
36
class SubjectsView(XHTMLView):
50
37
    '''The view of the list of subjects.'''
51
 
    template = 'templates/subjects.html'
52
 
    tab = 'subjects'
 
38
    template = 'subjects.html'
 
39
    appname = 'subjects' # XXX
53
40
 
54
41
    def authorize(self, req):
55
42
        return req.user is not None
56
43
 
57
44
    def populate(self, req, ctx):
58
 
        ctx['user'] = req.user
59
 
        ctx['semesters'] = []
60
 
        for semester in req.store.find(Semester).order_by(Desc(Semester.year),
61
 
                                                     Desc(Semester.semester)):
62
 
            enrolments = semester.enrolments.find(user=req.user)
63
 
            if enrolments.count():
64
 
                ctx['semesters'].append((semester, enrolments))
65
 
 
66
 
 
67
 
class UserValidator(formencode.FancyValidator):
68
 
    """A FormEncode validator that turns a username into a user.
69
 
 
70
 
    The state must have a 'store' attribute, which is the Storm store
71
 
    to use."""
72
 
    def _to_python(self, value, state):
73
 
        user = User.get_by_login(state.store, value)
74
 
        if user:
75
 
            return user
76
 
        else:
77
 
            raise formencode.Invalid('User does not exist', value, state)
78
 
 
79
 
 
80
 
class NoEnrolmentValidator(formencode.FancyValidator):
81
 
    """A FormEncode validator that ensures absence of an enrolment.
82
 
 
83
 
    The state must have an 'offering' attribute.
84
 
    """
85
 
    def _to_python(self, value, state):
86
 
        if state.offering.get_enrolment(value):
87
 
            raise formencode.Invalid('User already enrolled', value, state)
88
 
        return value
89
 
 
90
 
 
91
 
class EnrolSchema(formencode.Schema):
92
 
    user = formencode.All(NoEnrolmentValidator(), UserValidator())
93
 
 
94
 
 
95
 
class EnrolView(XHTMLView):
96
 
    """A form to enrol a user in an offering."""
97
 
    template = 'templates/enrol.html'
98
 
    tab = 'subjects'
99
 
    permission = 'edit'
100
 
 
101
 
    def __init__(self, req, subject, year, semester):
102
 
        """Find the given offering by subject, year and semester."""
103
 
        self.context = req.store.find(Offering,
104
 
            Offering.subject_id == Subject.id,
105
 
            Subject.short_name == subject,
106
 
            Offering.semester_id == Semester.id,
107
 
            Semester.year == year,
108
 
            Semester.semester == semester).one()
109
 
 
110
 
        if not self.context:
111
 
            raise NotFound()
112
 
 
113
 
    def filter(self, stream, ctx):
114
 
        return stream | HTMLFormFiller(data=ctx['data'])
115
 
 
116
 
    def populate(self, req, ctx):
117
 
        if req.method == 'POST':
118
 
            data = dict(req.get_fieldstorage())
119
 
            try:
120
 
                validator = EnrolSchema()
121
 
                req.offering = self.context # XXX: Getting into state.
122
 
                data = validator.to_python(data, state=req)
123
 
                self.context.enrol(data['user'])
124
 
                req.store.commit()
125
 
                req.throw_redirect(req.uri)
126
 
            except formencode.Invalid, e:
127
 
                errors = e.unpack_errors()
128
 
        else:
129
 
            data = {}
130
 
            errors = {}
131
 
 
132
 
        ctx['data'] = data or {}
133
 
        ctx['offering'] = self.context
134
 
        ctx['errors'] = errors
135
 
 
136
 
class OfferingProjectsView(XHTMLView):
137
 
    """View the projects for an offering."""
138
 
    template = 'templates/offering_projects.html'
139
 
    permission = 'edit'
140
 
    tab = 'subjects'
141
 
    
142
 
    def __init__(self, req, subject, year, semester):
143
 
        self.context = req.store.find(Offering,
144
 
            Offering.subject_id == Subject.id,
145
 
            Subject.short_name == subject,
146
 
            Offering.semester_id == Semester.id,
147
 
            Semester.year == year,
148
 
            Semester.semester == semester).one()
149
 
 
150
 
        if not self.context:
151
 
            raise NotFound()
152
 
 
153
 
    def project_url(self, projectset, project):
154
 
        return "/subjects/%s/%s/%s/+projects/%s" % (
155
 
                    self.context.subject.short_name,
156
 
                    self.context.semester.year,
157
 
                    self.context.semester.semester,
158
 
                    project.short_name
159
 
                    )
160
 
 
161
 
    def new_project_url(self, projectset):
162
 
        return "/api/subjects/" + self.context.subject.short_name + "/" +\
163
 
                self.context.semester.year + "/" + \
164
 
                self.context.semester.semester + "/+projectsets/" +\
165
 
                str(projectset.id) + "/+projects/+new"
166
 
    
167
 
    def populate(self, req, ctx):
168
 
        self.plugin_styles[Plugin] = ["project.css"]
169
 
        self.plugin_scripts[Plugin] = ["project.js"]
170
 
        ctx['offering'] = self.context
171
 
        ctx['projectsets'] = []
172
 
 
173
 
        #Open the projectset Fragment, and render it for inclusion
174
 
        #into the ProjectSets page
175
 
        #XXX: This could be a lot cleaner
176
 
        loader = genshi.template.TemplateLoader(".", auto_reload=True)
177
 
 
178
 
        set_fragment = os.path.join(os.path.dirname(__file__),
179
 
                "templates/projectset_fragment.html")
180
 
        project_fragment = os.path.join(os.path.dirname(__file__),
181
 
                "templates/project_fragment.html")
182
 
 
183
 
        for projectset in self.context.project_sets:
184
 
            settmpl = loader.load(set_fragment)
185
 
            setCtx = Context()
186
 
            setCtx['projectset'] = projectset
187
 
            setCtx['new_project_url'] = self.new_project_url(projectset)
188
 
            setCtx['projects'] = []
189
 
 
190
 
            for project in projectset.projects:
191
 
                projecttmpl = loader.load(project_fragment)
192
 
                projectCtx = Context()
193
 
                projectCtx['project'] = project
194
 
                projectCtx['project_url'] = self.project_url(projectset, project)
195
 
 
196
 
                setCtx['projects'].append(
197
 
                        projecttmpl.generate(projectCtx))
198
 
 
199
 
            ctx['projectsets'].append(settmpl.generate(setCtx))
200
 
 
201
 
 
202
 
class ProjectView(XHTMLView):
203
 
    """View the submissions for a ProjectSet"""
204
 
    template = "templates/project.html"
205
 
    permission = "edit"
206
 
    tab = 'subjects'
207
 
 
208
 
    def __init__(self, req, subject, year, semester, project):
209
 
        self.context = req.store.find(Project,
210
 
                Project.short_name == project,
211
 
                Project.project_set_id == ProjectSet.id,
212
 
                ProjectSet.offering_id == Offering.id,
213
 
                Offering.semester_id == Semester.id,
214
 
                Semester.year == year,
215
 
                Semester.semester == semester,
216
 
                Offering.subject_id == Subject.id,
217
 
                Subject.short_name == subject).one()
218
 
        if self.context is None:
219
 
            raise NotFound()
220
 
 
221
 
    def populate(self, req, ctx):
222
 
        ctx['format_datetime_short'] = ivle.date.format_datetime_for_paragraph
223
 
 
224
 
        ctx['project'] = self.context
225
 
        ctx['assesseds'] = self.context.assesseds
226
 
 
227
 
        ctx['submissions'] = []
228
 
        for assessed in self.context.assesseds:
229
 
            if assessed.submissions.count() > 0:
230
 
                ctx['submissions'].append(
231
 
                        assessed.submissions.order_by(
232
 
                            ProjectSubmission.date_submitted)[-1])
233
 
        ctx['assigned'] = self.context.project_set.get_assigned()
 
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
        for subject in enrolled_subjects:
 
54
            new_subj = {}
 
55
            new_subj['name'] = subject.name
 
56
            new_subj['url'] = subject.url
 
57
            ctx['enrolled_subjects'].append(new_subj)
 
58
 
 
59
        if len(unenrolled_subjects) > 0:
 
60
            for subject in unenrolled_subjects:
 
61
                new_subj = {}
 
62
                new_subj['name'] = subject.name
 
63
                new_subj['url'] = subject.url
 
64
                ctx['other_subjects'].append(new_subj)
 
65
 
234
66
 
235
67
class Plugin(ViewPlugin, MediaPlugin):
236
68
    urls = [
237
69
        ('subjects/', SubjectsView),
238
 
        ('subjects/:subject/:year/:semester/+enrolments/+new', EnrolView),
239
 
        ('subjects/:subject/:year/:semester/+projects', OfferingProjectsView),
240
 
        ('subjects/:subject/:year/:semester/+projects/:project', ProjectView),
241
 
        #API Views
242
 
        ('api/subjects/:subject/:year/:semester/+projectsets/+new',
243
 
            OfferingRESTView),
244
 
        ('api/subjects/:subject/:year/:semester/+projectsets/:projectset/+projects/+new',
245
 
            ProjectSetRESTView),
246
 
        ('api/subjects/:subject/:year/:semester/+projects/:project', 
247
 
            ProjectRESTView),
248
 
 
249
70
    ]
250
71
 
251
72
    tabs = [
252
 
        ('subjects', 'Subjects',
253
 
         'View subject content and complete worksheets',
254
 
         'subjects.png', 'subjects', 5)
 
73
        ('subjects', 'Subjects', 'Announcements and information about the '
 
74
         'subjects you are enrolled in.', 'subjects.png', 'subjects', 5)
255
75
    ]
256
76
 
257
77
    media = 'subject-media'