184 lines
7.4 KiB
Python
184 lines
7.4 KiB
Python
"""
|
|
Views for the course_mode module
|
|
"""
|
|
|
|
import decimal
|
|
from django.core.urlresolvers import reverse
|
|
from django.http import HttpResponseBadRequest
|
|
from django.shortcuts import redirect
|
|
from django.views.generic.base import View
|
|
from django.utils.translation import ugettext as _
|
|
from django.contrib.auth.decorators import login_required
|
|
from django.utils.decorators import method_decorator
|
|
|
|
from edxmako.shortcuts import render_to_response
|
|
|
|
from course_modes.models import CourseMode
|
|
from courseware.access import has_access
|
|
from student.models import CourseEnrollment
|
|
from opaque_keys.edx.locations import SlashSeparatedCourseKey
|
|
from xmodule.modulestore.django import modulestore
|
|
|
|
|
|
class ChooseModeView(View):
|
|
"""
|
|
View used when the user is asked to pick a mode
|
|
|
|
When a get request is used, shows the selection page.
|
|
When a post request is used, assumes that it is a form submission
|
|
from the selection page, parses the response, and then sends user
|
|
to the next step in the flow
|
|
"""
|
|
|
|
@method_decorator(login_required)
|
|
def get(self, request, course_id, error=None):
|
|
""" Displays the course mode choice page
|
|
|
|
Args:
|
|
request (`Request`): The Django Request object.
|
|
course_id (unicode): The slash-separated course key.
|
|
|
|
Keyword Args:
|
|
error (unicode): If provided, display this error message
|
|
on the page.
|
|
|
|
Returns:
|
|
Response
|
|
|
|
"""
|
|
course_key = SlashSeparatedCourseKey.from_deprecated_string(course_id)
|
|
|
|
enrollment_mode, is_active = CourseEnrollment.enrollment_mode_for_user(request.user, course_key)
|
|
upgrade = request.GET.get('upgrade', False)
|
|
request.session['attempting_upgrade'] = upgrade
|
|
|
|
# TODO (ECOM-16): Remove once the AB-test of auto-registration completes
|
|
auto_register = request.session.get('auto_register', False)
|
|
|
|
# Inactive users always need to re-register
|
|
# Verified and professional users do not need to register or upgrade
|
|
# Registered users who are not trying to upgrade do not need to re-register
|
|
if not auto_register:
|
|
go_to_dashboard = (
|
|
is_active and
|
|
(not upgrade or enrollment_mode in ['verified', 'professional'])
|
|
)
|
|
|
|
# If auto-registration is enabled, then students might already be registered,
|
|
# but we should still show them the "choose your track" page so they have
|
|
# the option to enter the verification/payment flow.
|
|
# TODO (ECOM-16): Based on the results of the AB-test, set the default behavior to
|
|
# either enable or disable auto-registration.
|
|
else:
|
|
go_to_dashboard = (
|
|
not upgrade and enrollment_mode in ['verified', 'professional']
|
|
)
|
|
|
|
if go_to_dashboard:
|
|
return redirect(reverse('dashboard'))
|
|
|
|
modes = CourseMode.modes_for_course_dict(course_key)
|
|
|
|
# We assume that, if 'professional' is one of the modes, it is the *only* mode.
|
|
# If we offer more modes alongside 'professional' in the future, this will need to route
|
|
# to the usual "choose your track" page.
|
|
if "professional" in modes:
|
|
return redirect(
|
|
reverse(
|
|
'verify_student_show_requirements',
|
|
kwargs={'course_id': course_key.to_deprecated_string()}
|
|
)
|
|
)
|
|
|
|
donation_for_course = request.session.get("donation_for_course", {})
|
|
chosen_price = donation_for_course.get(unicode(course_key), None)
|
|
|
|
course = modulestore().get_course(course_key)
|
|
context = {
|
|
"course_modes_choose_url": reverse("course_modes_choose", kwargs={'course_id': course_key.to_deprecated_string()}),
|
|
"modes": modes,
|
|
"course_name": course.display_name_with_default,
|
|
"course_org": course.display_org_with_default,
|
|
"course_num": course.display_number_with_default,
|
|
"chosen_price": chosen_price,
|
|
"error": error,
|
|
"upgrade": upgrade,
|
|
"can_audit": "audit" in modes,
|
|
"autoreg": auto_register
|
|
}
|
|
if "verified" in modes:
|
|
context["suggested_prices"] = [
|
|
decimal.Decimal(x.strip())
|
|
for x in modes["verified"].suggested_prices.split(",")
|
|
if x.strip()
|
|
]
|
|
context["currency"] = modes["verified"].currency.upper()
|
|
context["min_price"] = modes["verified"].min_price
|
|
context["verified_name"] = modes["verified"].name
|
|
context["verified_description"] = modes["verified"].description
|
|
|
|
return render_to_response("course_modes/choose.html", context)
|
|
|
|
@method_decorator(login_required)
|
|
def post(self, request, course_id):
|
|
""" Takes the form submission from the page and parses it """
|
|
course_key = SlashSeparatedCourseKey.from_deprecated_string(course_id)
|
|
user = request.user
|
|
|
|
# This is a bit redundant with logic in student.views.change_enrollment,
|
|
# but I don't really have the time to refactor it more nicely and test.
|
|
course = modulestore().get_course(course_key)
|
|
if not has_access(user, 'enroll', course):
|
|
error_msg = _("Enrollment is closed")
|
|
return self.get(request, course_id, error=error_msg)
|
|
|
|
upgrade = request.GET.get('upgrade', False)
|
|
|
|
requested_mode = self.get_requested_mode(request.POST)
|
|
|
|
allowed_modes = CourseMode.modes_for_course_dict(course_key)
|
|
if requested_mode not in allowed_modes:
|
|
return HttpResponseBadRequest(_("Enrollment mode not supported"))
|
|
|
|
if requested_mode in ("audit", "honor"):
|
|
CourseEnrollment.enroll(user, course_key, requested_mode)
|
|
return redirect('dashboard')
|
|
|
|
mode_info = allowed_modes[requested_mode]
|
|
|
|
if requested_mode == "verified":
|
|
amount = request.POST.get("contribution") or \
|
|
request.POST.get("contribution-other-amt") or 0
|
|
|
|
try:
|
|
# validate the amount passed in and force it into two digits
|
|
amount_value = decimal.Decimal(amount).quantize(decimal.Decimal('.01'), rounding=decimal.ROUND_DOWN)
|
|
except decimal.InvalidOperation:
|
|
error_msg = _("Invalid amount selected.")
|
|
return self.get(request, course_id, error=error_msg)
|
|
|
|
# Check for minimum pricing
|
|
if amount_value < mode_info.min_price:
|
|
error_msg = _("No selected price or selected price is too low.")
|
|
return self.get(request, course_id, error=error_msg)
|
|
|
|
donation_for_course = request.session.get("donation_for_course", {})
|
|
donation_for_course[unicode(course_key)] = amount_value
|
|
request.session["donation_for_course"] = donation_for_course
|
|
|
|
return redirect(
|
|
reverse('verify_student_show_requirements',
|
|
kwargs={'course_id': course_key.to_deprecated_string()}) + "?upgrade={}".format(upgrade))
|
|
|
|
def get_requested_mode(self, request_dict):
|
|
"""
|
|
Given the request object of `user_choice`, return the
|
|
corresponding course mode slug
|
|
"""
|
|
if 'audit_mode' in request_dict:
|
|
return 'audit'
|
|
if 'certificate_mode' and request_dict.get("honor-code"):
|
|
return 'honor'
|
|
if 'certificate_mode' in request_dict:
|
|
return 'verified'
|