X-Git-Url: https://git.auder.net/?a=blobdiff_plain;f=public%2Fjavascripts%2Fassessment.js;h=95cd4d74ca5cffe0c9ca479f8206a661a1b7553d;hb=db5571d67f6b6b3b841fa1eeaf2099cecf2bcffc;hp=c574f1fa76253f8a14b7ce1c343353b792afc725;hpb=cc7c0f5e225138cd1ba29e872d4e36fa79a67a59;p=qomet.git diff --git a/public/javascripts/assessment.js b/public/javascripts/assessment.js index c574f1f..95cd4d7 100644 --- a/public/javascripts/assessment.js +++ b/public/javascripts/assessment.js @@ -1,6 +1,3 @@ -// TODO: if display == "all", les envois devraient être non définitifs (possibilité de corriger) -// Et, blur sur une (sous-)question devrait envoyer la version courante de la sous-question - let socket = null; //monitor answers in real time if (assessment.mode == "secure" && !checkWindowSize()) @@ -11,15 +8,16 @@ function checkWindowSize() // NOTE: temporarily accept smartphone (security hole: pretend being a smartphone on desktop browser...) if (navigator.userAgent.match(/(iPhone|iPod|iPad|Android|BlackBerry)/)) return true; - return window.innerWidth == screen.width && window.innerHeight == screen.height; + // 3 is arbitrary, but a small tolerance is required (e.g. in Firefox) + return window.innerWidth >= screen.width-3 && window.innerHeight >= screen.height-3; }; new Vue({ el: "#assessment", data: { assessment: assessment, - inputs: [ ], //student's answers - student: { }, //filled later + answers: { }, //filled later with answering parameters + student: { }, //filled later (name, password) // Stage 0: unauthenticated (number), // 1: authenticated (got a name, unvalidated) // 2: locked: password set, exam started @@ -27,207 +25,7 @@ new Vue({ // 4: show answers stage: assessment.mode != "open" ? 0 : 1, remainingTime: 0, //global, in seconds - }, - components: { - "statements": { - props: ['assessment','inputs','student','stage'], - data: function() { - return { - index: 0, //current question index in assessment.indices - }; - }, - // TODO: general render function for nested exercises - // TODO: with answer if stage==4 : class "wrong" if ticked AND stage==4 AND received answers - // class "right" if stage == 4 AND received answers (background-color: red / green) - // There should be a questions navigator below, or next (visible if display=='all') - // Full questions tree is rendered, but some parts hidden depending on display settings - render(h) { - let self = this; - let questions = assessment.questions.map( (q,i) => { - let questionContent = [ ]; - questionContent.push( - h( - "div", - { - "class": { - wording: true, - }, - domProps: { - innerHTML: q.wording, - }, - } - ) - ); - let optionsOrder = _.range(q.options.length); - if (!q.fixed) - optionsOrder = _.shuffle(optionsOrder); - let optionList = [ ]; - optionsOrder.forEach( idx => { - let option = [ ]; - option.push( - h( - "input", - { - domProps: { - checked: this.inputs[i][idx], - }, - attrs: { - id: this.inputId(i,idx), - type: "checkbox", - }, - on: { - change: e => { this.inputs[i][idx] = e.target.checked; }, - }, - }, - ) - ); - option.push( - h( - "label", - { - domProps: { - innerHTML: q.options[idx], - }, - attrs: { - "for": this.inputId(i,idx), - }, - } - ) - ); - optionList.push( - h( - "div", - { - "class": { - option: true, - choiceCorrect: this.stage == 4 && assessment.questions[i].answer.includes(idx), - choiceWrong: this.stage == 4 && this.inputs[i][idx] && !assessment.questions[i].answer.includes(idx), - }, - }, - option - ) - ); - }); - questionContent.push( - h( - "div", - { - "class": { - optionList: true, - }, - }, - optionList - ) - ); - return h( - "div", - { - "class": { - "question": true, - "hide": this.stage == 2 && assessment.display == 'one' && assessment.indices[this.index] != i, - }, - }, - questionContent - ); - }); - if (this.stage == 2) - { - // TODO: one button per question - questions.unshift( - h( - "button", - { - "class": { - "waves-effect": true, - "waves-light": true, - "btn": true, - }, - style: { - "display": "block", - "margin-left": "auto", - "margin-right": "auto", - }, - on: { - click: () => this.sendAnswer(assessment.indices[this.index]), - }, - }, - "Send" - ) - ); - } - return h( - "div", - { - attrs: { - id: "statements", - }, - }, - questions - ); - }, - mounted: function() { - if (assessment.mode != "secure") - return; - window.addEventListener("keydown", e => { - // (Try to) Ignore F11 + F12 (avoid accidental window resize) - // NOTE: in Chromium at least, exiting fullscreen mode with F11 cannot be prevented. - // Workaround: disable key at higher level. Possible xbindkey config: - // "false" - // m:0x10 + c:95 - // Mod2 + F11 - if ([122,123].includes(e.keyCode)) - e.preventDefault(); - }, false); - window.addEventListener("blur", () => { - this.trySendCurrentAnswer(); - document.location.href= "/noblur"; - }, false); - window.addEventListener("resize", e => { - this.trySendCurrentAnswer(); - document.location.href= "/fullscreen"; - }, false); - }, - methods: { - inputId: function(i,j) { - return "q" + i + "_" + "input" + j; - }, - trySendCurrentAnswer: function() { - if (this.stage == 2) - this.sendAnswer(assessment.indices[this.index]); - }, - // stage 2 - sendAnswer: function(realIndex) { - if (this.index == assessment.questions.length - 1) - this.$emit("gameover"); - else - this.index++; - if (assessment.mode == "open") - return; //only local - let answerData = { - aid: assessment._id, - answer: JSON.stringify({ - index:realIndex.toString(), - input:this.inputs[realIndex] - .map( (tf,i) => { return {val:tf,idx:i}; } ) - .filter( item => { return item.val; }) - .map( item => { return item.idx; }) - }), - number: this.student.number, - password: this.student.password, - }; - $.ajax("/send/answer", { - method: "GET", - data: answerData, - dataType: "json", - success: ret => { - if (!!ret.errmsg) - return alert(ret.errmsg); - //socket.emit(message.newAnswer, answer); - }, - }); - }, - }, - }, + warnMsg: "", }, computed: { countdown: function() { @@ -235,13 +33,76 @@ new Vue({ let minutes = Math.floor(this.remainingTime / 60); return this.padWithZero(minutes) + ":" + this.padWithZero(seconds); }, + showAnswers: function() { + return this.stage == 4; + }, + }, + mounted: function() { + $(".modal").modal(); + if (assessment.mode != "open") + { + window.addEventListener("keydown", e => { + // Ignore F12 (avoid accidental window resize due to devtools) + // NOTE: in Chromium at least, fullscreen mode exit with F11 cannot be prevented. + // Workaround: disable key at higher level. Possible xbindkey config: + // "false" + // m:0x10 + c:95 + // Mod2 + F11 + if (e.keyCode == 123) + e.preventDefault(); + }, false); + } + window.addEventListener("blur", () => { + if (!socket) + return; + if (assessment.mode == "secure") + { + this.trySendCurrentAnswer(); + document.location.href= "/noblur"; + } + else if (assessment.mode == "exam") + socket.emit(message.studentBlur, {number:this.student.number}); + }, false); + if (assessment.mode == "exam") + { + window.addEventListener("focus", () => { + if (!socket) + return; + socket.emit(message.studentFocus, {number:this.student.number}); + }, false); + } + window.addEventListener("resize", e => { + if (!socket) + return; + if (assessment.mode == "secure") + { + this.trySendCurrentAnswer(); + document.location.href= "/fullscreen"; + } + else if (assessment.mode == "exam") + { + if (checkWindowSize()) + socket.emit(message.studentFullscreen, {number:this.student.number}); + else + socket.emit(message.studentResize, {number:this.student.number}); + } + }, false); }, methods: { + // In case of AJAX errors + showWarning: function(message) { + this.warnMsg = message; + $("#warning").modal("open"); + }, padWithZero: function(x) { if (x < 10) return "0" + x; return x; }, + trySendCurrentAnswer: function() { + if (this.stage == 2) + this.sendAnswer(); + }, // stage 0 --> 1 getStudent: function(cb) { $.ajax("/get/student", { @@ -253,7 +114,7 @@ new Vue({ dataType: "json", success: s => { if (!!s.errmsg) - return alert(s.errmsg); + return this.showWarning(s.errmsg); this.stage = 1; this.student = s.student; Vue.nextTick( () => { Materialize.updateTextFields(); }); @@ -268,34 +129,40 @@ new Vue({ }, // stage 1 --> 2 (get all questions, set password) startAssessment: function() { - let initializeStage2 = questions => { + let initializeStage2 = (questions,paper) => { $("#leftButton, #rightButton").hide(); if (assessment.time > 0) { - this.remainingTime = assessment.time * 60; + const deltaTime = !!paper ? Date.now() - paper.startTime : 0; + this.remainingTime = assessment.time * 60 - Math.round(deltaTime / 1000); this.runTimer(); } // Initialize structured answer(s) based on questions type and nesting (TODO: more general) if (!!questions) assessment.questions = questions; + this.answers.inputs = [ ]; for (let q of assessment.questions) - this.inputs.push( _(q.options.length).times( _.constant(false) ) ); - assessment.indices = assessment.fixed - ? _.range(assessment.questions.length) - : _.shuffle( _.range(assessment.questions.length) ); + this.answers.inputs.push( _(q.options.length).times( _.constant(false) ) ); + if (!paper) + { + this.answers.indices = assessment.fixed + ? _.range(assessment.questions.length) + : _.shuffle( _.range(assessment.questions.length) ); + } + else + { + // Resuming + let indices = paper.inputs.map( input => { return input.index; }); + let remainingIndices = _.difference( _.range(assessment.questions.length).map(String), indices ); + this.answers.indices = indices.concat( _.shuffle(remainingIndices) ); + } + this.answers.index = !!paper ? paper.inputs.length : 0; + this.answers.displayAll = assessment.display == "all"; + this.answers.showSolution = false; this.stage = 2; - Vue.nextTick( () => { - // Run Prism + MathJax on questions text - $("#statements").find("code[class^=language-]").each( (i,elem) => { - Prism.highlightElement(elem); - }); - MathJax.Hub.Queue(["Typeset",MathJax.Hub,"statements"]); - }); }; if (assessment.mode == "open") return initializeStage2(); - // TODO: if existing password cookie: get stored answers (papers[number cookie]), inject (inputs), set index+indices - // (instead of following ajax call) $.ajax("/start/assessment", { method: "GET", data: { @@ -305,18 +172,24 @@ new Vue({ dataType: "json", success: s => { if (!!s.errmsg) - return alert(s.errmsg); - this.student.password = s.password; - // Got password: students answers locked to this page until potential teacher - // action (power failure, computer down, ...) - // TODO: set password cookie - // TODO: password also exchanged by sockets to check identity - //socket = io.connect("/" + assessment.name, { - // query: "number=" + this.student.number + "&password=" + this.password - //}); - //socket.on(message.allAnswers, this.setAnswers); - //socket.on("disconnect", () => { }); //TODO: notify monitor (highlight red), redirect - initializeStage2(s.questions); + return this.showWarning(s.errmsg); + if (!!s.paper) + { + // Resuming: receive stored answers + startTime + this.student.password = s.paper.password; + this.answers.inputs = s.paper.inputs.map( inp => { return inp.input; }); + } + else + { + this.student.password = s.password; + // Got password: students answers locked to this page until potential teacher + // action (power failure, computer down, ...) + } + socket = io.connect("/", { + query: "aid=" + assessment._id + "&number=" + this.student.number + "&password=" + this.student.password + }); + socket.on(message.allAnswers, this.setAnswers); + initializeStage2(s.questions, s.paper); }, }); }, @@ -327,11 +200,57 @@ new Vue({ let self = this; setInterval( function() { self.remainingTime--; - if (self.remainingTime <= 0 || self.stage >= 4) - self.endAssessment(); + if (self.remainingTime <= 0) + { + if (self.stage == 2) + self.endAssessment(); clearInterval(this); + } }, 1000); }, + // stage 2 + sendOneAnswer: function() { + const realIndex = this.answers.indices[this.answers.index]; + let gotoNext = () => { + if (this.answers.index == assessment.questions.length - 1) + this.endAssessment(); + else + this.answers.index++; + this.$children[0].$forceUpdate(); //TODO: bad HACK, and shouldn't be required... + }; + if (assessment.mode == "open") + return gotoNext(); //only local + let answerData = { + aid: assessment._id, + answer: JSON.stringify({ + index: realIndex.toString(), + input: this.answers.inputs[realIndex] + .map( (tf,i) => { return {val:tf,idx:i}; } ) + .filter( item => { return item.val; }) + .map( item => { return item.idx; }) + }), + number: this.student.number, + password: this.student.password, + }; + $.ajax("/send/answer", { + method: "GET", + data: answerData, + dataType: "json", + success: ret => { + if (!!ret.errmsg) + return this.showWarning(ret.errmsg); + gotoNext(); + socket.emit(message.newAnswer, answerData); + }, + }); + }, + // TODO: I don't like that + sending should not be definitive in exam mode with display = all + sendAnswer: function() { + if (assessment.display == "one") + this.sendOneAnswer(); + else + assessment.questions.forEach(this.sendOneAnswer); + }, // stage 2 --> 3 (or 4) // from a message by statements component, or time over endAssessment: function() { @@ -340,6 +259,8 @@ new Vue({ if (assessment.mode == "open") { this.stage = 4; + this.answers.showSolution = true; + this.answers.displayAll = true; return; } $.ajax("/end/assessment", { @@ -352,20 +273,22 @@ new Vue({ dataType: "json", success: ret => { if (!!ret.errmsg) - return alert(ret.errmsg); - assessment.conclusion = ret.conclusion; + return this.showWarning(ret.errmsg); this.stage = 3; delete this.student["password"]; //unable to send new answers now - //socket.disconnect(); - //socket = null; }, }); }, // stage 3 --> 4 (on socket message "feedback") - setAnswers: function(answers) { + setAnswers: function(m) { + const answers = JSON.parse(m.answers); for (let i=0; i