blob: 7561d73d207ad7f461fb5862bcf8771909f92144 [file] [log] [blame]
scmmishra35bf5612018-11-13 16:36:22 +05301from __future__ import unicode_literals
scmmishrababb68d2018-11-19 16:13:21 +05302import erpnext.education.utils as utils
scmmishra35bf5612018-11-13 16:36:22 +05303import frappe
scmmishra936f36c2019-04-22 18:55:19 +05304from frappe import _
scmmishra35bf5612018-11-13 16:36:22 +05305
scmmishra11925292018-11-20 17:38:01 +05306# LMS Utils to Update State for Vue Store
7@frappe.whitelist()
8def get_program_enrollments():
scmmishrab3154ef2018-12-06 20:13:20 +05309 student = utils.get_current_student()
10 if student == None:
11 return None
scmmishra327334a2019-04-22 12:03:17 +053012 return student.get_program_enrollments()
scmmishra11925292018-11-20 17:38:01 +053013
14@frappe.whitelist()
15def get_all_course_enrollments():
16 student = utils.get_current_student()
17 if student == None:
18 return None
scmmishra327334a2019-04-22 12:03:17 +053019 return student.get_all_course_enrollments()
scmmishra11925292018-11-20 17:38:01 +053020
21# Vue Client Functions
scmmishra35bf5612018-11-13 16:36:22 +053022@frappe.whitelist(allow_guest=True)
23def get_portal_details():
scmmishrababb68d2018-11-19 16:13:21 +053024 """
25 Returns portal details from Education Settings Doctype. This contains the Title and Description for LMS amoung other things.
26 """
scmmishrac22eef22019-03-18 15:36:49 +053027 from erpnext import get_default_company
28
scmmishra35bf5612018-11-13 16:36:22 +053029 settings = frappe.get_doc("Education Settings")
scmmishrac22eef22019-03-18 15:36:49 +053030 title = settings.portal_title or get_default_company()
scmmishra35bf5612018-11-13 16:36:22 +053031 description = settings.description
32 return dict(title=title, description=description)
33
scmmishra35bf5612018-11-13 16:36:22 +053034@frappe.whitelist(allow_guest=True)
35def get_featured_programs():
36 featured_program_names = frappe.get_all("Program", filters={"is_published": True, "is_featured": True})
37 if featured_program_names:
scmmishrada2c90c2019-04-22 12:19:39 +053038 featured_list = [utils.get_program_and_enrollment_status(program['name']) for program in featured_program_names]
scmmishra35bf5612018-11-13 16:36:22 +053039 return featured_list
40 else:
scmmishra621cad82019-03-18 15:40:47 +053041 return get_all_programs()[:2]
scmmishra35bf5612018-11-13 16:36:22 +053042
scmmishra85c2fee2018-11-14 14:23:06 +053043@frappe.whitelist(allow_guest=True)
44def get_all_programs():
45 program_names = frappe.get_all("Program", filters={"is_published": True})
46 if program_names:
scmmishrada2c90c2019-04-22 12:19:39 +053047 program_list = [utils.get_program_and_enrollment_status(program['name']) for program in program_names]
scmmishra621cad82019-03-18 15:40:47 +053048 return program_list
scmmishra85c2fee2018-11-14 14:23:06 +053049
scmmishra35bf5612018-11-13 16:36:22 +053050@frappe.whitelist(allow_guest=True)
scmmishrada2c90c2019-04-22 12:19:39 +053051def get_program(program_name):
scmmishra35bf5612018-11-13 16:36:22 +053052 try:
scmmishrada2c90c2019-04-22 12:19:39 +053053 return frappe.get_doc('Program', program_name)
54 except frappe.DoesNotExistError:
55 frappe.throw(_("Program {0} does not exist.".format(program_name)))
scmmishra35bf5612018-11-13 16:36:22 +053056
scmmishra35bf5612018-11-13 16:36:22 +053057# Functions to get program & course details
58@frappe.whitelist(allow_guest=True)
59def get_courses(program_name):
60 program = frappe.get_doc('Program', program_name)
61 courses = program.get_course_list()
scmmishrafdbabde2018-11-22 15:33:30 +053062 return courses
scmmishra35bf5612018-11-13 16:36:22 +053063
scmmishra35bf5612018-11-13 16:36:22 +053064@frappe.whitelist()
scmmishrada39da62018-12-13 11:51:31 +053065def get_next_content(current_content, current_content_type, topic):
scmmishra35bf5612018-11-13 16:36:22 +053066 if frappe.session.user == "Guest":
67 return None
scmmishrada39da62018-12-13 11:51:31 +053068 topic = frappe.get_doc("Topic", topic)
69 content_list = [{'content_type':item.doctype, 'content':item.name} for item in topic.get_contents()]
70 current_index = content_list.index({'content': current_content, 'content_type': current_content_type})
scmmishra35bf5612018-11-13 16:36:22 +053071 try:
72 return content_list[current_index + 1]
73 except IndexError:
74 return None
75
76def get_quiz_with_answers(quiz_name):
77 try:
78 quiz = frappe.get_doc("Quiz", quiz_name).get_questions()
79 quiz_output = [{'name':question.name, 'question':question.question, 'options':[{'name': option.name, 'option':option.option, 'is_correct':option.is_correct} for option in question.options]} for question in quiz]
80 return quiz_output
81 except:
82 frappe.throw("Quiz {0} does not exist".format(quiz_name))
83 return None
84
85@frappe.whitelist()
scmmishra2b7e1582019-03-29 12:45:08 +053086def get_quiz_without_answers(quiz_name, course_name):
scmmishra35bf5612018-11-13 16:36:22 +053087 try:
scmmishra2b7e1582019-03-29 12:45:08 +053088 quiz = frappe.get_doc("Quiz", quiz_name)
89 questions = quiz.get_questions()
scmmishra35bf5612018-11-13 16:36:22 +053090 except:
91 frappe.throw("Quiz {0} does not exist".format(quiz_name))
92 return None
93
scmmishra3673b252019-04-22 18:10:19 +053094 if utils.check_super_access():
95 quiz_output = [{'name':question.name, 'question':question.question, 'type': question.type, 'options':[{'name': option.name, 'option':option.option} for option in question.options]} for question in questions]
96 return { 'quizData': quiz_output, 'status': None}
97
scmmishra2b7e1582019-03-29 12:45:08 +053098 enrollment = utils.get_course_enrollment(course_name).name
scmmishra66d23922019-04-22 12:54:43 +053099 quiz_progress = {}
100 quiz_progress['is_complete'], quiz_progress['score'], quiz_progress['result'] = utils.check_quiz_completion(quiz, enrollment)
scmmishra2b7e1582019-03-29 12:45:08 +0530101 quiz_output = [{'name':question.name, 'question':question.question, 'type': question.type, 'options':[{'name': option.name, 'option':option.option} for option in question.options]} for question in questions]
scmmishra66d23922019-04-22 12:54:43 +0530102 return { 'quizData': quiz_output, 'status': quiz_progress}
scmmishra2b7e1582019-03-29 12:45:08 +0530103
scmmishra35bf5612018-11-13 16:36:22 +0530104@frappe.whitelist()
scmmishraab8fc8c2019-02-28 16:42:25 +0530105def evaluate_quiz(course, quiz_response, quiz_name):
scmmishra35bf5612018-11-13 16:36:22 +0530106 """LMS Function: Evaluates a simple multiple choice quiz.
scmmishra66d23922019-04-22 12:54:43 +0530107 :param course: name of the course
scmmishra35bf5612018-11-13 16:36:22 +0530108 :param quiz_response: contains user selected choices for a quiz in the form of a string formatted as a dictionary. The function uses `json.loads()` to convert it to a python dictionary.
scmmishra66d23922019-04-22 12:54:43 +0530109 :param quiz_name: Name of the quiz attempted
scmmishra35bf5612018-11-13 16:36:22 +0530110 """
111 import json
112 quiz_response = json.loads(quiz_response)
113 quiz = frappe.get_doc("Quiz", quiz_name)
114 answers, score, status = quiz.evaluate(quiz_response, quiz_name)
scmmishraac8d3472019-04-22 14:30:55 +0530115 print(answers)
scmmishra35bf5612018-11-13 16:36:22 +0530116
scmmishraac8d3472019-04-22 14:30:55 +0530117 course_enrollment = utils.get_course_enrollment(course)
118 if course_enrollment:
119 course_enrollment.add_quiz_activity(quiz_name, quiz_response, answers, score, status)
scmmishra35bf5612018-11-13 16:36:22 +0530120
scmmishraac8d3472019-04-22 14:30:55 +0530121 return score
scmmishra35bf5612018-11-13 16:36:22 +0530122
123@frappe.whitelist()
scmmishra35bf5612018-11-13 16:36:22 +0530124def enroll_in_program(program_name):
scmmishra327334a2019-04-22 12:03:17 +0530125 student = utils.get_current_student()
126 if not student:
scmmishra5079e6c2019-04-22 14:46:26 +0530127 student = utils.create_student_from_current_user()
scmmishra35bf5612018-11-13 16:36:22 +0530128 program_enrollment = student.enroll_in_program(program_name)
scmmishra0a4902f2018-11-15 11:16:53 +0530129 return program_name
scmmishra35bf5612018-11-13 16:36:22 +0530130
scmmishraac8d3472019-04-22 14:30:55 +0530131# Academdy Activity
scmmishra35bf5612018-11-13 16:36:22 +0530132@frappe.whitelist()
scmmishraa592f702018-11-20 18:37:01 +0530133def add_activity(course, content_type, content):
scmmishra000e7062019-03-19 12:30:43 +0530134 if not utils.get_current_student():
135 return
scmmishraa592f702018-11-20 18:37:01 +0530136 enrollment = utils.get_course_enrollment(course)
scmmishraac8d3472019-04-22 14:30:55 +0530137 enrollment.add_activity(content_type, content)
scmmishra35bf5612018-11-13 16:36:22 +0530138
scmmishrafdbabde2018-11-22 15:33:30 +0530139@frappe.whitelist()
scmmishrad5973fe2019-04-22 12:05:22 +0530140def get_student_course_details(course_name, program_name):
scmmishrada39da62018-12-13 11:51:31 +0530141 """
142 Return the porgress of a course in a program as well as the content to continue from.
scmmishra95bb3c02019-02-26 16:49:58 +0530143 :param course_name:
144 :param program_name:
scmmishrada39da62018-12-13 11:51:31 +0530145 """
scmmishra327334a2019-04-22 12:03:17 +0530146 student = utils.get_current_student()
147 if not student:
scmmishra000e7062019-03-19 12:30:43 +0530148 return {'flag':'Start Course' }
scmmishra3673b252019-04-22 18:10:19 +0530149
scmmishrafdbabde2018-11-22 15:33:30 +0530150 course_enrollment = utils.get_course_enrollment(course_name)
scmmishrab3154ef2018-12-06 20:13:20 +0530151 program_enrollment = utils.get_program_enrollment(program_name)
scmmishra3673b252019-04-22 18:10:19 +0530152
scmmishrab3154ef2018-12-06 20:13:20 +0530153 if not program_enrollment:
154 return None
scmmishra3673b252019-04-22 18:10:19 +0530155
scmmishra97c994f2018-11-26 14:41:15 +0530156 if not course_enrollment:
scmmishra5c646e62019-03-18 18:37:26 +0530157 course_enrollment = utils.enroll_in_course(course_name, program_name)
scmmishra3673b252019-04-22 18:10:19 +0530158
scmmishrada39da62018-12-13 11:51:31 +0530159 progress = course_enrollment.get_progress(student)
160 count = sum([activity['is_complete'] for activity in progress])
scmmishraa592f702018-11-20 18:37:01 +0530161 if count == 0:
scmmishra000e7062019-03-19 12:30:43 +0530162 return {'flag':'Start Course'}
scmmishraa592f702018-11-20 18:37:01 +0530163 elif count == len(progress):
scmmishra000e7062019-03-19 12:30:43 +0530164 return {'flag':'Completed'}
scmmishraa592f702018-11-20 18:37:01 +0530165 elif count < len(progress):
166 next_item = next(item for item in progress if item['is_complete']==False)
scmmishra000e7062019-03-19 12:30:43 +0530167 return {'flag':'Continue'}
scmmishra95bb3c02019-02-26 16:49:58 +0530168
scmmishraaffbfe72018-11-26 11:59:25 +0530169@frappe.whitelist()
scmmishrad78c3262019-04-22 12:06:28 +0530170def get_student_topic_details(topic_name, course_name):
scmmishra7e1678e2019-02-26 17:11:01 +0530171 """
172 Return the porgress of a course in a program as well as the content to continue from.
173 :param topic_name:
174 :param course_name:
scmmishra7e1678e2019-02-26 17:11:01 +0530175 """
scmmishra000e7062019-03-19 12:30:43 +0530176 topic = frappe.get_doc("Topic", topic_name)
scmmishra327334a2019-04-22 12:03:17 +0530177 student = utils.get_current_student()
178 if not student:
scmmishra000e7062019-03-19 12:30:43 +0530179 topic_content = topic.get_all_children()
180 if topic_content:
181 return {'flag':'Start Course', 'content_type': topic_content[0].content_type, 'content': topic_content[0].content}
182 else:
183 return None
scmmishra7e1678e2019-02-26 17:11:01 +0530184 course_enrollment = utils.get_course_enrollment(course_name)
scmmishrade5f71a2019-02-28 15:40:49 +0530185 progress = student.get_topic_progress(course_enrollment.name, topic)
scmmishra3726f8a2019-02-28 16:33:53 +0530186 if not progress:
187 return { 'flag':'Start Topic', 'content_type': None, 'content': None }
scmmishra7e1678e2019-02-26 17:11:01 +0530188 count = sum([activity['is_complete'] for activity in progress])
189 if count == 0:
scmmishrade5f71a2019-02-28 15:40:49 +0530190 return {'flag':'Start Topic', 'content_type': progress[0]['content_type'], 'content': progress[0]['content']}
scmmishra7e1678e2019-02-26 17:11:01 +0530191 elif count == len(progress):
192 return {'flag':'Completed', 'content_type': progress[0]['content_type'], 'content': progress[0]['content']}
193 elif count < len(progress):
194 next_item = next(item for item in progress if item['is_complete']==False)
195 return {'flag':'Continue', 'content_type': next_item['content_type'], 'content': next_item['content']}
196
197@frappe.whitelist()
scmmishra201fec32018-11-26 16:52:45 +0530198def get_program_progress(program_name):
scmmishra68b32032019-04-22 18:40:46 +0530199 program_enrollment = frappe.get_doc("Program Enrollment", utils.get_program_enrollment(program_name))
scmmishra97c994f2018-11-26 14:41:15 +0530200 if not program_enrollment:
201 return None
202 else:
scmmishra68b32032019-04-22 18:40:46 +0530203 return program_enrollment.get_program_progress()
scmmishra29558512018-11-26 19:16:54 +0530204
205@frappe.whitelist()
206def get_joining_date():
scmmishra5079e6c2019-04-22 14:46:26 +0530207 student = utils.get_current_student()
208 if student:
scmmishra9b7ac3e2019-03-28 14:47:22 +0530209 return student.joining_date
scmmishra4d102292018-12-07 17:41:40 +0530210
211@frappe.whitelist()
scmmishra68b32032019-04-22 18:40:46 +0530212def get_quiz_progress_of_program(program_name):
213 program_enrollment = frappe.get_doc("Program Enrollment", utils.get_program_enrollment(program_name))
scmmishra4d102292018-12-07 17:41:40 +0530214 if not program_enrollment:
215 return None
216 else:
scmmishra68b32032019-04-22 18:40:46 +0530217 return program_enrollment.get_quiz_progress()
scmmishra95bb3c02019-02-26 16:49:58 +0530218
219
220@frappe.whitelist(allow_guest=True)
221def get_course_details(course_name):
222 try:
scmmishra64d2fe02019-04-22 12:30:08 +0530223 course = frappe.get_doc('Course', course_name)
scmmishra95bb3c02019-02-26 16:49:58 +0530224 return course
225 except:
226 return None
227
228# Functions to get program & course details
229@frappe.whitelist(allow_guest=True)
230def get_topics(course_name):
scmmishra4add8262019-04-22 12:28:13 +0530231 try:
232 course = frappe.get_doc('Course', course_name)
233 return course.get_topics()
234 except frappe.DoesNotExistError:
235 frappe.throw(_("Course {0} does not exist.".format(course_name)))
scmmishra23880ae2019-02-27 12:09:57 +0530236
237@frappe.whitelist()
scmmishra4add8262019-04-22 12:28:13 +0530238def get_content(content_type, content):
scmmishra23880ae2019-02-27 12:09:57 +0530239 try:
scmmishra4add8262019-04-22 12:28:13 +0530240 return frappe.get_doc(content_type, content)
241 except frappe.DoesNotExistError:
242 frappe.throw(_("{0} {1} does not exist.".format(content_type, content)))