refactor: refactored the course page

- simplified the portal page for course
- added mentods to LMS Course and Community Member to reduce custom code
  in portal pages
- included lessons in the ChapterTeaser
This commit is contained in:
Anand Chitipothu
2021-05-03 12:05:17 +05:30
parent 9e103af8b5
commit b9d94df4d8
7 changed files with 157 additions and 150 deletions

View File

@@ -11,54 +11,71 @@ import random
class CommunityMember(Document): class CommunityMember(Document):
def validate(self): def validate(self):
self.validate_username() self.validate_username()
self.abbr = ("").join([ s[0] for s in self.full_name.split() ]) self.abbr = ("").join([ s[0] for s in self.full_name.split() ])
if self.route != self.username: if self.route != self.username:
self.route = self.username self.route = self.username
def validate_username(self): def validate_username(self):
if not self.username: if not self.username:
self.username = create_username_from_email(self.email) self.username = create_username_from_email(self.email)
if self.username: if self.username:
if len(self.username) < 4: if len(self.username) < 4:
frappe.throw(_("Username must be atleast 4 characters long.")) frappe.throw(_("Username must be atleast 4 characters long."))
if not re.match("^[A-Za-z0-9_]*$", self.username): if not re.match("^[A-Za-z0-9_]*$", self.username):
frappe.throw(_("Username can only contain alphabets, numbers and underscore.")) frappe.throw(_("Username can only contain alphabets, numbers and underscore."))
self.username = self.username.lower() self.username = self.username.lower()
def __repr__(self): def get_course_count(self) -> int:
return f"<CommunityMember: {self.email}>" """Returns the number of courses authored by this user.
"""
return frappe.db.count(
'LMS Course', {
'owner': self.email
})
def get_batch_count(self) -> int:
"""Returns the number of batches authored by this user.
"""
return frappe.db.count(
'LMS Batch Membership', {
'member': self.name,
'member_role': 'Mentor'
})
def __repr__(self):
return f"<CommunityMember: {self.email}>"
def create_member_from_user(doc, method): def create_member_from_user(doc, method):
username = doc.username username = doc.username
if ( doc.username and username_exists(doc.username)) or not doc.username: if ( doc.username and username_exists(doc.username)) or not doc.username:
username = create_username_from_email(doc.email) username = create_username_from_email(doc.email)
elif len(doc.username) < 4: elif len(doc.username) < 4:
username = adjust_username(doc.username) username = adjust_username(doc.username)
if username_exists(username): if username_exists(username):
username = username + str(random.randint(0,9)) username = username + str(random.randint(0,9))
member = frappe.get_doc({ member = frappe.get_doc({
"doctype": "Community Member", "doctype": "Community Member",
"full_name": doc.full_name, "full_name": doc.full_name,
"username": username, "username": username,
"email": doc.email, "email": doc.email,
"route": doc.username, "route": doc.username,
"owner": doc.email "owner": doc.email
}) })
member.save(ignore_permissions=True) member.save(ignore_permissions=True)
def username_exists(username): def username_exists(username):
return frappe.db.exists("Community Member", dict(username=username)) return frappe.db.exists("Community Member", dict(username=username))
def create_username_from_email(email): def create_username_from_email(email):
string = email.split("@")[0] string = email.split("@")[0]
return ''.join(e for e in string if e.isalnum()) return ''.join(e for e in string if e.isalnum())
def adjust_username(username): def adjust_username(username):
return username.ljust(4, str(random.randint(0,9))) return username.ljust(4, str(random.randint(0,9)))

View File

@@ -3,8 +3,12 @@
# For license information, please see license.txt # For license information, please see license.txt
from __future__ import unicode_literals from __future__ import unicode_literals
# import frappe import frappe
from frappe.model.document import Document from frappe.model.document import Document
class Chapter(Document): class Chapter(Document):
pass def get_lessons(self):
rows = frappe.db.get_all("Lesson",
filters={"chapter": self.name},
fields='*')
return [frappe.get_doc(dict(row, doctype='Lesson')) for row in rows]

View File

@@ -8,6 +8,18 @@ from frappe.model.document import Document
from ...utils import slugify from ...utils import slugify
class LMSCourse(Document): class LMSCourse(Document):
@staticmethod
def find(slug):
"""Returns the course with specified slug.
"""
return find("LMS Course", is_published=True, slug=slug)
@staticmethod
def find_all():
"""Returns all published courses.
"""
return find_all("LMS Course", is_published=True)
def before_save(self): def before_save(self):
if not self.slug: if not self.slug:
self.slug = self.generate_slug(title=self.title) self.slug = self.generate_slug(title=self.title)
@@ -50,7 +62,7 @@ class LMSCourse(Document):
"""Returns the name of Community Member document for a give user. """Returns the name of Community Member document for a give user.
""" """
try: try:
return frappe.db.get_value("Community Member", {"email": email}, ["name"]) return frappe.db.get_value("Community Member", {"email": email}, "name")
except frappe.DoesNotExistError: except frappe.DoesNotExistError:
return None return None
@@ -85,18 +97,65 @@ class LMSCourse(Document):
for mentor in mentors: for mentor in mentors:
member = frappe.get_doc("Community Member", mentor.mentor) member = frappe.get_doc("Community Member", mentor.mentor)
# TODO: change this to count query # TODO: change this to count query
member.batch_count = len(frappe.get_all("LMS Batch Membership", {"member": member.name, "member_type": "Mentor"})) member.batch_count = len(frappe.get_all("LMS Batch Membership", {"member": member, "member_type": "Mentor"}))
course_mentors.append(member) course_mentors.append(member)
return course_mentors return course_mentors
def get_instructor(self): def is_mentor(self, email):
return frappe.get_doc("User", self.owner) """Checks if given user is a mentor for this course.
@staticmethod
def find_all():
"""Returns all published courses.
""" """
rows = frappe.db.get_all("LMS Course", if not email:
filters={"is_published": True}, return False
fields='*') member = self.get_community_member(email)
return [frappe.get_doc(dict(row, doctype='LMS Course')) for row in rows] return frappe.db.exists({
'doctype': 'LMS Course Mentor Mapping',
"course": self.name,
"member": member
})
def get_instructor(self):
member_name = self.get_community_member(self.owner)
return frappe.get_doc("Community Member", member_name)
def get_chapters(self):
"""Returns all chapters of this course.
"""
# TODO: chapters should have a way to specify the order
return find_all("Chapter", course=self.name, order_by="creation")
def get_batches(self, mentor=None):
batches = find_all("LMS Batch", course=self.name)
if mentor:
# TODO: optimize this
member = self.get_community_member(email=mentor)
memberships = frappe.db.get_all(
"LMS Batch Membership",
{"member": member},
["name"], as_dict=1)
batch_names = {m.batch for m in memberships}
return [b for b in batches if b.name in batch_names]
def get_upcoming_batches(self):
now = frappe.utils.nowdate()
return find_all("LMS Batch",
course=self.name,
start_date=[">", now])
def find_all(doctype, order_by=None, **filters):
"""Queries the database for documents of a doctype matching given filters.
"""
rows = frappe.db.get_all(doctype,
filters=filters,
fields='*',
order_by=order_by)
return [frappe.get_doc(dict(row, doctype=doctype)) for row in rows]
def find(doctype, **filters):
"""Queries the database for a document of given doctype matching given filters.
"""
rows = frappe.db.get_all(doctype,
filters=filters,
fields='*')
if rows:
row = rows[0]
return frappe.get_doc(dict(row, doctype=doctype))

View File

@@ -4,5 +4,12 @@
<div class="chapter-description"> <div class="chapter-description">
{{ chapter.description or "" }} {{ chapter.description or "" }}
</div> </div>
<div class="chapter-lessons">
{% for lesson in chapter.get_lessons() %}
<div class="lesson-teaser">
{{lesson.title}}
</div>
{% endfor %}
</div>
</div> </div>
</div> </div>

View File

@@ -17,18 +17,18 @@
<div class="col-lg-9 col-md-12"> <div class="col-lg-9 col-md-12">
<div class="course-details"> <div class="course-details">
{{ CourseDescription(course) }} {{ CourseDescription(course) }}
{{ BatchSection(course, is_mentor, upcoming_batches, mentor_batches) }} {{ BatchSection(course) }}
{{ CourseOutline(course) }} {{ CourseOutline(course) }}
</div> </div>
</div> </div>
<div class="col-lg-3 col-md-12"> <div class="col-lg-3 col-md-12">
<div class="sidebar"> <div class="sidebar">
{{ InstructorsSection(instructor) }} {{ InstructorsSection(course.get_instructor()) }}
</div> </div>
<div class="sidebar"> <div class="sidebar">
{{ MentorsSection(mentors, is_mentor) }} {{ MentorsSection(course.get_mentors(), course.is_mentor(frappe.session.user)) }}
</div> </div>
</div> </div>
</div> </div>
@@ -57,11 +57,11 @@
{% endif %} {% endif %}
{% endmacro %} {% endmacro %}
{% macro BatchSection(course, is_mentor, upcoming_batches, mentor_batches) %} {% macro BatchSection(course) %}
{% if is_mentor %} {% if course.is_mentor(frappe.session.user) %}
{{ BatchSectionForMentors(course, mentor_batches) }} {{ BatchSectionForMentors(course, course.get_batches(mentor=frappe.session.user)) }}
{% else %} {% else %}
{{ BatchSectionForStudents(course, upcoming_batches) }} {{ BatchSectionForStudents(course, course.get_upcoming_batches()) }}
{% endif %} {% endif %}
{% endmacro %} {% endmacro %}
@@ -133,7 +133,7 @@
{% macro CourseOutline(course) %} {% macro CourseOutline(course) %}
<h2>Course Outline</h2> <h2>Course Outline</h2>
{% for chapter in course.chapters %} {% for chapter in course.get_chapters() %}
{{ widgets.ChapterTeaser(chapter=chapter)}} {{ widgets.ChapterTeaser(chapter=chapter)}}
{% endfor %} {% endfor %}
{% endmacro %} {% endmacro %}

View File

@@ -1,101 +1,21 @@
import frappe import frappe
from community.www.courses.utils import get_instructor from community.www.courses.utils import get_instructor
from frappe.utils import nowdate, getdate from frappe.utils import nowdate, getdate
from community.lms.models import Course
def get_context(context): def get_context(context):
context.no_cache = 1 context.no_cache = 1
try: try:
course_id = frappe.form_dict["course"] course_slug = frappe.form_dict["course"]
except KeyError: except KeyError:
frappe.local.flags.redirect_location = "/courses" frappe.local.flags.redirect_location = "/courses"
raise frappe.Redirect raise frappe.Redirect
context.course = get_course(course_id) course = Course.find(course_slug)
context.batches = get_course_batches(context.course.name) if course is None:
context.is_mentor = is_mentor(context.course.name) frappe.local.flags.redirect_location = "/courses"
context.memberships = get_membership(context.batches)
if len(context.memberships) and not context.is_mentor:
frappe.local.flags.redirect_location = "/courses/" + course_id + "/" + context.memberships[0].code + "/learn"
raise frappe.Redirect raise frappe.Redirect
context.upcoming_batches = get_upcoming_batches(context.course.name)
context.instructor = get_instructor(context.course.owner)
context.mentors = get_mentors(context.course.name)
if context.is_mentor: context.course = course
context.mentor_batches = get_mentor_batches(context.memberships) # Your Bacthes for mentor
def get_course(slug):
course = frappe.db.get_value("LMS Course", {"slug": slug},
["name", "slug", "title", "description", "short_introduction", "video_link", "owner"], as_dict=1)
course["chapters"] = frappe.db.get_all("Chapter",
filters={
"course": course["name"]
},
fields=["name", "title", "description"],
order_by="creation"
)
return course
def get_upcoming_batches(course):
batches = frappe.get_all("LMS Batch", {"course": course, "start_date": [">", nowdate()]}, ["start_date", "start_time", "end_time", "sessions_on", "name"])
batches = get_batch_mentors(batches)
return batches
def get_batch_mentors(batches):
for batch in batches:
batch.mentors = []
mentors = frappe.get_all("LMS Batch Membership", {"batch": batch.name, "member_type": "Mentor"}, ["member"])
for mentor in mentors:
member = frappe.db.get_value("Community Member", mentor.member, ["full_name", "photo", "abbr"], as_dict=1)
batch.mentors.append(member)
return batches
def get_membership(batches):
memberships = []
member = frappe.db.get_value("Community Member", {"email": frappe.session.user}, "name")
for batch in batches:
membership = frappe.db.get_value("LMS Batch Membership", {"member": member, "batch": batch.name}, ["batch", "member", "member_type"], as_dict=1)
if membership:
membership.code = batch.code
memberships.append(membership)
return memberships
def get_mentors(course):
course_mentors = []
mentors = frappe.get_all("LMS Course Mentor Mapping", {"course": course}, ["mentor"])
for mentor in mentors:
member = frappe.get_doc("Community Member", mentor.mentor)
member.batch_count = len(frappe.get_all("LMS Batch Membership", {"member": member.name, "member_type": "Mentor"}))
course_mentors.append(member)
return course_mentors
def get_course_batches(course):
return frappe.get_all("LMS Batch", {"course": course}, ["name", "code"])
def get_mentor_batches(memberships):
mentor_batches = []
memberships_as_mentor = list(filter(lambda x: x.member_type == "Mentor", memberships))
for membership in memberships_as_mentor:
batch = frappe.get_doc("LMS Batch", membership.batch)
mentor_batches.append(batch)
for batch in mentor_batches:
if getdate(batch.start_date) < getdate():
batch.status = "active"
batch.badge_class = "green_badge"
else:
batch.status = "scheduled"
batch.badge_class = "yellow_badge"
mentor_batches = get_batch_mentors(mentor_batches)
return mentor_batches
def is_mentor(course):
try:
member = frappe.db.get_value("Community Member", {"email": frappe.session.user}, ["name"])
except frappe.DoesNotExistError:
return False
mapping = frappe.get_all("LMS Course Mentor Mapping", {"course": course, "mentor": member})
if len(mapping):
return True

View File

@@ -2,7 +2,7 @@
<h3>Instructor</h3> <h3>Instructor</h3>
<div class="instructor"> <div class="instructor">
<div class="instructor-title">{{instructor.full_name}}</div> <div class="instructor-title">{{instructor.full_name}}</div>
<div class="instructor-subtitle">Created {{instructor.course_count}} courses</div> <div class="instructor-subtitle">Created {{instructor.get_course_count()}} courses</div>
</div> </div>
{% endmacro %} {% endmacro %}
@@ -11,7 +11,7 @@
{% for m in mentors %} {% for m in mentors %}
<div class="instructor"> <div class="instructor">
<div class="instructor-title">{{m.full_name}}</div> <div class="instructor-title">{{m.full_name}}</div>
<div class="instructor-subtitle">Mentored {{m.batch_count}} batches</div> <div class="instructor-subtitle">Mentored {{m.get_batch_count()}} batches</div>
</div> </div>
{% endfor %} {% endfor %}
{% if not is_mentor %} {% if not is_mentor %}