2019-06-03 12:57:38 +05:30
|
|
|
class Quiz {
|
|
|
|
constructor(wrapper, options) {
|
|
|
|
this.wrapper = wrapper;
|
|
|
|
Object.assign(this, options);
|
|
|
|
this.questions = []
|
|
|
|
this.refresh();
|
|
|
|
}
|
|
|
|
|
|
|
|
refresh() {
|
|
|
|
this.get_quiz();
|
|
|
|
}
|
|
|
|
|
|
|
|
get_quiz() {
|
|
|
|
frappe.call('erpnext.education.utils.get_quiz', {
|
|
|
|
quiz_name: this.name,
|
|
|
|
course: this.course
|
|
|
|
}).then(res => {
|
|
|
|
this.make(res.message)
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
make(data) {
|
|
|
|
data.questions.forEach(question_data => {
|
|
|
|
let question_wrapper = document.createElement('div');
|
|
|
|
let question = new Question({
|
|
|
|
wrapper: question_wrapper,
|
|
|
|
...question_data
|
|
|
|
});
|
|
|
|
this.questions.push(question)
|
|
|
|
this.wrapper.appendChild(question_wrapper);
|
|
|
|
})
|
2020-04-03 10:08:19 +05:30
|
|
|
if (data.activity && data.activity.is_complete) {
|
2019-06-03 12:57:38 +05:30
|
|
|
this.disable()
|
|
|
|
let indicator = 'red'
|
|
|
|
let message = 'Your are not allowed to attempt the quiz again.'
|
|
|
|
if (data.activity.result == 'Pass') {
|
|
|
|
indicator = 'green'
|
|
|
|
message = 'You have already cleared the quiz.'
|
|
|
|
}
|
|
|
|
|
|
|
|
this.set_quiz_footer(message, indicator, data.activity.score)
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
this.make_actions();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
make_actions() {
|
|
|
|
const button = document.createElement("button");
|
|
|
|
button.classList.add("btn", "btn-primary", "mt-5", "mr-2");
|
|
|
|
|
|
|
|
button.id = 'submit-button';
|
|
|
|
button.innerText = 'Submit';
|
|
|
|
button.onclick = () => this.submit();
|
|
|
|
this.submit_btn = button
|
|
|
|
this.wrapper.appendChild(button);
|
|
|
|
}
|
|
|
|
|
|
|
|
submit() {
|
|
|
|
this.submit_btn.innerText = 'Evaluating..'
|
|
|
|
this.submit_btn.disabled = true
|
|
|
|
this.disable()
|
|
|
|
frappe.call('erpnext.education.utils.evaluate_quiz', {
|
|
|
|
quiz_name: this.name,
|
|
|
|
quiz_response: this.get_selected(),
|
2019-06-03 14:40:52 +05:30
|
|
|
course: this.course,
|
|
|
|
program: this.program
|
2019-06-03 12:57:38 +05:30
|
|
|
}).then(res => {
|
|
|
|
this.submit_btn.remove()
|
|
|
|
if (!res.message) {
|
2019-07-03 15:15:08 +05:30
|
|
|
frappe.throw(__("Something went wrong while evaluating the quiz."))
|
2019-06-03 12:57:38 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
let indicator = 'red'
|
|
|
|
let message = 'Fail'
|
|
|
|
if (res.message.status == 'Pass') {
|
|
|
|
indicator = 'green'
|
|
|
|
message = 'Congratulations, you cleared the quiz.'
|
|
|
|
}
|
|
|
|
|
|
|
|
this.set_quiz_footer(message, indicator, res.message.score)
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
set_quiz_footer(message, indicator, score) {
|
|
|
|
const div = document.createElement("div");
|
|
|
|
div.classList.add("mt-5");
|
|
|
|
div.innerHTML = `<div class="row">
|
|
|
|
<div class="col-md-8">
|
|
|
|
<h4>${message}</h4>
|
|
|
|
<h5 class="text-muted"><span class="indicator ${indicator}">Score: ${score}/100</span></h5>
|
|
|
|
</div>
|
|
|
|
<div class="col-md-4">
|
|
|
|
<a href="${this.next_url}" class="btn btn-primary pull-right">${this.quiz_exit_button}</a>
|
|
|
|
</div>
|
|
|
|
</div>`
|
|
|
|
|
|
|
|
this.wrapper.appendChild(div)
|
|
|
|
}
|
|
|
|
|
|
|
|
disable() {
|
|
|
|
this.questions.forEach(que => que.disable())
|
|
|
|
}
|
|
|
|
|
|
|
|
get_selected() {
|
|
|
|
let que = {}
|
|
|
|
this.questions.forEach(question => {
|
|
|
|
que[question.name] = question.get_selected()
|
|
|
|
})
|
|
|
|
return que
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class Question {
|
|
|
|
constructor(opts) {
|
|
|
|
Object.assign(this, opts);
|
|
|
|
this.make();
|
|
|
|
}
|
|
|
|
|
|
|
|
make() {
|
|
|
|
this.make_question()
|
|
|
|
this.make_options()
|
|
|
|
}
|
|
|
|
|
|
|
|
get_selected() {
|
|
|
|
let selected = this.options.filter(opt => opt.input.checked)
|
|
|
|
if (this.type == 'Single Correct Answer') {
|
|
|
|
if (selected[0]) return selected[0].name
|
|
|
|
}
|
|
|
|
if (this.type == 'Multiple Correct Answer') {
|
|
|
|
return selected.map(opt => opt.name)
|
|
|
|
}
|
|
|
|
return null
|
|
|
|
}
|
|
|
|
|
|
|
|
disable() {
|
|
|
|
let selected = this.options.forEach(opt => opt.input.disabled = true)
|
|
|
|
}
|
|
|
|
|
|
|
|
make_question() {
|
|
|
|
let question_wrapper = document.createElement('h5');
|
|
|
|
question_wrapper.classList.add('mt-3');
|
|
|
|
question_wrapper.innerText = this.question;
|
|
|
|
this.wrapper.appendChild(question_wrapper);
|
|
|
|
}
|
|
|
|
|
|
|
|
make_options() {
|
|
|
|
let make_input = (name, value) => {
|
|
|
|
let input = document.createElement('input');
|
|
|
|
input.id = name;
|
|
|
|
input.name = this.name;
|
|
|
|
input.value = value;
|
|
|
|
input.type = 'radio';
|
|
|
|
if (this.type == 'Multiple Correct Answer')
|
|
|
|
input.type = 'checkbox';
|
|
|
|
input.classList.add('form-check-input');
|
|
|
|
return input;
|
|
|
|
}
|
|
|
|
|
|
|
|
let make_label = function(name, value) {
|
|
|
|
let label = document.createElement('label');
|
|
|
|
label.classList.add('form-check-label');
|
|
|
|
label.htmlFor = name;
|
|
|
|
label.innerText = value;
|
|
|
|
return label
|
|
|
|
}
|
|
|
|
|
|
|
|
let make_option = function (wrapper, option) {
|
|
|
|
let option_div = document.createElement('div')
|
|
|
|
option_div.classList.add('form-check', 'pb-1')
|
|
|
|
let input = make_input(option.name, option.option);
|
|
|
|
let label = make_label(option.name, option.option);
|
|
|
|
option_div.appendChild(input)
|
|
|
|
option_div.appendChild(label)
|
|
|
|
wrapper.appendChild(option_div)
|
|
|
|
return {input: input, ...option}
|
|
|
|
}
|
|
|
|
|
|
|
|
let options_wrapper = document.createElement('div')
|
|
|
|
options_wrapper.classList.add('ml-2')
|
|
|
|
let option_list = []
|
|
|
|
this.options.forEach(opt => option_list.push(make_option(options_wrapper, opt)))
|
|
|
|
this.options = option_list
|
|
|
|
this.wrapper.appendChild(options_wrapper)
|
|
|
|
}
|
|
|
|
}
|