return obj or dflt
-def instructs(user, course):
+def instructs(user, site):
try:
- mbr = models.Member.objects.get(user=user, course=course)
+ mbr = models.Member.objects.get(user=user, site=site)
return mbr.role in ('INSTR', 'PROXY')
except:
return False
// code is a SELECT, so we add a callback to lookup titles.
$('#id_code').change(function() {
$('#id_title')[0].disabled=true;
- $.getJSON(ROOT + '/course/new/ajax_title', {course_code: $(this).val()},
+ $.getJSON(ROOT + '/site/new/ajax_title', {course_code: $(this).val()},
function(resp) {
$('#id_title').val(resp.title)
$('#id_title')[0].disabled=false;
span.final_item { font-weight: bold; font-size: 110%; }
-/* item trees (tree of headings and items in a course */
+/* item trees (tree of headings and items in a site */
#sidepanel { width: 183px; float: right; text-align: right;}
#sidepanel div { margin: 6px 0; }
.menublockopener { margin-left: 0.25em; color: #bbb !important; font-weight: normal !important; }
.menublock { background-color: #f2e4cc; font-size: 95%; padding: 1px 4px; }
-#coursebanner { background-color: #f2e4cc; margin: -12px -12px 12px -12px; padding: 8px; }
-#coursesearch { float: right; }
-#coursebanner h1 { margin: 12px 0; font-size: 125%; }
+#sitebanner { background-color: #f2e4cc; margin: -12px -12px 12px -12px; padding: 8px; }
+#sitesearch { float: right; }
+#sitebanner h1 { margin: 12px 0; font-size: 125%; }
-#edit_course_link { margin: 8px 0 8px 0; font-size: 95%; }
+#edit_site_link { margin: 8px 0 8px 0; font-size: 95%; }
#breadcrumbs { margin: 8px 0px 16px 0; width: 716px;
// fixme, I need to rename menublocks.js to something more like
-// 'course-item-stuff.js'.
+// 'site-item-stuff.js'.
// this is some item resequencing code.
# General methods that return sets of users (e.g. all instructors, all
# staff) -- let's put those on the UserProfile class, as class
-# methods. For now, let's put personal methods (e.g. my courses) on
+# methods. For now, let's put personal methods (e.g. my sites) on
# the User object (although UserProfile would be another logical
# candidate).
def item_tree(self, subtree=None):
"""
- Return a list, representing a tree of the course items, in
+ Return a list, representing a tree of the site items, in
display order. Every element of the list is an (Item, [Item])
tuple, where the second element is a list of sub-elements (if
a heading) or None (if an item).
walk(subtree, out)
return out
- def course_url(self, suffix=''):
+ def site_url(self, suffix=''):
# I'm not fond of this being here. I think I'll leave this and
# item_url non-implemented, and monkey-patch them in views.py.
req = genshi_locals.get_request()
prefix = req.META['SCRIPT_NAME']
- return '%s/course/%d/%s' % (prefix, self.id, suffix)
+ return '%s/site/%d/%s' % (prefix, self.id, suffix)
def generate_new_passkey(self):
# todo: have a pluggable passkey algorithm.
def is_joinable_by(self, user):
"""Return True if the user could feasibly register into this
- course: she's not already in it, and the course allows open
+ site: she's not already in it, and the site allows open
registration."""
return user.is_authenticated() \
and self.access in ('ANON', 'LOGIN') \
def item_url(self, suffix='', force_local_url=False):
# I'm not fond of this being here. I think I'll leave this and
- # course_url non-implemented, and monkey-patch them in views.py.
+ # site_url non-implemented, and monkey-patch them in views.py.
req = genshi_locals.get_request()
prefix = req.META['SCRIPT_NAME']
if self.item_type == 'ELEC' and suffix == '':
- return '%s/course/%d/item/%d/dl/%s' % (
- prefix, self.course_id, self.id,
+ return '%s/site/%d/item/%d/dl/%s' % (
+ prefix, self.site_id, self.id,
self.fileobj.name.split('/')[-1])
if self.item_type == 'URL' and suffix == '' and not force_local_url:
return self.url
else:
- return '%s/course/%d/item/%d/%s' % (
- prefix, self.course_id, self.id, suffix)
+ return '%s/site/%d/item/%d/%s' % (
+ prefix, self.site_id, self.id, suffix)
def parent_url(self, suffix=''):
if self.parent_heading:
return self.parent_heading.item_url()
else:
- return self.course.course_url()
+ return self.site.site_url()
def describe_physical_item_status(self):
"""Return a (bool,str) tuple: whether the item is available,
# I'm not ready to break items out into their own urls.py, but I do
# want to cut down on the common boilerplate in the urlpatterns below.
-ITEM_PREFIX = r'^course/(?P<course_id>\d+)/item/(?P<item_id>\d+)/'
+ITEM_PREFIX = r'^site/(?P<site_id>\d+)/item/(?P<item_id>\d+)/'
GENERIC_REGEX = r'((?P<obj_id>\d+)/)?(?P<action>.+)?$'
urlpatterns = patterns('conifer.syrup.views',
(r'^$', 'welcome'),
- (r'^course/$', 'my_courses'),
- (r'^course/new/$', 'add_new_course'),
- (r'^course/new/ajax_title$', 'add_new_course_ajax_title'),
- (r'^course/invitation/$', 'course_invitation'),
+ (r'^site/$', 'my_sites'),
+ (r'^site/new/$', 'add_new_site'),
+ (r'^site/new/ajax_title$', 'add_new_site_ajax_title'),
+ (r'^site/invitation/$', 'site_invitation'),
(r'^browse/$', 'browse'),
(r'^browse/(?P<browse_option>.*)/$', 'browse'),
(r'^prefs/$', 'user_prefs'),
(r'^z3950test/$', 'z3950_test'),
- #MARK: propose we kill open_courses, we have browse.
- (r'^opencourse/$', 'open_courses'),
+ #MARK: propose we kill open_sites, we have browse.
+ (r'^opensite/$', 'open_sites'),
(r'^search/$', 'search'),
(r'^zsearch/$', 'zsearch'),
#MARK: propose we kill instructors, we have browse
(r'^instructors/search/(?P<instructor>.*)$', 'instructor_search'),
#MARK: propose we kill departments, we have browse
(r'^departments/$', 'departments'),
- (r'^course/(?P<course_id>\d+)/$', 'course_detail'),
+ (r'^site/(?P<site_id>\d+)/$', 'site_detail'),
(r'^instructor/(?P<instructor_id>.*)/$', 'instructor_detail'),
(r'^department/(?P<department_id>.*)/$', 'department_detail'),
- (r'^course/(?P<course_id>\d+)/search/$', 'course_search'),
- (r'^course/(?P<course_id>\d+)/edit/$', 'edit_course'),
- (r'^course/(?P<course_id>\d+)/edit/delete/$', 'delete_course'),
- (r'^course/(?P<course_id>\d+)/edit/permission/$', 'edit_course_permissions'),
- (r'^course/(?P<course_id>\d+)/feeds/(?P<feed_type>.*)$', 'course_feeds'),
- (r'^course/(?P<course_id>\d+)/join/$', 'course_join'),
+ (r'^site/(?P<site_id>\d+)/search/$', 'site_search'),
+ (r'^site/(?P<site_id>\d+)/edit/$', 'edit_site'),
+ (r'^site/(?P<site_id>\d+)/edit/delete/$', 'delete_site'),
+ (r'^site/(?P<site_id>\d+)/edit/permission/$', 'edit_site_permissions'),
+ (r'^site/(?P<site_id>\d+)/feeds/(?P<feed_type>.*)$', 'site_feeds'),
+ (r'^site/(?P<site_id>\d+)/join/$', 'site_join'),
(ITEM_PREFIX + r'$', 'item_detail'),
(ITEM_PREFIX + r'dl/(?P<filename>.*)$', 'item_download'),
(ITEM_PREFIX + r'meta$', 'item_metadata'),
(r'^phys/mark_arrived/match/$', 'phys_mark_arrived_match'),
(r'^phys/circlist/$', 'phys_circlist'),
- (r'^course/(?P<course_id>\d+)/reseq$', 'course_reseq'),
+ (r'^site/(?P<site_id>\d+)/reseq$', 'site_reseq'),
(ITEM_PREFIX + r'reseq', 'item_heading_reseq'),
(ITEM_PREFIX + r'relocate/', 'item_relocate'), # move to new subheading
# (r'^admin/terms/(?P<term_id>\d+)/$', 'admin_term_edit'),
from conifer.integration import hooks
from general import *
-from courses import *
+from sites import *
from items import *
from search import *
from admin import *
#-----------------------------------------------------------------------------
# Authorization
-def _fast_user_membership_query(user_id, course_id, where=None):
+# TODO: this _fast_user_membership_query is broken.
+
+def _fast_user_membership_query(user_id, site_id, where=None):
# I use a raw SQL query here because I want the lookup to be as
# fast as possible. Caching would help too, but let's try this
# first. (todo, review later.)
query = ('select count(*) from syrup_member '
- 'where user_id=%s and course_id=%s ')
+ 'where user_id=%s and site_id=%s ')
if where:
query += (' and ' + where)
cursor = django.db.connection.cursor()
- cursor.execute(query, [user_id, int(course_id)])
+ cursor.execute(query, [user_id, int(site_id)])
res = cursor.fetchall()
cursor.close()
allowed = bool(res[0][0])
# decorator
def instructors_only(handler):
- def hdlr(request, course_id, *args, **kwargs):
+ def hdlr(request, site_id, *args, **kwargs):
allowed = request.user.is_superuser
if not allowed:
allowed = _fast_user_membership_query(
- request.user.id, course_id, "role in ('INSTR','PROXY')")
+ request.user.id, site_id, "role in ('INSTR','ASSIST')")
if allowed:
- return handler(request, course_id, *args, **kwargs)
+ return handler(request, site_id, *args, **kwargs)
else:
return _access_denied(request,
_('Only instructors are allowed here.'))
# decorator
def members_only(handler):
- def hdlr(request, course_id, *args, **kwargs):
+ def hdlr(request, site_id, *args, **kwargs):
user = request.user
allowed = user.is_superuser
if not allowed:
- course = models.ReadingList.objects.get(pk=course_id)
- allowed = course.access=='ANON' or \
- (user.is_authenticated() and course.access=='LOGIN')
+ site = models.Site.objects.get(pk=site_id)
+ allowed = site.access=='ANON' or \
+ (user.is_authenticated() and site.access=='LOGIN')
if not allowed:
- allowed = _fast_user_membership_query(user.id, course_id)
+ allowed = _fast_user_membership_query(user.id, site_id)
if allowed:
- return handler(request, course_id, *args, **kwargs)
+ return handler(request, site_id, *args, **kwargs)
else:
- if course.access=='LOGIN':
+ if site.access=='LOGIN':
msg = _('Please log in, so that you can enter this site.')
else:
- msg = _('Only course members are allowed here.')
+ msg = _('Only site members are allowed here.')
return _access_denied(request, msg)
return hdlr
#-----------------------------------------------------------
def user_filters(user):
- """Returns a dict of filters for Item, ReadingList, etc. querysets,
+ """Returns a dict of filters for Item, Site, etc. querysets,
based on the given user's permissions."""
# TODO, figure out a way of EXPLAIN'ing these queries! I have no
# idea of their complexity.
if user.is_anonymous():
- # then only anonymous-access courses are available.
- filters = {'items': Q(course__access='ANON'),
- 'courses': Q(access='ANON'),
+ # then only anonymous-access sites are available.
+ filters = {'items': Q(site__access='ANON'),
+ 'sites': Q(access='ANON'),
'instructors': Q(), # TODO: do we really need a filter here?
}
else:
- # logged-in users have access to courses which are of the
+ # logged-in users have access to sites which are of the
# LOGIN class ('all logged-in users') or in which they
# have explicit Member-ship.
filters = {
- 'items': (Q(course__access__in=('LOGIN','ANON')) \
- | Q(course__member__user=user)),
- 'courses': (Q(access__in=('LOGIN','ANON')) | Q(member__user=user)),
+ 'items': (Q(site__access__in=('LOGIN','ANON')) \
+ | Q(site__member__user=user)),
+ 'sites': (Q(access__in=('LOGIN','ANON')) | Q(member__user=user)),
'instructors': Q(), # TODO: do we really need a filter here?
}
return filters
from django.utils.translation import ugettext as _
#-----------------------------------------------------------------------------
-# Course feeds
+# Site feeds
@public # and proud of it!
-def course_feeds(request, course_id, feed_type):
- course = get_object_or_404(models.Course, pk=course_id)
+def site_feeds(request, site_id, feed_type):
+ site = get_object_or_404(models.Site, pk=site_id)
if feed_type == '':
- return g.render('feeds/course_feed_index.xhtml',
- course=course)
+ return g.render('feeds/site_feed_index.xhtml',
+ site=site)
else:
- items = course.items()
+ items = site.items()
def render_title(item):
return item.title
if feed_type == 'top-level':
acc.append(item)
flatten(kids, acc)
return acc
- items = flatten(course.item_tree(), [])
+ items = flatten(site.item_tree(), [])
def render_title(item):
if item.parent_heading:
return '%s :: %s' % (item.parent_heading.title, item.title)
else:
return item.title
lastmod = items and max(i.last_modified for i in items) or datetime.now()
- resp = g.render('feeds/course_atom.xml',
- course=course,
+ resp = g.render('feeds/site_atom.xml',
+ site=site,
feed_type=feed_type,
lastmod=lastmod,
render_title=render_title,
def welcome(request):
return HttpResponseRedirect('browse/')
-# MARK: propose we get rid of this. We already have a 'Courses' browser.
-def open_courses(request):
+# MARK: propose we get rid of this. We already have a 'Sites' browser.
+def open_sites(request):
page_num = int(request.GET.get('page', 1))
count = int(request.GET.get('count', 5))
- paginator = Paginator(models.ReadingList.objects.all(), count) # fixme, what filter?
- return g.render('open_courses.xhtml', paginator=paginator,
+ paginator = Paginator(models.Site.objects.all(), count) # fixme, what filter?
+ return g.render('open_sites.xhtml', paginator=paginator,
page_num=page_num,
count=count)
# MARK: propose we drop this too. We have a browse.
if action == 'join':
paginator = Paginator(models.User.active_instructors(), count)
elif action == 'drop':
- paginator = Paginator(models.ReadingList.objects.all(), count) # fixme, what filter?
+ paginator = Paginator(models.Site.objects.all(), count) # fixme, what filter?
else:
- paginator = Paginator(models.ReadingList.objects.all(), count) # fixme, what filter?
+ paginator = Paginator(models.Site.objects.all(), count) # fixme, what filter?
return g.render('instructors.xhtml', paginator=paginator,
page_num=page_num,
elif browse_option == 'courses':
# fixme, course filter should not be (active=True) but based on user identity.
for_courses = user_filters(request.user)['courses']
- queryset = models.ReadingList.objects.filter(for_courses)
+ queryset = models.Site.objects.filter(for_courses)
template = 'courses.xhtml'
queryset = queryset and queryset.distinct()
count=count)
@login_required
-def my_courses(request):
- return g.render('my_courses.xhtml')
+def my_sites(request):
+ return g.render('my_sites.xhtml')
def instructor_detail(request, instructor_id):
page_num = int(request.GET.get('page', 1))
count = int(request.GET.get('count', 5))
'''
i am not sure this is the best way to go from instructor
- to course
+ to site
'''
- courses = models.ReadingList.objects.filter(member__user=instructor_id,
+ sites = models.Site.objects.filter(member__user=instructor_id,
member__role='INSTR')
filters = user_filters(request.user)
courses = courses.filter(filters['courses'])
page_num = int(request.GET.get('page', 1))
count = int(request.GET.get('count', 5))
- paginator = Paginator(models.ReadingList.objects.
+ paginator = Paginator(models.Site.objects.
filter(department__id=department_id).
order_by('title'), count)
from django.utils.translation import ugettext as _
@members_only
-def item_detail(request, course_id, item_id):
+def item_detail(request, site_id, item_id):
"""Display an item (however that makes sense)."""
# really, displaying an item will vary based on what type of item
# it is -- e.g. a URL item would redirect to the target URL. I'd
# like this URL to be the generic dispatcher, but for now let's
# just display some metadata about the item.
- item = get_object_or_404(models.Item, pk=item_id, course__id=course_id)
+ item = get_object_or_404(models.Item, pk=item_id, site__id=site_id)
if item.url:
return _heading_url(request, item)
else:
- return item_metadata(request, course_id, item_id)
+ return item_metadata(request, site_id, item_id)
@members_only
-def item_metadata(request, course_id, item_id):
+def item_metadata(request, site_id, item_id):
"""Display a metadata page for the item."""
- item = get_object_or_404(models.Item, pk=item_id, course__id=course_id)
+ item = get_object_or_404(models.Item, pk=item_id, site__id=site_id)
if item.item_type == 'HEADING':
return _heading_detail(request, item)
else:
- return g.render('item/item_metadata.xhtml', course=item.course,
+ return g.render('item/item_metadata.xhtml', site=item.site,
item=item)
def _heading_url(request, item):
@instructors_only
-def item_add(request, course_id, item_id):
+def item_add(request, site_id, item_id):
# The parent_item_id is the id for the parent-heading item. Zero
# represents 'top-level', i.e. the new item should have no
# heading.
parent_item_id = item_id
if parent_item_id=='0':
parent_item = None
- course = get_object_or_404(models.Course, pk=course_id)
- siblings = course.item_set.filter(parent_heading=None)
+ site = get_object_or_404(models.Site, pk=site_id)
+ siblings = site.item_set.filter(parent_heading=None)
else:
- parent_item = get_object_or_404(models.Item, pk=parent_item_id, course__id=course_id)
+ parent_item = get_object_or_404(models.Item, pk=parent_item_id, site__id=site_id)
assert parent_item.item_type == 'HEADING', _('You can only add items to headings!')
- course = parent_item.course
- siblings = course.item_set.filter(parent_heading=parent_item)
+ site = parent_item.site
+ siblings = site.item_set.filter(parent_heading=parent_item)
try:
next_order = 1 + max(i.sort_order for i in siblings)
except:
next_order = 0
- if not course.can_edit(request.user):
+ if not site.can_edit(request.user):
return _access_denied(_('You are not an editor.'))
item_type = request.GET.get('item_type')
return HttpResponseRedirect(request.get_full_path())
else:
item = models.Item(
- course=course,
+ site=site,
item_type='HEADING',
sort_order = next_order,
parent_heading=parent_item,
return HttpResponseRedirect(request.get_full_path())
else:
item = models.Item(
- course=course,
+ site=site,
item_type='URL',
parent_heading=parent_item,
sort_order = next_order,
# fixme, better error handling.
return HttpResponseRedirect(request.get_full_path())
item = models.Item(
- course=course,
+ site=site,
item_type='ELEC',
parent_heading=parent_item,
sort_order = next_order,
if parent_item:
return HttpResponseRedirect(parent_item.item_url('meta'))
else:
- return HttpResponseRedirect(course.course_url())
+ return HttpResponseRedirect(site.site_url())
@instructors_only
-def item_add_cat_search(request, course_id, item_id):
+def item_add_cat_search(request, site_id, item_id):
# this chunk stolen from item_add(). Refactor.
parent_item_id = item_id
if parent_item_id=='0':
parent_item = None
- course = get_object_or_404(models.Course, pk=course_id)
- siblings = course.item_set.filter(parent_heading=None)
+ site = get_object_or_404(models.Site, pk=site_id)
+ siblings = site.item_set.filter(parent_heading=None)
else:
- parent_item = get_object_or_404(models.Item, pk=parent_item_id, course__id=course_id)
+ parent_item = get_object_or_404(models.Item, pk=parent_item_id, site__id=site_id)
assert parent_item.item_type == 'HEADING', _('You can only add items to headings!')
- course = parent_item.course
- siblings = course.item_set.filter(parent_heading=parent_item)
+ site = parent_item.site
+ siblings = site.item_set.filter(parent_heading=parent_item)
try:
next_order = 1 + max(i.sort_order for i in siblings)
if request.method != 'POST':
if not 'query' in request.GET:
return g.render('item/item_add_cat_search.xhtml', results=[], query='',
- course=course, parent_item=parent_item)
+ site=site, parent_item=parent_item)
query = request.GET.get('query','').strip()
start, limit = (int(request.GET.get(k,v)) for k,v in (('start',1),('limit',10)))
results, numhits = lib_integration.cat_search(query, start, limit)
return g.render('item/item_add_cat_search.xhtml',
results=results, query=query,
start=start, limit=limit, numhits=numhits,
- course=course, parent_item=parent_item)
+ site=site, parent_item=parent_item)
else:
- # User has selected an item; add it to course site.
+ # User has selected an item; add it to site.
raw_pickitem = request.POST.get('pickitem', '').strip()
#fixme, this block copied from item_add. refactor.
parent_item_id = item_id
if parent_item_id == '0':
# no heading (toplevel)
parent_item = None
- course = get_object_or_404(models.Course, pk=course_id)
+ site = get_object_or_404(models.Site, pk=site_id)
else:
- parent_item = get_object_or_404(models.Item, pk=parent_item_id, course__id=course_id)
+ parent_item = get_object_or_404(models.Item, pk=parent_item_id, site__id=site_id)
assert parent_item.item_type == 'HEADING', _('You can only add items to headings!')
- course = parent_item.course
- if not course.can_edit(request.user):
+ site = parent_item.site
+ if not site.can_edit(request.user):
return _access_denied(_('You are not an editor.'))
pickitem = simplejson.loads(raw_pickitem)
else:
dct = dict(item_type='PHYS')
- item = course.item_set.create(parent_heading=parent_item,
+ item = site.item_set.create(parent_heading=parent_item,
sort_order=next_order,
title=dublin.get('dc:title','Untitled'),
**dct)
#------------------------------------------------------------
@instructors_only
-def item_edit(request, course_id, item_id):
- course = get_object_or_404(models.Course, pk=course_id)
- item = get_object_or_404(models.Item, pk=item_id, course__id=course_id)
+def item_edit(request, site_id, item_id):
+ site = get_object_or_404(models.Site, pk=site_id)
+ item = get_object_or_404(models.Item, pk=item_id, site__id=site_id)
item_type = item.item_type
template = 'item/item_add_%s.xhtml' % item_type.lower()
parent_item = item.parent_heading
return HttpResponseRedirect(item.parent_url())
@instructors_only
-def item_delete(request, course_id, item_id):
- course = get_object_or_404(models.Course, pk=course_id)
- item = get_object_or_404(models.Item, pk=item_id, course__id=course_id)
+def item_delete(request, site_id, item_id):
+ site = get_object_or_404(models.Site, pk=site_id)
+ item = get_object_or_404(models.Item, pk=item_id, site__id=site_id)
if request.method != 'POST':
return g.render('item/item_delete_confirm.xhtml', **locals())
else:
if item.parent_heading:
redir = HttpResponseRedirect(item.parent_heading.item_url('meta'))
else:
- redir = HttpResponseRedirect(course.course_url())
+ redir = HttpResponseRedirect(site.site_url())
item.delete()
return redir
else:
return HttpResponseRedirect('../meta')
@members_only
-def item_download(request, course_id, item_id, filename):
- course = get_object_or_404(models.Course, pk=course_id)
- item = get_object_or_404(models.Item, pk=item_id, course__id=course_id)
+def item_download(request, site_id, item_id, filename):
+ site = get_object_or_404(models.Site, pk=site_id)
+ item = get_object_or_404(models.Item, pk=item_id, site__id=site_id)
assert item.item_type == 'ELEC', _('Can only download ELEC documents!')
fileiter = item.fileobj.chunks()
resp = HttpResponse(fileiter)
#------------------------------------------------------------
# resequencing items
-def _reseq(request, course, parent_heading):
+def _reseq(request, site, parent_heading):
new_order = request.POST['new_order'].strip().split(' ')
# new_order is now a list like this: ['item_3', 'item_8', 'item_1', ...].
# get at the ints.
new_order = [int(n.split('_')[1]) for n in new_order]
print >> sys.stderr, new_order
- the_items = list(course.item_set.filter(parent_heading=parent_heading).order_by('sort_order'))
+ the_items = list(site.item_set.filter(parent_heading=parent_heading).order_by('sort_order'))
# sort the items by position in new_order
the_items.sort(key=lambda item: new_order.index(item.id))
for newnum, item in enumerate(the_items):
return HttpResponse("'ok'");
@instructors_only
-def course_reseq(request, course_id):
- course = get_object_or_404(models.Course, pk=course_id)
+def site_reseq(request, site_id):
+ site = get_object_or_404(models.Site, pk=site_id)
parent_heading = None
- return _reseq(request, course, parent_heading)
+ return _reseq(request, site, parent_heading)
@instructors_only
-def item_heading_reseq(request, course_id, item_id):
- course = get_object_or_404(models.Course, pk=course_id)
- item = get_object_or_404(models.Item, pk=item_id, course__id=course_id)
+def item_heading_reseq(request, site_id, item_id):
+ site = get_object_or_404(models.Site, pk=site_id)
+ item = get_object_or_404(models.Item, pk=item_id, site__id=site_id)
parent_heading = item
- return _reseq(request, course, parent_heading)
+ return _reseq(request, site, parent_heading)
@instructors_only
-def item_relocate(request, course_id, item_id):
+def item_relocate(request, site_id, item_id):
"""Move an item from its current subheading to another one."""
- course = get_object_or_404(models.Course, pk=course_id)
- item = get_object_or_404(models.Item, pk=item_id, course__id=course_id)
+ site = get_object_or_404(models.Site, pk=site_id)
+ item = get_object_or_404(models.Item, pk=item_id, site__id=site_id)
if request.method != 'POST':
return g.render('item/item_relocate.xhtml', **locals())
else:
if newheading == 0:
new_parent = None
else:
- new_parent = course.item_set.get(pk=newheading)
+ new_parent = site.item_set.get(pk=newheading)
if item in new_parent.hierarchy():
# then we would create a cycle. Bail out.
return simple_message(_('Impossible item-move!'),
if new_parent:
return HttpResponseRedirect(new_parent.item_url('meta'))
else:
- return HttpResponseRedirect(course.course_url())
+ return HttpResponseRedirect(site.site_url())
cursor.close()
wanted = models.Item.objects.filter(
- item_type='PHYS', course__term=term).select_related('metadata')
+ item_type='PHYS', site__term=term).select_related('metadata')
wanted = [w for w in wanted if w.id not in bad_ids]
return g.render('phys/circlist_for_term.xhtml',
term=term,
#-----------------------------------------------------------------------------
# Search and search support
-def search(request, in_course=None, with_instructor=None):
+def search(request, in_site=None, with_instructor=None):
''' Need to work on this, the basic idea is
- - put an entry point for instructor and course listings
+ - put an entry point for instructor and site listings
- page through item entries
- If in_course is provided, then limit search to the contents of the specified course.
+ If in_site is provided, then limit search to the contents of the specified site.
If with_instructor is provided, then limit search to instructors
'''
- print("in_couse is %s" % in_course)
+ print("in_couse is %s" % in_site)
print("with_instructor is %s" % with_instructor)
found_entries = None
page_num = int(request.GET.get('page', 1))
# Next, we filter based on user permissions.
flt = user_filters(request.user)
- user_filter_for_items, user_filter_for_courses = flt['items'], flt['courses']
+ user_filter_for_items, user_filter_for_sites = flt['items'], flt['sites']
# Note, we haven't user-filtered anything yet; we've just set
# up the filters.
#need to think about sort order here, probably better by author (will make sortable at display level)
results_list = models.Item.objects.filter(item_query)
- if in_course:
- # For an in-course search, we know the user has
- # permissions to view the course; no need for
+ if in_site:
+ # For an in-site search, we know the user has
+ # permissions to view the site; no need for
# user_filter_for_items.
- results_list = results_list.filter(course=in_course)
+ results_list = results_list.filter(site=in_site)
elif with_instructor:
print("in instructor")
results_list = results_list.filter(instructor=with_instructor)
results_len = len(results_list)
paginator = Paginator(results_list, count)
- #course search
- if in_course:
- # then no course search is necessary.
- course_list = []; course_len = 0
+ #site search
+ if in_site:
+ # then no site search is necessary.
+ site_list = []; site_len = 0
else:
- course_query = get_query(query_string, ['title', 'department__name'])
+ site_query = get_query(query_string, ['title', 'department__name'])
# apply the search-filter and the user-filter
- course_results = models.ReadingList.objects.filter(course_query).filter(user_filter_for_courses)
- course_list = course_results.order_by('title')
- course_len = len(course_results)
+ site_results = models.Site.objects.filter(site_query).filter(user_filter_for_sites)
+ site_list = site_results.order_by('title')
+ site_len = len(site_results)
#instructor search
- if in_course:
+ if in_site:
instructor_list = []; instr_len = 0
else:
instr_query = get_query(query_string, ['user__last_name'])
instructor_results = models.Member.objects.filter(instr_query).filter(role='INSTR')
- if in_course:
- instructor_results = instructor_results.filter(course=in_course)
+ if in_site:
+ instructor_results = instructor_results.filter(site=in_site)
instructor_list = instructor_results.order_by('user__last_name')[0:5]
instr_len = len(instructor_results)
- elif in_course:
- # we are in a course, but have no query? Return to the course-home page.
+ elif in_site:
+ # we are in a site, but have no query? Return to the site-home page.
return HttpResponseRedirect('../')
else:
results_list = models.Item.objects.order_by('title')
results_len = len(results_list)
paginator = Paginator( results_list,
count)
- course_results = models.ReadingList.objects.filter(active=True)
- course_list = course_results.order_by('title')[0:5]
- course_len = len(course_results)
+ site_results = models.Site.objects.filter(active=True)
+ site_list = site_results.order_by('title')[0:5]
+ site_len = len(site_results)
instructor_results = models.Member.objects.filter(role='INSTR')
instructor_list = instructor_results.order_by('user__last_name')[0:5]
instr_len = len(instructor_results)
from search import *
#-----------------------------------------------------------------------------
-# Creating a new course
+# Creating a new site
-class NewCourseForm(ModelForm):
+class NewSiteForm(ModelForm):
class Meta:
- model = models.Course
+ model = models.Site
exclude = ('passkey','access')
def clean_code(self):
choices.sort()
empty_label = u'---------'
choices.insert(0, ('', empty_label))
- NewCourseForm.base_fields['code'].widget = Select(
+ NewSiteForm.base_fields['code'].widget = Select(
choices = choices)
- NewCourseForm.base_fields['code'].empty_label = empty_label
+ NewSiteForm.base_fields['code'].empty_label = empty_label
#--------------------
@login_required
-def add_new_course(request):
- if not request.user.can_create_reading_lists():
- return _access_denied(_('You are not allowed to create course sites.'))
- return _add_or_edit_course(request)
+def add_new_site(request):
+ if not request.user.can_create_sites():
+ return _access_denied(_('You are not allowed to create sites.'))
+ return _add_or_edit_site(request)
@instructors_only
-def edit_course(request, course_id):
- instance = get_object_or_404(models.Course, pk=course_id)
- return _add_or_edit_course(request, instance=instance)
+def edit_site(request, site_id):
+ instance = get_object_or_404(models.Site, pk=site_id)
+ return _add_or_edit_site(request, instance=instance)
-def _add_or_edit_course(request, instance=None):
+def _add_or_edit_site(request, instance=None):
is_add = (instance is None)
if is_add:
- instance = models.Course()
+ instance = models.Site()
current_access_level = not is_add and instance.access or None
example = models.campus.course_code_example
if request.method != 'POST':
- form = NewCourseForm(instance=instance)
- return g.render('edit_course.xhtml', **locals())
+ form = NewSiteForm(instance=instance)
+ return g.render('edit_site.xhtml', **locals())
else:
- form = NewCourseForm(request.POST, instance=instance)
+ form = NewSiteForm(request.POST, instance=instance)
if not form.is_valid():
- return g.render('edit_course.xhtml', **locals())
+ return g.render('edit_site.xhtml', **locals())
else:
form.save()
- course = form.instance
- if course.access == u'INVIT' and not course.passkey:
- course.generate_new_passkey()
- course.save()
- assert course.id
- user_in_course = models.Member.objects.filter(user=request.user,course=course)
- if not user_in_course: # for edits, might already be!
- mbr = course.member_set.create(user=request.user, role='INSTR')
+ site = form.instance
+ if site.access == u'INVIT' and not site.passkey:
+ site.generate_new_passkey()
+ site.save()
+ assert site.id
+ user_in_site = models.Member.objects.filter(user=request.user,site=site)
+ if not user_in_site: # for edits, might already be!
+ mbr = site.member_set.create(user=request.user, role='INSTR')
mbr.save()
- if is_add or (current_access_level != course.access):
+ if is_add or (current_access_level != site.access):
# we need to configure permissions.
- return HttpResponseRedirect(course.course_url('edit/permission/'))
+ return HttpResponseRedirect(site.site_url('edit/permission/'))
else:
return HttpResponseRedirect('../') # back to main view.
# no access-control needed to protect title lookup.
-def add_new_course_ajax_title(request):
+def add_new_site_ajax_title(request):
course_code = request.GET['course_code']
title = models.campus.course_code_lookup_title(course_code)
return HttpResponse(simplejson.dumps({'title':title}))
@instructors_only
-def edit_course_permissions(request, course_id):
- course = get_object_or_404(models.Course, pk=course_id)
+def edit_site_permissions(request, site_id):
+ site = get_object_or_404(models.Site, pk=site_id)
# choices: make the access-choice labels more personalized than
# the ones in 'models'.
choices = [
choose_access = django.forms.Select(choices=choices)
if request.method != 'POST':
- return g.render('edit_course_permissions.xhtml', **locals())
+ return g.render('edit_site_permissions.xhtml', **locals())
else:
POST = request.POST
if 'action_change_code' in POST:
# update invitation code -------------------------------------
- course.generate_new_passkey()
- course.access = u'INVIT'
- course.save()
+ site.generate_new_passkey()
+ site.access = u'INVIT'
+ site.save()
return HttpResponseRedirect('.#student_access')
elif 'action_save_instructor' in POST:
instr = models.maybe_initialize_user(iname)
if instr:
try:
- return models.Member.objects.get(user=instr, course=course)
+ return models.Member.objects.get(user=instr, site=site)
except models.Member.DoesNotExist:
- return models.Member.objects.create(user=instr, course=course)
+ return models.Member.objects.create(user=instr, site=site)
# add a new instructor
if iname:
instr.save()
else:
instructor_error = 'No such user: %s' % iname
- return g.render('edit_course_permissions.xhtml', **locals())
+ return g.render('edit_site_permissions.xhtml', **locals())
# removing and changing roles of instructors
for name in POST if name.startswith('instructor_remove_')]
for instr_id, newrole in to_change_role:
if not instr_id in to_remove:
- instr = models.Member.objects.get(pk=instr_id, course=course)
+ instr = models.Member.objects.get(pk=instr_id, site=site)
instr.role = newrole
instr.save()
for instr_id in to_remove:
# todo, should warn if deleting yourself!
- instr = models.Member.objects.get(pk=instr_id, course=course)
+ instr = models.Member.objects.get(pk=instr_id, site=site)
instr.delete()
# todo, should have some error-reporting.
return HttpResponseRedirect('.')
elif 'action_save_student' in POST:
# update student details ------------------------------------
access = POST.get('access')
- course.access = access
+ site.access = access
# drop all provided users. fixme, this could be optimized to do add/drops.
- models.Member.objects.filter(course=course, provided=True).delete()
- if course.access == u'STUDT':
- initial_sections = course.sections()
+ models.Member.objects.filter(site=site, provided=True).delete()
+ if site.access == u'STUDT':
+ initial_sections = site.sections()
# add the 'new section' if any
new_sec = request.POST.get('add_section')
new_sec = models.section_decode_safe(new_sec)
if new_sec:
- course.add_sections(new_sec)
+ site.add_sections(new_sec)
# remove the sections to be dropped
to_remove = [models.section_decode_safe(name.rsplit('_',1)[1]) \
for name in POST \
if name.startswith('remove_section_')]
- course.drop_sections(*to_remove)
- student_names = models.campus.students_in(*course.sections())
+ site.drop_sections(*to_remove)
+ student_names = models.campus.students_in(*site.sections())
for name in student_names:
user = models.maybe_initialize_user(name)
if user:
- if not models.Member.objects.filter(course=course, user=user):
+ if not models.Member.objects.filter(site=site, user=user):
mbr = models.Member.objects.create(
- course=course, user=user,
+ site=site, user=user,
role='STUDT', provided=True)
mbr.save()
else:
pass
- course.save()
+ site.save()
return HttpResponseRedirect('.#student_access')
@instructors_only
-def delete_course(request, course_id):
- course = get_object_or_404(models.Course, pk=course_id)
+def delete_site(request, site_id):
+ site = get_object_or_404(models.Site, pk=site_id)
if request.POST.get('confirm_delete'):
- course.delete()
- return HttpResponseRedirect(reverse('my_courses'))
+ site.delete()
+ return HttpResponseRedirect(reverse('my_sites'))
else:
return HttpResponseRedirect('../')
#-----------------------------------------------------------------------------
-# Course Invitation Code handler
+# Site Invitation Code handler
@login_required # must be, to avoid/audit brute force attacks.
-def course_invitation(request):
+def site_invitation(request):
if request.method != 'POST':
- return g.render('course_invitation.xhtml', code='', error='',
+ return g.render('site_invitation.xhtml', code='', error='',
**locals())
else:
code = request.POST.get('code', '').strip()
return HttpResponseRedirect('.')
try:
# note, we only allow the passkey if access='INVIT'.
- crs = models.ReadingList.objects.filter(access='INVIT').get(passkey=code)
- except models.ReadingList.DoesNotExist:
+ crs = models.Site.objects.filter(access='INVIT').get(passkey=code)
+ except models.Site.DoesNotExist:
# todo, do we need a formal logging system? Or a table for
# invitation failures? They should be captured somehow, I
# think. Should we temporarily disable accounts after
log('WARN', 'Invitation failure, user %r gave code %r' % \
(request.user.username, code))
error = _('The code you provided is not valid.')
- return g.render('course_invitation.xhtml', **locals())
+ return g.render('site_invitation.xhtml', **locals())
# the passkey is good; add the user if not already a member.
- if not models.Member.objects.filter(user=request.user, course=crs):
- mbr = models.Member.objects.create(user=request.user, course=crs,
+ if not models.Member.objects.filter(user=request.user, site=crs):
+ mbr = models.Member.objects.create(user=request.user, site=crs,
role='STUDT')
mbr.save()
- return HttpResponseRedirect(crs.course_url())
+ return HttpResponseRedirect(crs.site_url())
#-----------------------------------------------------------------------------
-# ReadingList-instance handlers
+# Site-instance handlers
@members_only
-def course_detail(request, course_id):
- course = get_object_or_404(models.ReadingList, pk=course_id)
- return g.render('course_detail.xhtml', course=course)
+def site_detail(request, site_id):
+ site = get_object_or_404(models.Site, pk=site_id)
+ return g.render('site_detail.xhtml', site=site)
@members_only
-def course_search(request, course_id):
- course = get_object_or_404(models.ReadingList, pk=course_id)
- return search(request, in_course=course)
+def site_search(request, site_id):
+ site = get_object_or_404(models.Site, pk=site_id)
+ return search(request, in_site=site)
@login_required
-def course_join(request, course_id):
- """Self-register into an open-registration course."""
- course = get_object_or_404(models.ReadingList, pk=course_id)
- if not course.is_joinable_by(request.user):
+def site_join(request, site_id):
+ """Self-register into an open-registration site."""
+ site = get_object_or_404(models.Site, pk=site_id)
+ if not site.is_joinable_by(request.user):
# user should never see this.
- return simple_message(_('You cannot join this course.'),
- _('Sorry, but you cannot join this course at this time.'))
+ return simple_message(_('You cannot join this site.'),
+ _('Sorry, but you cannot join this site at this time.'))
elif request.method != 'POST':
- return g.render('course_join.xhtml', course=course)
+ return g.render('site_join.xhtml', site=site)
else:
- mbr = models.Member.objects.create(user=request.user, course=course, role='STUDT')
+ mbr = models.Member.objects.create(user=request.user, site=site, role='STUDT')
mbr.save()
- return HttpResponseRedirect(course.course_url())
+ return HttpResponseRedirect(site.site_url())
</head>
<body>
<h1>${title}</h1>
- (Note: some course materials may require you
+ (Note: some reserve materials may require you
to <a href="${ROOT}/accounts/login/?next=${ROOT}/">log in</a>)
<h2>Choose from one of the options below:</h2>
<div class="itemadd">
<?python
-searchtext = _('search this course...')
+searchtext = _('search this site...')
?>
<html xmlns="http://www.w3.org/1999/xhtml"
xmlns:py="http://genshi.edgewall.org/"
xmlns:xi="http://www.w3.org/2001/XInclude"
py:strip="">
- <div py:def="course_search(course)" id="coursesearch" style="float: right;">
- <form method="get" action="${course.course_url('search')}"
+ <div py:def="site_search(site)" id="sitesearch" style="float: right;">
+ <form method="get" action="${site.site_url('search')}"
onsubmit="if(q.value.replace(/^\s*/, '').replace(/\s*$/, '') =='') return false;">
<input id="q" name="q" maxlength="100" size="25" type="text"
value="${searchtext}" onblur="if(this.value=='') this.value='${searchtext}';"
</form>
</div>
- <div py:def="course_banner(course)" py:strip="True">
- <div id="coursebanner">
- ${course_search(course)}
- <div class="deptident">${course.department}</div>
- <h1><a href="${course.course_url()}"><span py:if="course.code">${course.code}: </span>${course.title}</a></h1>
+ <div py:def="site_banner(site)" py:strip="True">
+ <div id="sitebanner">
+ ${site_search(site)}
+ <div class="deptident">${site.department}</div>
+ <h1><a href="${site.site_url()}"><span py:if="site.code">${site.code}: </span>${site.title}</a></h1>
</div>
</div>
py:with="hier=item.hierarchy()[:-1]; lastnum=len(hier)"
class="nestedtitle">
<div id="breadcrumbs">
- <span><a href="${item.course.course_url()}">Top</a></span> »
+ <span><a href="${item.site.site_url()}">Top</a></span> »
<span py:for="n, x in enumerate(hier)"><a href="${x.item_url()}">${x.title}</a> » </span>
<span class="final_item">${item.title}</span>
</div>
--- /dev/null
+<?python
+title = defined('custom_title') and custom_title or _('Courses')
+?>
+<html xmlns="http://www.w3.org/1999/xhtml"
+ xmlns:xi="http://www.w3.org/2001/XInclude"
+ xmlns:py="http://genshi.edgewall.org/">
+<xi:include href="master.xhtml"/>
+<xi:include href="paginate.xhtml"/>
+<head>
+ <title>${title}</title>
+ <script type="text/javascript">
+ <!-- !This ought to be in paginate.xhtml, not here. how to do? -->
+ $(function() { $('.pagetable').tablesorter(); });
+ </script>
+</head>
+<body>
+ <h1>${title}</h1>
+ <tr py:def="pageheader()">
+ <th>Name</th>
+ </tr>
+ <span py:def="pagerow(item)">
+ <td><a href="${item.course_url()}">${item.title}, ${item.term}</a></td>
+ </span>
+ ${pagetable(paginator, count, pagerow, pageheader)}
+</body>
+</html>
<?python
if instance.id:
- title = _('Edit course details')
+ title = _('Edit site details')
else:
- title = _('Create a new course site')
+ title = _('Create a new site')
?>
<html xmlns="http://www.w3.org/1999/xhtml"
xmlns:xi="http://www.w3.org/2001/XInclude"
xmlns:py="http://genshi.edgewall.org/">
<xi:include href="master.xhtml"/>
-<xi:include href="components/course.xhtml"/>
+<xi:include href="components/site.xhtml"/>
<head>
<title>${title}</title>
- <script type="text/javascript" src="${ROOT}/static/edit_course.js"/>
+ <script type="text/javascript" src="${ROOT}/static/edit_site.js"/>
</head>
<body>
- <div py:if="instance.id">${course_banner(instance)}</div>
+ <div py:if="instance.id">${site_banner(instance)}</div>
<h1>${title}</h1>
- <p py:if="instance.id"><a href="permission/">Edit course permissions</a> • <a href="${instance.course_url()}">Return to course page</a></p>
+ <p py:if="instance.id"><a href="permission/">Edit site permissions</a> • <a href="${instance.site_url()}">Return to site page</a></p>
<form action="." method="POST">
<tr py:def="field_row(field, example=None)">
<th>${field.label}</th>
</form>
<div class="gap"/>
<div py:if="instance.id">
- <h2>Delete this course</h2>
+ <h2>Delete this site</h2>
<form action="delete/" method="POST">
<p><input type="checkbox" name="confirm_delete" id="confirm_delete"/>
- <label for="confirm_delete">Yes, I want to delete this course site and all of its contents.</label>
+ <label for="confirm_delete">Yes, I want to delete this site and all of its contents.</label>
</p>
- <p><input type="submit" value="Delete this course"/> ${go_back_link()}</p>
+ <p><input type="submit" value="Delete this site"/> ${go_back_link()}</p>
</form>
</div>
</body>
<?python
-title = _('Edit course permissions')
-instructors = [m for m in models.Member.objects.filter(course=course) if m.role in ('PROXY', 'INSTR')]
+title = _('Edit site permissions')
+instructors = [m for m in models.Member.objects.filter(site=site) if m.role in ('PROXY', 'INSTR')]
?>
<html xmlns="http://www.w3.org/1999/xhtml"
xmlns:xi="http://www.w3.org/2001/XInclude"
xmlns:py="http://genshi.edgewall.org/">
<xi:include href="master.xhtml"/>
- <xi:include href="components/course.xhtml"/>
+ <xi:include href="components/site.xhtml"/>
<head>
<title>${title}</title>
- <script type="text/javascript" src="${ROOT}/static/edit_course_permissions.js"/>
+ <script type="text/javascript" src="${ROOT}/static/edit_site_permissions.js"/>
</head>
<body>
- ${course_banner(course)}
+ ${site_banner(site)}
<h1>${title}</h1>
- <p><a href="../">Edit course details</a> • <a href="${course.course_url()}">Return to course page</a></p>
+ <p><a href="../">Edit site details</a> • <a href="${site.site_url()}">Return to site page</a></p>
<form action="." method="POST">
<h2>Instructor Access</h2>
<div py:if="defined('instructor_error')" class="errors">${instructor_error}</div>
<div class="gap"/>
<h2 id="student_access">Student Access</h2>
<p>Who will have student-level access to this site?
- <span style="margin-left: 12;"/>${Markup(choose_access.render('access', course.access, {'id':'id_access'}, []))}</p>
+ <span style="margin-left: 12;"/>${Markup(choose_access.render('access', site.access, {'id':'id_access'}, []))}</p>
<div id="INVIT_panel" class="specific">
- <h3>Course Invitation Code</h3>
- <p style="font-size: larger;">Your Course Invitation Code is: <strong>${course.passkey}</strong>
+ <h3>Site Invitation Code</h3>
+ <p style="font-size: larger;">Your Site Invitation Code is: <strong>${site.passkey}</strong>
<span style="margin-left: 16;">
<input type="submit" name="action_change_code" value="Select a new code"/>
</span>
</p>
<p>This invitation code will enable your students to join this
- course site. Share it only with your students: anyone who has
+ site. Share it only with your students: anyone who has
the code can join your site.</p>
<p>You may change the code at any time. This will not block
students who have already joined, but will prevent new students
<div class="gap"/>
<h2>Class List</h2>
- <p>The following users have student-level access in this course site.</p>
+ <p>The following users have student-level access in this site.</p>
<ol>
- <li py:for="student in course.get_students()">
+ <li py:for="student in site.get_students()">
${student.get_full_name()} (${student.email})
</li>
</ol>
<feed xmlns="http://www.w3.org/2005/Atom"
xmlns:xi="http://www.w3.org/2001/XInclude"
xmlns:py="http://genshi.edgewall.org/">
- <title>Reserves (${feed_type}) for ${course.list_display()}</title>
- <link href="${root}${course.course_url()}"
+ <title>Reserves (${feed_type}) for ${site.list_display()}</title>
+ <link href="${root}${site.site_url()}"
rel="alternate"/>
- <link href="${root}${course.course_url('feeds/' + feed_type)}"
+ <link href="${root}${site.site_url('feeds/' + feed_type)}"
rel="self"/>
- <id>${root}${course.course_url('feeds/') + feed_type}</id>
+ <id>${root}${site.site_url('feeds/') + feed_type}</id>
<updated>${lastmod.strftime("%Y-%m-%dT%H:%M:%SZ")}</updated>
<entry py:for="item in items">
<title>${render_title(item)}</title>
xmlns:xi="http://www.w3.org/2001/XInclude"
xmlns:py="http://genshi.edgewall.org/">
<xi:include href="../master.xhtml"/>
- <xi:include href="../components/course.xhtml"/>
+ <xi:include href="../components/site.xhtml"/>
<head>
<title>${title}</title>
<script type="text/javascript" src="${ROOT}/static/menublocks.js"/>
</head>
<body>
- ${course_banner(course)}
+ ${site_banner(site)}
<h2>Available Feeds</h2>
<ul>
<li><a href="recent-changes">Recent changes</a></li>
xmlns:py="http://genshi.edgewall.org/">
<xi:include href="../master.xhtml"/>
<xi:include href="../paginate.xhtml"/>
-<xi:include href="../components/course.xhtml"/>
+<xi:include href="../components/site.xhtml"/>
<head>
<title>${title}</title>
<script type="text/javascript">
</script>
</head>
<body>
- ${course_banner(course)}
+ ${site_banner(site)}
${nested_title(parent_item)}
<h2>${title}</h2>
<div class="helptext">
<?python
is_edit = bool(item.id)
title = is_edit and _('Edit an electronic document') or _('Add a new electronic document')
-course_title = '%s: %s (%s)' % (course.code, course.title, course.term)
+site_title = '%s: %s (%s)' % (site.code, site.title, site.term)
?>
<html xmlns="http://www.w3.org/1999/xhtml"
xmlns:xi="http://www.w3.org/2001/XInclude"
xmlns:py="http://genshi.edgewall.org/">
<xi:include href="../master.xhtml"/>
- <xi:include href="../components/course.xhtml"/>
+ <xi:include href="../components/site.xhtml"/>
<head>
<title>${title}</title>
<script type="text/javascript">
${item_metadata_formset_header()}
</head>
<body>
- ${course_banner(course)}
+ ${site_banner(site)}
${nested_title(parent_item)}
${offer_to_delete(item)}
<h2>${title}</h2>
<?python
is_edit = bool(item.id)
title = is_edit and _('Edit a subheading') or _('Add a new subheading')
-course_title = '%s: %s (%s)' % (course.code, course.title, course.term)
+site_title = '%s: %s (%s)' % (site.code, site.title, site.term)
?>
<html xmlns="http://www.w3.org/1999/xhtml"
xmlns:xi="http://www.w3.org/2001/XInclude"
xmlns:py="http://genshi.edgewall.org/">
<xi:include href="../master.xhtml"/>
- <xi:include href="../components/course.xhtml"/>
+ <xi:include href="../components/site.xhtml"/>
<head>
<title>${title}</title>
<script type="text/javascript">
${item_metadata_formset_header()}
</head>
<body>
- ${course_banner(course)}
+ ${site_banner(site)}
${nested_title(parent_item)}
${offer_to_delete(item)}
<h2>${title}</h2>
<?python
is_edit = bool(item.id)
title = is_edit and _('Edit a physical-item request') or _('Add a physical-item request')
-course_title = '%s: %s (%s)' % (course.code, course.title, course.term)
+site_title = '%s: %s (%s)' % (site.code, site.title, site.term)
?>
<html xmlns="http://www.w3.org/1999/xhtml"
xmlns:xi="http://www.w3.org/2001/XInclude"
xmlns:py="http://genshi.edgewall.org/">
<xi:include href="../master.xhtml"/>
- <xi:include href="../components/course.xhtml"/>
+ <xi:include href="../components/site.xhtml"/>
<head>
<title>${title}</title>
<script type="text/javascript">
</script>
</head>
<body>
- ${course_banner(course)}
+ ${site_banner(site)}
${nested_title(parent_item)}
<div id="sidepanel">${offer_to_delete(item)}</div>
<h2>${title}</h2>
<?python
is_edit = bool(item.id)
title = is_edit and _('Edit a URL') or _('Add a new URL')
-course_title = '%s: %s (%s)' % (course.code, course.title, course.term)
+site_title = '%s: %s (%s)' % (site.code, site.title, site.term)
?>
<html xmlns="http://www.w3.org/1999/xhtml"
xmlns:xi="http://www.w3.org/2001/XInclude"
xmlns:py="http://genshi.edgewall.org/">
<xi:include href="../master.xhtml"/>
- <xi:include href="../components/course.xhtml"/>
+ <xi:include href="../components/site.xhtml"/>
<head>
<title>${title}</title>
<script type="text/javascript">
${item_metadata_formset_header()}
</head>
<body>
- ${course_banner(course)}
+ ${site_banner(site)}
${nested_title(parent_item)}
${offer_to_delete(item)}
<h2>${title}</h2>
<?python
-course_title = '%s: %s (%s)' % (course.code, course.title, course.term)
+site_title = '%s: %s (%s)' % (site.code, site.title, site.term)
hier = item.hierarchy()[:-1]
if item.item_type == 'HEADING':
title = _('Delete this heading?')
xmlns:xi="http://www.w3.org/2001/XInclude"
xmlns:py="http://genshi.edgewall.org/">
<xi:include href="../master.xhtml"/>
- <xi:include href="../components/course.xhtml"/>
+ <xi:include href="../components/site.xhtml"/>
<head>
<title>${title}</title>
</head>
<body>
- ${course_banner(course)}
+ ${site_banner(site)}
${nested_title(item)}
<h1>${title}</h1>
<p>Are you sure you want to delete this?</p>
<?python
-course = item.course
+site = item.site
title = item.title
-course_title = '%s: %s (%s)' % (course.code, course.title, course.term)
-is_editor = course.can_edit(request.user)
-item_tree = course.item_tree(subtree=item)
+site_title = '%s: %s (%s)' % (site.code, site.title, site.term)
+is_editor = site.can_edit(request.user)
+item_tree = site.item_tree(subtree=item)
?>
<html xmlns="http://www.w3.org/1999/xhtml"
xmlns:xi="http://www.w3.org/2001/XInclude"
xmlns:py="http://genshi.edgewall.org/">
<xi:include href="../master.xhtml"/>
- <xi:include href="../components/course.xhtml"/>
+ <xi:include href="../components/site.xhtml"/>
<head>
<title>${title}</title>
<script type="text/javascript" src="${ROOT}/static/menublocks.js"/>
</head>
<body>
- ${course_banner(course)}
+ ${site_banner(site)}
<div py:if="is_editor" id="sidepanel" class="little_action_panel">
<div>
<a href="edit/">Edit this heading</a>
<?python
from django.utils.simplejson import loads
-course_title = '%s: %s (%s)' % (course.code, course.title, course.term)
+site_title = '%s: %s (%s)' % (site.code, site.title, site.term)
hier = item.hierarchy()[:-1]
title = item.title
smallint = item.smallint()
metadata = item.metadata_set.all()
-is_editor = course.can_edit(request.user)
+is_editor = site.can_edit(request.user)
?>
<html xmlns="http://www.w3.org/1999/xhtml"
xmlns:xi="http://www.w3.org/2001/XInclude"
xmlns:py="http://genshi.edgewall.org/">
<xi:include href="../master.xhtml"/>
- <xi:include href="../components/course.xhtml"/>
+ <xi:include href="../components/site.xhtml"/>
<head>
<title>${title}</title>
<style>
</style>
</head>
<body>
- ${course_banner(course)}
+ ${site_banner(site)}
${nested_title(item)}
<div py:if="is_editor" id="sidepanel">
<div>
xmlns:xi="http://www.w3.org/2001/XInclude"
xmlns:py="http://genshi.edgewall.org/">
<xi:include href="../master.xhtml"/>
- <xi:include href="../components/course.xhtml"/>
+ <xi:include href="../components/site.xhtml"/>
<head>
<title>${title}</title>
<script type="text/javascript" src="${ROOT}/static/menublocks.js"/>
</head>
<body>
- ${course_banner(course)}
+ ${site_banner(site)}
${nested_title(item)}
<h2>${title}</h2>
<p>Please choose the heading under which this item should be moved:</p>
<div class="gap"/></td></tr>
<tr><th>New heading:</th>
<td>
- ${heading_tree(course.item_tree(), stop_at=item)}
+ ${heading_tree(site.item_tree(), stop_at=item)}
<div class="gap"/>
</td>
</tr>
<h1>${title}</h1>
<?python
if user.is_authenticated():
- my_courses = user.reading_lists()
+ my_sites = user.sites()
else:
- my_courses = []
+ my_sites = []
?>
- <p py:if="not my_courses">You are not part of any course reserves at this time.</p>
- <p py:for="course in my_courses" style="font-size: large;">
- <a href="${course.id}/">${course.list_display()}</a>
+ <p py:if="not my_sites">You are not part of any course-reserve sites at this time.</p>
+ <p py:for="site in my_sites" style="font-size: large;">
+ <a href="${site.id}/">${site.list_display()}</a>
</p>
<div class="gap"/>
<div class="itemadd">
- <p py:if="user.can_create_reading_lists()"><a href="new/">Create a new course reserves list</a></p>
+ <p py:if="user.can_create_sites()"><a href="new/">Create a new course-reserves site</a></p>
<p><a href="invitation/">Join a reserves list using an Invitation Code</a></p>
</div>
</body>
<?python
-title = _('Open Courses')
+title = _('Open Sites')
?>
<html xmlns="http://www.w3.org/1999/xhtml"
xmlns:xi="http://www.w3.org/2001/XInclude"
</tr>
<span py:def="pagerow(item)">
<td>${item.term}</td>
- <td><a href="../course/${item.id}/">${item.code}</a></td>
- <td><a href="../course/${item.id}/">${item.title}</a></td>
+ <td><a href="../site/${item.id}/">${item.code}</a></td>
+ <td><a href="../site/${item.id}/">${item.title}</a></td>
</span>
<!--
${pagetable(paginator, start, count, pagerow, pageheader)}
<input type="hidden" name="barcode" value="${barcode}"/>
<table class="metadata_table" width="100%;">
<thead style="font-size: 70%;">
- <tr><th py:for="v in 'Select Title Author Course Instructor Score'.split(' ')">${v}</th></tr>
+ <tr><th py:for="v in 'Select Title Author Site Instructor Score'.split(' ')">${v}</th></tr>
</thead>
<tbody py:for="n, (score, item) in enumerate(ranked)"
class="${score < 5 and 'likely' or score < 50 and 'maybe' or 'doubtful'}"
<span py:if="marc"><a href="javascript:$('#marcdata${n}').toggle();void(0);">detail</a></span>
</td>
<td>${item.author() or '—'}</td>
- <td>${item.course.title}</td>
- <td>${','.join(n.last_name for n in item.course.get_instructors())}</td>
+ <td>${item.site.title}</td>
+ <td>${','.join(n.last_name for n in item.site.get_instructors())}</td>
<td>${repr(score)}</td>
</tr>
<tr py:if="marc" id="marcdata${n}" style="display: none;">
<p><b>This doesn't work yet.</b></p>
<input type="checkbox" name="wants_email_notices" id="id_wants_email_notices"
py:attrs="{'checked':user.get_profile().wants_email_notices or None}"/>
- <label for="id_wants_email_notices">I would like to receive emails when new items are added to my course sites.</label>
+ <label for="id_wants_email_notices">I would like to receive emails when new items are added to my course-reserve sites.</label>
<p><input type="submit" value="Update notification preferences"/></p>
</form>
</div>
<span py:def="pagerow(item)">
<td>${Markup(item.author_hl(norm_query))}</td>
<td><a href="${item.item_url('meta')}">${Markup(item.title_hl(norm_query))}</a></td>
- <td><a href="${item.course.course_url()}">${item.course.title}</a></td>
+ <td><a href="${item.site.site_url()}">${item.site.title}</a></td>
<td><span py:if="item.item_type=='PHYS'">${item.smallint()} • ${item.barcode()}</span></td>
</span>
${pagetable(paginator, count, pagerow, pageheader, query=query_string)}
<?python
-title = '%s: %s (%s)' % (course.code, course.title, course.term)
-item_tree = course.item_tree()
-is_editor = course.can_edit(request.user)
-is_joinable = course.is_joinable_by(request.user)
+title = '%s: %s (%s)' % (site.code, site.title, site.term)
+item_tree = site.item_tree()
+is_editor = site.can_edit(request.user)
+is_joinable = site.is_joinable_by(request.user)
?>
<html xmlns="http://www.w3.org/1999/xhtml"
xmlns:xi="http://www.w3.org/2001/XInclude"
xmlns:py="http://genshi.edgewall.org/">
<xi:include href="master.xhtml"/>
- <xi:include href="components/course.xhtml"/>
+ <xi:include href="components/site.xhtml"/>
<head>
<title>${title}</title>
<script type="text/javascript" src="${ROOT}/static/menublocks.js"/>
</head>
<body>
- ${course_banner(course)}
+ ${site_banner(site)}
<p py:if="not item_tree">
- There are no items associated with this course yet.
+ There are no items associated with this site yet.
</p>
<div id="sidepanel">
- <div py:if="is_editor" id="edit_course" class="little_action_panel">
- <a href="${course.course_url()}edit/">Edit course details</a>
+ <div py:if="is_editor" id="edit_site" class="little_action_panel">
+ <a href="${site.site_url()}edit/">Edit site details</a>
</div>
<div py:if="is_joinable">
- <a href="${course.course_url()}join/">Join this course</a>
+ <a href="${site.site_url()}join/">Join this site</a>
</div>
<div id="showtree_panel" class="little_action_panel">
<a href="javascript:doToggleItemTree();">Show/hide tree</a>
<?python
-title = _('Join a course using an Invitation Code')
+title = _('Join a site using an Invitation Code')
?>
<html xmlns="http://www.w3.org/1999/xhtml"
xmlns:xi="http://www.w3.org/2001/XInclude"
<?python
-title = _('Join this course?')
+title = _('Join this site?')
?>
<html xmlns="http://www.w3.org/1999/xhtml"
xmlns:xi="http://www.w3.org/2001/XInclude"
xmlns:py="http://genshi.edgewall.org/">
<xi:include href="master.xhtml"/>
- <xi:include href="components/course.xhtml"/>
+ <xi:include href="components/site.xhtml"/>
<head>
<title>${title}</title>
<script type="text/javascript" src="${ROOT}/static/menublocks.js"/>
</head>
<body>
- ${course_banner(course)}
+ ${site_banner(site)}
<h2>${title}</h2>
- <p>If you choose to join this course, it will be added to your "My
- Courses" list.</p>
+ <p>If you choose to join this site, it will be added to your "My
+ Sites" list.</p>
<form action="." method="POST">
- <p><input type="submit" value="Yes, join this course"/> ${go_back_link()}</p>
+ <p><input type="submit" value="Yes, join this site"/> ${go_back_link()}</p>
</form>
</body>
+++ /dev/null
-<?python
-title = defined('custom_title') and custom_title or _('Courses')
-?>
-<html xmlns="http://www.w3.org/1999/xhtml"
- xmlns:xi="http://www.w3.org/2001/XInclude"
- xmlns:py="http://genshi.edgewall.org/">
-<xi:include href="master.xhtml"/>
-<xi:include href="paginate.xhtml"/>
-<head>
- <title>${title}</title>
- <script type="text/javascript">
- <!-- !This ought to be in paginate.xhtml, not here. how to do? -->
- $(function() { $('.pagetable').tablesorter(); });
- </script>
-</head>
-<body>
- <h1>${title}</h1>
- <tr py:def="pageheader()">
- <th>Name</th>
- </tr>
- <span py:def="pagerow(item)">
- <td><a href="${item.course_url()}">${item.title}, ${item.term}</a></td>
- </span>
- ${pagetable(paginator, count, pagerow, pageheader)}
-</body>
-</html>
<ul id="tabbar">
<!-- <li><a href="${ROOT}/">Home</a></li> -->
<li><a href="${ROOT}/browse/">Browse</a></li>
- <li class="active"><a href="${ROOT}/course/">My Reserves</a></li>
+ <li class="active"><a href="${ROOT}/site/">My Reserves</a></li>
<div py:strip="True"
py:if="request.user.is_staff">
<li><a href="${ROOT}/admin/">Admin Options</a></li>