blob: 9327175e1187b2dfd8371b2ea3cf611803b29572 [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
4
scmmishra11925292018-11-20 17:38:01 +05305# LMS Utils to Update State for Vue Store
6@frappe.whitelist()
7def get_program_enrollments():
scmmishrab3154ef2018-12-06 20:13:20 +05308 student = utils.get_current_student()
9 if student == None:
10 return None
scmmishra11925292018-11-20 17:38:01 +053011 try:
scmmishrab3154ef2018-12-06 20:13:20 +053012 student = frappe.get_doc("Student", student)
scmmishra11925292018-11-20 17:38:01 +053013 return student.get_program_enrollments()
14 except:
15 return None
16
17@frappe.whitelist()
18def get_all_course_enrollments():
19 student = utils.get_current_student()
20 if student == None:
21 return None
22 try:
23 student = frappe.get_doc("Student", student)
24 return student.get_all_course_enrollments()
25 except:
26 return None
27
28# Vue Client Functions
scmmishra35bf5612018-11-13 16:36:22 +053029@frappe.whitelist(allow_guest=True)
30def get_portal_details():
scmmishrababb68d2018-11-19 16:13:21 +053031 """
32 Returns portal details from Education Settings Doctype. This contains the Title and Description for LMS amoung other things.
33 """
scmmishra35bf5612018-11-13 16:36:22 +053034 settings = frappe.get_doc("Education Settings")
35 title = settings.portal_title
36 description = settings.description
37 return dict(title=title, description=description)
38
scmmishra35bf5612018-11-13 16:36:22 +053039@frappe.whitelist(allow_guest=True)
40def get_featured_programs():
41 featured_program_names = frappe.get_all("Program", filters={"is_published": True, "is_featured": True})
42 if featured_program_names:
scmmishrababb68d2018-11-19 16:13:21 +053043 featured_list = [utils.get_program(program['name']) for program in featured_program_names]
scmmishra35bf5612018-11-13 16:36:22 +053044 return featured_list
45 else:
46 return None
47
scmmishra85c2fee2018-11-14 14:23:06 +053048@frappe.whitelist(allow_guest=True)
49def get_all_programs():
50 program_names = frappe.get_all("Program", filters={"is_published": True})
51 if program_names:
scmmishrababb68d2018-11-19 16:13:21 +053052 featured_list = [utils.get_program(program['name']) for program in program_names]
scmmishra85c2fee2018-11-14 14:23:06 +053053 return featured_list
54 else:
55 return None
56
scmmishra35bf5612018-11-13 16:36:22 +053057@frappe.whitelist(allow_guest=True)
58def get_program_details(program_name):
59 try:
60 program = frappe.get_doc('Program', program_name)
61 return program
62 except:
63 return None
64
scmmishra35bf5612018-11-13 16:36:22 +053065# Functions to get program & course details
66@frappe.whitelist(allow_guest=True)
67def get_courses(program_name):
68 program = frappe.get_doc('Program', program_name)
69 courses = program.get_course_list()
scmmishrafdbabde2018-11-22 15:33:30 +053070 return courses
scmmishra35bf5612018-11-13 16:36:22 +053071
scmmishra35bf5612018-11-13 16:36:22 +053072@frappe.whitelist()
scmmishrada39da62018-12-13 11:51:31 +053073def get_next_content(current_content, current_content_type, topic):
scmmishra35bf5612018-11-13 16:36:22 +053074 if frappe.session.user == "Guest":
75 return None
scmmishrada39da62018-12-13 11:51:31 +053076 topic = frappe.get_doc("Topic", topic)
77 content_list = [{'content_type':item.doctype, 'content':item.name} for item in topic.get_contents()]
78 current_index = content_list.index({'content': current_content, 'content_type': current_content_type})
scmmishra35bf5612018-11-13 16:36:22 +053079 try:
80 return content_list[current_index + 1]
81 except IndexError:
82 return None
83
84def get_quiz_with_answers(quiz_name):
85 try:
86 quiz = frappe.get_doc("Quiz", quiz_name).get_questions()
87 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]
88 return quiz_output
89 except:
90 frappe.throw("Quiz {0} does not exist".format(quiz_name))
91 return None
92
93@frappe.whitelist()
94def get_quiz_without_answers(quiz_name):
95 try:
96 quiz = frappe.get_doc("Quiz", quiz_name).get_questions()
97 quiz_output = [{'name':question.name, 'question':question.question, 'options':[{'name': option.name, 'option':option.option} for option in question.options]} for question in quiz]
98 return quiz_output
99 except:
100 frappe.throw("Quiz {0} does not exist".format(quiz_name))
101 return None
102
103@frappe.whitelist()
scmmishraab8fc8c2019-02-28 16:42:25 +0530104def evaluate_quiz(course, quiz_response, quiz_name):
scmmishra35bf5612018-11-13 16:36:22 +0530105 """LMS Function: Evaluates a simple multiple choice quiz.
scmmishra35bf5612018-11-13 16:36:22 +0530106 :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.
107 """
108 import json
109 quiz_response = json.loads(quiz_response)
110 quiz = frappe.get_doc("Quiz", quiz_name)
scmmishraab8fc8c2019-02-28 16:42:25 +0530111 enrollment = utils.get_course_enrollment(course).name
scmmishra35bf5612018-11-13 16:36:22 +0530112 answers, score, status = quiz.evaluate(quiz_response, quiz_name)
113
114 result = {k: ('Correct' if v else 'Wrong') for k,v in answers.items()}
115 result_data = []
116 for key in answers:
117 item = {}
118 item['question'] = key
119 item['quiz_result'] = result[key]
120 try:
121 item['selected_option'] = frappe.get_value('Options', quiz_response[key], 'option')
122 except:
123 item['selected_option'] = "Unattempted"
124 result_data.append(item)
125 # result_data = [{'question': key, 'selected_option': frappe.get_value('Options', quiz_response[key], 'option'), 'quiz_result': result[key]} for key in answers]
126
127 add_quiz_activity(enrollment, quiz_name, result_data, score, status)
128 return(score)
129
scmmishrababb68d2018-11-19 16:13:21 +0530130def add_quiz_activity(enrollment, quiz_name, result_data, score, status):
131 quiz_activity = frappe.get_doc({
132 "doctype": "Quiz Activity",
133 "enrollment": enrollment,
134 "quiz": quiz_name,
135 "activity_date": frappe.utils.datetime.datetime.now(),
136 "result": result_data,
137 "score": score,
138 "status": status
139 })
140 quiz_activity.save()
141 frappe.db.commit()
scmmishra35bf5612018-11-13 16:36:22 +0530142
143@frappe.whitelist()
scmmishra35bf5612018-11-13 16:36:22 +0530144def enroll_in_program(program_name):
scmmishrababb68d2018-11-19 16:13:21 +0530145 if(not utils.get_current_student()):
scmmishra082e3c92018-11-23 17:16:22 +0530146 utils.create_student()
scmmishrababb68d2018-11-19 16:13:21 +0530147 student = frappe.get_doc("Student", utils.get_current_student())
scmmishra35bf5612018-11-13 16:36:22 +0530148 program_enrollment = student.enroll_in_program(program_name)
scmmishra0a4902f2018-11-15 11:16:53 +0530149 return program_name
scmmishra35bf5612018-11-13 16:36:22 +0530150
scmmishra95bb3c02019-02-26 16:49:58 +0530151# Academty Activity
scmmishra35bf5612018-11-13 16:36:22 +0530152@frappe.whitelist()
scmmishraa592f702018-11-20 18:37:01 +0530153def add_activity(course, content_type, content):
154 enrollment = utils.get_course_enrollment(course)
155 if(utils.check_activity_exists(enrollment.name, content_type, content)):
scmmishra35bf5612018-11-13 16:36:22 +0530156 pass
157 else:
158 activity = frappe.get_doc({
159 "doctype": "Course Activity",
scmmishraa592f702018-11-20 18:37:01 +0530160 "enrollment": enrollment.name,
scmmishra35bf5612018-11-13 16:36:22 +0530161 "content_type": content_type,
162 "content": content,
163 "activity_date": frappe.utils.datetime.datetime.now()
164 })
165 activity.save()
166 frappe.db.commit()
167
scmmishrafdbabde2018-11-22 15:33:30 +0530168@frappe.whitelist()
scmmishra97c994f2018-11-26 14:41:15 +0530169def get_course_meta(course_name, program_name):
scmmishrada39da62018-12-13 11:51:31 +0530170 """
171 Return the porgress of a course in a program as well as the content to continue from.
scmmishra95bb3c02019-02-26 16:49:58 +0530172 :param course_name:
173 :param program_name:
scmmishrada39da62018-12-13 11:51:31 +0530174 """
scmmishrafdbabde2018-11-22 15:33:30 +0530175 course_enrollment = utils.get_course_enrollment(course_name)
scmmishrab3154ef2018-12-06 20:13:20 +0530176 program_enrollment = utils.get_program_enrollment(program_name)
scmmishrada39da62018-12-13 11:51:31 +0530177 student = frappe.get_doc("Student", utils.get_current_student())
scmmishrab3154ef2018-12-06 20:13:20 +0530178 if not program_enrollment:
179 return None
scmmishra97c994f2018-11-26 14:41:15 +0530180 if not course_enrollment:
181 utils.enroll_in_course(course_name, program_name)
scmmishrada39da62018-12-13 11:51:31 +0530182 progress = course_enrollment.get_progress(student)
183 count = sum([activity['is_complete'] for activity in progress])
scmmishraa592f702018-11-20 18:37:01 +0530184 if count == 0:
185 return {'flag':'Start Course', 'content_type': progress[0]['content_type'], 'content': progress[0]['content']}
186 elif count == len(progress):
scmmishra798f13b2018-12-05 10:35:48 +0530187 return {'flag':'Completed', 'content_type': progress[0]['content_type'], 'content': progress[0]['content']}
scmmishraa592f702018-11-20 18:37:01 +0530188 elif count < len(progress):
189 next_item = next(item for item in progress if item['is_complete']==False)
190 return {'flag':'Continue', 'content_type': next_item['content_type'], 'content': next_item['content']}
scmmishra95bb3c02019-02-26 16:49:58 +0530191
scmmishraaffbfe72018-11-26 11:59:25 +0530192@frappe.whitelist()
scmmishrade5f71a2019-02-28 15:40:49 +0530193def get_topic_meta(topic_name, course_name):
scmmishra7e1678e2019-02-26 17:11:01 +0530194 """
195 Return the porgress of a course in a program as well as the content to continue from.
196 :param topic_name:
197 :param course_name:
scmmishra7e1678e2019-02-26 17:11:01 +0530198 """
scmmishra7e1678e2019-02-26 17:11:01 +0530199 course_enrollment = utils.get_course_enrollment(course_name)
scmmishra7e1678e2019-02-26 17:11:01 +0530200 student = frappe.get_doc("Student", utils.get_current_student())
scmmishrade5f71a2019-02-28 15:40:49 +0530201 topic = frappe.get_doc("Topic", topic_name)
202 progress = student.get_topic_progress(course_enrollment.name, topic)
scmmishra3726f8a2019-02-28 16:33:53 +0530203 if not progress:
204 return { 'flag':'Start Topic', 'content_type': None, 'content': None }
scmmishra7e1678e2019-02-26 17:11:01 +0530205 count = sum([activity['is_complete'] for activity in progress])
206 if count == 0:
scmmishrade5f71a2019-02-28 15:40:49 +0530207 return {'flag':'Start Topic', 'content_type': progress[0]['content_type'], 'content': progress[0]['content']}
scmmishra7e1678e2019-02-26 17:11:01 +0530208 elif count == len(progress):
209 return {'flag':'Completed', 'content_type': progress[0]['content_type'], 'content': progress[0]['content']}
210 elif count < len(progress):
211 next_item = next(item for item in progress if item['is_complete']==False)
212 return {'flag':'Continue', 'content_type': next_item['content_type'], 'content': next_item['content']}
213
214@frappe.whitelist()
scmmishra201fec32018-11-26 16:52:45 +0530215def get_program_progress(program_name):
scmmishra29558512018-11-26 19:16:54 +0530216 import math
scmmishraaffbfe72018-11-26 11:59:25 +0530217 program = frappe.get_doc("Program", program_name)
scmmishrada39da62018-12-13 11:51:31 +0530218 program_enrollment = utils.get_program_enrollment(program_name)
scmmishra201fec32018-11-26 16:52:45 +0530219 program_meta = {}
scmmishra97c994f2018-11-26 14:41:15 +0530220 if not program_enrollment:
221 return None
222 else:
scmmishra201fec32018-11-26 16:52:45 +0530223 progress = []
scmmishra97c994f2018-11-26 14:41:15 +0530224 for course in program.get_all_children():
scmmishra201fec32018-11-26 16:52:45 +0530225 meta = get_course_meta(course.course, program_name)
226 is_complete = False
scmmishra4d102292018-12-07 17:41:40 +0530227 if meta['flag'] == "Completed":
scmmishra201fec32018-11-26 16:52:45 +0530228 is_complete = True
229 progress.append({'course_name': course.course_name, 'name': course.course, 'is_complete': is_complete})
230 program_meta['progress'] = progress
231 program_meta['name'] = program_name
232 program_meta['program'] = program.program_name
scmmishra29558512018-11-26 19:16:54 +0530233 program_meta['percentage'] = math.ceil((sum([item['is_complete'] for item in progress] * 100)/len(progress)))
234 return program_meta
235
236@frappe.whitelist()
237def get_joining_date():
238 student = frappe.get_doc("Student", utils.get_current_student())
scmmishra4d102292018-12-07 17:41:40 +0530239 return student.joining_date
240
241@frappe.whitelist()
242def get_quiz_progress(program_name):
243 program = frappe.get_doc("Program", program_name)
scmmishrada39da62018-12-13 11:51:31 +0530244 program_enrollment = utils.get_program_enrollment(program_name)
scmmishra4d102292018-12-07 17:41:40 +0530245 quiz_meta = frappe._dict()
scmmishrada39da62018-12-13 11:51:31 +0530246 student = frappe.get_doc("Student", utils.get_current_student())
scmmishra4d102292018-12-07 17:41:40 +0530247 if not program_enrollment:
248 return None
249 else:
250 progress_list = []
251 for course in program.get_all_children():
252 course_enrollment = utils.get_course_enrollment(course.course)
scmmishrada39da62018-12-13 11:51:31 +0530253 meta = course_enrollment.get_progress(student)
scmmishra4d102292018-12-07 17:41:40 +0530254 for progress_item in meta:
scmmishra87df23b2018-12-09 19:57:12 +0530255 # if progress_item['content_type'] == "Quiz" and progress_item['is_complete'] == True:
scmmishra4d102292018-12-07 17:41:40 +0530256 if progress_item['content_type'] == "Quiz":
scmmishra87df23b2018-12-09 19:57:12 +0530257 progress_item['course'] = course.course_name
scmmishra4d102292018-12-07 17:41:40 +0530258 progress_list.append(progress_item)
259 quiz_meta.quiz_attempt = progress_list
260 quiz_meta.name = program_name
261 quiz_meta.program = program.program_name
scmmishra95bb3c02019-02-26 16:49:58 +0530262 return quiz_meta
263
264
265@frappe.whitelist(allow_guest=True)
266def get_course_details(course_name):
267 try:
268 course = frappe.get_doc('Course', course_name)
269 return course
270 except:
271 return None
272
273# Functions to get program & course details
274@frappe.whitelist(allow_guest=True)
275def get_topics(course_name):
276 course = frappe.get_doc('Course', course_name)
277 topics = course.get_topics()
scmmishra23880ae2019-02-27 12:09:57 +0530278 return topics
279
280@frappe.whitelist()
281def get_content(type, content):
282 try:
283 return frappe.get_doc(type, content)
284 except:
285 return None