WikiLerni/front/public/JS/quiz.app.js

1104 lines
139 KiB
JavaScript

/******/ (function(modules) { // webpackBootstrap
/******/ // The module cache
/******/ var installedModules = {};
/******/
/******/ // The require function
/******/ function __webpack_require__(moduleId) {
/******/
/******/ // Check if module is in cache
/******/ if(installedModules[moduleId]) {
/******/ return installedModules[moduleId].exports;
/******/ }
/******/ // Create a new module (and put it into the cache)
/******/ var module = installedModules[moduleId] = {
/******/ i: moduleId,
/******/ l: false,
/******/ exports: {}
/******/ };
/******/
/******/ // Execute the module function
/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
/******/
/******/ // Flag the module as loaded
/******/ module.l = true;
/******/
/******/ // Return the exports of the module
/******/ return module.exports;
/******/ }
/******/
/******/
/******/ // expose the modules object (__webpack_modules__)
/******/ __webpack_require__.m = modules;
/******/
/******/ // expose the module cache
/******/ __webpack_require__.c = installedModules;
/******/
/******/ // define getter function for harmony exports
/******/ __webpack_require__.d = function(exports, name, getter) {
/******/ if(!__webpack_require__.o(exports, name)) {
/******/ Object.defineProperty(exports, name, { enumerable: true, get: getter });
/******/ }
/******/ };
/******/
/******/ // define __esModule on exports
/******/ __webpack_require__.r = function(exports) {
/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
/******/ }
/******/ Object.defineProperty(exports, '__esModule', { value: true });
/******/ };
/******/
/******/ // create a fake namespace object
/******/ // mode & 1: value is a module id, require it
/******/ // mode & 2: merge all properties of value into the ns
/******/ // mode & 4: return value when already ns object
/******/ // mode & 8|1: behave like require
/******/ __webpack_require__.t = function(value, mode) {
/******/ if(mode & 1) value = __webpack_require__(value);
/******/ if(mode & 8) return value;
/******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;
/******/ var ns = Object.create(null);
/******/ __webpack_require__.r(ns);
/******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value });
/******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));
/******/ return ns;
/******/ };
/******/
/******/ // getDefaultExport function for compatibility with non-harmony modules
/******/ __webpack_require__.n = function(module) {
/******/ var getter = module && module.__esModule ?
/******/ function getDefault() { return module['default']; } :
/******/ function getModuleExports() { return module; };
/******/ __webpack_require__.d(getter, 'a', getter);
/******/ return getter;
/******/ };
/******/
/******/ // Object.prototype.hasOwnProperty.call
/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
/******/
/******/ // __webpack_public_path__
/******/ __webpack_require__.p = "";
/******/
/******/
/******/ // Load entry module and return exports
/******/ return __webpack_require__(__webpack_require__.s = "./src/quiz.js");
/******/ })
/************************************************************************/
/******/ ({
/***/ "../lang sync recursive ^\\.\\/.*\\/answer$":
/*!*************************************!*\
!*** ../lang sync ^\.\/.*\/answer$ ***!
\*************************************/
/*! no static exports found */
/***/ (function(module, exports, __webpack_require__) {
var map = {
"./fr/answer": "../lang/fr/answer.js"
};
function webpackContext(req) {
var id = webpackContextResolve(req);
return __webpack_require__(id);
}
function webpackContextResolve(req) {
if(!__webpack_require__.o(map, req)) {
var e = new Error("Cannot find module '" + req + "'");
e.code = 'MODULE_NOT_FOUND';
throw e;
}
return map[req];
}
webpackContext.keys = function webpackContextKeys() {
return Object.keys(map);
};
webpackContext.resolve = webpackContextResolve;
module.exports = webpackContext;
webpackContext.id = "../lang sync recursive ^\\.\\/.*\\/answer$";
/***/ }),
/***/ "../lang sync recursive ^\\.\\/.*\\/general$":
/*!**************************************!*\
!*** ../lang sync ^\.\/.*\/general$ ***!
\**************************************/
/*! no static exports found */
/***/ (function(module, exports, __webpack_require__) {
var map = {
"./fr/general": "../lang/fr/general.js"
};
function webpackContext(req) {
var id = webpackContextResolve(req);
return __webpack_require__(id);
}
function webpackContextResolve(req) {
if(!__webpack_require__.o(map, req)) {
var e = new Error("Cannot find module '" + req + "'");
e.code = 'MODULE_NOT_FOUND';
throw e;
}
return map[req];
}
webpackContext.keys = function webpackContextKeys() {
return Object.keys(map);
};
webpackContext.resolve = webpackContextResolve;
module.exports = webpackContext;
webpackContext.id = "../lang sync recursive ^\\.\\/.*\\/general$";
/***/ }),
/***/ "../lang/fr/answer.js":
/*!****************************!*\
!*** ../lang/fr/answer.js ***!
\****************************/
/*! no static exports found */
/***/ (function(module, exports) {
module.exports = {
needNumberUserResponses: "Le nombre de questions auxquelles l'utilisateur a répondu doit être fourni.",
needIntegerNumberUserResponses: "Le nombre de questions auxquelles l'utilisateur a répondu doit être un nombre entier.",
needMinNumberUserResponses: "Le nombre de questions auxquelles l'utilisateur a répondu ne peut être inférieur à 1.",
needMaxNumberUserResponses: "Le nombre de questions auxquelles l'utilisateur a répondu ne peut être aussi élevé.",
needNumberCorrectResponses: "Le nombre de réponses correctes doit être fourni.",
needIntegerNumberCorrectResponses: "Le nombre de réponses correctes doit être un nombre entier.",
needMinNumberCorrectResponses: "Le nombre de réponses correctes ne peut être négatif.",
needMaxNumberCorrectResponses: "Le nombre de réponses correctes ne peut être supérieur au nombre de questions.",
needIntegerNumberSecondesResponse: "La durée de la réponse doit être un nombre entier de secondes.",
needMinNumberSecondesResponse: "La durée de la réponse ne peut être négative.",
checkResponsesOuputFail: "Vous avez répondu en DURATION secondes et avez NBCORRECTANSWERS bonne(s) réponse(s) sur NBQUESTIONS. C'est certain, vous ferez mieux la prochaine fois !",
checkResponsesOuputMedium: "Vous avez répondu en DURATION secondes et avez NBCORRECTANSWERS bonne(s) réponse(s) sur NBQUESTIONS. C'est pas mal du tout !",
checkResponsesOuputSuccess: "Vous avez répondu en DURATION secondes et avez NBCORRECTANSWERS bonne(s) réponse(s) sur NBQUESTIONS. Bravo ! Rien ne vous échappe !",
wantToSaveResponses: "Si vous le souhaitez, vous pouvez sauvegarder votre résultat en créant votre compte.\nCela vous permettra de tester de nouveau ce quiz dans l'avenir pour voir si vous avez mémoriser les bonnes réponses. Vous pourrez ainsi aussi recevoir régulièrement de nouveaux quizs.",
responseSavedMessage: "Votre résultat a été enregistré. <a href='/#URL'>Accèder à tous vos quizs</a>.",
responseSavedError: "Cependant une erreur a été rencontrée durant l'enregistrement de votre résultat. <a href='/#URL'>Accèder à tous vos quizs</a>.",
noQuestionnaireWithoudAnswer: "Bravo ! Vous avez répondu à tous les quizs qui vous on été proposés !<br>Il y en a des nouveaux publiés régulièrement et qui vous seront proposés.<br>En attendant vous pouvez peut-être essayer de répondre de nouveau à certain quiz pour voir si vous vous souvenez des bonnes réponses ?",
nbQuestionnaireWithoudAnswer: "Il y a #NB quizs qui vous ont été proposés et auxquels vous n'avez pas répondu. Voici les derniers :!",
statsUser: "Vous avez enregistré NBANSWERS réponses à <b>NBQUESTIONNAIRES questionnaires différents</b> sur les NBTOTQUESTIONNAIRES proposés par le site.<br>En moyenne, vous avez mis AVGDURATION secondes à répondre et avez <b>correctement répondu à AVGCORRECTANSWERS % des questions</b>.",
previousAnswersTitle: "Vos résultats précédents",
previousAnswersStats: "En moyenne, vous avez répondu à ce quiz en AVGDURATION secondes, en ayant <b>AVGCORRECTANSWERS % de bonnes réponses</b>.",
previousAnswersLine: "Le DATEANSWER, vous avez répondu correctement à NBCORRECTANSWERS questions en AVGDURATION secondes."
};
/***/ }),
/***/ "../lang/fr/general.js":
/*!*****************************!*\
!*** ../lang/fr/general.js ***!
\*****************************/
/*! no static exports found */
/***/ (function(module, exports) {
module.exports = {
siteHTMLTitle: "WikiLerni : qu'allez-vous apprendre aujourd'hui ?",
siteMetaDescription: "Chaque jour, testez vos connaissances et apprenez de nouvelles choses avec WikiLerni.",
scriptTimingInfo: "Durée de la réponse : SCRIPT_TIMING millisecondes, route : SCRIPT_URL",
scriptTimingAlert: "*** Script lent : SCRIPT_TIMING millisecondes, route : SCRIPT_URL",
serverError: "Désolé. Une erreur imprévue est survenue. Si cela persiste, n'hésitez à prévenir l'administrateur du site.",
serverErrorAdmin: "Bug de l'application :",
neededParams: "Des paramètres nécessaires manquants sont manquants.",
badUrl: "Tentative d'accès à une page n'existant pas :",
notValidFormat: "Format non valide.",
notAllowed: "Vous n'avez pas les droits nécessaires pour cette action.",
notRequired: "Facultatif.",
updateBtnTxt: "Modifier",
addBtnTxt: "Ajouter",
deleteBtnTxt: "Supprimer",
addOkMessage: "Les données ont bien été enregistrées.",
updateOkMessage: "La mise à jour à jour a bien été enregistrée.",
deleteOkMessage: "La suppression a bien été enregistrée.",
failAuth: "Erreur d'authentification.",
failAuthHeader: "Absence de header Authorization.",
failAuthToken: "Token invalide ou utilisateur non trouvé.",
failAuthId: "Identifiant non valide : ",
failAuthCron: "Tentative de lancement d'un cron sans le bon token.",
previousPage: "Page précédente",
nextPage: "Page suivante",
btnLinkToQuestionnaire: "Aller au quiz !",
statsAdmin: "Durant les dernières 24h : NB_USERS_24H comptes ont été créés, NB_SUBSCRIPTIONS_24H validés et NB_USERS_DELETED_24H supprimés. NB_ANSWERS_24H réponses aux quizs ont été enregistrées.<br>En tout, il y a : NB_USERS_TOT comptes, dont NB_SUBSCRIPTIONS_TOT validés et NB_SUBSCRIPTIONS_PREMIUM comptes prémium. NB_ANSWERS_TOT réponses aux quizs ont été enregistrées.<br>Parmi les NB_USERS_DELETED_TOT comptes supprimés, NB_USERS_DELETED_VALIDED avaient validé leur compte et NB_USERS_DELETED_PREMIUM avaient souscrit un compte prémium."
};
/***/ }),
/***/ "../tools/main.js":
/*!************************!*\
!*** ../tools/main.js ***!
\************************/
/*! no static exports found */
/***/ (function(module, exports) {
function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
function _defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } }
function _createClass(Constructor, protoProps, staticProps) { if (protoProps) _defineProperties(Constructor.prototype, protoProps); if (staticProps) _defineProperties(Constructor, staticProps); return Constructor; }
// Quelques fonctions utiles pour les chaînes
var Tool = /*#__PURE__*/function () {
function Tool() {
_classCallCheck(this, Tool);
}
_createClass(Tool, null, [{
key: "isEmpty",
value: function isEmpty(myVar) {
if (myVar === undefined || myVar === null) return true;else {
myVar += ""; // si autre chose qu'une chaîne envoyé...
myVar = myVar.trim();
if (myVar === "") return true;else return false;
}
}
}, {
key: "trimIfNotNull",
value: function trimIfNotNull(myString) {
if (Tool.isEmpty(myString)) myString = null;else {
myString += ""; // si autre chose qu'une chaîne envoyé...
myString = myString.trim();
}
return myString;
}
}, {
key: "shortenIfLongerThan",
value: function shortenIfLongerThan(myString, max) {
myString += ""; // au cas où cela ne serait pas une chaîne...
if (myString.length > max) myString = myString.substring(0, max - 3) + "...";
return myString;
} // source : https://stackoverflow.com/questions/15604140/replace-multiple-strings-with-multiple-other-strings
}, {
key: "replaceAll",
value: function replaceAll(myString, mapObj) {
var replaceElts = new RegExp(Object.keys(mapObj).join("|"), "gi");
return myString.replace(replaceElts, function (matched) {
return mapObj[matched];
});
} // source : https://developer.mozilla.org/fr/docs/Web/JavaScript/Reference/Objets_globaux/Math/random
}, {
key: "getRandomInt",
value: function getRandomInt(min, max) {
min = Math.ceil(min);
max = Math.floor(max);
return Math.floor(Math.random() * (max - min)) + min;
} // à compléter : https://en.wikipedia.org/wiki/Date_format_by_country
}, {
key: "dateFormat",
value: function dateFormat(dateString) {
var lang = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : "fr";
if (Tool.isEmpty(dateString)) return "";
var myDate = new Date(dateString);
var myDay = myDate.getDate() + "";
if (myDay.length === 1) myDay = "0" + myDay;
var myMounth = myDate.getMonth() + 1 + "";
if (myMounth.length === 1) myMounth = "0" + myMounth;
var myYear = myDate.getFullYear();
if (lang === "fr") return myDay + "/" + myMounth + "/" + myYear;else if (lang === "form") // 2014-02-09
return myYear + "-" + myMounth + "-" + myDay;else return myMounth + "/" + myDay + "/" + myYear;
}
}]);
return Tool;
}();
module.exports = Tool;
/***/ }),
/***/ "./src/config/general.js":
/*!*******************************!*\
!*** ./src/config/general.js ***!
\*******************************/
/*! no static exports found */
/***/ (function(module, exports) {
module.exports = {
apiUrl: "http://localhost:3000/api",
usersGetConfigUrl: "/user/getconfig",
lang: "fr",
userHomePage: "accueil.html",
adminHomePage: "admin.html",
managerHomePage: "gestion.html",
subscribePage: "inscription.html",
connectionPage: "connexion.html",
accountPage: "compte.html",
questionnairesManagementPage: "gestion-quizs.html",
usersManagementPage: "gestion-utilisateurs.html",
nbQuestionnairesUserHomePage: 10,
illustrationDir: "/img/quizs/"
}; // à terme fusionner avec les déclarations backend concernant le front, tout étant saisi dans l'admin
/***/ }),
/***/ "./src/quiz.js":
/*!*********************!*\
!*** ./src/quiz.js ***!
\*********************/
/*! no exports provided */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony import */ var _tools_answers_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./tools/answers.js */ "./src/tools/answers.js");
/* harmony import */ var _tools_dom_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./tools/dom.js */ "./src/tools/dom.js");
/* harmony import */ var _tools_everywhere_js__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./tools/everywhere.js */ "./src/tools/everywhere.js");
/* harmony import */ var _tools_clientstorage_js__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./tools/clientstorage.js */ "./src/tools/clientstorage.js");
/* harmony import */ var _tools_forms_js__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ./tools/forms.js */ "./src/tools/forms.js");
/* harmony import */ var _tools_main__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ../../tools/main */ "../tools/main.js");
/* harmony import */ var _tools_main__WEBPACK_IMPORTED_MODULE_5___default = /*#__PURE__*/__webpack_require__.n(_tools_main__WEBPACK_IMPORTED_MODULE_5__);
/* harmony import */ var _tools_users_js__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! ./tools/users.js */ "./src/tools/users.js");
function asyncGeneratorStep(gen, resolve, reject, _next, _throw, key, arg) { try { var info = gen[key](arg); var value = info.value; } catch (error) { reject(error); return; } if (info.done) { resolve(value); } else { Promise.resolve(value).then(_next, _throw); } }
function _asyncToGenerator(fn) { return function () { var self = this, args = arguments; return new Promise(function (resolve, reject) { var gen = fn.apply(self, args); function _next(value) { asyncGeneratorStep(gen, resolve, reject, _next, _throw, "next", value); } function _throw(err) { asyncGeneratorStep(gen, resolve, reject, _next, _throw, "throw", err); } _next(undefined); }); }; }
// -- GESTION DU FORMULAIRE PERMETTANT D'AFFICHER ET RÉPONDRE À UN QUIZ
/// Il n'est pas nécessaire d'être connecté pour répondre au quiz et voir son résultat.
/// Mais si pas connecté, je propose à l'internaute de se connecter ou de créer un compte.
/// Dans ce but son résultat est alors stocké dans son navigateur
/// Si il est connecté, l'enregistrement de son résultat se fait automatiquement et ses précédents résultats affichés
// Fichier de configuration côté client :
var configFrontEnd = __webpack_require__(/*! ./config/general */ "./src/config/general.js"); // Importation des fonctions utiles au script :
// Dictionnaires :
var txt = __webpack_require__("../lang sync recursive ^\\.\\/.*\\/general$")("./" + configFrontEnd.lang + "/general");
var txtAnswers = __webpack_require__("../lang sync recursive ^\\.\\/.*\\/answer$")("./" + configFrontEnd.lang + "/answer"); // Principaux éléments du DOM manipulés :
var myForm = document.getElementById("questionnaire");
var divResponse = document.getElementById("response");
var btnShow = document.getElementById("showQuestionnaire");
var btnSubmit = document.getElementById("checkResponses");
var explanationsTitle = document.getElementById("explanationsTitle");
var explanationsContent = document.getElementById("explanationsContent");
Object(_tools_everywhere_js__WEBPACK_IMPORTED_MODULE_2__["helloDev"])(); // Chargement des configurations fournies par le backend + test de connexion de l'utilisateur :
var config, isConnected, user;
var initialise = /*#__PURE__*/function () {
var _ref = _asyncToGenerator( /*#__PURE__*/regeneratorRuntime.mark(function _callee() {
return regeneratorRuntime.wrap(function _callee$(_context) {
while (1) {
switch (_context.prev = _context.next) {
case 0:
_context.prev = 0;
_context.next = 3;
return Object(_tools_users_js__WEBPACK_IMPORTED_MODULE_6__["getConfig"])();
case 3:
config = _context.sent;
if (config) {
_context.next = 8;
break;
}
Object(_tools_dom_js__WEBPACK_IMPORTED_MODULE_1__["addElement"])(divResponse, "p", txt.serverError, "", ["error"]);
_context.next = 13;
break;
case 8:
btnShow.style.display = "inline";
_context.next = 11;
return Object(_tools_users_js__WEBPACK_IMPORTED_MODULE_6__["checkSession"])(config, ["user"]);
case 11:
isConnected = _context.sent;
// seuls les simples utilisateurs peuvent répondre aux quizs
// Si l'utilisateur est connecté, je vérifie sur il a déjà répondu à ce quiz et si oui affiche ses précédentes réponses dans l'encart servant à expliquer le topo aux nouveaux
if (isConnected) {
user = Object(_tools_clientstorage_js__WEBPACK_IMPORTED_MODULE_3__["getLocaly"])("user", true);
checkPreviousResponses(config, user);
}
case 13:
_context.next = 18;
break;
case 15:
_context.prev = 15;
_context.t0 = _context["catch"](0);
Object(_tools_dom_js__WEBPACK_IMPORTED_MODULE_1__["addElement"])(divResponse, "p", txt.serverError, "", ["error"]);
case 18:
case "end":
return _context.stop();
}
}
}, _callee, null, [[0, 15]]);
}));
return function initialise() {
return _ref.apply(this, arguments);
};
}();
initialise(); // Affichage du questionnaire si l'utilisateur clique sur le bouton ou si l'id du formulaire est passée par l'url
// Déclenche en même temps le chronomètre mesurant la durée de la réponse aux questions.
var showQuestionnaire = function showQuestionnaire() {
chronoBegin = Date.now();
myForm.style.display = "block";
btnShow.setAttribute("disabled", true);
window.location.assign("#questionnaire");
};
var chronoBegin = 0;
btnShow.addEventListener("click", function (e) {
try {
e.preventDefault();
showQuestionnaire();
} catch (e) {
Object(_tools_dom_js__WEBPACK_IMPORTED_MODULE_1__["addElement"])(divResponse, "p", txt.serverError, "", ["error"]);
}
});
if (location.hash != undefined && location.hash === "#questionnaire") showQuestionnaire(); // Traitement de l'envoi de sa réponse par l'utilisateur :
var answer = {};
myForm.addEventListener("submit", function (e) {
try {
e.preventDefault();
btnSubmit.setAttribute("disabled", true); // seulement un envoi à la fois :)
divResponse.innerHTML = ""; // supprime les éventuels messages déjà affichés
var userResponses = Object(_tools_forms_js__WEBPACK_IMPORTED_MODULE_4__["getDatasFromInputs"])(myForm);
answer.duration = Math.round((Date.now() - chronoBegin) / 1000);
answer.nbCorrectAnswers = 0;
answer.QuestionnaireId = document.getElementById("questionnaireId").value; // Récupére les réponses du quiz sur le serveur :
var xhr = new XMLHttpRequest();
xhr.open("GET", configFrontEnd.apiUrl + config.questionnaireRoutes + config.getQuestionnaireRoutes + "/" + userResponses.questionnaireId);
var idChoice, goodResponse;
xhr.onreadystatechange = function () {
if (this.readyState == XMLHttpRequest.DONE) {
var response = JSON.parse(this.responseText);
if (this.status === 200 && Array.isArray(response.Questions)) {
answer.nbQuestions = response.Questions.length; // Puis on passe en revue les réponses pour afficher celles qui sont correctes et détecter les erreurs
// Il peut y avoir plusieurs bonnes réponses pour une même question (QCM)
for (var i in response.Questions) {
goodResponse = true;
for (var j in response.Questions[i].Choices) {
idChoice = response.Questions[i].Choices[j].id;
if (response.Questions[i].Choices[j].isCorrect === true) {
document.getElementById("response_" + idChoice).parentNode.classList.add("success");
if (userResponses["response_" + idChoice] === undefined) // bonne réponse non sélectionnée
goodResponse = false;
} else {
if (userResponses["response_" + idChoice] === "on") // réponse cochée n'étant pas une des bonnes
{
goodResponse = false;
document.getElementById("response_" + idChoice).parentNode.classList.add("error");
}
}
}
if (goodResponse) answer.nbCorrectAnswers++;
} // Affichage du résultat, suivant si l'utilisateur est connecté ou pas et son score :
var getOuput = Object(_tools_answers_js__WEBPACK_IMPORTED_MODULE_0__["checkAnswerOuput"])(answer);
if (isConnected) {
// si l'utilisateur est connecté on enregistre son résultat sur le serveur.
var xhr2 = new XMLHttpRequest();
xhr2.open("POST", configFrontEnd.apiUrl + config.questionnaireRoutes + config.saveAnswersRoute);
xhr2.onreadystatechange = function () {
if (this.readyState == XMLHttpRequest.DONE) {
var response2 = JSON.parse(this.responseText);
if (this.status === 201 && response2.message) {
getOuput += "<br>" + response2.message.replace("#URL", configFrontEnd.userHomePage);
checkPreviousResponses(config, user);
} else getOuput += "<br>" + txtAnswers.responseSavedError.replace("#URL", configFrontEnd.userHomePage);
Object(_tools_dom_js__WEBPACK_IMPORTED_MODULE_1__["addElement"])(divResponse, "p", getOuput, "", ["information"]);
}
};
xhr2.setRequestHeader("Authorization", "Bearer " + user.token);
xhr2.setRequestHeader("Content-Type", "application/json");
answer.timeDifference = Object(_tools_users_js__WEBPACK_IMPORTED_MODULE_6__["getTimeDifference"])(config); // on en profite pour mettre les pendules à l'heure
xhr2.send(JSON.stringify(answer));
} else {
// sinon je l'enregistre côté client pour permettre de le retrouver au moment de la création du compte ou de la connexion
if (Object(_tools_answers_js__WEBPACK_IMPORTED_MODULE_0__["saveAnswer"])(answer)) {
getOuput += "<br>" + txtAnswers.wantToSaveResponses;
Object(_tools_dom_js__WEBPACK_IMPORTED_MODULE_1__["addElement"])(divResponse, "p", getOuput, "", ["information"]);
document.querySelector(".subcribeBtns").style.display = "block";
} else // pas la peine de proposer de créer un compte si le stockage local ne fonctionne pas
Object(_tools_dom_js__WEBPACK_IMPORTED_MODULE_1__["addElement"])(divResponse, "p", getOuput, "", ["information"]);
} // affichage des textes d'explications pour chaque question
var explanations = document.querySelectorAll(".help");
for (var _i in explanations) {
explanations[_i].style.display = "block";
} // affiche une erreur "TypeError: explanations[i].style is undefined" mais tout fonctionne ?
} else Object(_tools_dom_js__WEBPACK_IMPORTED_MODULE_1__["addElement"])(divResponse, "p", txt.serverError, "", ["error"]);
}
};
xhr.send();
} catch (e) {
Object(_tools_dom_js__WEBPACK_IMPORTED_MODULE_1__["addElement"])(divResponse, "p", txt.serverError, "", ["error"]);
}
}); // Fonction vérifiant les précédentes réponses de l'utilisateur
// Utile si connecté lors du premier chargement de la page, puis après une nouvelle réponse
var checkPreviousResponses = function checkPreviousResponses(config, user) {
var xhrPreviousRes = new XMLHttpRequest();
xhrPreviousRes.open("GET", configFrontEnd.apiUrl + config.questionnaireRoutes + config.getPreviousAnswers + user.id + "/" + document.getElementById("questionnaireId").value);
xhrPreviousRes.onreadystatechange = function () {
if (this.readyState == XMLHttpRequest.DONE) {
var response = JSON.parse(this.responseText);
if (this.status === 200) {
var nbResponses = response.length;
var previousAnswersContent = "";
if (nbResponses !== 0) {
Object(_tools_dom_js__WEBPACK_IMPORTED_MODULE_1__["addElement"])(explanationsTitle, "span", txtAnswers.previousAnswersTitle);
var totNbQuestions = 0,
totNbCorrectAnswers = 0,
totDuration = 0,
mapLineContent;
for (var i in response) {
totNbQuestions += response[i].nbQuestions;
totNbCorrectAnswers += response[i].nbCorrectAnswers;
totDuration += response[i].duration;
mapLineContent = {
DATEANSWER: Object(_tools_main__WEBPACK_IMPORTED_MODULE_5__["dateFormat"])(response[i].createdAt),
// revoir problème de la langue
NBCORRECTANSWERS: response[i].nbCorrectAnswers,
AVGDURATION: response[i].duration
};
previousAnswersContent += "<li>" + Object(_tools_main__WEBPACK_IMPORTED_MODULE_5__["replaceAll"])(txtAnswers.previousAnswersLine, mapLineContent) + "</li>";
}
mapLineContent = {
AVGDURATION: Math.round(totDuration / nbResponses),
AVGCORRECTANSWERS: Math.round(totNbCorrectAnswers / totNbQuestions * 100)
};
previousAnswersContent = "<h5>" + Object(_tools_main__WEBPACK_IMPORTED_MODULE_5__["replaceAll"])(txtAnswers.previousAnswersStats, mapLineContent) + "</h5>" + previousAnswersContent;
Object(_tools_dom_js__WEBPACK_IMPORTED_MODULE_1__["addElement"])(explanationsContent, "ul", previousAnswersContent);
}
} else Object(_tools_dom_js__WEBPACK_IMPORTED_MODULE_1__["addElement"])(divResponse, "p", txt.serverError, "", ["error"]);
}
};
xhrPreviousRes.setRequestHeader("Authorization", "Bearer " + user.token);
xhrPreviousRes.send();
};
/***/ }),
/***/ "./src/tools/answers.js":
/*!******************************!*\
!*** ./src/tools/answers.js ***!
\******************************/
/*! exports provided: saveAnswer, checkAnswerOuput */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "saveAnswer", function() { return saveAnswer; });
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "checkAnswerOuput", function() { return checkAnswerOuput; });
/* harmony import */ var _clientstorage_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./clientstorage.js */ "./src/tools/clientstorage.js");
/* harmony import */ var _tools_main__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../../../tools/main */ "../tools/main.js");
/* harmony import */ var _tools_main__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_tools_main__WEBPACK_IMPORTED_MODULE_1__);
var configFrontEnd = __webpack_require__(/*! ../config/general */ "./src/config/general.js");
var txt = __webpack_require__("../lang sync recursive ^\\.\\/.*\\/answer$")("./" + configFrontEnd.lang + "/answer"); // Enregistrement côté client du dernier résultat à un quiz en attendant d'être connecté
var saveAnswer = function saveAnswer(answer) {
if (!Object(_tools_main__WEBPACK_IMPORTED_MODULE_1__["isEmpty"])(answer.duration) && !Object(_tools_main__WEBPACK_IMPORTED_MODULE_1__["isEmpty"])(answer.nbCorrectAnswers) && !Object(_tools_main__WEBPACK_IMPORTED_MODULE_1__["isEmpty"])(answer.nbQuestions) && !Object(_tools_main__WEBPACK_IMPORTED_MODULE_1__["isEmpty"])(answer.QuestionnaireId)) {
Object(_clientstorage_js__WEBPACK_IMPORTED_MODULE_0__["saveLocaly"])("lastAnswer", answer);
return true;
} else return false;
}; // Retourne le texte suivant le nombre de bonnes réponses
var checkAnswerOuput = function checkAnswerOuput(answer) {
if (!Object(_tools_main__WEBPACK_IMPORTED_MODULE_1__["isEmpty"])(answer.duration) && !Object(_tools_main__WEBPACK_IMPORTED_MODULE_1__["isEmpty"])(answer.nbCorrectAnswers) && !Object(_tools_main__WEBPACK_IMPORTED_MODULE_1__["isEmpty"])(answer.nbQuestions)) {
var ratio = answer.nbCorrectAnswers / answer.nbQuestions;
var mapObj = {
DURATION: answer.duration,
NBCORRECTANSWERS: answer.nbCorrectAnswers,
NBQUESTIONS: answer.nbQuestions
};
var output = "";
if (ratio < 0.4) output = Object(_tools_main__WEBPACK_IMPORTED_MODULE_1__["replaceAll"])(txt.checkResponsesOuputFail, mapObj);else if (ratio < 0.8) output = Object(_tools_main__WEBPACK_IMPORTED_MODULE_1__["replaceAll"])(txt.checkResponsesOuputMedium, mapObj);else output = Object(_tools_main__WEBPACK_IMPORTED_MODULE_1__["replaceAll"])(txt.checkResponsesOuputSuccess, mapObj);
if (output) return output;else return "";
} else return "";
};
/*
export const checkSession = async (config) =>
{
return new Promise((resolve, reject) =>
{
if(isEmpty(localStorage.getItem("user")))
resolve(false);
else
{
const user=JSON.parse(localStorage.getItem("user"));
if(user.duration===undefined || user.duration < Date.now())
{
localStorage.removeItem("user");
resolve(false);
}
else
{
const xhr = new XMLHttpRequest();
xhr.open("GET", configFrontEnd.apiUrl+config.userRoutes+config.checkLoginRoute+user.token);
xhr.onload = () =>
{
let response=JSON.parse(xhr.responseText);
if (xhr.status === 200 && response.isValid && response.id != undefined)
{
if(response.id===user.id)
resolve(true);
else
{
localStorage.removeItem("user");
resolve(false);
}
}
else
{
localStorage.removeItem("user");
resolve(false);
}
}
xhr.onerror = () => reject(xhr.statusText);
xhr.send();
}
}
});
}*/
/***/ }),
/***/ "./src/tools/clientstorage.js":
/*!************************************!*\
!*** ./src/tools/clientstorage.js ***!
\************************************/
/*! exports provided: saveLocaly, getLocaly, removeLocaly */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "saveLocaly", function() { return saveLocaly; });
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "getLocaly", function() { return getLocaly; });
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "removeLocaly", function() { return removeLocaly; });
// FONCTIONS UTILES AU STOCKAGE LOCAL (SESSION, COOKIES, INDEXDB, ETC.)
// Revenir pour gérer le cas où local.storage n'est pas connu pour utiliser cookie
var saveLocaly = function saveLocaly(name, data) {
localStorage.setItem(name, JSON.stringify(data));
};
var getLocaly = function getLocaly(name) {
var json = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : false;
if (json) return JSON.parse(localStorage.getItem(name));else return localStorage.getItem(name);
};
var removeLocaly = function removeLocaly(name) {
localStorage.removeItem(name);
};
/***/ }),
/***/ "./src/tools/dom.js":
/*!**************************!*\
!*** ./src/tools/dom.js ***!
\**************************/
/*! exports provided: addElement */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "addElement", function() { return addElement; });
/* harmony import */ var _tools_main__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../../../tools/main */ "../tools/main.js");
/* harmony import */ var _tools_main__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_tools_main__WEBPACK_IMPORTED_MODULE_0__);
function _typeof(obj) { "@babel/helpers - typeof"; if (typeof Symbol === "function" && typeof Symbol.iterator === "symbol") { _typeof = function _typeof(obj) { return typeof obj; }; } else { _typeof = function _typeof(obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; }; } return _typeof(obj); }
// Fonction associant les attributs fournis à un champ de formulaire
var addElement = function addElement(eltParent, eltType) {
var eltContent = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : "";
var eltId = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : "";
var eltClass = arguments.length > 4 && arguments[4] !== undefined ? arguments[4] : [];
var eltAttributes = arguments.length > 5 && arguments[5] !== undefined ? arguments[5] : {};
var replace = arguments.length > 6 && arguments[6] !== undefined ? arguments[6] : true;
if (Object(_tools_main__WEBPACK_IMPORTED_MODULE_0__["isEmpty"])(eltType) || Object(_tools_main__WEBPACK_IMPORTED_MODULE_0__["isEmpty"])(eltParent)) return false;else {
var newElement = document.createElement(eltType);
if (!Object(_tools_main__WEBPACK_IMPORTED_MODULE_0__["isEmpty"])(eltId)) // tester si l'id n'est pas déjà utilisé dans le DOM ?
newElement.id = eltId;
if (Array.isArray(eltClass) && eltClass.length != 0) {
for (var i in eltClass) {
newElement.classList.add(eltClass[i]);
}
}
if (_typeof(eltAttributes) === "object") // !! tous les objets ne sont pas ok
{
for (var attributName in eltAttributes) {
newElement.setAttribute(attributName, eltAttributes[attributName]);
}
}
if (!Object(_tools_main__WEBPACK_IMPORTED_MODULE_0__["isEmpty"])(eltContent)) newElement.innerHTML = eltContent.replace(/\n/g, "<br>"); // innerHTML permet d'ajouter du texte ayant lui-même des balises, etc.
if (replace) eltParent.innerHTML = "";
eltParent.appendChild(newElement);
}
};
/***/ }),
/***/ "./src/tools/everywhere.js":
/*!*********************************!*\
!*** ./src/tools/everywhere.js ***!
\*********************************/
/*! exports provided: helloDev */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "helloDev", function() { return helloDev; });
// Ce script fournit des fonctions utilisées sur toutes les pages du site
var helloDev = function helloDev() {
console.log("**** Hello ami développeur :-)\n Le code de WikiLerni est libre et est lisible sur gitlab : \n Bonne lecture :-) \n Pour les suggestions d'amélioration ou questions : dev@wililerni.com ****");
return true;
};
/***/ }),
/***/ "./src/tools/forms.js":
/*!****************************!*\
!*** ./src/tools/forms.js ***!
\****************************/
/*! exports provided: setAttributesToInputs, getDatasFromInputs, empyForm, empyAndHideForm */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "setAttributesToInputs", function() { return setAttributesToInputs; });
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "getDatasFromInputs", function() { return getDatasFromInputs; });
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "empyForm", function() { return empyForm; });
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "empyAndHideForm", function() { return empyAndHideForm; });
/* harmony import */ var _tools_main__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../../../tools/main */ "../tools/main.js");
/* harmony import */ var _tools_main__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_tools_main__WEBPACK_IMPORTED_MODULE_0__);
function _createForOfIteratorHelper(o, allowArrayLike) { var it; if (typeof Symbol === "undefined" || o[Symbol.iterator] == null) { if (Array.isArray(o) || (it = _unsupportedIterableToArray(o)) || allowArrayLike && o && typeof o.length === "number") { if (it) o = it; var i = 0; var F = function F() {}; return { s: F, n: function n() { if (i >= o.length) return { done: true }; return { done: false, value: o[i++] }; }, e: function e(_e) { throw _e; }, f: F }; } throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); } var normalCompletion = true, didErr = false, err; return { s: function s() { it = o[Symbol.iterator](); }, n: function n() { var step = it.next(); normalCompletion = step.done; return step; }, e: function e(_e2) { didErr = true; err = _e2; }, f: function f() { try { if (!normalCompletion && it.return != null) it.return(); } finally { if (didErr) throw err; } } }; }
function _unsupportedIterableToArray(o, minLen) { if (!o) return; if (typeof o === "string") return _arrayLikeToArray(o, minLen); var n = Object.prototype.toString.call(o).slice(8, -1); if (n === "Object" && o.constructor) n = o.constructor.name; if (n === "Map" || n === "Set") return Array.from(o); if (n === "Arguments" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen); }
function _arrayLikeToArray(arr, len) { if (len == null || len > arr.length) len = arr.length; for (var i = 0, arr2 = new Array(len); i < len; i++) { arr2[i] = arr[i]; } return arr2; }
// Fonction associant les attributs fournis à un champ de formulaire
var setAttributesToInputs = function setAttributesToInputs(inputsConf, myForm) {
for (var i in myForm.elements) {
if (!Object(_tools_main__WEBPACK_IMPORTED_MODULE_0__["isEmpty"])(myForm.elements[i].id)) {
var idInput = myForm.elements[i].id;
if (inputsConf[idInput] !== undefined) {
var inputHTML = document.getElementById(idInput);
for (var attribute in inputsConf[idInput]) {
inputHTML.setAttribute(attribute, inputsConf[idInput][attribute]);
}
}
}
}
return true;
}; // Récupère toutes les valeurs de champs en omettant les checkbox non cochées, etc.
var getDatasFromInputs = function getDatasFromInputs(myForm) {
var datas = {};
var formData = new FormData(myForm);
var _iterator = _createForOfIteratorHelper(formData.entries()),
_step;
try {
for (_iterator.s(); !(_step = _iterator.n()).done;) {
var entrie = _step.value;
datas[entrie[0]] = entrie[1];
}
} catch (err) {
_iterator.e(err);
} finally {
_iterator.f();
}
return datas;
}; // Vide tous les champs d'un formulaire, y compris hidden, checkbox, etc.
// Revoir pour les select
var empyForm = function empyForm(myForm) {
var formData = new FormData(myForm);
var _iterator2 = _createForOfIteratorHelper(formData.entries()),
_step2;
try {
for (_iterator2.s(); !(_step2 = _iterator2.n()).done;) {
var entrie = _step2.value;
if (myForm.elements[entrie[0]].type == "checkbox" || myForm.elements[entrie[0]].type == "radio") myForm.elements[entrie[0]].checked = false;else myForm.elements[entrie[0]].value = "";
}
} catch (err) {
_iterator2.e(err);
} finally {
_iterator2.f();
}
return true;
}; // Vide et cache le formulaire
var empyAndHideForm = function empyAndHideForm(myForm) {
empyForm(myForm);
myForm.style.display = "none";
};
/***/ }),
/***/ "./src/tools/users.js":
/*!****************************!*\
!*** ./src/tools/users.js ***!
\****************************/
/*! exports provided: getConfig, getTimeDifference, getPassword, setSession, checkAnswerDatas, checkSession */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "getConfig", function() { return getConfig; });
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "getTimeDifference", function() { return getTimeDifference; });
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "getPassword", function() { return getPassword; });
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "setSession", function() { return setSession; });
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "checkAnswerDatas", function() { return checkAnswerDatas; });
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "checkSession", function() { return checkSession; });
/* harmony import */ var _clientstorage_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./clientstorage.js */ "./src/tools/clientstorage.js");
/* harmony import */ var _tools_main__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../../../tools/main */ "../tools/main.js");
/* harmony import */ var _tools_main__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_tools_main__WEBPACK_IMPORTED_MODULE_1__);
function asyncGeneratorStep(gen, resolve, reject, _next, _throw, key, arg) { try { var info = gen[key](arg); var value = info.value; } catch (error) { reject(error); return; } if (info.done) { resolve(value); } else { Promise.resolve(value).then(_next, _throw); } }
function _asyncToGenerator(fn) { return function () { var self = this, args = arguments; return new Promise(function (resolve, reject) { var gen = fn.apply(self, args); function _next(value) { asyncGeneratorStep(gen, resolve, reject, _next, _throw, "next", value); } function _throw(err) { asyncGeneratorStep(gen, resolve, reject, _next, _throw, "throw", err); } _next(undefined); }); }; }
var configFrontEnd = __webpack_require__(/*! ../config/general */ "./src/config/general.js");
// Récupère les données de configuration des utilisateurs
// À terme, un fichier statique importable comme module devrait éviter une requête ajax
var getConfig = /*#__PURE__*/function () {
var _ref = _asyncToGenerator( /*#__PURE__*/regeneratorRuntime.mark(function _callee() {
return regeneratorRuntime.wrap(function _callee$(_context) {
while (1) {
switch (_context.prev = _context.next) {
case 0:
return _context.abrupt("return", new Promise(function (resolve, reject) {
var xhr = new XMLHttpRequest();
xhr.open("GET", configFrontEnd.apiUrl + configFrontEnd.usersGetConfigUrl);
xhr.onload = function () {
return resolve(JSON.parse(xhr.responseText));
};
xhr.onerror = function () {
return reject(xhr.statusText);
};
xhr.send();
}));
case 1:
case "end":
return _context.stop();
}
}
}, _callee);
}));
return function getConfig() {
return _ref.apply(this, arguments);
};
}();
var getTimeDifference = function getTimeDifference(config) {
var timeLocal = new Date().getTimezoneOffset();
if (timeLocal > config.timeDifferenceMax || timeLocal < config.timeDifferenceMin) return 0;else return timeLocal;
}; // On enlève volontairement les 0/O pour éviter les confusions !
// Et mieux vaut aussi débuter et finir par une lettre simple.
var getPassword = function getPassword(nbCarMin, nbCarMax) {
var nbCar = nbCarMin + Math.floor(Math.random() * (nbCarMax - nbCarMin));
var letters = "ABCDEFGHIJKLMNPQRSTUVWXYZabcdefghijklmnpqrstuvwxyz";
var others = "123456789!?.*-_%@&ÉÀÈÙ€$ÂÊÛÎ";
var password = letters[Math.floor(Math.random() * letters.length)];
for (var i = 1; i < nbCar - 1; i++) {
if (i % 2 === 1) password += others[Math.floor(Math.random() * others.length)];else password += letters[Math.floor(Math.random() * letters.length)];
}
password += letters[Math.floor(Math.random() * letters.length)];
return password;
}; // J'utilise le stockage local du navigateur pour enregistrer les données permettant de reconnaître l'utilisateur par la suite
// Seul le serveur pourra vérifier que les identifiants sont (toujours) valides.
var setSession = function setSession(userId, token, durationTS) {
var storageUser = {
id: userId,
token: token,
duration: durationTS
};
Object(_clientstorage_js__WEBPACK_IMPORTED_MODULE_0__["saveLocaly"])("user", storageUser);
}; // Vérifie qu'il y a des données locales concernant le résultat d'un quiz
// Et les ajoute aux données envoyées par les formulaires d'inscription/connexion si c'est le cas
var checkAnswerDatas = function checkAnswerDatas(datas) {
var lastAnswer = Object(_clientstorage_js__WEBPACK_IMPORTED_MODULE_0__["getLocaly"])("lastAnswer");
if (!Object(_tools_main__WEBPACK_IMPORTED_MODULE_1__["isEmpty"])(lastAnswer)) {
var answer = JSON.parse(lastAnswer);
if (!Object(_tools_main__WEBPACK_IMPORTED_MODULE_1__["isEmpty"])(answer.duration) && !Object(_tools_main__WEBPACK_IMPORTED_MODULE_1__["isEmpty"])(answer.nbCorrectAnswers) && !Object(_tools_main__WEBPACK_IMPORTED_MODULE_1__["isEmpty"])(answer.QuestionnaireId) && !Object(_tools_main__WEBPACK_IMPORTED_MODULE_1__["isEmpty"])(answer.nbQuestions)) {
datas.duration = answer.duration;
datas.nbCorrectAnswers = answer.nbCorrectAnswers;
datas.QuestionnaireId = answer.QuestionnaireId;
datas.nbQuestions = answer.nbQuestions;
}
}
return datas;
}; // Cette fonction teste la connexion de l'utilisateur d'une page
// On peut fournis une liste de statuts acceptés (si vide = tous), ainsi qu'une url de redirection si non connecté, un message d'erreur à afficher sur la page de destination et l'url sur laquelle revenir une fois connecté
var checkSession = /*#__PURE__*/function () {
var _ref2 = _asyncToGenerator( /*#__PURE__*/regeneratorRuntime.mark(function _callee2(config) {
var status,
urlRedirection,
message,
urlWanted,
_args2 = arguments;
return regeneratorRuntime.wrap(function _callee2$(_context2) {
while (1) {
switch (_context2.prev = _context2.next) {
case 0:
status = _args2.length > 1 && _args2[1] !== undefined ? _args2[1] : [];
urlRedirection = _args2.length > 2 ? _args2[2] : undefined;
message = _args2.length > 3 ? _args2[3] : undefined;
urlWanted = _args2.length > 4 ? _args2[4] : undefined;
return _context2.abrupt("return", new Promise(function (resolve, reject) {
var userDatas = Object(_clientstorage_js__WEBPACK_IMPORTED_MODULE_0__["getLocaly"])("user");
if (Object(_tools_main__WEBPACK_IMPORTED_MODULE_1__["isEmpty"])(userDatas)) {
redirectUser(urlRedirection, message, urlWanted);
resolve(false);
} else {
var user = JSON.parse(userDatas);
if (Object(_tools_main__WEBPACK_IMPORTED_MODULE_1__["isEmpty"])(user.id) || Object(_tools_main__WEBPACK_IMPORTED_MODULE_1__["isEmpty"])(user.token) || Object(_tools_main__WEBPACK_IMPORTED_MODULE_1__["isEmpty"])(user.duration) || user.duration < Date.now()) {
Object(_clientstorage_js__WEBPACK_IMPORTED_MODULE_0__["removeLocaly"])("user");
redirectUser(urlRedirection, message, urlWanted);
resolve(false);
} else {
var xhr = new XMLHttpRequest();
xhr.open("GET", configFrontEnd.apiUrl + config.userRoutes + config.checkLoginRoute + user.token);
xhr.onload = function () {
var response = JSON.parse(xhr.responseText);
if (xhr.status === 200 && response.isValid && response.id != undefined) {
if (response.id === user.id) {
user.name = response.name;
user.language = response.language;
user.timeDifference = response.timeDifference;
user.status = response.status; // le token et de toute façon vérifier à chaque requête à l'API
Object(_clientstorage_js__WEBPACK_IMPORTED_MODULE_0__["saveLocaly"])("user", user); // si il s'agit d'un "user" et que son abonnement a expiré, je le redirige vers la caisse :-)
if (response.status === "user" && response.nbDaysOk <= 0) {
var urlAccount = config.siteUrl + "/" + configFrontEnd.accountPage;
if (window.location.href.indexOf(urlAccount) === -1) window.location.assign("/" + configFrontEnd.accountPage); // passée directement ici, l'ancre #subscribe ne fonctionne pas !?
resolve(true);
} else {
if (status.length !== 0 && status.indexOf(response.status) === -1) {
redirectUser(urlRedirection, message, urlWanted);
resolve(false);
} else resolve(true);
}
} else {
Object(_clientstorage_js__WEBPACK_IMPORTED_MODULE_0__["removeLocaly"])("user");
redirectUser(urlRedirection, message, urlWanted);
resolve(false);
}
} else {
Object(_clientstorage_js__WEBPACK_IMPORTED_MODULE_0__["removeLocaly"])("user");
redirectUser(urlRedirection, message, urlWanted);
resolve(false);
}
};
xhr.onerror = function () {
return reject(xhr.statusText);
};
xhr.send();
}
}
}));
case 5:
case "end":
return _context2.stop();
}
}
}, _callee2);
}));
return function checkSession(_x) {
return _ref2.apply(this, arguments);
};
}(); // Cette fonction sert à la précédente en cas de connexion non valide
var redirectUser = function redirectUser(urlRedirection, message, urlWanted) {
if (!Object(_tools_main__WEBPACK_IMPORTED_MODULE_1__["isEmpty"])(message)) Object(_clientstorage_js__WEBPACK_IMPORTED_MODULE_0__["saveLocaly"])("message", message);
if (!Object(_tools_main__WEBPACK_IMPORTED_MODULE_1__["isEmpty"])(urlWanted)) Object(_clientstorage_js__WEBPACK_IMPORTED_MODULE_0__["saveLocaly"])("url", urlWanted);
if (!Object(_tools_main__WEBPACK_IMPORTED_MODULE_1__["isEmpty"])(urlRedirection)) window.location.assign(urlRedirection);
};
/***/ })
/******/ });
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack:///webpack/bootstrap","webpack:///../lang sync ^\\.\\/.*\\/answer$","webpack:///../lang sync ^\\.\\/.*\\/general$","webpack:///../lang/fr/answer.js","webpack:///../lang/fr/general.js","webpack:///../tools/main.js","webpack:///./src/config/general.js","webpack:///./src/quiz.js","webpack:///./src/tools/answers.js","webpack:///./src/tools/clientstorage.js","webpack:///./src/tools/dom.js","webpack:///./src/tools/everywhere.js","webpack:///./src/tools/forms.js","webpack:///./src/tools/users.js"],"names":["module","exports","needNumberUserResponses","needIntegerNumberUserResponses","needMinNumberUserResponses","needMaxNumberUserResponses","needNumberCorrectResponses","needIntegerNumberCorrectResponses","needMinNumberCorrectResponses","needMaxNumberCorrectResponses","needIntegerNumberSecondesResponse","needMinNumberSecondesResponse","checkResponsesOuputFail","checkResponsesOuputMedium","checkResponsesOuputSuccess","wantToSaveResponses","responseSavedMessage","responseSavedError","noQuestionnaireWithoudAnswer","nbQuestionnaireWithoudAnswer","statsUser","previousAnswersTitle","previousAnswersStats","previousAnswersLine","siteHTMLTitle","siteMetaDescription","scriptTimingInfo","scriptTimingAlert","serverError","serverErrorAdmin","neededParams","badUrl","notValidFormat","notAllowed","notRequired","updateBtnTxt","addBtnTxt","deleteBtnTxt","addOkMessage","updateOkMessage","deleteOkMessage","failAuth","failAuthHeader","failAuthToken","failAuthId","failAuthCron","previousPage","nextPage","btnLinkToQuestionnaire","statsAdmin","Tool","myVar","undefined","trim","myString","isEmpty","max","length","substring","mapObj","replaceElts","RegExp","Object","keys","join","replace","matched","min","Math","ceil","floor","random","dateString","lang","myDate","Date","myDay","getDate","myMounth","getMonth","myYear","getFullYear","apiUrl","usersGetConfigUrl","userHomePage","adminHomePage","managerHomePage","subscribePage","connectionPage","accountPage","questionnairesManagementPage","usersManagementPage","nbQuestionnairesUserHomePage","illustrationDir","configFrontEnd","require","txt","txtAnswers","myForm","document","getElementById","divResponse","btnShow","btnSubmit","explanationsTitle","explanationsContent","helloDev","config","isConnected","user","initialise","getConfig","addElement","style","display","checkSession","getLocaly","checkPreviousResponses","showQuestionnaire","chronoBegin","now","setAttribute","window","location","assign","addEventListener","e","preventDefault","hash","answer","innerHTML","userResponses","getDatasFromInputs","duration","round","nbCorrectAnswers","QuestionnaireId","value","xhr","XMLHttpRequest","open","questionnaireRoutes","getQuestionnaireRoutes","questionnaireId","idChoice","goodResponse","onreadystatechange","readyState","DONE","response","JSON","parse","responseText","status","Array","isArray","Questions","nbQuestions","i","j","Choices","id","isCorrect","parentNode","classList","add","getOuput","checkAnswerOuput","xhr2","saveAnswersRoute","response2","message","setRequestHeader","token","timeDifference","getTimeDifference","send","stringify","saveAnswer","querySelector","explanations","querySelectorAll","xhrPreviousRes","getPreviousAnswers","nbResponses","previousAnswersContent","totNbQuestions","totNbCorrectAnswers","totDuration","mapLineContent","DATEANSWER","dateFormat","createdAt","NBCORRECTANSWERS","AVGDURATION","replaceAll","AVGCORRECTANSWERS","saveLocaly","ratio","DURATION","NBQUESTIONS","output","name","data","localStorage","setItem","json","getItem","removeLocaly","removeItem","eltParent","eltType","eltContent","eltId","eltClass","eltAttributes","newElement","createElement","attributName","appendChild","console","log","setAttributesToInputs","inputsConf","elements","idInput","inputHTML","attribute","datas","formData","FormData","entries","entrie","empyForm","type","checked","empyAndHideForm","Promise","resolve","reject","onload","onerror","statusText","timeLocal","getTimezoneOffset","timeDifferenceMax","timeDifferenceMin","getPassword","nbCarMin","nbCarMax","nbCar","letters","others","password","setSession","userId","durationTS","storageUser","checkAnswerDatas","lastAnswer","urlRedirection","urlWanted","userDatas","redirectUser","userRoutes","checkLoginRoute","isValid","language","nbDaysOk","urlAccount","siteUrl","href","indexOf"],"mappings":"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","file":"./JS/quiz.app.js","sourcesContent":[" \t// The module cache\n \tvar installedModules = {};\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId]) {\n \t\t\treturn installedModules[moduleId].exports;\n \t\t}\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\ti: moduleId,\n \t\t\tl: false,\n \t\t\texports: {}\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.l = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// define getter function for harmony exports\n \t__webpack_require__.d = function(exports, name, getter) {\n \t\tif(!__webpack_require__.o(exports, name)) {\n \t\t\tObject.defineProperty(exports, name, { enumerable: true, get: getter });\n \t\t}\n \t};\n\n \t// define __esModule on exports\n \t__webpack_require__.r = function(exports) {\n \t\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n \t\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n \t\t}\n \t\tObject.defineProperty(exports, '__esModule', { value: true });\n \t};\n\n \t// create a fake namespace object\n \t// mode & 1: value is a module id, require it\n \t// mode & 2: merge all properties of value into the ns\n \t// mode & 4: return value when already ns object\n \t// mode & 8|1: behave like require\n \t__webpack_require__.t = function(value, mode) {\n \t\tif(mode & 1) value = __webpack_require__(value);\n \t\tif(mode & 8) return value;\n \t\tif((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;\n \t\tvar ns = Object.create(null);\n \t\t__webpack_require__.r(ns);\n \t\tObject.defineProperty(ns, 'default', { enumerable: true, value: value });\n \t\tif(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));\n \t\treturn ns;\n \t};\n\n \t// getDefaultExport function for compatibility with non-harmony modules\n \t__webpack_require__.n = function(module) {\n \t\tvar getter = module && module.__esModule ?\n \t\t\tfunction getDefault() { return module['default']; } :\n \t\t\tfunction getModuleExports() { return module; };\n \t\t__webpack_require__.d(getter, 'a', getter);\n \t\treturn getter;\n \t};\n\n \t// Object.prototype.hasOwnProperty.call\n \t__webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"\";\n\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(__webpack_require__.s = \"./src/quiz.js\");\n","var map = {\n\t\"./fr/answer\": \"../lang/fr/answer.js\"\n};\n\n\nfunction webpackContext(req) {\n\tvar id = webpackContextResolve(req);\n\treturn __webpack_require__(id);\n}\nfunction webpackContextResolve(req) {\n\tif(!__webpack_require__.o(map, req)) {\n\t\tvar e = new Error(\"Cannot find module '\" + req + \"'\");\n\t\te.code = 'MODULE_NOT_FOUND';\n\t\tthrow e;\n\t}\n\treturn map[req];\n}\nwebpackContext.keys = function webpackContextKeys() {\n\treturn Object.keys(map);\n};\nwebpackContext.resolve = webpackContextResolve;\nmodule.exports = webpackContext;\nwebpackContext.id = \"../lang sync recursive ^\\\\.\\\\/.*\\\\/answer$\";","var map = {\n\t\"./fr/general\": \"../lang/fr/general.js\"\n};\n\n\nfunction webpackContext(req) {\n\tvar id = webpackContextResolve(req);\n\treturn __webpack_require__(id);\n}\nfunction webpackContextResolve(req) {\n\tif(!__webpack_require__.o(map, req)) {\n\t\tvar e = new Error(\"Cannot find module '\" + req + \"'\");\n\t\te.code = 'MODULE_NOT_FOUND';\n\t\tthrow e;\n\t}\n\treturn map[req];\n}\nwebpackContext.keys = function webpackContextKeys() {\n\treturn Object.keys(map);\n};\nwebpackContext.resolve = webpackContextResolve;\nmodule.exports = webpackContext;\nwebpackContext.id = \"../lang sync recursive ^\\\\.\\\\/.*\\\\/general$\";","module.exports =\n{\n    needNumberUserResponses : \"Le nombre de questions auxquelles l'utilisateur a répondu doit être fourni.\",\n    needIntegerNumberUserResponses : \"Le nombre de questions auxquelles l'utilisateur a répondu doit être un nombre entier.\",\n    needMinNumberUserResponses : \"Le nombre de questions auxquelles l'utilisateur a répondu ne peut être inférieur à 1.\",\n    needMaxNumberUserResponses : \"Le nombre de questions auxquelles l'utilisateur a répondu ne peut être aussi élevé.\",\n    needNumberCorrectResponses : \"Le nombre de réponses correctes doit être fourni.\",\n    needIntegerNumberCorrectResponses : \"Le nombre de réponses correctes doit être un nombre entier.\",\n    needMinNumberCorrectResponses : \"Le nombre de réponses correctes ne peut être négatif.\",\n    needMaxNumberCorrectResponses : \"Le nombre de réponses correctes ne peut être supérieur au nombre de questions.\",\n    needIntegerNumberSecondesResponse : \"La durée de la réponse doit être un nombre entier de secondes.\",\n    needMinNumberSecondesResponse : \"La durée de la réponse ne peut être négative.\",\n    checkResponsesOuputFail : \"Vous avez répondu en DURATION secondes et avez NBCORRECTANSWERS bonne(s) réponse(s) sur NBQUESTIONS. C'est certain, vous ferez mieux la prochaine fois !\",\n    checkResponsesOuputMedium : \"Vous avez répondu en DURATION secondes et avez NBCORRECTANSWERS bonne(s) réponse(s) sur NBQUESTIONS. C'est pas mal du tout !\",\n    checkResponsesOuputSuccess : \"Vous avez répondu en DURATION secondes et avez NBCORRECTANSWERS bonne(s) réponse(s) sur NBQUESTIONS. Bravo ! Rien ne vous échappe !\",\n    wantToSaveResponses: \"Si vous le souhaitez, vous pouvez sauvegarder votre résultat en créant votre compte.\\nCela vous permettra de tester de nouveau ce quiz dans l'avenir pour voir si vous avez mémoriser les bonnes réponses. Vous pourrez ainsi aussi recevoir régulièrement de nouveaux quizs.\",\n    responseSavedMessage : \"Votre résultat a été enregistré. <a href='/#URL'>Accèder à tous vos quizs</a>.\",\n    responseSavedError : \"Cependant une erreur a été rencontrée durant l'enregistrement de votre résultat. <a href='/#URL'>Accèder à tous vos quizs</a>.\",\n    noQuestionnaireWithoudAnswer: \"Bravo ! Vous avez répondu à tous les quizs qui vous on été proposés !<br>Il y en a des nouveaux publiés régulièrement et qui vous seront proposés.<br>En attendant vous pouvez peut-être essayer de répondre de nouveau à certain quiz pour voir si vous vous souvenez des bonnes réponses ?\",\n    nbQuestionnaireWithoudAnswer: \"Il y a #NB quizs qui vous ont été proposés et auxquels vous n'avez pas répondu. Voici les derniers :!\",\n    statsUser: \"Vous avez enregistré NBANSWERS réponses à <b>NBQUESTIONNAIRES questionnaires différents</b> sur les NBTOTQUESTIONNAIRES proposés par le site.<br>En moyenne, vous avez mis AVGDURATION secondes à répondre et avez <b>correctement répondu à AVGCORRECTANSWERS % des questions</b>.\",\n    previousAnswersTitle: \"Vos résultats précédents\",\n    previousAnswersStats: \"En moyenne, vous avez répondu à ce quiz en AVGDURATION secondes, en ayant <b>AVGCORRECTANSWERS % de bonnes réponses</b>.\",\n    previousAnswersLine: \"Le DATEANSWER, vous avez répondu correctement à NBCORRECTANSWERS questions en AVGDURATION secondes.\"\n};\n","module.exports =\n{\n    siteHTMLTitle : \"WikiLerni : qu'allez-vous apprendre aujourd'hui ?\",\n    siteMetaDescription : \"Chaque jour, testez vos connaissances et apprenez de nouvelles choses avec WikiLerni.\",\n    scriptTimingInfo : \"Durée de la réponse : SCRIPT_TIMING millisecondes, route : SCRIPT_URL\",\n    scriptTimingAlert : \"*** Script lent : SCRIPT_TIMING millisecondes, route : SCRIPT_URL\",\n    serverError : \"Désolé. Une erreur imprévue est survenue. Si cela persiste, n'hésitez à prévenir l'administrateur du site.\",\n    serverErrorAdmin : \"Bug de l'application :\",\n    neededParams : \"Des paramètres nécessaires manquants sont manquants.\",\n    badUrl : \"Tentative d'accès à une page n'existant pas :\",\n    notValidFormat : \"Format non valide.\",\n    notAllowed : \"Vous n'avez pas les droits nécessaires pour cette action.\",\n    notRequired : \"Facultatif.\",\n    updateBtnTxt: \"Modifier\",\n    addBtnTxt: \"Ajouter\",\n    deleteBtnTxt: \"Supprimer\",\n    addOkMessage : \"Les données ont bien été enregistrées.\",\n    updateOkMessage : \"La mise à jour à jour a bien été enregistrée.\",\n    deleteOkMessage : \"La suppression a bien été enregistrée.\",\n    failAuth : \"Erreur d'authentification.\",\n    failAuthHeader : \"Absence de header Authorization.\",\n    failAuthToken : \"Token invalide ou utilisateur non trouvé.\",\n    failAuthId : \"Identifiant non valide : \",\n    failAuthCron : \"Tentative de lancement d'un cron sans le bon token.\",\n    previousPage : \"Page précédente\",\n    nextPage : \"Page suivante\",\n    btnLinkToQuestionnaire : \"Aller au quiz !\",\n    statsAdmin : \"Durant les dernières 24h : NB_USERS_24H comptes ont été créés, NB_SUBSCRIPTIONS_24H validés et NB_USERS_DELETED_24H supprimés. NB_ANSWERS_24H réponses aux quizs ont été enregistrées.<br>En tout, il y a : NB_USERS_TOT comptes, dont NB_SUBSCRIPTIONS_TOT validés et NB_SUBSCRIPTIONS_PREMIUM comptes prémium. NB_ANSWERS_TOT réponses aux quizs ont été enregistrées.<br>Parmi les NB_USERS_DELETED_TOT comptes supprimés, NB_USERS_DELETED_VALIDED avaient validé leur compte et NB_USERS_DELETED_PREMIUM avaient souscrit un compte prémium.\"\n};","// Quelques fonctions utiles pour les chaînes\n\nclass Tool\n{\n    static isEmpty(myVar)\n    {\n        if(myVar===undefined || myVar===null)\n            return true;\n        else\n        {\n            myVar+=\"\";// si autre chose qu'une chaîne envoyé...\n            myVar=myVar.trim();\n            if(myVar===\"\")\n                return true;\n            else\n                return false;\n        }\n    }\n    \n    static trimIfNotNull(myString)\n    {\n        if(Tool.isEmpty(myString))\n            myString=null;\n        else\n        {\n            myString+=\"\";// si autre chose qu'une chaîne envoyé...\n            myString=myString.trim();\n        }\n        return myString;\n    }\n\n\n    static shortenIfLongerThan(myString, max)\n    {\n        myString+=\"\";// au cas où cela ne serait pas une chaîne...\n       if(myString.length > max)\n            myString=myString.substring(0, (max-3))+\"...\";\n        return myString;\n    }\n\n    // source : https://stackoverflow.com/questions/15604140/replace-multiple-strings-with-multiple-other-strings\n    static replaceAll(myString, mapObj)\n    {\n        const replaceElts = new RegExp(Object.keys(mapObj).join(\"|\"),\"gi\");\n        return myString.replace(replaceElts, (matched) =>\n        {\n            return mapObj[matched];\n        });\n    }\n\n    // source : https://developer.mozilla.org/fr/docs/Web/JavaScript/Reference/Objets_globaux/Math/random\n    static getRandomInt(min, max)\n    {\n        min = Math.ceil(min);\n        max = Math.floor(max);\n        return Math.floor(Math.random() * (max - min)) + min;\n    }\n\n    // à compléter : https://en.wikipedia.org/wiki/Date_format_by_country\n    static dateFormat(dateString, lang=\"fr\")\n    {\n        if(Tool.isEmpty(dateString))\n            return \"\";\n        let myDate=new Date(dateString);\n        let myDay=myDate.getDate()+\"\";\n        if(myDay.length===1)\n            myDay=\"0\"+myDay;\n        let myMounth=(myDate.getMonth()+1)+\"\";\n        if(myMounth.length===1)\n            myMounth=\"0\"+myMounth;\n        let myYear=myDate.getFullYear();\n        if(lang===\"fr\")\n            return myDay+\"/\"+myMounth+\"/\"+myYear;\n        else if (lang===\"form\")// 2014-02-09\n            return myYear+\"-\"+myMounth+\"-\"+myDay;\n        else\n            return myMounth+\"/\"+myDay+\"/\"+myYear;\n    }\n}\n\nmodule.exports = Tool;","module.exports =\n{\n    apiUrl : \"http://localhost:3000/api\",\n    usersGetConfigUrl : \"/user/getconfig\",\n    lang : \"fr\",\n    userHomePage : \"accueil.html\",\n    adminHomePage : \"admin.html\",\n    managerHomePage : \"gestion.html\",\n    subscribePage : \"inscription.html\",\n    connectionPage : \"connexion.html\",\n    accountPage: \"compte.html\",\n    questionnairesManagementPage: \"gestion-quizs.html\",\n    usersManagementPage: \"gestion-utilisateurs.html\",\n    nbQuestionnairesUserHomePage : 10,\n    illustrationDir : \"/img/quizs/\"\n};\n// à terme fusionner avec les déclarations backend concernant le front, tout étant saisi dans l'admin","// -- GESTION DU FORMULAIRE PERMETTANT D'AFFICHER ET RÉPONDRE À UN QUIZ\n\n/// Il n'est pas nécessaire d'être connecté pour répondre au quiz et voir son résultat.\n/// Mais si pas connecté, je propose à l'internaute de se connecter ou de créer un compte.\n/// Dans ce but son résultat est alors stocké dans son navigateur\n/// Si il est connecté, l'enregistrement de son résultat se fait automatiquement et ses précédents résultats affichés\n\n// Fichier de configuration côté client :\nconst configFrontEnd = require(\"./config/general\");\n\n// Importation des fonctions utiles au script :\nimport { checkAnswerOuput, saveAnswer } from \"./tools/answers.js\";\nimport { addElement } from \"./tools/dom.js\";\nimport { helloDev } from \"./tools/everywhere.js\";\nimport { getLocaly } from \"./tools/clientstorage.js\";\nimport { getDatasFromInputs } from \"./tools/forms.js\";\nimport { dateFormat, isEmpty, replaceAll } from \"../../tools/main\";\nimport { checkSession, getConfig, getTimeDifference } from \"./tools/users.js\";\n\n// Dictionnaires :\nconst txt = require(\"../../lang/\"+configFrontEnd.lang+\"/general\");\nconst txtAnswers = require(\"../../lang/\"+configFrontEnd.lang+\"/answer\");\n\n// Principaux éléments du DOM manipulés :\nconst myForm = document.getElementById(\"questionnaire\");\nconst divResponse = document.getElementById(\"response\");\nconst btnShow = document.getElementById(\"showQuestionnaire\");\nconst btnSubmit = document.getElementById(\"checkResponses\");\nconst explanationsTitle = document.getElementById(\"explanationsTitle\");\nconst explanationsContent = document.getElementById(\"explanationsContent\");\n\nhelloDev();\n\n// Chargement des configurations fournies par le backend + test de connexion de l'utilisateur :\nlet config, isConnected, user;\nconst initialise = async () =>\n{\n    try\n    {\n        config = await getConfig();\n        if(!config)\n            addElement(divResponse, \"p\", txt.serverError, \"\", [\"error\"]);\n        else\n        {\n            btnShow.style.display=\"inline\";\n            isConnected=await checkSession(config, [\"user\"]);// seuls les simples utilisateurs peuvent répondre aux quizs\n            // Si l'utilisateur est connecté, je vérifie sur il a déjà répondu à ce quiz et si oui affiche ses précédentes réponses dans l'encart servant à expliquer le topo aux nouveaux\n            if(isConnected)\n            {\n                user=getLocaly(\"user\", true);\n                checkPreviousResponses(config, user);\n            }\n        }\n    }\n    catch(e)\n    {\n        addElement(divResponse, \"p\", txt.serverError, \"\", [\"error\"]);\n    }\n}\ninitialise();\n\n// Affichage du questionnaire si l'utilisateur clique sur le bouton ou si l'id du formulaire est passée par l'url\n// Déclenche en même temps le chronomètre mesurant la durée de la réponse aux questions.\nconst showQuestionnaire = () =>\n{\n    chronoBegin=Date.now();\n    myForm.style.display=\"block\";\n    btnShow.setAttribute(\"disabled\", true);\n    window.location.assign(\"#questionnaire\");\n}\nlet chronoBegin=0;\nbtnShow.addEventListener(\"click\", function(e)\n{\n    try\n    {\n        e.preventDefault();\n        showQuestionnaire();\n    }\n    catch(e)\n    {\n        addElement(divResponse, \"p\", txt.serverError, \"\", [\"error\"]);\n    }\n});\nif(location.hash!=undefined && location.hash===\"#questionnaire\")\n    showQuestionnaire();\n\n// Traitement de l'envoi de sa réponse par l'utilisateur :\nlet answer = {};\nmyForm.addEventListener(\"submit\", function(e)\n{\n    try\n    {\n        e.preventDefault();       \n        btnSubmit.setAttribute(\"disabled\", true);// seulement un envoi à la fois :)\n        divResponse.innerHTML=\"\";// supprime les éventuels messages déjà affichés\n        const userResponses=getDatasFromInputs(myForm);\n        answer.duration=Math.round((Date.now()-chronoBegin)/1000);\n        answer.nbCorrectAnswers=0;\n        answer.QuestionnaireId=document.getElementById(\"questionnaireId\").value;\n        // Récupére les réponses du quiz sur le serveur :\n        const xhr = new XMLHttpRequest();\n        xhr.open(\"GET\", configFrontEnd.apiUrl+config.questionnaireRoutes+config.getQuestionnaireRoutes+\"/\"+userResponses.questionnaireId);\n        let idChoice, goodResponse;\n        xhr.onreadystatechange = function()\n        {\n            if (this.readyState == XMLHttpRequest.DONE)\n            {\n                let response=JSON.parse(this.responseText);\n                if (this.status === 200 && Array.isArray(response.Questions))\n                {\n                    answer.nbQuestions=response.Questions.length;\n                    // Puis on passe en revue les réponses pour afficher celles qui sont correctes et détecter les erreurs\n                    // Il peut y avoir plusieurs bonnes réponses pour une même question (QCM)\n                    for(let i in response.Questions)\n                    {\n                        goodResponse=true;\n                        for(let j in response.Questions[i].Choices)\n                        {\n                            idChoice=response.Questions[i].Choices[j].id;\n                            if(response.Questions[i].Choices[j].isCorrect===true)\n                            {\n                                document.getElementById(\"response_\"+idChoice).parentNode.classList.add(\"success\");\n                                if(userResponses[\"response_\"+idChoice]===undefined)// bonne réponse non sélectionnée\n                                    goodResponse=false;\n                            }\n                            else\n                            {\n                                if(userResponses[\"response_\"+idChoice]===\"on\")// réponse cochée n'étant pas une des bonnes\n                                {\n                                    goodResponse=false;\n                                    document.getElementById(\"response_\"+idChoice).parentNode.classList.add(\"error\");\n                                }\n                            }\n                        }\n                        if(goodResponse)\n                            answer.nbCorrectAnswers++;\n                    }\n                    // Affichage du résultat, suivant si l'utilisateur est connecté ou pas et son score :\n                    let getOuput=checkAnswerOuput(answer);\n                    if(isConnected)\n                    {\n                        // si l'utilisateur est connecté on enregistre son résultat sur le serveur.\n                        const xhr2 = new XMLHttpRequest();\n                        xhr2.open(\"POST\", configFrontEnd.apiUrl+config.questionnaireRoutes+config.saveAnswersRoute);\n                        xhr2.onreadystatechange = function()\n                        {\n                            if (this.readyState == XMLHttpRequest.DONE)\n                            {\n                                let response2=JSON.parse(this.responseText);\n                                if (this.status === 201 && (response2.message))\n                                {\n                                    getOuput+=\"<br>\"+response2.message.replace(\"#URL\", configFrontEnd.userHomePage);\n                                    checkPreviousResponses(config, user);\n                                }\n                                else\n                                    getOuput+=\"<br>\"+txtAnswers.responseSavedError.replace(\"#URL\", configFrontEnd.userHomePage);\n                                addElement(divResponse, \"p\", getOuput, \"\", [\"information\"]);\n                            }\n                        }\n                        xhr2.setRequestHeader(\"Authorization\", \"Bearer \"+user.token);\n                        xhr2.setRequestHeader(\"Content-Type\", \"application/json\");\n                        answer.timeDifference=getTimeDifference(config);// on en profite pour mettre les pendules à l'heure\n                        xhr2.send(JSON.stringify(answer));\n                    }\n                    else\n                    {   // sinon je l'enregistre côté client pour permettre de le retrouver au moment de la création du compte ou de la connexion\n                        if(saveAnswer(answer))\n                        {\n                             getOuput+=\"<br>\"+txtAnswers.wantToSaveResponses;\n                             addElement(divResponse, \"p\", getOuput, \"\", [\"information\"]);\n                             document.querySelector(\".subcribeBtns\").style.display=\"block\";\n                        }\n                        else // pas la peine de proposer de créer un compte si le stockage local ne fonctionne pas\n                            addElement(divResponse, \"p\", getOuput, \"\", [\"information\"]);\n                    } \n                    // affichage des textes d'explications pour chaque question\n                    const explanations=document.querySelectorAll(\".help\");\n                    for(let i in explanations)\n                        explanations[i].style.display=\"block\";// affiche une erreur \"TypeError: explanations[i].style is undefined\" mais tout fonctionne ?\n                }\n                else\n                    addElement(divResponse, \"p\", txt.serverError, \"\", [\"error\"]);\n            }\n        }\n        xhr.send();\n    }\n    catch(e)\n    {\n        addElement(divResponse, \"p\", txt.serverError, \"\", [\"error\"]);\n    }\n})\n\n// Fonction vérifiant les précédentes réponses de l'utilisateur\n// Utile si connecté lors du premier chargement de la page, puis après une nouvelle réponse\nconst checkPreviousResponses = (config, user) =>\n{\n    const xhrPreviousRes = new XMLHttpRequest();\n    xhrPreviousRes.open(\"GET\", configFrontEnd.apiUrl+config.questionnaireRoutes+config.getPreviousAnswers+user.id+\"/\"+document.getElementById(\"questionnaireId\").value);\n    xhrPreviousRes.onreadystatechange = function()\n    {\n        if (this.readyState == XMLHttpRequest.DONE)\n        {\n            let response=JSON.parse(this.responseText);\n            if (this.status === 200)\n            {\n                const nbResponses=response.length;\n                let previousAnswersContent=\"\";\n                if(nbResponses!==0)\n                {\n                    addElement(explanationsTitle, \"span\", txtAnswers.previousAnswersTitle);\n                    let totNbQuestions=0, totNbCorrectAnswers=0, totDuration=0, mapLineContent;\n                    for(let i in response)\n                    {\n                        totNbQuestions+=response[i].nbQuestions;\n                        totNbCorrectAnswers+=response[i].nbCorrectAnswers;\n                        totDuration+=response[i].duration;\n                        mapLineContent =\n                        {\n                            DATEANSWER : dateFormat(response[i].createdAt),// revoir problème de la langue\n                            NBCORRECTANSWERS : response[i].nbCorrectAnswers,\n                            AVGDURATION : response[i].duration\n                        };\n                        previousAnswersContent+=\"<li>\"+replaceAll(txtAnswers.previousAnswersLine, mapLineContent)+\"</li>\";\n                    }\n                    mapLineContent =\n                    {\n                        AVGDURATION : Math.round(totDuration/nbResponses),\n                        AVGCORRECTANSWERS : Math.round(totNbCorrectAnswers/totNbQuestions*100)\n                    };\n                    previousAnswersContent=\"<h5>\"+replaceAll(txtAnswers.previousAnswersStats, mapLineContent)+\"</h5>\"+previousAnswersContent;\n                    addElement(explanationsContent, \"ul\", previousAnswersContent);\n               }\n            }\n            else\n                addElement(divResponse, \"p\", txt.serverError, \"\", [\"error\"]);\n        }\n    }\n    xhrPreviousRes.setRequestHeader(\"Authorization\", \"Bearer \"+user.token);\n    xhrPreviousRes.send();\n}","const configFrontEnd = require(\"../config/general\");\n\nimport { saveLocaly } from \"./clientstorage.js\";\nimport { isEmpty, replaceAll } from \"../../../tools/main\";\n\nconst txt = require(\"../../../lang/\"+configFrontEnd.lang+\"/answer\");\n\n// Enregistrement côté client du dernier résultat à un quiz en attendant d'être connecté\nexport const saveAnswer = (answer) =>\n{\n    if(!isEmpty(answer.duration) && !isEmpty(answer.nbCorrectAnswers) && !isEmpty(answer.nbQuestions) && !isEmpty(answer.QuestionnaireId))\n    {\n        saveLocaly(\"lastAnswer\", answer);\n        return true;\n    }\n    else\n        return false;\n}\n\n// Retourne le texte suivant le nombre de bonnes réponses\nexport const checkAnswerOuput = (answer) =>\n{\n    if(!isEmpty(answer.duration) && !isEmpty(answer.nbCorrectAnswers) && !isEmpty(answer.nbQuestions))\n    {\n        const ratio=answer.nbCorrectAnswers/answer.nbQuestions;\n        const mapObj=\n        {\n            DURATION: answer.duration,\n            NBCORRECTANSWERS: answer.nbCorrectAnswers,\n            NBQUESTIONS: answer.nbQuestions\n        }\n        let output=\"\";\n        if(ratio < 0.4)\n            output=replaceAll(txt.checkResponsesOuputFail, mapObj);\n        else if(ratio < 0.8)\n            output=replaceAll(txt.checkResponsesOuputMedium, mapObj);\n        else\n            output=replaceAll(txt.checkResponsesOuputSuccess, mapObj);\n        if(output)\n            return output;\n        else\n            return \"\";\n    }\n    else\n        return \"\";\n}\n\n/*\nexport const checkSession = async (config) =>\n{\n    return new Promise((resolve, reject) =>\n    {\n        if(isEmpty(localStorage.getItem(\"user\")))\n            resolve(false);\n        else\n        {\n            const user=JSON.parse(localStorage.getItem(\"user\"));\n            if(user.duration===undefined || user.duration < Date.now())\n            {\n                localStorage.removeItem(\"user\");\n                resolve(false);\n            }\n            else\n            {\n                const xhr = new XMLHttpRequest();\n                xhr.open(\"GET\", configFrontEnd.apiUrl+config.userRoutes+config.checkLoginRoute+user.token);\n                xhr.onload = () =>\n                {\n                    let response=JSON.parse(xhr.responseText);\n                    if (xhr.status === 200 && response.isValid && response.id != undefined)\n                    {\n                        if(response.id===user.id)\n                            resolve(true);\n                        else\n                        {\n                            localStorage.removeItem(\"user\");\n                            resolve(false);\n                        }\n                    }\n                    else\n                    {\n                        localStorage.removeItem(\"user\");\n                        resolve(false);\n                    }\n                }\n                xhr.onerror = () => reject(xhr.statusText);\n                xhr.send();\n            }\n        }\n    });\n}*/","// FONCTIONS UTILES AU STOCKAGE LOCAL (SESSION, COOKIES, INDEXDB, ETC.)\n// Revenir pour gérer le cas où local.storage n'est pas connu pour utiliser cookie\n    \nexport const saveLocaly = (name, data) =>\n{\n    localStorage.setItem(name, JSON.stringify(data));\n}\n\nexport const getLocaly = (name, json=false) =>\n{\n    if(json)\n        return JSON.parse(localStorage.getItem(name));\n    else\n        return localStorage.getItem(name);\n}\n\nexport const removeLocaly = (name) =>\n{\n    localStorage.removeItem(name);\n}","import { isEmpty } from \"../../../tools/main\";\n\n// Fonction associant les attributs fournis à un champ de formulaire\nexport const addElement = (eltParent, eltType, eltContent=\"\", eltId=\"\", eltClass=[], eltAttributes={}, replace=true) =>\n{\n    if(isEmpty(eltType) || isEmpty(eltParent))\n        return false;\n    else\n    {\n        const newElement=document.createElement(eltType);\n        \n        if(!isEmpty(eltId))// tester si l'id n'est pas déjà utilisé dans le DOM ?\n            newElement.id=eltId;\n\n        if(Array.isArray(eltClass) && eltClass.length!=0)\n        {\n            for(let i in eltClass)\n                newElement.classList.add(eltClass[i]);\n        }\n\n        if(typeof eltAttributes === \"object\") // !! tous les objets ne sont pas ok\n        {\n            for(let attributName in eltAttributes)\n                newElement.setAttribute(attributName, eltAttributes[attributName]);\n        }\n\n        if(!isEmpty(eltContent))\n            newElement.innerHTML=eltContent.replace(/\\n/g,\"<br>\");// innerHTML permet d'ajouter du texte ayant lui-même des balises, etc.\n            \n        if(replace)\n            eltParent.innerHTML=\"\";\n        eltParent.appendChild(newElement);\n    }    \n}","// Ce script fournit des fonctions utilisées sur toutes les pages du site\n\nexport const helloDev = () =>\n{\n    console.log(\"**** Hello ami développeur :-)\\n Le code de WikiLerni est libre et est lisible sur gitlab : \\n Bonne lecture :-) \\n Pour les suggestions d'amélioration ou questions : dev@wililerni.com ****\");\n    return true;\n}","import { isEmpty } from \"../../../tools/main\";\n\n// Fonction associant les attributs fournis à un champ de formulaire\nexport const setAttributesToInputs = (inputsConf, myForm) =>\n{\n    for(let i in myForm.elements)\n    {\n        if(!isEmpty(myForm.elements[i].id))\n        {\n            let idInput=myForm.elements[i].id;\n            if(inputsConf[idInput]!==undefined)\n            {\n                let inputHTML=document.getElementById(idInput);\n                for (let attribute in inputsConf[idInput])\n                    inputHTML.setAttribute(attribute, inputsConf[idInput][attribute]);\n            }\n        }\n    }\n    return true;\n}\n\n// Récupère toutes les valeurs de champs en omettant les checkbox non cochées, etc.\nexport const getDatasFromInputs = (myForm) =>\n{\n    const datas={};\n    const formData = new FormData(myForm);\n    for(let entrie of formData.entries())\n       datas[entrie[0]]=entrie[1];\n    return datas;\n}\n\n// Vide tous les champs d'un formulaire, y compris hidden, checkbox, etc.\n// Revoir pour les select\nexport const empyForm = (myForm) =>\n{\n    const formData = new FormData(myForm);\n    for(let entrie of formData.entries())\n    {\n        if(myForm.elements[entrie[0]].type==\"checkbox\" || myForm.elements[entrie[0]].type==\"radio\")\n            myForm.elements[entrie[0]].checked=false;\n        else\n            myForm.elements[entrie[0]].value=\"\";\n    }\n    return true;\n}\n// Vide et cache le formulaire\nexport const empyAndHideForm = (myForm) =>\n{\n    empyForm(myForm);\n    myForm.style.display=\"none\";\n}","const configFrontEnd = require(\"../config/general\");\n\nimport { getLocaly, removeLocaly, saveLocaly } from \"./clientstorage.js\";\nimport { isEmpty } from \"../../../tools/main\";\n\n// Récupère les données de configuration des utilisateurs\n// À terme, un fichier statique importable comme module devrait éviter une requête ajax\nexport const getConfig =  async () =>\n{\n    return new Promise((resolve, reject) =>\n    {\n        const xhr = new XMLHttpRequest();\n        xhr.open(\"GET\", configFrontEnd.apiUrl+configFrontEnd.usersGetConfigUrl);\n        xhr.onload = () => resolve(JSON.parse(xhr.responseText));\n        xhr.onerror = () => reject(xhr.statusText);\n        xhr.send();\n    });\n}\n\nexport const getTimeDifference = (config) =>\n{\n    const timeLocal=new Date().getTimezoneOffset();\n    if(timeLocal > config.timeDifferenceMax || timeLocal < config.timeDifferenceMin)\n        return 0;\n    else return timeLocal;\n}\n\n// On enlève volontairement les 0/O pour éviter les confusions !\n// Et mieux vaut aussi débuter et finir par une lettre simple.\nexport const getPassword = (nbCarMin, nbCarMax) =>\n{\n    const nbCar=nbCarMin+Math.floor(Math.random()*(nbCarMax-nbCarMin));\n    const letters=\"ABCDEFGHIJKLMNPQRSTUVWXYZabcdefghijklmnpqrstuvwxyz\";\n    const others=\"123456789!?.*-_%@&ÉÀÈÙ€$ÂÊÛÎ\";\n    let password=letters[Math.floor(Math.random()*letters.length)];\n    for(let i=1;i<(nbCar-1);i++)\n    {\n        if((i % 2) ===1)\n            password+=others[Math.floor(Math.random()*others.length)];\n        else\n            password+=letters[Math.floor(Math.random()*letters.length)];   \n    }\n    password+=letters[Math.floor(Math.random()*letters.length)];\n    return password;\n}\n\n// J'utilise le stockage local du navigateur pour enregistrer les données permettant de reconnaître l'utilisateur par la suite\n// Seul le serveur pourra vérifier que les identifiants sont (toujours) valides.\nexport const setSession = (userId, token, durationTS) =>\n{\n    const storageUser=\n    {\n        id: userId,\n        token: token,\n        duration : durationTS\n    }\n    saveLocaly(\"user\", storageUser);\n}\n\n// Vérifie qu'il y a des données locales concernant le résultat d'un quiz\n// Et les ajoute aux données envoyées par les formulaires d'inscription/connexion si c'est le cas\nexport const checkAnswerDatas = (datas) =>\n{\n    const lastAnswer=getLocaly(\"lastAnswer\");\n    if(!isEmpty(lastAnswer))\n    {\n        const answer=JSON.parse(lastAnswer);\n        if(!isEmpty(answer.duration) && !isEmpty(answer.nbCorrectAnswers) && !isEmpty(answer.QuestionnaireId) && !isEmpty(answer.nbQuestions))\n        {\n            datas.duration=answer.duration;\n            datas.nbCorrectAnswers=answer.nbCorrectAnswers;\n            datas.QuestionnaireId=answer.QuestionnaireId;\n            datas.nbQuestions=answer.nbQuestions;\n        }\n    }\n    return datas;\n}\n\n// Cette fonction teste la connexion de l'utilisateur d'une page\n// On peut fournis une liste de statuts acceptés (si vide = tous), ainsi qu'une url de redirection si non connecté, un message d'erreur à afficher sur la page de destination et l'url sur laquelle revenir une fois connecté\nexport const checkSession = async (config, status=[], urlRedirection, message, urlWanted) =>\n{\n    return new Promise((resolve, reject) =>\n    {\n        const userDatas=getLocaly(\"user\");\n        if(isEmpty(userDatas))\n        {\n            redirectUser(urlRedirection, message, urlWanted);\n            resolve(false);\n        }\n        else\n        {\n            const user=JSON.parse(userDatas);\n            if(isEmpty(user.id) || isEmpty(user.token) || isEmpty(user.duration) || user.duration < Date.now())\n            {\n                removeLocaly(\"user\");\n                redirectUser(urlRedirection, message, urlWanted);\n                resolve(false);\n            }\n            else\n            {\n                const xhr = new XMLHttpRequest();\n                xhr.open(\"GET\", configFrontEnd.apiUrl+config.userRoutes+config.checkLoginRoute+user.token);\n                xhr.onload = () =>\n                {\n                    let response=JSON.parse(xhr.responseText);\n                    if (xhr.status === 200 && response.isValid && response.id != undefined)\n                    {\n                        if(response.id===user.id)\n                        {\n                            user.name=response.name;\n                            user.language=response.language;\n                            user.timeDifference=response.timeDifference;\n                            user.status=response.status;// le token et de toute façon vérifier à chaque requête à l'API\n                            saveLocaly(\"user\", user);\n                            // si il s'agit d'un \"user\" et que son abonnement a expiré, je le redirige vers la caisse :-)\n                            if(response.status===\"user\" && response.nbDaysOk <= 0)\n                            {\n                                const urlAccount=config.siteUrl+\"/\"+configFrontEnd.accountPage;\n                                if(window.location.href.indexOf(urlAccount)===-1)\n                                    window.location.assign(\"/\"+configFrontEnd.accountPage);// passée directement ici, l'ancre #subscribe ne fonctionne pas !?\n                                resolve(true);\n                            }\n                            else\n                            {\n                                if(status.length!==0 && status.indexOf(response.status)===-1)\n                                {\n                                    redirectUser(urlRedirection, message, urlWanted);\n                                    resolve(false);\n                                }\n                                else\n                                    resolve(true);\n                            }\n                        }\n                        else\n                        {\n                            removeLocaly(\"user\");\n                            redirectUser(urlRedirection, message, urlWanted);\n                            resolve(false);\n                        }\n                    }\n                    else\n                    {\n                        removeLocaly(\"user\");\n                        redirectUser(urlRedirection, message, urlWanted);\n                        resolve(false);\n                    }\n                }\n                xhr.onerror = () => reject(xhr.statusText);\n                xhr.send();\n            }\n        }\n    });\n}\n// Cette fonction sert à la précédente en cas de connexion non valide\nconst redirectUser = (urlRedirection, message, urlWanted) =>\n{\n    if(!isEmpty(message))\n        saveLocaly(\"message\", message);\n    if(!isEmpty(urlWanted))\n        saveLocaly(\"url\", urlWanted);\n    if(!isEmpty(urlRedirection))\n        window.location.assign(urlRedirection);\n}"],"sourceRoot":""}