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

1679 lines
210 KiB
JavaScript
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/******/ (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/questionnaire.js");
/******/ })
/************************************************************************/
/******/ ({
/***/ "../config/instance.js":
/*!*****************************!*\
!*** ../config/instance.js ***!
\*****************************/
/*! no static exports found */
/***/ (function(module, exports, __webpack_require__) {
var _module$exports;
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
var users = __webpack_require__(/*! ./users */ "../config/users.js");
var questionnaires = __webpack_require__(/*! ./questionnaires */ "../config/questionnaires.js");
module.exports = (_module$exports = {
apiUrl: "http://localhost:3000/api",
siteUrl: "http://localhost:8080",
adminName: "Fabrice",
adminEmail: "dev@wikilerni.com",
senderName: "WikiLerni (local)",
senderEmail: "bonjour@wikilerni.com",
adminLang: "fr",
theme: "wikilerni",
// le thème utilisé (dans /views) pour générer les pages HTML. Contient ses propres fichiers de configuration.
availableLangs: ["fr"],
// Languages in which the site is available. The first one is the default one.
siteName: "WikiLerni (local)",
beginCodeGodfather: "WL",
// case-sensitive and can't contain "@" !
defaultReceiptDays: "147",
// Par défaut, quel(s) jour(s) de la semaine, l'utilisateur reçoit-il quelque chose ? (1=dimanche, 2=lundi... 7=samedi)
cronTimingAlertInSeconde: 120,
// for logs
responseTimingAlertInSeconde: 3,
// idem
tokenSignupValidationTimeInHours: "48h",
// https://github.com/zeit/ms
tokenLoginLinkTimeInHours: "1h",
tokenConnexionMinTimeInHours: "24h",
tokenConnexionMaxTimeInDays: "180 days",
tokenLoginChangingTimeInHours: "1h",
// for email & password changing
tokenDeleteUserTimeInHours: "1h",
tokenUnsubscribeLinkTimeInDays: "7 days",
// token send with subscription's emails
freeAccountTimingInDays: 0,
// if 0 = unlimited
freeAccountExpirationNotificationInDays: 3,
accountExpirationFirstNotificationInDays: 10,
accountExpirationSecondNotificationInDays: 3,
inactiveAccountTimeToDeleteInDays: 180,
// Questionnaires:
nbQuestionsMin: 1,
// minimum number of questions for the questionnaire to be publishable
nbQuestionsMax: 2,
// if 0 = not maximum
nbChoicesMax: 10,
nbNewQuestionnaires: 12,
// for RSS, etc.
hourGiveNewQuestionnaireBegin: 7,
// in user local time
hourGiveNewQuestionnaireEnd: 21,
// idem
maxQuestionnaireSendedAtSameTime: 50,
// for subscription's e-mailing
minSearchQuestionnaires: 3,
fieldNewQuestionnaires: "publishingAt",
// field to be used to create the list of the last questionnaires, can be "createdAt", "updatedAt" or "publishingAt"
// Groups :
nbQuestionnairesByGroupMin: 1,
nbQuestionnairesByGroupMax: 0,
// Illustrations:
nbIllustrationsMin: 1,
nbIllustrationsMax: 1,
maxIllustrationSizeinOctet: 1000000,
// pas contrôlé pour l'instant. À revoir.
mimeTypesForIllustration: ["image/jpg", "image/jpeg", "image/png", "image/gif", "image/png"],
// -- Upload and resize:
illustrationsWidthMaxInPx: 500,
illustrationsMiniaturesWidthMaxInPx: 200,
// Links:
nbLinksMin: 1,
nbLinksMax: 0
}, _defineProperty(_module$exports, "nbQuestionsMin", 1), _defineProperty(_module$exports, "nbQuestionsMax", 0), _defineProperty(_module$exports, "nbChoicesMax", 10), _defineProperty(_module$exports, "passwordMinLength", users.password.minlength), _defineProperty(_module$exports, "dirCacheUsers", users.dirCacheUsers), _defineProperty(_module$exports, "dirCacheUsersAnswers", users.dirCacheUsersAnswers), _defineProperty(_module$exports, "dirCacheQuestionnaires", questionnaires.dirCacheQuestionnaires), _defineProperty(_module$exports, "dirCacheQuestions", questionnaires.dirCacheQuestions), _defineProperty(_module$exports, "dirCacheUsersQuestionnaires", questionnaires.dirCacheUsersQuestionnaires), _defineProperty(_module$exports, "dirHTMLQuestionnaires", questionnaires.dirHTMLQuestionnaires), _defineProperty(_module$exports, "dirWebQuestionnaires", questionnaires.dirWebQuestionnaires), _module$exports);
/***/ }),
/***/ "../config/matomo.js":
/*!***************************!*\
!*** ../config/matomo.js ***!
\***************************/
/*! no static exports found */
/***/ (function(module, exports) {
// Si non utilisé, supprimer l'appel des JS front-end puis recompiler.
// Revoir comment réorganiser cela sous-forme de module indépendant.
module.exports = {
// Stats Matomo :
matomo: {
url: "https://stats.le-fab-lab.com/",
siteId: "5"
}
};
/***/ }),
/***/ "../config/questionnaires.js":
/*!***********************************!*\
!*** ../config/questionnaires.js ***!
\***********************************/
/*! no static exports found */
/***/ (function(module, exports) {
module.exports = {
// API'routes (after "apiUrl" defined in instance.js)
questionnaireRoutes: "/questionnaire",
// la base à laquelle s'ajoute les routes suivantes
getListNextQuestionnaires: "/getlistnextquestionnaires/",
getQuestionnaireRoutes: "/get",
getRandomQuestionnairesRoute: "/getrandom",
getStatsQuestionnaires: "/stats/",
previewQuestionnaireRoutes: "/preview",
publishedQuestionnaireRoutes: "/quiz/",
regenerateHTML: "/htmlregenerated",
searchAdminQuestionnairesRoute: "/searchadmin",
searchQuestionnairesRoute: "/search",
// -- groupes :
groupRoutes: "/group",
getGroupRoute: "/get/",
previewGroupRoutes: "/preview",
searchGroupsRoute: "/search",
// -- questions & choices :
questionsRoute: "/question/",
// -- tags :
tagsSearchRoute: "/tags/search/",
// -- answers :
getAdminStats: "/getadminstats/",
getPreviousAnswers: "/user/answers/",
getStatsAnswers: "/user/anwswers/stats/",
// fonctionne aussi pour les groupes
saveAnswersRoute: "/answer/",
// idem
// forms : à compléter avec valeurs par défaut, etc. cf modèle
Questionnaire: {
title: {
maxlength: 255,
required: true
},
slug: {
maxlength: 150
},
// champ requis mais calculé à partir du titre qd laissé vide dans le formulaire
introduction: {
required: true
}
},
searchQuestionnaires: {
minlength: 3,
required: true
},
Group: {
title: {
maxlength: 255,
required: true
},
slug: {
maxlength: 150
} // champ requis mais calculé à partir du titre qd laissé vide dans le formulaire
},
Question: {
text: {
maxlength: 255,
required: true
},
rank: {
required: true,
min: 1,
defaultValue: 1
}
},
Choice: {
text: {
maxlength: 255,
required: true
}
},
search: {
minlength: 3,
required: true
},
searchGroups: {
minlength: 3,
required: true
},
// Emplacement des fichiers JSON générés :
dirCacheGroups: "datas/questionnaires/groups",
dirCacheQuestionnaires: "datas/questionnaires",
dirCacheQuestions: "datas/questionnaires/questions",
dirCacheTags: "datas/questionnaires/tags",
dirCacheUsersQuestionnaires: "datas/users/questionnaires",
// Emplacement des fichiers HTML générés :
dirHTMLGroups: "front/public/quiz/gp",
dirHTMLQuestionnaires: "front/public/quiz",
dirHTMLNews: "front/public/quizs",
dirHTMLTags: "front/public/quizs",
// Idem mais pour urls :
dirWebGroups: "quiz/gp",
dirWebQuestionnaires: "quiz",
dirWebNews: "quizs/",
dirWebTags: "quizs/",
// limite des résultat du moteur de recherche, quand demande de résultats au hasard :
nbRandomResults: 3,
/* Valeurs en fait définies dans instance.js donc à supprimer quand plus utilisées ailleurs : */
nbQuestionsMin: 1,
nbQuestionsMax: 0,
nbChoicesMax: 10,
nbTagsMin: 0,
nbTagsMax: 0 // 0 = not max
};
/***/ }),
/***/ "../config/users.js":
/*!**************************!*\
!*** ../config/users.js ***!
\**************************/
/*! no static exports found */
/***/ (function(module, exports) {
module.exports = {
// API'routes (after "apiUrl" defined in instance.js)
userRoutes: "/user",
checkDeleteLinkRoute: "/confirmdelete/",
checkIfIsEmailfreeRoute: "/isemailfree",
checkLoginRoute: "/checklogin/",
checkNewLoginLinkRoute: "/confirmnewlogin/",
checkSubscribeTokenRoute: "/validation/",
connectionRoute: "/login",
connectionWithLinkRoute: "/checkloginlink",
createUserRoute: "/create",
getAdminStats: "/getadminstats/",
getGodChilds: "/getgodchilds/",
getGodfatherRoute: "/getgodfatherid",
getLoginLinkRoute: "/getloginlink",
getPayments: "/payment/getforoneuser/",
getUserInfos: "/get/",
getUsersQuestionnairesRoute: "/getusersquestionnaires/",
// les questionnaires auxquels l'utilisateur a déjà eu accès via son abonnement
searchUserRoute: "/search/",
signupCompletionRoute: "/signupcompletion/",
subscribeRoute: "/signup",
unsubscribeRoute: "/subscription/stop/",
updateUserInfos: "/modify/",
validateUserRoute: "/validate/",
// forms : à compléter avec valeurs par défaut, etc. cf modèle
name: {
maxlength: 70,
required: true
},
email: {
maxlength: 255,
required: true
},
password: {
minlength: 8,
maxlength: 72,
required: true
},
// https://www.npmjs.com/package/bcrypt#security-issues-and-concerns
newPassword: {
minlength: 8,
maxlength: 72
},
codeGodfather: {
maxlength: 255
},
cguOk: {
value: "true",
required: true
},
search: {
minlength: 1,
required: true
},
timeDifferenceMin: -720,
timeDifferenceMax: 840,
// JSON dir
dirCacheUsers: "datas/users",
dirCacheUsersAnswers: "datas/users/questionnaires/answers",
dirCacheUsersWithoutAnswers: "datas/users/questionnaires/without"
};
/***/ }),
/***/ "../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 = {
checkResponsesOuputFail: "Vous avez répondu en DURATION secondes et avez <u><b>NBCORRECTANSWERS bonne(s) réponse(s) sur NBQUESTIONS questions</b></u>. C'est certain, vous ferez mieux la prochaine fois !",
checkResponsesOuputMedium: "Vous avez répondu en DURATION secondes et avez <u><b>NBCORRECTANSWERS bonne(s) réponse(s) sur NBQUESTIONS questions</b></u>. C'est pas mal du tout !",
checkResponsesOuputSuccess: "Vous avez répondu en DURATION secondes et avez <u><b>NBCORRECTANSWERS bonne(s) réponse(s) sur NBQUESTIONS questions</b></u>. Bravo ! Rien ne vous échappe !",
needIntegerNumberCorrectResponses: "Le nombre de réponses correctes doit être un nombre entier.",
needIntegerNumberSecondesResponse: "La durée de la réponse doit être un nombre entier de secondes.",
needIntegerNumberUserResponses: "Le nombre de questions auxquelles l'utilisateur a répondu doit être un nombre entier.",
needMaxNumberUserResponses: "Le nombre de questions auxquelles l'utilisateur a répondu ne peut être aussi élevé.",
needMinNumberUserResponses: "Le nombre de questions auxquelles l'utilisateur a répondu ne peut être inférieur à 1.",
needNumberCorrectResponses: "Le nombre de réponses correctes doit être fourni.",
needNumberUserResponses: "Le nombre de questions auxquelles l'utilisateur a répondu doit être fourni.",
needMaxNumberCorrectResponses: "Le nombre de réponses correctes ne peut être supérieur au nombre de questions.",
needMinNumberCorrectResponses: "Le nombre de réponses correctes ne peut être négatif.",
needMinNumberSecondesResponse: "La durée de la réponse ne peut être négative.",
noPreviousAnswer: "On dirait que c'est la première fois que vous répondez à ce quiz. Bonne lecture !",
previousAnswersLine: "Le DATEANSWER, vous avez répondu correctement à NBCORRECTANSWERS questions sur NBQUESTIONS en AVGDURATION secondes.",
previousAnswersStats: "En moyenne, vous avez répondu à ce quiz en AVGDURATION secondes, en ayant <b>AVGCORRECTANSWERS % de bonnes réponses</b>.",
previousAnswersTitle: "Bonjour #NOM, voici vos précédents résultats à ce quiz",
responseSavedError: "Cependant une erreur a été rencontrée durant l'enregistrement de votre résultat. <a href='/#URL'>Accèder à tous vos quizs</a>.",
responseSavedMessage: "Votre résultat a été enregistré. <a href='/#URL'>Accèder à tous vos quizs</a>.",
statsUser: "Vous avez enregistré <b>NBANSWERS réponseS1 à NBQUESTIONNAIRES quizS2 différentS3</b> sur les NBTOTQUESTIONNAIRES proposéS4 par le site.<br>En moyenne, vous avez mis AVGDURATION secondes à répondre et avez <b>correctement répondu à AVGCORRECTANSWERS % des questions</b>.",
wantToSaveResponses: "Si vous le souhaitez, vous pouvez <u><b>sauvegarder votre résultat</b></u> en créant votre compte ci-dessous. Cela vous permettra aussi de <u><b>recevoir régulièrement de nouvelles \"graines de culture\"</b></u> directement sur votre e-mail."
};
/***/ }),
/***/ "../lang/fr/general.js":
/*!*****************************!*\
!*** ../lang/fr/general.js ***!
\*****************************/
/*! no static exports found */
/***/ (function(module, exports) {
module.exports = {
addBtnTxt: "Ajouter",
addOkMessage: "Les données ont bien été enregistrées.",
alertNewWindow: "nouvelle fenêtre",
badUrl: "Tentative d'accès à une page n'existant pas :",
btnLinkToQuestionnaire: "Afficher !",
btnProposeConnection: "Je me connecte.",
btnProposeSubscribe: "Je crée mon compte.",
btnShowOnWebSite: "Lire la suite sur #SITE_NAME",
deleteBtnTxt: "Supprimer",
deleteFailMessage: "La suppression de l'enregistrement #ID a échoué.",
deleteOkMessage: "La suppression a bien été enregistrée.",
failAuth: "Erreur d'authentification.",
failAuthCron: "Tentative de lancement d'un cron sans le bon token.",
failAuthHeader: "Absence de header Authorization.",
failAuthId: "Identifiant non valide : ",
failAuthToken: "Token invalide ou utilisateur non trouvé.",
neededParams: "Des paramètres nécessaires manquants sont manquants.",
nextPage: "Page suivante",
notAllowed: "Vous n'avez pas les droits nécessaires pour cette action.",
notRequired: "Facultatif.",
notValidFormat: "Format non valide.",
previousPage: "Page précédente",
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 :",
siteHTMLTitle: "WikiLerni : la culture générale en liberté",
siteMetaDescription: "Avec WikiLerni, vous apprenez régulièrement de nouvelles choses. Vous recevez de courts articles lisibles en quelques minutes. Des quizs vous permettent ensuite de tester ce que vous avez retenu.",
scriptTimingAlert: "*** Script lent : SCRIPT_TIMING millisecondes, route : SCRIPT_URL",
scriptTimingInfo: "Durée de la réponse : SCRIPT_TIMING millisecondes, route : SCRIPT_URL",
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.",
subscriptionCall: "Inscrivez-vous !",
updateBtnTxt: "Modifier",
updateOkMessage: "La mise à jour à jour a bien été enregistrée."
};
/***/ }),
/***/ "../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;
} // On enlève volontairement les 0/O pour éviter les confusions !
// Et mieux vaut aussi débuter et finir par une lettre simple.
}, {
key: "getPassword",
value: 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;
}
}]);
return Tool;
}();
module.exports = Tool;
/***/ }),
/***/ "../views sync recursive ^\\.\\/.*\\.js$":
/*!**********************************!*\
!*** ../views sync ^\.\/.*\.js$ ***!
\**********************************/
/*! no static exports found */
/***/ (function(module, exports, __webpack_require__) {
var map = {
"./default/config/fr.js": "../views/default/config/fr.js",
"./wikilerni/config/fr.js": "../views/wikilerni/config/fr.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 = "../views sync recursive ^\\.\\/.*\\.js$";
/***/ }),
/***/ "../views/default/config/fr.js":
/*!*************************************!*\
!*** ../views/default/config/fr.js ***!
\*************************************/
/*! no static exports found */
/***/ (function(module, exports) {
module.exports = {
headLinks: [{
anchor: "Accueil",
attributes: {
href: "/"
}
}, {
anchor: "Mon compte",
attributes: {
href: "/connexion.html",
id: "accountHeadLink"
}
}, {
anchor: "À propos",
attributes: {
href: "/a-propos.html"
}
}, {
anchor: "Contact",
attributes: {
href: "/contact.html"
}
}],
footLinks: [{
anchor: "Crédits",
attributes: {
href: "/credits.html"
}
}, {
anchor: "Mentions légales",
attributes: {
href: "/mentions-legales.html",
rel: "nofollow"
}
}, {
anchor: "Données personnelles",
attributes: {
href: "/donnees.html"
}
}, {
anchor: "C.G.U.",
attributes: {
href: "/cgu.html",
rel: "nofollow"
}
}, {
anchor: "C.G.V.",
attributes: {
href: "/cgv.html",
rel: "nofollow"
}
}],
maxQuestionnairesByPage: 10,
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/",
siteSlogan: "Cultivons notre jardin !",
homeTitle1: "De nature curieuse ?",
homeP1: "Avec WikiLerni vous apprenez chaque jour de nouvelles choses.<br>Des articles de Wikipédia sont sélectionnés pour vous et sont suivis d'un quiz vous permettant de tester ce que vous en avez retenu.<br>De jour en jour de nouvelles graines de savoir sont ainsi semées dans votre \"jardin\".",
homeTitle2: "La culture en liberté",
homeP2: "Tout comme sur Wikipédia (*), le logiciel et le contenu partagé sur WikiLerni sont libres.<br>Vous pouvez les utiliser, les modifier et les diffuser selon votre souhait.<br>Sur WikiLerni, pas de publicité, ni de commercialisation de vos données personnelles.<br>Vous pouvez venir y \"cultiver votre jardin\" en toute tranquillité.<br><br><small><em>(*) Bien que partageant ses valeurs, WikiLerni est un projet indépendant de la fondation Wikipédia.</em></small>",
newQuestionnairesTitle: "Les derniers quizs publiés sur WikiLerni",
newQuestionnairesIntro: "Liste des derniers quizs publiés sur WikiLerni.",
explanationTitle: "Vous découvrez WikiLerni ?",
explanationTxt: "Le principe est simple : vous commencez par lire l'article Wikipédia dont le lien vous est proposé.<br>Puis vous afficher le quiz pour vérifier ce que vous en avez retenu de votre lecture. Vous obtenez alors votre résultat immédiatement.<br><br>Toutes les réponses se trouvent dans l'article proposé à la lecture. <b>Vous êtes ici pour apprendre de nouvelles choses</b>, mais libre à vous d'essayer d'y répondre immédiatement.<br><br>Quand le sujet s'y prête, ne vous étonnez pas si certaines des réponses proposées peuvent être un peu décalées, absurdes... On peut apprendre avec le sourire, non ? :-)<br><br>Une fois votre résultat obtenu, il vous sera proposé de créer un compte pour le sauvegarder.<br>Ce compte vous permettra de tester de nouveau ce quiz pour vérifier ce que vous en avez retenu plusieurs jours, semaines, mois... Et de recevoir régulièrement de nouvelles suggestions de lectures.<br><br>Mais <b>la création de ce compte est facultative</b> et <a href='/quizs/' title='Les derniers quizs publiés'>vous pouvez parcourir WikiLerni librement</a>.",
noJSNotification: "Désolé, mais pour l'instant, l'utilisation de WikiLerni nécessite l'activation du JavaScript.",
tagsListTxt: "Parcourir les rubriques :"
};
/***/ }),
/***/ "../views/wikilerni/config/fr.js":
/*!***************************************!*\
!*** ../views/wikilerni/config/fr.js ***!
\***************************************/
/*! no static exports found */
/***/ (function(module, exports) {
module.exports = {
// liens de l'interface
headLinks: [{
anchor: "Contact",
attributes: {
href: "/contact.html",
rel: "nofollow"
}
}, {
anchor: "Parcourir",
attributes: {
href: "/quizs/",
id: "indexHeadLink",
title: "Les dernières publications"
}
}, {
anchor: "Mon compte",
attributes: {
href: "/connexion.html",
id: "accountHeadLink",
title: "Accéder ou créez votre compte WikiLerni"
}
}, {
anchor: "À propos",
attributes: {
href: "/a-propos.html",
title: "En savoir + sur WikiLerni"
}
}, {
anchor: "Accueil",
attributes: {
href: "/",
title: "Page d'accueil"
}
}],
footLinks: [{
anchor: "Blog",
attributes: {
href: "https://framasphere.org/people/7e54b7a0b53201389eef2a0000053625",
title: "Le blog WikiLerni sur diaspora*"
}
}, {
anchor: "Crédits",
attributes: {
href: "/credits.html",
title: "Qui a créé WikiLerni ? Quels sont vos droits ?"
}
}, {
anchor: "Mentions légales",
attributes: {
href: "/mentions-legales.html",
rel: "nofollow"
}
}, {
anchor: "Données personnelles",
attributes: {
href: "/donnees.html",
title: "Vos données personnelles sur WikiLerni"
}
}, {
anchor: "CGV & CGU",
attributes: {
href: "/CGV-CGU.html",
rel: "nofollow"
}
}],
accountPage: "compte.html",
aboutPage: "a-propos.html",
adminHomePage: "admin.html",
cguPage: "CGV-CGU.html",
connectionPage: "connexion.html",
deleteLinkPage: "aurevoir.html?t=",
loginLinkPage: "login.html?t=",
managerHomePage: "gestion.html",
newLoginLinkPage: "newlogin.html?t=",
questionnairesManagementPage: "gestion-quizs.html",
stopMailPage: "stop-mail.html?t=",
subscribePage: "inscription.html",
updateAccountPage: "compte.html",
userHomePage: "accueil.html",
userHomePageTxt: "Ma page d'accueil.",
usersManagementPage: "gestion-utilisateurs.html",
validationLinkPage: "validation.html?t=",
/* Textes (général) */
siteSlogan: "Cultivons notre jardin !",
noJSNotification: "Désolé, mais pour l'instant, l'utilisation de WikiLerni nécessite l'activation du JavaScript.",
mailRecipientTxt: "Message envoyé à :",
licenceTxt: "@copyleft Le contenu de WikiLerni <a href=\"/credits.html\" title=\"En savoir plus ?\">est libre</a> et vous est offert sans publicité. Vous pouvez <a href=\"/participer-financement.html\" title=\"Financement participatif avec contre-parties\">participer à son financement en cliquant ici</a>.",
/* Page d'accueil */
homePageTxt: "Page d'accueil",
homeTitle1: "De nature curieuse ?",
homeP1: "<b>Avec WikiLerni, vous apprenez régulièrement de nouvelles choses</b>.<br>Vous recevez de courts articles, lisibles en quelques minutes.<br>Des quizs vous permettent ensuite de tester ce que vous avez retenu.<br>De jour en jour <b>de nouvelles graines de culture sont ainsi semées dans votre jardin</b>.",
homeTitle2: "La culture générale en liberté",
homeP2: "<b>Pas de faits alternatifs</b>, tous les contenus sont <b>sourcés par des articles Wikipédia</b>.<br>Et tout comme sur Wikipédia, le logiciel et le contenu publié sur WikiLerni <a href=\"/credits.html\" title=\"En savoir plus sur ce sujet\">sont partagés sous licences libres</a>.<br>Le tout sans publicité, ni commercialisation de vos données.<br><b>Sur WikiLerni, vous cultivez votre jardin en toute tranquillité.</b>",
homeBtnAboutTxt: "En savoir plus sur WikiLerni ?",
homeBtnSubscribeTxt: "Testez WikiLerni",
homeSubcriptionFormTitle: "Recevez les prochains articles WikiLerni",
/* Page dernières publications... */
newQuestionnairesTitle: "Culture générale - apprenez de nouvelles choses avec WikiLerni",
newQuestionnairesIntro: "WikiLerni : testez vos connaissances et apprenez de nouvelles choses avec WikiLerni.",
newsListTitle: "<p><b>Avec WikiLerni, vous apprenez chaque jour quelque chose de nouveau</b><br>Si dessous les dernières publications. Vous pouvez aussi <a href='/quizs/themes.html'>parcourir le site par thèmes ou mots-clés</a>.</p>",
/* Plan du site, liste des tags */
tagListTitle: "Culture générale - des articles et quizs sur de nombreux thèmes !",
tagListMetaDesc: "WikiLerni : découvrir les différents thèmes abordés par WikiLerni. Inxex du site.",
tagListIntro: "<h3>Avec WikiLerni, devenez fort en thèmes... Oui mais quels thèmes ? :)</h3><blockquote>Aristote: «Lhomme a naturellement la passion de connaître…»</blockquote>",
/* Page quizs */
answersExplanationsLinkText: "Relire",
quizElementLinksIntro: "En savoir plus",
quizElementSubcriptionFormTitle: "Recevez les prochains articles WikiLerni",
explanationTitle: "Vous découvrez WikiLerni?",
explanationTxt: "<p>Le principe est simple: <b>vous commencez par lire larticle Wikipédia dont le lien vous est proposé</b>. Puis vous <b>afficher le quiz pour vérifier ce que vous avez retenu de votre lecture</b>. Suivant les questions, <b>une ou plusieurs réponses peuvent être correctes</b> et doivent donc être cochées. Cest toujours <b>le contenu de larticle Wikipédia qui fait foi</b> concernant les «bonnes» réponses. Cest une façon de tester à la fois votre capacité dattention et votre mémoire. Les articles de Wikipédia peuvent évoluer, donc nhésitez pas <a href='/contact.html'>à me signaler une erreur</a>.</p><p><b>WikiLerni vous propose dautres solutions pour améliorer votre culture générale</b>. Pour en savoir plus, cliquez sur le bouton ci-dessous.</p>",
explanationElementTxt: "<p>WikiLerni vous propose de <b>recevoir par e-mail à la fréquence choisie, de courts articles lisibles en quelques minutes</b> et portant sur des sujets très variés de <b>culture générale</b> (arts, histoire, littérature, sciences, etc.).</p><p>Ces articles sont basés sur <b>une ou plusieurs pages de Wikipédia</b> (fournies en lien), dont <b>ils extraient certaines informations</b>.</p><p>Chaque série darticles est <b>suivie dun quiz</b> permettant de tester ce que vous en avez retenu.</p><p><b>Vous apprenez ainsi régulièrement de nouvelles choses</b> très simplement.</p>",
/* Autres */
illustrationDir: "/img/quizs/",
twitterAccount: "WikiLerni",
maxQuestionnairesByPage: 12,
maxQuestionnairesSiteHomePage: 3,
nbQuestionnairesUserHomePage: 3
};
/***/ }),
/***/ "./src/config/general.js":
/*!*******************************!*\
!*** ./src/config/general.js ***!
\*******************************/
/*! no static exports found */
/***/ (function(module, exports) {
// à supprimer une fois que tout récupérer du backend :
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/"
};
/***/ }),
/***/ "./src/questionnaire.js":
/*!******************************!*\
!*** ./src/questionnaire.js ***!
\******************************/
/*! no exports provided */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony import */ var _config_instance_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../../config/instance.js */ "../config/instance.js");
/* harmony import */ var _config_instance_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_config_instance_js__WEBPACK_IMPORTED_MODULE_0__);
/* harmony import */ var _config_questionnaires_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../../config/questionnaires.js */ "../config/questionnaires.js");
/* harmony import */ var _config_questionnaires_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_config_questionnaires_js__WEBPACK_IMPORTED_MODULE_1__);
/* harmony import */ var _tools_answers_js__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./tools/answers.js */ "./src/tools/answers.js");
/* harmony import */ var _tools_dom_js__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./tools/dom.js */ "./src/tools/dom.js");
/* harmony import */ var _tools_everywhere_js__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ./tools/everywhere.js */ "./src/tools/everywhere.js");
/* harmony import */ var _tools_clientstorage_js__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ./tools/clientstorage.js */ "./src/tools/clientstorage.js");
/* harmony import */ var _tools_forms_js__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! ./tools/forms.js */ "./src/tools/forms.js");
/* harmony import */ var _tools_main__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! ../../tools/main */ "../tools/main.js");
/* harmony import */ var _tools_main__WEBPACK_IMPORTED_MODULE_7___default = /*#__PURE__*/__webpack_require__.n(_tools_main__WEBPACK_IMPORTED_MODULE_7__);
/* harmony import */ var _tools_matomo_js__WEBPACK_IMPORTED_MODULE_8__ = __webpack_require__(/*! ./tools/matomo.js */ "./src/tools/matomo.js");
/* harmony import */ var _tools_users_js__WEBPACK_IMPORTED_MODULE_9__ = __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é, on propose à l'internaute de se connecter ou de créer un compte pour sauvegarder son résultat.
/// Dans ce but son résultat est stocké dans son navigateur.
/// Si il est connecté, l'enregistrement de son résultat se fait automatiquement côté serveur et ses éventuels précédents résultats sont affichés.
// Fichier de configuration tirés du backend :
var lang = _config_instance_js__WEBPACK_IMPORTED_MODULE_0__["availableLangs"][0];
var configTemplate = __webpack_require__("../views sync recursive ^\\.\\/.*\\.js$")("./" + _config_instance_js__WEBPACK_IMPORTED_MODULE_0__["theme"] + "/config/" + lang + ".js");
// Dictionnaires :
var _require = __webpack_require__("../lang sync recursive ^\\.\\/.*\\/answer$")("./" + lang + "/answer"),
noPreviousAnswer = _require.noPreviousAnswer,
previousAnswersLine = _require.previousAnswersLine,
previousAnswersStats = _require.previousAnswersStats,
previousAnswersTitle = _require.previousAnswersTitle,
responseSavedError = _require.responseSavedError,
wantToSaveResponses = _require.wantToSaveResponses;
var _require2 = __webpack_require__("../lang sync recursive ^\\.\\/.*\\/general$")("./" + lang + "/general"),
serverError = _require2.serverError; // 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");
var 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;
btnShow.style.display = "inline"; // bouton caché si JS inactif, car JS nécessaire pour vérifier les réponses
_context.next = 4;
return Object(_tools_users_js__WEBPACK_IMPORTED_MODULE_9__["checkSession"])(["user"]);
case 4:
isConnected = _context.sent;
// "user" car seuls les utilisateurs de base peuvent enregistrer leurs réponses aux quizs
// Si l'utilisateur est connecté et a déjà répondu à ce quiz, on affiche ses précédentes réponses à la place du texte servant à expliquer le topo aux nouveaux
if (isConnected) {
user = Object(_tools_clientstorage_js__WEBPACK_IMPORTED_MODULE_5__["getLocaly"])("user", true);
Object(_tools_everywhere_js__WEBPACK_IMPORTED_MODULE_4__["updateAccountLink"])(user.status, configTemplate); // lien vers le compte adapté pour les utilisateurs connectés
checkPreviousResponses(user);
} else Object(_tools_matomo_js__WEBPACK_IMPORTED_MODULE_8__["loadMatomo"])();
_context.next = 11;
break;
case 8:
_context.prev = 8;
_context.t0 = _context["catch"](0);
console.error(_context.t0);
case 11:
case "end":
return _context.stop();
}
}
}, _callee, null, [[0, 8]]);
}));
return function initialise() {
return _ref.apply(this, arguments);
};
}();
initialise();
Object(_tools_everywhere_js__WEBPACK_IMPORTED_MODULE_4__["helloDev"])(); // Affichage du questionnaire quand 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.style.display = "none";
var here = window.location; // window.location à ajouter pour ne pas quitter la page en mode "preview".
if (window.location.hash !== "") {
window.location.hash = ""; // ! le "#" reste
window.location.assign(here + "questionnaire");
} else window.location.assign(here + "#questionnaire");
};
var chronoBegin = 0;
btnShow.addEventListener("click", function (e) {
try {
e.preventDefault();
showQuestionnaire();
} catch (e) {
Object(_tools_dom_js__WEBPACK_IMPORTED_MODULE_3__["addElement"])(divResponse, "p", serverError, "", ["error"]);
console.error(e);
}
}); // Lien passé par mail pour voir directement le quiz
if (location.hash != "" && location.hash === "#questionnaire") showQuestionnaire(); // Traitement de l'envoi de la réponse de l'utilisateur :
var answer = {};
myForm.addEventListener("submit", function (e) {
try {
e.preventDefault();
btnSubmit.style.display = "none"; // seulement un envoi à la fois, SVP :)
divResponse.innerHTML = ""; // supprime les éventuels messages déjà affichés
var userResponses = Object(_tools_forms_js__WEBPACK_IMPORTED_MODULE_6__["getDatasFromInputs"])(myForm);
answer.duration = Math.round((Date.now() - chronoBegin) / 1000);
answer.nbQuestions = 0;
answer.nbCorrectAnswers = 0;
answer.QuestionnaireId = document.getElementById("questionnaireId").value; // Les réponses sont regroupées par question, donc quand idQuestion change, on connaît le résultat pour la question précédente.
// Pour qu'une réponse soit bonne, il faut cocher toutes les bonnes réponses (si QCM) à la question ET cocher aucune des mauvaises.
var idChoice,
idQuestion = "",
goodResponse = false;
for (var item in userResponses) {
if (item.startsWith("isCorrect_response_")) // = Nouvelle réponse possible.
{
idChoice = item.substring(item.lastIndexOf("_") + 1); // si on change de queston
if (userResponses["question_id_response_" + idChoice] != idQuestion) // on commence à traiter une nouvelle question
{
idQuestion = userResponses["question_id_response_" + idChoice];
answer.nbQuestions++;
if (goodResponse) // résultat de la question précédente
answer.nbCorrectAnswers++;
goodResponse = true; // réponse bonne jusqu'à la première erreur...
}
if (userResponses[item] == "true") {
document.getElementById("response_" + idChoice).parentNode.classList.add("isCorrect");
if (userResponses["response_" + idChoice] === undefined) // une bonne réponse n'a pas été sélectionnée
goodResponse = false;
} else {
if (userResponses["response_" + idChoice] === "on") // réponse cochée ne faisant pas partie des bonnes
{
goodResponse = false;
document.getElementById("response_" + idChoice).parentNode.classList.add("isNotCorrect");
}
}
}
} // si j'ai bien répondu à la dernière question, il faut le compter ici, car je suis sorti de la boucle :
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_2__["checkAnswerOuput"])(answer);
if (isConnected) {
// Si l'utilisateur est connecté, on enregistre son résultat sur le serveur.
var xhrSaveAnswer = new XMLHttpRequest();
xhrSaveAnswer.open("POST", _config_instance_js__WEBPACK_IMPORTED_MODULE_0__["apiUrl"] + _config_questionnaires_js__WEBPACK_IMPORTED_MODULE_1__["questionnaireRoutes"] + _config_questionnaires_js__WEBPACK_IMPORTED_MODULE_1__["saveAnswersRoute"]);
xhrSaveAnswer.onreadystatechange = function () {
if (this.readyState == XMLHttpRequest.DONE) {
var xhrResponse = JSON.parse(this.responseText);
if (this.status === 201 && xhrResponse.message) {
getOuput += "<br>" + xhrResponse.message.replace("#URL", configTemplate.userHomePage);
checkPreviousResponses(user);
} else getOuput += "<br>" + responseSavedError.replace("#URL", configTemplate.userHomePage); // on redirige vers le résultat
window.location.hash = "";
var here = window.location; // window.location à ajouter pour ne pas quitter la page en mode "preview"...
window.location.assign(here + "explanations");
}
};
xhrSaveAnswer.setRequestHeader("Authorization", "Bearer " + user.token);
xhrSaveAnswer.setRequestHeader("Content-Type", "application/json");
answer.timeDifference = Object(_tools_users_js__WEBPACK_IMPORTED_MODULE_9__["getTimeDifference"])(); // on en profite pour mettre les pendules à l'heure.
xhrSaveAnswer.send(JSON.stringify(answer));
} else {
// si pas connecté, on enregistre le résultat 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_2__["saveAnswer"])(answer)) {
getOuput += "<br><br>" + wantToSaveResponses;
Object(_tools_dom_js__WEBPACK_IMPORTED_MODULE_3__["addElement"])(divResponse, "p", getOuput, "", ["info"]);
document.querySelector(".subscribeBtns").style.display = "block";
} else // inutile de proposer de créer un compte si le stockage local ne fonctionne pas
Object(_tools_dom_js__WEBPACK_IMPORTED_MODULE_3__["addElement"])(divResponse, "p", getOuput, "", ["info"]); // on redirige vers le résultat
window.location.hash = "";
var here = window.location; // window.location à ajouter pour ne pas quitter la page en mode "preview"...
window.location.assign(here + "response");
} // + affichage des textes d'explications pour chaque question
var explanations = document.querySelectorAll(".help");
for (var i in explanations) {
if (explanations[i].style != undefined) // sinon, la console affiche une erreur "TypeError: explanations[i].style is undefined", bien que tout fonctionne (?)
explanations[i].style.display = "block";
}
} catch (e) {
Object(_tools_dom_js__WEBPACK_IMPORTED_MODULE_3__["addElement"])(divResponse, "p", serverError, "", ["error"]);
console.error(e);
}
}); // 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(user) {
var xhrPreviousRes = new XMLHttpRequest();
xhrPreviousRes.open("GET", _config_instance_js__WEBPACK_IMPORTED_MODULE_0__["apiUrl"] + _config_questionnaires_js__WEBPACK_IMPORTED_MODULE_1__["questionnaireRoutes"] + _config_questionnaires_js__WEBPACK_IMPORTED_MODULE_1__["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 = "";
Object(_tools_dom_js__WEBPACK_IMPORTED_MODULE_3__["addElement"])(explanationsTitle, "span", previousAnswersTitle.replace("#NOM", user.name));
if (nbResponses !== 0) {
var totNbQuestions = 0,
totNbCorrectAnswers = 0,
totDuration = 0,
mapLineContent;
for (var i in response) {
totNbQuestions += response[i].nbQuestions; // ! on ne peut se baser sur la version actuelle du quiz, car le nombre de questions a pu évoluer.
totNbCorrectAnswers += response[i].nbCorrectAnswers;
totDuration += response[i].duration;
mapLineContent = {
DATEANSWER: Object(_tools_main__WEBPACK_IMPORTED_MODULE_7__["dateFormat"])(response[i].createdAt, lang),
NBCORRECTANSWERS: response[i].nbCorrectAnswers,
NBQUESTIONS: response[i].nbQuestions,
AVGDURATION: response[i].duration
};
previousAnswersContent += "<li>" + Object(_tools_main__WEBPACK_IMPORTED_MODULE_7__["replaceAll"])(previousAnswersLine, mapLineContent) + "</li>";
}
mapLineContent = {
AVGDURATION: Math.round(totDuration / nbResponses),
AVGCORRECTANSWERS: Math.round(totNbCorrectAnswers / totNbQuestions * 100)
};
previousAnswersContent = "<h5>" + Object(_tools_main__WEBPACK_IMPORTED_MODULE_7__["replaceAll"])(previousAnswersStats, mapLineContent) + "</h5>" + previousAnswersContent;
Object(_tools_dom_js__WEBPACK_IMPORTED_MODULE_3__["addElement"])(explanationsContent, "ul", previousAnswersContent);
} else Object(_tools_dom_js__WEBPACK_IMPORTED_MODULE_3__["addElement"])(explanationsContent, "ul", noPreviousAnswer); // dans un cas comme dans l'autre, bouton pour revenir à l'accueil du compte
Object(_tools_dom_js__WEBPACK_IMPORTED_MODULE_3__["addElement"])(explanationsContent, "p", "<a href=\"/" + configTemplate.userHomePage + "\" class=\"button cardboard\">" + configTemplate.userHomePageTxt + "</a>", "", ["btn"], "", false);
}
}
};
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(_tools_main__WEBPACK_IMPORTED_MODULE_1__["isEmpty"])(answer.GroupId))) {
Object(_clientstorage_js__WEBPACK_IMPORTED_MODULE_0__["saveLocaly"])("lastAnswer", answer);
return true;
} else return false;
}; // Retourne un 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 "";
};
/***/ }),
/***/ "./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, updateAccountLink */
/***/ (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; });
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "updateAccountLink", function() { return updateAccountLink; });
// Ce script fournit des fonctions utilisées sur toutes les pages du site
var helloDev = function helloDev() {
console.log("**** Hello ami développeur :-)\n\nLe code de WikiLerni est libre et vous pouvez le trouver à cette adresse :\nhttps://forge.chapril.org/Fab_Blab/WikiLerni\n\nPour les suggestions d'amélioration ou questions : dev@wililerni.com ****");
return true;
};
var updateAccountLink = function updateAccountLink(status, configTemplate) {
var link = document.getElementById("accountHeadLink");
var homePage = status + "HomePage";
link.href = "/" + configTemplate[homePage];
};
/***/ }),
/***/ "./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/matomo.js":
/*!*****************************!*\
!*** ./src/tools/matomo.js ***!
\*****************************/
/*! exports provided: loadMatomo */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "loadMatomo", function() { return loadMatomo; });
/* harmony import */ var _config_matomo_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../../../config/matomo.js */ "../config/matomo.js");
/* harmony import */ var _config_matomo_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_config_matomo_js__WEBPACK_IMPORTED_MODULE_0__);
// Fonction chargeant le code matomo
var loadMatomo = function loadMatomo() {
// chargement de matomo
var _paq = window._paq = window._paq || []; //ajout cnil :
_paq.push([function () {
var self = this;
function getOriginalVisitorCookieTimeout() {
var now = new Date(),
nowTs = Math.round(now.getTime() / 1000),
visitorInfo = self.getVisitorInfo();
var createTs = parseInt(visitorInfo[2]);
var cookieTimeout = 33696000; // 13 mois en secondes
var originalTimeout = createTs + cookieTimeout - nowTs;
return originalTimeout;
}
this.setVisitorCookieTimeout(getOriginalVisitorCookieTimeout());
}]); //--fin ajout cnil
_paq.push(['trackPageView']);
_paq.push(['enableLinkTracking']);
(function () {
var u = _config_matomo_js__WEBPACK_IMPORTED_MODULE_0__["matomo"].url;
_paq.push(['setTrackerUrl', u + 'matomo.php']);
_paq.push(['setSiteId', _config_matomo_js__WEBPACK_IMPORTED_MODULE_0__["matomo"].siteId]);
var d = document,
g = d.createElement('script'),
s = d.getElementsByTagName('script')[0];
g.type = 'text/javascript';
g.async = true;
g.src = u + 'matomo.js';
s.parentNode.insertBefore(g, s);
})();
};
/***/ }),
/***/ "./src/tools/users.js":
/*!****************************!*\
!*** ./src/tools/users.js ***!
\****************************/
/*! exports provided: getTimeDifference, setSession, checkAnswerDatas, checkSession */
/***/ (function(module, __webpack_exports__, __webpack_require__) {
"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "getTimeDifference", function() { return getTimeDifference; });
/* 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 _config_instance_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../../../config/instance.js */ "../config/instance.js");
/* harmony import */ var _config_instance_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_config_instance_js__WEBPACK_IMPORTED_MODULE_0__);
/* harmony import */ var _config_users_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../../../config/users.js */ "../config/users.js");
/* harmony import */ var _config_users_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_config_users_js__WEBPACK_IMPORTED_MODULE_1__);
/* harmony import */ var _clientstorage_js__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./clientstorage.js */ "./src/tools/clientstorage.js");
/* harmony import */ var _tools_main__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ../../../tools/main */ "../tools/main.js");
/* harmony import */ var _tools_main__WEBPACK_IMPORTED_MODULE_3___default = /*#__PURE__*/__webpack_require__.n(_tools_main__WEBPACK_IMPORTED_MODULE_3__);
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 lang = _config_instance_js__WEBPACK_IMPORTED_MODULE_0__["availableLangs"][0];
var configTemplate = __webpack_require__("../views sync recursive ^\\.\\/.*\\.js$")("./" + _config_instance_js__WEBPACK_IMPORTED_MODULE_0__["theme"] + "/config/" + lang + ".js");
var getTimeDifference = function getTimeDifference() {
// multiplier par -1, car c'est ce qu'il faut "ajouter" à l'heure UTC pour revenir en heure locale qui m'intéresse et non l'inverse
var timeLocal = new Date().getTimezoneOffset() * -1;
if (timeLocal > _config_users_js__WEBPACK_IMPORTED_MODULE_1__["timeDifferenceMax"] || timeLocal < _config_users_js__WEBPACK_IMPORTED_MODULE_1__["timeDifferenceMin"]) return 0;else return timeLocal;
}; // 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_2__["saveLocaly"])("user", storageUser);
}; // Vérifie qu'il y a des données locales concernant le résultat d'un quiz ou d'un groupe de quizs
// 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_2__["getLocaly"])("lastAnswer");
if (!Object(_tools_main__WEBPACK_IMPORTED_MODULE_3__["isEmpty"])(lastAnswer)) {
var answer = JSON.parse(lastAnswer);
if (!Object(_tools_main__WEBPACK_IMPORTED_MODULE_3__["isEmpty"])(answer.duration) && !Object(_tools_main__WEBPACK_IMPORTED_MODULE_3__["isEmpty"])(answer.nbCorrectAnswers) && !Object(_tools_main__WEBPACK_IMPORTED_MODULE_3__["isEmpty"])(answer.nbQuestions) && (!Object(_tools_main__WEBPACK_IMPORTED_MODULE_3__["isEmpty"])(answer.QuestionnaireId) || !Object(_tools_main__WEBPACK_IMPORTED_MODULE_3__["isEmpty"])(answer.GroupId))) {
datas.duration = answer.duration;
datas.nbCorrectAnswers = answer.nbCorrectAnswers;
datas.nbQuestions = answer.nbQuestions;
if (!Object(_tools_main__WEBPACK_IMPORTED_MODULE_3__["isEmpty"])(answer.QuestionnaireId)) datas.QuestionnaireId = answer.QuestionnaireId;else datas.GroupId = answer.GroupId;
}
}
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 _ref = _asyncToGenerator( /*#__PURE__*/regeneratorRuntime.mark(function _callee() {
var status,
urlRedirection,
message,
urlWanted,
_args = arguments;
return regeneratorRuntime.wrap(function _callee$(_context) {
while (1) {
switch (_context.prev = _context.next) {
case 0:
status = _args.length > 0 && _args[0] !== undefined ? _args[0] : [];
urlRedirection = _args.length > 1 ? _args[1] : undefined;
message = _args.length > 2 ? _args[2] : undefined;
urlWanted = _args.length > 3 ? _args[3] : undefined;
return _context.abrupt("return", new Promise(function (resolve, reject) {
var userDatas = Object(_clientstorage_js__WEBPACK_IMPORTED_MODULE_2__["getLocaly"])("user");
if (Object(_tools_main__WEBPACK_IMPORTED_MODULE_3__["isEmpty"])(userDatas)) {
redirectUser(urlRedirection, message, urlWanted);
resolve(false);
} else {
var user = JSON.parse(userDatas);
if (Object(_tools_main__WEBPACK_IMPORTED_MODULE_3__["isEmpty"])(user.id) || Object(_tools_main__WEBPACK_IMPORTED_MODULE_3__["isEmpty"])(user.token) || Object(_tools_main__WEBPACK_IMPORTED_MODULE_3__["isEmpty"])(user.duration) || user.duration < Date.now()) {
Object(_clientstorage_js__WEBPACK_IMPORTED_MODULE_2__["removeLocaly"])("user");
redirectUser(urlRedirection, message, urlWanted);
resolve(false);
} else {
var xhr = new XMLHttpRequest();
xhr.open("GET", _config_instance_js__WEBPACK_IMPORTED_MODULE_0__["apiUrl"] + _config_users_js__WEBPACK_IMPORTED_MODULE_1__["userRoutes"] + _config_users_js__WEBPACK_IMPORTED_MODULE_1__["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; // c'est le token qui sert à vérifier le statut à chaque requête à l'API
Object(_clientstorage_js__WEBPACK_IMPORTED_MODULE_2__["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_instance_js__WEBPACK_IMPORTED_MODULE_0__["siteUrl"] + "/" + configTemplate.accountPage;
if (window.location.href.indexOf(urlAccount) === -1) window.location.assign("/" + configTemplate.accountPage + "#subscribe"); // 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_2__["removeLocaly"])("user");
redirectUser(urlRedirection, message, urlWanted);
resolve(false);
}
} else {
Object(_clientstorage_js__WEBPACK_IMPORTED_MODULE_2__["removeLocaly"])("user");
redirectUser(urlRedirection, message, urlWanted);
resolve(false);
}
};
xhr.onerror = function () {
return reject(xhr.statusText);
};
xhr.send();
}
}
}));
case 5:
case "end":
return _context.stop();
}
}
}, _callee);
}));
return function checkSession() {
return _ref.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_3__["isEmpty"])(message)) Object(_clientstorage_js__WEBPACK_IMPORTED_MODULE_2__["saveLocaly"])("message", message);
if (!Object(_tools_main__WEBPACK_IMPORTED_MODULE_3__["isEmpty"])(urlWanted)) Object(_clientstorage_js__WEBPACK_IMPORTED_MODULE_2__["saveLocaly"])("url", urlWanted);
if (!Object(_tools_main__WEBPACK_IMPORTED_MODULE_3__["isEmpty"])(urlRedirection)) window.location.assign(urlRedirection);
};
/***/ })
/******/ });
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack:///webpack/bootstrap","webpack:///../config/instance.js","webpack:///../config/matomo.js","webpack:///../config/questionnaires.js","webpack:///../config/users.js","webpack:///../lang sync ^\\.\\/.*\\/answer$","webpack:///../lang sync ^\\.\\/.*\\/general$","webpack:///../lang/fr/answer.js","webpack:///../lang/fr/general.js","webpack:///../tools/main.js","webpack:///../views sync ^\\.\\/.*\\.js$","webpack:///../views/default/config/fr.js","webpack:///../views/wikilerni/config/fr.js","webpack:///./src/config/general.js","webpack:///./src/questionnaire.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/matomo.js","webpack:///./src/tools/users.js"],"names":["users","require","questionnaires","module","exports","apiUrl","siteUrl","adminName","adminEmail","senderName","senderEmail","adminLang","theme","availableLangs","siteName","beginCodeGodfather","defaultReceiptDays","cronTimingAlertInSeconde","responseTimingAlertInSeconde","tokenSignupValidationTimeInHours","tokenLoginLinkTimeInHours","tokenConnexionMinTimeInHours","tokenConnexionMaxTimeInDays","tokenLoginChangingTimeInHours","tokenDeleteUserTimeInHours","tokenUnsubscribeLinkTimeInDays","freeAccountTimingInDays","freeAccountExpirationNotificationInDays","accountExpirationFirstNotificationInDays","accountExpirationSecondNotificationInDays","inactiveAccountTimeToDeleteInDays","nbQuestionsMin","nbQuestionsMax","nbChoicesMax","nbNewQuestionnaires","hourGiveNewQuestionnaireBegin","hourGiveNewQuestionnaireEnd","maxQuestionnaireSendedAtSameTime","minSearchQuestionnaires","fieldNewQuestionnaires","nbQuestionnairesByGroupMin","nbQuestionnairesByGroupMax","nbIllustrationsMin","nbIllustrationsMax","maxIllustrationSizeinOctet","mimeTypesForIllustration","illustrationsWidthMaxInPx","illustrationsMiniaturesWidthMaxInPx","nbLinksMin","nbLinksMax","password","minlength","dirCacheUsers","dirCacheUsersAnswers","dirCacheQuestionnaires","dirCacheQuestions","dirCacheUsersQuestionnaires","dirHTMLQuestionnaires","dirWebQuestionnaires","matomo","url","siteId","questionnaireRoutes","getListNextQuestionnaires","getQuestionnaireRoutes","getRandomQuestionnairesRoute","getStatsQuestionnaires","previewQuestionnaireRoutes","publishedQuestionnaireRoutes","regenerateHTML","searchAdminQuestionnairesRoute","searchQuestionnairesRoute","groupRoutes","getGroupRoute","previewGroupRoutes","searchGroupsRoute","questionsRoute","tagsSearchRoute","getAdminStats","getPreviousAnswers","getStatsAnswers","saveAnswersRoute","Questionnaire","title","maxlength","required","slug","introduction","searchQuestionnaires","Group","Question","text","rank","min","defaultValue","Choice","search","searchGroups","dirCacheGroups","dirCacheTags","dirHTMLGroups","dirHTMLNews","dirHTMLTags","dirWebGroups","dirWebNews","dirWebTags","nbRandomResults","nbTagsMin","nbTagsMax","userRoutes","checkDeleteLinkRoute","checkIfIsEmailfreeRoute","checkLoginRoute","checkNewLoginLinkRoute","checkSubscribeTokenRoute","connectionRoute","connectionWithLinkRoute","createUserRoute","getGodChilds","getGodfatherRoute","getLoginLinkRoute","getPayments","getUserInfos","getUsersQuestionnairesRoute","searchUserRoute","signupCompletionRoute","subscribeRoute","unsubscribeRoute","updateUserInfos","validateUserRoute","name","email","newPassword","codeGodfather","cguOk","value","timeDifferenceMin","timeDifferenceMax","dirCacheUsersWithoutAnswers","checkResponsesOuputFail","checkResponsesOuputMedium","checkResponsesOuputSuccess","needIntegerNumberCorrectResponses","needIntegerNumberSecondesResponse","needIntegerNumberUserResponses","needMaxNumberUserResponses","needMinNumberUserResponses","needNumberCorrectResponses","needNumberUserResponses","needMaxNumberCorrectResponses","needMinNumberCorrectResponses","needMinNumberSecondesResponse","noPreviousAnswer","previousAnswersLine","previousAnswersStats","previousAnswersTitle","responseSavedError","responseSavedMessage","statsUser","wantToSaveResponses","addBtnTxt","addOkMessage","alertNewWindow","badUrl","btnLinkToQuestionnaire","btnProposeConnection","btnProposeSubscribe","btnShowOnWebSite","deleteBtnTxt","deleteFailMessage","deleteOkMessage","failAuth","failAuthCron","failAuthHeader","failAuthId","failAuthToken","neededParams","nextPage","notAllowed","notRequired","notValidFormat","previousPage","serverError","serverErrorAdmin","siteHTMLTitle","siteMetaDescription","scriptTimingAlert","scriptTimingInfo","statsAdmin","subscriptionCall","updateBtnTxt","updateOkMessage","Tool","myVar","undefined","trim","myString","isEmpty","max","length","substring","mapObj","replaceElts","RegExp","Object","keys","join","replace","matched","Math","ceil","floor","random","dateString","lang","myDate","Date","myDay","getDate","myMounth","getMonth","myYear","getFullYear","nbCarMin","nbCarMax","nbCar","letters","others","i","headLinks","anchor","attributes","href","id","footLinks","rel","maxQuestionnairesByPage","userHomePage","adminHomePage","managerHomePage","subscribePage","connectionPage","accountPage","questionnairesManagementPage","usersManagementPage","nbQuestionnairesUserHomePage","illustrationDir","siteSlogan","homeTitle1","homeP1","homeTitle2","homeP2","newQuestionnairesTitle","newQuestionnairesIntro","explanationTitle","explanationTxt","noJSNotification","tagsListTxt","aboutPage","cguPage","deleteLinkPage","loginLinkPage","newLoginLinkPage","stopMailPage","updateAccountPage","userHomePageTxt","validationLinkPage","mailRecipientTxt","licenceTxt","homePageTxt","homeBtnAboutTxt","homeBtnSubscribeTxt","homeSubcriptionFormTitle","newsListTitle","tagListTitle","tagListMetaDesc","tagListIntro","answersExplanationsLinkText","quizElementLinksIntro","quizElementSubcriptionFormTitle","explanationElementTxt","twitterAccount","maxQuestionnairesSiteHomePage","usersGetConfigUrl","configTemplate","myForm","document","getElementById","divResponse","btnShow","btnSubmit","explanationsTitle","explanationsContent","isConnected","user","initialise","style","display","checkSession","getLocaly","updateAccountLink","status","checkPreviousResponses","loadMatomo","console","error","helloDev","showQuestionnaire","chronoBegin","now","here","window","location","hash","assign","addEventListener","e","preventDefault","addElement","answer","innerHTML","userResponses","getDatasFromInputs","duration","round","nbQuestions","nbCorrectAnswers","QuestionnaireId","idChoice","idQuestion","goodResponse","item","startsWith","lastIndexOf","parentNode","classList","add","getOuput","checkAnswerOuput","xhrSaveAnswer","XMLHttpRequest","open","onreadystatechange","readyState","DONE","xhrResponse","JSON","parse","responseText","message","setRequestHeader","token","timeDifference","getTimeDifference","send","stringify","saveAnswer","querySelector","explanations","querySelectorAll","xhrPreviousRes","response","nbResponses","previousAnswersContent","totNbQuestions","totNbCorrectAnswers","totDuration","mapLineContent","DATEANSWER","dateFormat","createdAt","NBCORRECTANSWERS","NBQUESTIONS","AVGDURATION","replaceAll","AVGCORRECTANSWERS","configFrontEnd","txt","GroupId","saveLocaly","ratio","DURATION","output","data","localStorage","setItem","json","getItem","removeLocaly","removeItem","eltParent","eltType","eltContent","eltId","eltClass","eltAttributes","newElement","createElement","Array","isArray","attributName","setAttribute","appendChild","log","link","homePage","setAttributesToInputs","inputsConf","elements","idInput","inputHTML","attribute","datas","formData","FormData","entries","entrie","empyForm","type","checked","empyAndHideForm","_paq","push","self","getOriginalVisitorCookieTimeout","nowTs","getTime","visitorInfo","getVisitorInfo","createTs","parseInt","cookieTimeout","originalTimeout","setVisitorCookieTimeout","u","d","g","s","getElementsByTagName","async","src","insertBefore","timeLocal","getTimezoneOffset","setSession","userId","durationTS","storageUser","checkAnswerDatas","lastAnswer","urlRedirection","urlWanted","Promise","resolve","reject","userDatas","redirectUser","xhr","onload","isValid","language","nbDaysOk","urlAccount","indexOf","onerror","statusText"],"mappings":"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","file":"./JS/questionnaire.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/questionnaire.js\");\n","const users = require(\"./users\");\nconst questionnaires = require(\"./questionnaires\");\n\nmodule.exports =\n{\n    apiUrl: \"http://localhost:3000/api\",\n    siteUrl: \"http://localhost:8080\",\n    adminName: \"Fabrice\",\n    adminEmail: \"dev@wikilerni.com\",\n    senderName: \"WikiLerni (local)\",\n    senderEmail: \"bonjour@wikilerni.com\",\n    adminLang: \"fr\",\n    theme: \"wikilerni\", // le thème utilisé (dans /views) pour générer les pages HTML. Contient ses propres fichiers de configuration.\n    availableLangs: [\"fr\"],// Languages in which the site is available. The first one is the default one.\n    siteName: \"WikiLerni (local)\",\n    beginCodeGodfather: \"WL\", // case-sensitive and can't contain \"@\" !\n    defaultReceiptDays: \"147\", // Par défaut, quel(s) jour(s) de la semaine, l'utilisateur reçoit-il quelque chose ? (1=dimanche, 2=lundi... 7=samedi)\n    cronTimingAlertInSeconde: 120, // for logs\n    responseTimingAlertInSeconde: 3, // idem\n    tokenSignupValidationTimeInHours: \"48h\", // https://github.com/zeit/ms\n    tokenLoginLinkTimeInHours: \"1h\",\n    tokenConnexionMinTimeInHours: \"24h\",\n    tokenConnexionMaxTimeInDays: \"180 days\",\n    tokenLoginChangingTimeInHours: \"1h\",// for email & password changing\n    tokenDeleteUserTimeInHours: \"1h\",\n    tokenUnsubscribeLinkTimeInDays: \"7 days\", // token send with subscription's emails\n    freeAccountTimingInDays: 0,// if 0 = unlimited\n    freeAccountExpirationNotificationInDays: 3,\n    accountExpirationFirstNotificationInDays: 10,\n    accountExpirationSecondNotificationInDays: 3,\n    inactiveAccountTimeToDeleteInDays: 180,\n    // Questionnaires:\n    nbQuestionsMin: 1, // minimum number of questions for the questionnaire to be publishable\n    nbQuestionsMax: 2, // if 0 = not maximum\n    nbChoicesMax: 10,\n    nbNewQuestionnaires: 12,// for RSS, etc.\n    hourGiveNewQuestionnaireBegin:7, // in user local time\n    hourGiveNewQuestionnaireEnd:21, // idem\n    maxQuestionnaireSendedAtSameTime: 50, // for subscription's e-mailing\n    minSearchQuestionnaires: 3,\n    fieldNewQuestionnaires : \"publishingAt\", // field to be used to create the list of the last questionnaires, can be \"createdAt\", \"updatedAt\" or \"publishingAt\"\n    // Groups :\n    nbQuestionnairesByGroupMin: 1,\n    nbQuestionnairesByGroupMax: 0,\n    // Illustrations:\n    nbIllustrationsMin: 1,\n    nbIllustrationsMax: 1,\n    maxIllustrationSizeinOctet: 1000000,// pas contrôlé pour l'instant. À revoir.\n    mimeTypesForIllustration: [ \"image/jpg\", \"image/jpeg\", \"image/png\", \"image/gif\", \"image/png\" ],\n    // -- Upload and resize:\n    illustrationsWidthMaxInPx: 500,\n    illustrationsMiniaturesWidthMaxInPx: 200,\n    // Links:\n    nbLinksMin: 1,\n    nbLinksMax: 0,\n    // Questions & responses:\n    nbQuestionsMin: 1,\n    nbQuestionsMax: 0,\n    nbChoicesMax: 10,\n    // à supprimer quand tous les \"require\" à jour:\n    passwordMinLength: users.password.minlength,\n    dirCacheUsers: users.dirCacheUsers,\n    dirCacheUsersAnswers: users.dirCacheUsersAnswers,\n    dirCacheQuestionnaires: questionnaires.dirCacheQuestionnaires,\n    dirCacheQuestions: questionnaires.dirCacheQuestions,\n    dirCacheUsersQuestionnaires: questionnaires.dirCacheUsersQuestionnaires,\n    dirHTMLQuestionnaires: questionnaires.dirHTMLQuestionnaires,\n    dirWebQuestionnaires: questionnaires.dirWebQuestionnaires\n};","// Si non utilisé, supprimer l'appel des JS front-end puis recompiler.\n// Revoir comment réorganiser cela sous-forme de module indépendant.\nmodule.exports =\n{\n    // Stats Matomo :\n    matomo: { url: \"https://stats.le-fab-lab.com/\", siteId: \"5\" },\n};","module.exports =\n{    \n    // API'routes (after \"apiUrl\" defined in instance.js)\n    questionnaireRoutes: \"/questionnaire\",// la base à laquelle s'ajoute les routes suivantes\n    getListNextQuestionnaires: \"/getlistnextquestionnaires/\",\n    getQuestionnaireRoutes: \"/get\",\n    getRandomQuestionnairesRoute : \"/getrandom\", \n    getStatsQuestionnaires : \"/stats/\",\n    previewQuestionnaireRoutes: \"/preview\",\n    publishedQuestionnaireRoutes: \"/quiz/\",\n    regenerateHTML: \"/htmlregenerated\",\n    searchAdminQuestionnairesRoute : \"/searchadmin\",\n    searchQuestionnairesRoute : \"/search\",\n    // -- groupes :\n    groupRoutes: \"/group\",\n    getGroupRoute: \"/get/\",\n    previewGroupRoutes: \"/preview\",\n    searchGroupsRoute : \"/search\",\n    // -- questions & choices :\n    questionsRoute: \"/question/\",\n    // -- tags :\n    tagsSearchRoute: \"/tags/search/\",\n    // -- answers :\n    getAdminStats: \"/getadminstats/\",\n    getPreviousAnswers: \"/user/answers/\",\n    getStatsAnswers : \"/user/anwswers/stats/\",// fonctionne aussi pour les groupes\n    saveAnswersRoute: \"/answer/\",// idem\n    // forms : à compléter avec valeurs par défaut, etc. cf modèle\n    Questionnaire :\n    {\n        title: { maxlength: 255, required: true },\n        slug: { maxlength: 150 }, // champ requis mais calculé à partir du titre qd laissé vide dans le formulaire\n        introduction: { required: true }\n    },\n    searchQuestionnaires : { minlength: 3, required: true },\n    Group :\n    {\n        title: { maxlength: 255, required: true },\n        slug: { maxlength: 150 }, // champ requis mais calculé à partir du titre qd laissé vide dans le formulaire\n    },\n    Question :\n    {\n        text: { maxlength: 255, required: true },\n        rank: { required: true, min:1, defaultValue:1 }\n    },          \n    Choice :\n    {\n        text: { maxlength: 255, required: true }\n    },\n    search: { minlength: 3, required: true },\n    searchGroups: { minlength: 3, required: true },\n    // Emplacement des fichiers JSON générés :\n    dirCacheGroups : \"datas/questionnaires/groups\",\n    dirCacheQuestionnaires : \"datas/questionnaires\",\n    dirCacheQuestions : \"datas/questionnaires/questions\",\n    dirCacheTags : \"datas/questionnaires/tags\",\n    dirCacheUsersQuestionnaires : \"datas/users/questionnaires\",\n    // Emplacement des fichiers HTML générés :\n    dirHTMLGroups : \"front/public/quiz/gp\",\n    dirHTMLQuestionnaires : \"front/public/quiz\",\n    dirHTMLNews : \"front/public/quizs\",\n    dirHTMLTags : \"front/public/quizs\",\n    // Idem mais pour urls :\n    dirWebGroups : \"quiz/gp\",\n    dirWebQuestionnaires : \"quiz\",\n    dirWebNews : \"quizs/\",\n    dirWebTags : \"quizs/\",\n    // limite des résultat du moteur de recherche, quand demande de résultats au hasard :\n    nbRandomResults : 3,\n    /* Valeurs en fait définies dans instance.js donc à supprimer quand plus utilisées ailleurs : */\n    nbQuestionsMin: 1,\n    nbQuestionsMax: 0,\n    nbChoicesMax: 10,\n    nbTagsMin: 0,\n    nbTagsMax: 0, // 0 = not max    \n};","module.exports =\n{\n    // API'routes (after \"apiUrl\" defined in instance.js)\n    userRoutes: \"/user\",\n    checkDeleteLinkRoute: \"/confirmdelete/\",   \n    checkIfIsEmailfreeRoute: \"/isemailfree\",\n    checkLoginRoute: \"/checklogin/\",\n    checkNewLoginLinkRoute: \"/confirmnewlogin/\",\n    checkSubscribeTokenRoute: \"/validation/\",\n    connectionRoute: \"/login\",\n    connectionWithLinkRoute: \"/checkloginlink\",\n    createUserRoute: \"/create\",\n    getAdminStats: \"/getadminstats/\",\n    getGodChilds: \"/getgodchilds/\",    \n    getGodfatherRoute: \"/getgodfatherid\",\n    getLoginLinkRoute: \"/getloginlink\",\n    getPayments: \"/payment/getforoneuser/\",\n    getUserInfos: \"/get/\",\n    getUsersQuestionnairesRoute: \"/getusersquestionnaires/\",// les questionnaires auxquels l'utilisateur a déjà eu accès via son abonnement    \n    searchUserRoute: \"/search/\",\n    signupCompletionRoute: \"/signupcompletion/\",\n    subscribeRoute: \"/signup\",\n    unsubscribeRoute: \"/subscription/stop/\",\n    updateUserInfos: \"/modify/\",\n    validateUserRoute: \"/validate/\",\n    // forms : à compléter avec valeurs par défaut, etc. cf modèle\n    name: { maxlength: 70, required: true },\n    email: { maxlength: 255, required: true },\n    password: { minlength: 8, maxlength:72, required: true }, // https://www.npmjs.com/package/bcrypt#security-issues-and-concerns\n    newPassword: { minlength: 8, maxlength:72 },\n    codeGodfather: { maxlength: 255 },\n    cguOk: { value: \"true\", required: true },\n    search: { minlength: 1, required: true },\n    timeDifferenceMin: -720,\n    timeDifferenceMax: 840,\n    // JSON dir\n    dirCacheUsers : \"datas/users\",\n    dirCacheUsersAnswers : \"datas/users/questionnaires/answers\",\n    dirCacheUsersWithoutAnswers : \"datas/users/questionnaires/without\"\n};\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    checkResponsesOuputFail : \"Vous avez répondu en DURATION secondes et avez <u><b>NBCORRECTANSWERS bonne(s) réponse(s) sur NBQUESTIONS questions</b></u>. C'est certain, vous ferez mieux la prochaine fois !\",\n    checkResponsesOuputMedium : \"Vous avez répondu en DURATION secondes et avez <u><b>NBCORRECTANSWERS bonne(s) réponse(s) sur NBQUESTIONS questions</b></u>. C'est pas mal du tout !\",\n    checkResponsesOuputSuccess : \"Vous avez répondu en DURATION secondes et avez <u><b>NBCORRECTANSWERS bonne(s) réponse(s) sur NBQUESTIONS questions</b></u>. Bravo ! Rien ne vous échappe !\",\n    needIntegerNumberCorrectResponses : \"Le nombre de réponses correctes doit être un nombre entier.\",\n    needIntegerNumberSecondesResponse : \"La durée de la réponse doit être un nombre entier de secondes.\",\n    needIntegerNumberUserResponses : \"Le nombre de questions auxquelles l'utilisateur a répondu doit être un nombre entier.\",\n    needMaxNumberUserResponses : \"Le nombre de questions auxquelles l'utilisateur a répondu ne peut être aussi élevé.\",\n    needMinNumberUserResponses : \"Le nombre de questions auxquelles l'utilisateur a répondu ne peut être inférieur à 1.\",\n    needNumberCorrectResponses : \"Le nombre de réponses correctes doit être fourni.\",\n    needNumberUserResponses : \"Le nombre de questions auxquelles l'utilisateur a répondu doit être fourni.\",\n    needMaxNumberCorrectResponses : \"Le nombre de réponses correctes ne peut être supérieur au nombre de questions.\",\n    needMinNumberCorrectResponses : \"Le nombre de réponses correctes ne peut être négatif.\",\n    needMinNumberSecondesResponse : \"La durée de la réponse ne peut être négative.\",\n    noPreviousAnswer: \"On dirait que c'est la première fois que vous répondez à ce quiz. Bonne lecture !\",\n    previousAnswersLine: \"Le DATEANSWER, vous avez répondu correctement à NBCORRECTANSWERS questions sur NBQUESTIONS en AVGDURATION secondes.\",\n    previousAnswersStats: \"En moyenne, vous avez répondu à ce quiz en AVGDURATION secondes, en ayant <b>AVGCORRECTANSWERS % de bonnes réponses</b>.\",\n    previousAnswersTitle: \"Bonjour #NOM, voici vos précédents résultats à ce quiz\",\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    responseSavedMessage : \"Votre résultat a été enregistré. <a href='/#URL'>Accèder à tous vos quizs</a>.\",\n    statsUser: \"Vous avez enregistré <b>NBANSWERS réponseS1 à NBQUESTIONNAIRES quizS2 différentS3</b> sur les NBTOTQUESTIONNAIRES proposéS4 par le site.<br>En moyenne, vous avez mis AVGDURATION secondes à répondre et avez <b>correctement répondu à AVGCORRECTANSWERS % des questions</b>.\",\n    wantToSaveResponses: \"Si vous le souhaitez, vous pouvez <u><b>sauvegarder votre résultat</b></u> en créant votre compte ci-dessous. Cela vous permettra aussi de <u><b>recevoir régulièrement de nouvelles \\\"graines de culture\\\"</b></u> directement sur votre e-mail.\",\n};","module.exports =\n{\n    addBtnTxt: \"Ajouter\",\n    addOkMessage : \"Les données ont bien été enregistrées.\",\n    alertNewWindow: \"nouvelle fenêtre\",\n    badUrl : \"Tentative d'accès à une page n'existant pas :\",\n    btnLinkToQuestionnaire : \"Afficher !\",\n    btnProposeConnection: \"Je me connecte.\",\n    btnProposeSubscribe: \"Je crée mon compte.\",\n    btnShowOnWebSite: \"Lire la suite sur #SITE_NAME\",\n    deleteBtnTxt: \"Supprimer\",\n    deleteFailMessage : \"La suppression de l'enregistrement #ID a échoué.\",\n    deleteOkMessage : \"La suppression a bien été enregistrée.\",\n    failAuth : \"Erreur d'authentification.\",\n    failAuthCron : \"Tentative de lancement d'un cron sans le bon token.\",\n    failAuthHeader : \"Absence de header Authorization.\",\n    failAuthId : \"Identifiant non valide : \",\n    failAuthToken : \"Token invalide ou utilisateur non trouvé.\",\n    neededParams : \"Des paramètres nécessaires manquants sont manquants.\",\n    nextPage : \"Page suivante\",\n    notAllowed : \"Vous n'avez pas les droits nécessaires pour cette action.\",\n    notRequired : \"Facultatif.\",\n    notValidFormat : \"Format non valide.\",\n    previousPage : \"Page précédente\",\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    siteHTMLTitle : \"WikiLerni : la culture générale en liberté\",\n    siteMetaDescription : \"Avec WikiLerni, vous apprenez régulièrement de nouvelles choses. Vous recevez de courts articles lisibles en quelques minutes. Des quizs vous permettent ensuite de tester ce que vous avez retenu.\",\n    scriptTimingAlert : \"*** Script lent : SCRIPT_TIMING millisecondes, route : SCRIPT_URL\",\n    scriptTimingInfo : \"Durée de la réponse : SCRIPT_TIMING millisecondes, route : SCRIPT_URL\",\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    subscriptionCall: \"Inscrivez-vous !\",\n    updateBtnTxt: \"Modifier\",\n    updateOkMessage : \"La mise à jour à jour a bien été enregistrée.\"\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    // On enlève volontairement les 0/O pour éviter les confusions !\n    // Et mieux vaut aussi débuter et finir par une lettre simple.\n    static 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\nmodule.exports = Tool;","var map = {\n\t\"./default/config/fr.js\": \"../views/default/config/fr.js\",\n\t\"./wikilerni/config/fr.js\": \"../views/wikilerni/config/fr.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 = \"../views sync recursive ^\\\\.\\\\/.*\\\\.js$\";","module.exports =\n{\n    headLinks:\n    [\n        { anchor: \"Accueil\", attributes: { href:\"/\" } },\n        { anchor: \"Mon compte\", attributes: { href:\"/connexion.html\", id: \"accountHeadLink\" } },\n        { anchor: \"À propos\", attributes: { href:\"/a-propos.html\" } },\n        { anchor: \"Contact\", attributes: { href:\"/contact.html\" } },\n    ],\n    footLinks:\n    [\n        { anchor: \"Crédits\", attributes: { href:\"/credits.html\" } },\n        { anchor: \"Mentions légales\", attributes: { href:\"/mentions-legales.html\", rel: \"nofollow\" } },\n        { anchor: \"Données personnelles\", attributes: { href:\"/donnees.html\" } },\n        { anchor: \"C.G.U.\", attributes: { href:\"/cgu.html\", rel: \"nofollow\" } },\n        { anchor: \"C.G.V.\", attributes: { href:\"/cgv.html\", rel: \"nofollow\" } },\n    ],\n    maxQuestionnairesByPage: 10,\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    siteSlogan: \"Cultivons notre jardin !\",\n    homeTitle1: \"De nature curieuse ?\",\n    homeP1: \"Avec WikiLerni vous apprenez chaque jour de nouvelles choses.<br>Des articles de Wikipédia sont sélectionnés pour vous et sont suivis d'un quiz vous permettant de tester ce que vous en avez retenu.<br>De jour en jour de nouvelles graines de savoir sont ainsi semées dans votre \\\"jardin\\\".\",\n    homeTitle2: \"La culture en liberté\",\n    homeP2: \"Tout comme sur Wikipédia (*), le logiciel et le contenu partagé sur WikiLerni sont libres.<br>Vous pouvez les utiliser, les modifier et les diffuser selon votre souhait.<br>Sur WikiLerni, pas de publicité, ni de commercialisation de vos données personnelles.<br>Vous pouvez venir y \\\"cultiver votre jardin\\\" en toute tranquillité.<br><br><small><em>(*) Bien que partageant ses valeurs, WikiLerni est un projet indépendant de la fondation Wikipédia.</em></small>\",\n    newQuestionnairesTitle: \"Les derniers quizs publiés sur WikiLerni\",\n    newQuestionnairesIntro: \"Liste des derniers quizs publiés sur WikiLerni.\",\n    explanationTitle: \"Vous découvrez WikiLerni ?\",\n    explanationTxt: \"Le principe est simple : vous commencez par lire l'article Wikipédia dont le lien vous est proposé.<br>Puis vous afficher le quiz pour vérifier ce que vous en avez retenu de votre lecture. Vous obtenez alors votre résultat immédiatement.<br><br>Toutes les réponses se trouvent dans l'article proposé à la lecture. <b>Vous êtes ici pour apprendre de nouvelles choses</b>, mais libre à vous d'essayer d'y répondre immédiatement.<br><br>Quand le sujet s'y prête, ne vous étonnez pas si certaines des réponses proposées peuvent être un peu décalées, absurdes... On peut apprendre avec le sourire, non ? :-)<br><br>Une fois votre résultat obtenu, il vous sera proposé de créer un compte pour le sauvegarder.<br>Ce compte vous permettra de tester de nouveau ce quiz pour vérifier ce que vous en avez retenu plusieurs jours, semaines, mois... Et de recevoir régulièrement de nouvelles suggestions de lectures.<br><br>Mais <b>la création de ce compte est facultative</b> et <a href='/quizs/' title='Les derniers quizs publiés'>vous pouvez parcourir WikiLerni librement</a>.\",\n    noJSNotification: \"Désolé, mais pour l'instant, l'utilisation de WikiLerni nécessite l'activation du JavaScript.\",\n    tagsListTxt: \"Parcourir les rubriques :\",\n};","module.exports =\n{\n    // liens de l'interface\n    headLinks:\n    [\n        { anchor: \"Contact\", attributes: { href:\"/contact.html\", rel: \"nofollow\" } },\n        { anchor: \"Parcourir\", attributes: { href:\"/quizs/\", id:\"indexHeadLink\", title:\"Les dernières publications\" } },\n        { anchor: \"Mon compte\", attributes: { href:\"/connexion.html\", id: \"accountHeadLink\", title:\"Accéder ou créez votre compte WikiLerni\" } },\n        { anchor: \"À propos\", attributes: { href:\"/a-propos.html\", title:\"En savoir + sur WikiLerni\" } },\n        { anchor: \"Accueil\", attributes: { href:\"/\", title:\"Page d'accueil\" } }\n    ],\n    footLinks:\n    [\n        { anchor: \"Blog\", attributes: { href:\"https://framasphere.org/people/7e54b7a0b53201389eef2a0000053625\", title:\"Le blog WikiLerni sur diaspora*\" } },\n        { anchor: \"Crédits\", attributes: { href:\"/credits.html\", title:\"Qui a créé WikiLerni ? Quels sont vos droits ?\" } },\n        { anchor: \"Mentions légales\", attributes: { href:\"/mentions-legales.html\", rel: \"nofollow\" } },\n        { anchor: \"Données personnelles\", attributes: { href:\"/donnees.html\", title:\"Vos données personnelles sur WikiLerni\" } },\n        { anchor: \"CGV & CGU\", attributes: { href:\"/CGV-CGU.html\", rel: \"nofollow\" } }\n    ],\n    accountPage: \"compte.html\",\n    aboutPage: \"a-propos.html\",\n    adminHomePage: \"admin.html\",\n    cguPage: \"CGV-CGU.html\",\n    connectionPage : \"connexion.html\",\n    deleteLinkPage : \"aurevoir.html?t=\",\n    loginLinkPage : \"login.html?t=\",\n    managerHomePage : \"gestion.html\",\n    newLoginLinkPage : \"newlogin.html?t=\",\n    questionnairesManagementPage: \"gestion-quizs.html\",    \n    stopMailPage : \"stop-mail.html?t=\",\n    subscribePage : \"inscription.html\",\n    updateAccountPage: \"compte.html\",\n    userHomePage : \"accueil.html\",\n    userHomePageTxt : \"Ma page d'accueil.\",\n    usersManagementPage: \"gestion-utilisateurs.html\",\n    validationLinkPage : \"validation.html?t=\",\n    /* Textes (général) */\n    siteSlogan: \"Cultivons notre jardin !\",\n    noJSNotification: \"Désolé, mais pour l'instant, l'utilisation de WikiLerni nécessite l'activation du JavaScript.\",\n    mailRecipientTxt: \"Message envoyé à :\",\n    licenceTxt: \"@copyleft Le contenu de WikiLerni <a href=\\\"/credits.html\\\" title=\\\"En savoir plus ?\\\">est libre</a> et vous est offert sans publicité. Vous pouvez <a href=\\\"/participer-financement.html\\\" title=\\\"Financement participatif avec contre-parties\\\">participer à son financement en cliquant ici</a>.\",\n    /* Page d'accueil */\n    homePageTxt: \"Page d'accueil\",\n    homeTitle1: \"De nature curieuse ?\",\n    homeP1: \"<b>Avec WikiLerni, vous apprenez régulièrement de nouvelles choses</b>.<br>Vous recevez de courts articles, lisibles en quelques minutes.<br>Des quizs vous permettent ensuite de tester ce que vous avez retenu.<br>De jour en jour <b>de nouvelles graines de culture sont ainsi semées dans votre jardin</b>.\",\n    homeTitle2: \"La culture générale en liberté\",\n    homeP2: \"<b>Pas de faits alternatifs</b>, tous les contenus sont <b>sourcés par des articles Wikipédia</b>.<br>Et tout comme sur Wikipédia, le logiciel et le contenu publié sur WikiLerni <a href=\\\"/credits.html\\\" title=\\\"En savoir plus sur ce sujet\\\">sont partagés sous licences libres</a>.<br>Le tout sans publicité, ni commercialisation de vos données.<br><b>Sur WikiLerni, vous cultivez votre jardin en toute tranquillité.</b>\",\n    homeBtnAboutTxt: \"En savoir plus sur WikiLerni ?\",\n    homeBtnSubscribeTxt:  \"Testez WikiLerni\",\n    homeSubcriptionFormTitle:  \"Recevez les prochains articles WikiLerni\",\n    /* Page dernières publications... */   \n    newQuestionnairesTitle: \"Culture générale - apprenez de nouvelles choses avec WikiLerni\",\n    newQuestionnairesIntro: \"WikiLerni : testez vos connaissances et apprenez de nouvelles choses avec WikiLerni.\",\n    newsListTitle: \"<p><b>Avec WikiLerni, vous apprenez chaque jour quelque chose de nouveau</b><br>Si dessous les dernières publications. Vous pouvez aussi <a href='/quizs/themes.html'>parcourir le site par thèmes ou mots-clés</a>.</p>\",\n    /* Plan du site, liste des tags */   \n    tagListTitle: \"Culture générale - des articles et quizs sur de nombreux thèmes !\",\n    tagListMetaDesc: \"WikiLerni : découvrir les différents thèmes abordés par WikiLerni. Inxex du site.\",\n    tagListIntro: \"<h3>Avec WikiLerni, devenez fort en thèmes... Oui mais quels thèmes ? :)</h3><blockquote>Aristote : « L’homme a naturellement la passion de connaître… »</blockquote>\",\n    /* Page quizs */\n    answersExplanationsLinkText: \"Relire\",\n    quizElementLinksIntro: \"En savoir plus\",\n    quizElementSubcriptionFormTitle: \"Recevez les prochains articles WikiLerni\",\n    explanationTitle: \"Vous découvrez WikiLerni ?\",\n    explanationTxt: \"<p>Le principe est simple : <b>vous commencez par lire l’article Wikipédia dont le lien vous est proposé</b>. Puis vous <b>afficher le quiz pour vérifier ce que vous avez retenu de votre lecture</b>. Suivant les questions, <b>une ou plusieurs réponses peuvent être correctes</b> et doivent donc être cochées. C’est toujours <b>le contenu de l’article Wikipédia qui fait foi</b> concernant les « bonnes » réponses. C’est une façon de tester à la fois votre capacité d’attention et votre mémoire. Les articles de Wikipédia peuvent évoluer, donc n’hésitez pas <a href='/contact.html'>à me signaler une erreur</a>.</p><p><b>WikiLerni vous propose d’autres solutions pour améliorer votre culture générale</b>. Pour en savoir plus, cliquez sur le bouton ci-dessous.</p>\",\n    explanationElementTxt: \"<p>WikiLerni vous propose de <b>recevoir par e-mail à la fréquence choisie, de courts articles lisibles en quelques minutes</b> et portant sur des sujets très variés de <b>culture générale</b> (arts, histoire, littérature, sciences, etc.).</p><p>Ces articles sont basés sur <b>une ou plusieurs pages de Wikipédia</b> (fournies en lien), dont <b>ils extraient certaines informations</b>.</p><p>Chaque série d’articles est <b>suivie d’un quiz</b> permettant de tester ce que vous en avez retenu.</p><p><b>Vous apprenez ainsi régulièrement de nouvelles choses</b> très simplement.</p>\",\n    /* Autres */\n    illustrationDir : \"/img/quizs/\",\n    twitterAccount: \"WikiLerni\",\n    maxQuestionnairesByPage: 12,\n    maxQuestionnairesSiteHomePage: 3,\n    nbQuestionnairesUserHomePage : 3,\n};","// à supprimer une fois que tout récupérer du backend :\n\nmodule.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};","// -- 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é, on propose à l'internaute de se connecter ou de créer un compte pour sauvegarder son résultat.\n/// Dans ce but son résultat est stocké dans son navigateur.\n/// Si il est connecté, l'enregistrement de son résultat se fait automatiquement côté serveur et ses éventuels précédents résultats sont affichés.\n\n// Fichier de configuration tirés du backend :\nimport { apiUrl, availableLangs, theme } from \"../../config/instance.js\";\nconst lang=availableLangs[0];\nimport { getPreviousAnswers, questionnaireRoutes, saveAnswersRoute } from \"../../config/questionnaires.js\";\nconst configTemplate = require(\"../../views/\"+theme+\"/config/\"+lang+\".js\");\n\nimport { checkAnswerOuput, saveAnswer } from \"./tools/answers.js\";\nimport { addElement } from \"./tools/dom.js\";\nimport { helloDev, updateAccountLink } from \"./tools/everywhere.js\";\nimport { getLocaly } from \"./tools/clientstorage.js\";\nimport { getDatasFromInputs } from \"./tools/forms.js\";\nimport { dateFormat, replaceAll } from \"../../tools/main\";\nimport { loadMatomo } from \"./tools/matomo.js\";\nimport { checkSession, getTimeDifference } from \"./tools/users.js\";\n\n// Dictionnaires :\nconst { noPreviousAnswer, previousAnswersLine, previousAnswersStats, previousAnswersTitle, responseSavedError, wantToSaveResponses } = require(\"../../lang/\"+lang+\"/answer\");\nconst { serverError } = require(\"../../lang/\"+lang+\"/general\");\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\nlet isConnected, user;\nconst initialise = async () =>\n{\n    try\n    {\n        btnShow.style.display=\"inline\";// bouton caché si JS inactif, car JS nécessaire pour vérifier les réponses\n        isConnected=await checkSession([\"user\"]);// \"user\" car seuls les utilisateurs de base peuvent enregistrer leurs réponses aux quizs\n        // Si l'utilisateur est connecté et a déjà répondu à ce quiz, on affiche ses précédentes réponses à la place du texte servant à expliquer le topo aux nouveaux\n        if(isConnected)\n        {\n            user=getLocaly(\"user\", true);\n            updateAccountLink(user.status, configTemplate);// lien vers le compte adapté pour les utilisateurs connectés\n            checkPreviousResponses(user);\n        }\n        else\n            loadMatomo();\n    }\n    catch(e)\n    {\n        console.error(e);\n    }\n}\ninitialise();\nhelloDev();\n\n// Affichage du questionnaire quand 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.style.display=\"none\";\n    const here=window.location;// window.location à ajouter pour ne pas quitter la page en mode \"preview\".\n    if(window.location.hash!==\"\")\n    {\n        window.location.hash=\"\";// ! le \"#\" reste\n        window.location.assign(here+\"questionnaire\");\n    }\n    else\n        window.location.assign(here+\"#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\", serverError, \"\", [\"error\"]);\n        console.error(e);\n    }\n});\n// Lien passé par mail pour voir directement le quiz\nif(location.hash!=\"\" && location.hash===\"#questionnaire\")\n    showQuestionnaire();\n\n// Traitement de l'envoi de la réponse de l'utilisateur :\nlet answer = {};\nmyForm.addEventListener(\"submit\", function(e)\n{\n    try\n    {\n        e.preventDefault();       \n        btnSubmit.style.display=\"none\";// seulement un envoi à la fois, SVP :)\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.nbQuestions=0;\n        answer.nbCorrectAnswers=0;\n        answer.QuestionnaireId=document.getElementById(\"questionnaireId\").value;\n        // Les réponses sont regroupées par question, donc quand idQuestion change, on connaît le résultat pour la question précédente.\n        // Pour qu'une réponse soit bonne, il faut cocher toutes les bonnes réponses (si QCM) à la question ET cocher aucune des mauvaises.\n        let idChoice, idQuestion=\"\", goodResponse=false;\n        for(let item in userResponses)\n        {\n            if(item.startsWith(\"isCorrect_response_\"))// = Nouvelle réponse possible.\n            {\n                idChoice = item.substring(item.lastIndexOf(\"_\") + 1);\n                // si on change de queston\n                if(userResponses[\"question_id_response_\"+idChoice]!=idQuestion) // on commence à traiter une nouvelle question\n                {\n                    idQuestion=userResponses[\"question_id_response_\"+idChoice];\n                    answer.nbQuestions++;\n                    if(goodResponse) // résultat de la question précédente\n                        answer.nbCorrectAnswers++;\n                    goodResponse=true;// réponse bonne jusqu'à la première erreur...\n                }\n                if(userResponses[item]==\"true\")\n                {\n                    document.getElementById(\"response_\"+idChoice).parentNode.classList.add(\"isCorrect\");\n                    if(userResponses[\"response_\"+idChoice]===undefined)// une bonne réponse n'a pas été sélectionnée\n                        goodResponse=false;\n                }\n                else\n                {\n                    if(userResponses[\"response_\"+idChoice]===\"on\")// réponse cochée ne faisant pas partie des bonnes\n                    {\n                        goodResponse=false;\n                        document.getElementById(\"response_\"+idChoice).parentNode.classList.add(\"isNotCorrect\");\n                    }\n                }\n            }\n        }\n        // si j'ai bien répondu à la dernière question, il faut le compter ici, car je suis sorti de la boucle :\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 xhrSaveAnswer = new XMLHttpRequest();\n            xhrSaveAnswer.open(\"POST\", apiUrl+questionnaireRoutes+saveAnswersRoute);\n            xhrSaveAnswer.onreadystatechange = function()\n            {\n                if (this.readyState == XMLHttpRequest.DONE)\n                {\n                    let xhrResponse=JSON.parse(this.responseText);\n                    if (this.status === 201 && (xhrResponse.message))\n                    {\n                        getOuput+=\"<br>\"+xhrResponse.message.replace(\"#URL\", configTemplate.userHomePage);\n                        checkPreviousResponses(user);\n                    }\n                    else\n                        getOuput+=\"<br>\"+responseSavedError.replace(\"#URL\", configTemplate.userHomePage);\n                    // on redirige vers le résultat\n                    window.location.hash=\"\";\n                    const here=window.location;// window.location à ajouter pour ne pas quitter la page en mode \"preview\"...\n                    window.location.assign(here+\"explanations\");\n                }\n            }\n            xhrSaveAnswer.setRequestHeader(\"Authorization\", \"Bearer \"+user.token);\n            xhrSaveAnswer.setRequestHeader(\"Content-Type\", \"application/json\");\n            answer.timeDifference=getTimeDifference();// on en profite pour mettre les pendules à l'heure.\n            xhrSaveAnswer.send(JSON.stringify(answer));\n        }\n        else\n        {   // si pas connecté, on enregistre le résultat 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><br>\"+wantToSaveResponses;\n                 addElement(divResponse, \"p\", getOuput, \"\", [\"info\"]);\n                 document.querySelector(\".subscribeBtns\").style.display=\"block\";\n            }\n            else // inutile de proposer de créer un compte si le stockage local ne fonctionne pas\n                addElement(divResponse, \"p\", getOuput, \"\", [\"info\"]);\n            // on redirige vers le résultat\n            window.location.hash=\"\";\n            const here=window.location;// window.location à ajouter pour ne pas quitter la page en mode \"preview\"...\n            window.location.assign(here+\"response\");           \n        }\n        // + affichage des textes d'explications pour chaque question\n        const explanations=document.querySelectorAll(\".help\");\n        for(let i in explanations)\n        {\n            if(explanations[i].style!=undefined) // sinon, la console affiche une erreur \"TypeError: explanations[i].style is undefined\", bien que tout fonctionne (?)\n                explanations[i].style.display=\"block\";\n        }\n    }\n    catch(e)\n    {\n        addElement(divResponse, \"p\", serverError, \"\", [\"error\"]);\n        console.error(e);\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 = (user) =>\n{\n    const xhrPreviousRes = new XMLHttpRequest();\n    xhrPreviousRes.open(\"GET\", apiUrl+questionnaireRoutes+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                addElement(explanationsTitle, \"span\", previousAnswersTitle.replace(\"#NOM\", user.name));\n                if(nbResponses!==0)\n                {\n                    let totNbQuestions=0, totNbCorrectAnswers=0, totDuration=0, mapLineContent;\n                    for(let i in response)\n                    {\n                        totNbQuestions+=response[i].nbQuestions;// ! on ne peut se baser sur la version actuelle du quiz, car le nombre de questions a pu évoluer.\n                        totNbCorrectAnswers+=response[i].nbCorrectAnswers;\n                        totDuration+=response[i].duration;\n                        mapLineContent =\n                        {\n                            DATEANSWER : dateFormat(response[i].createdAt, lang),\n                            NBCORRECTANSWERS : response[i].nbCorrectAnswers,\n                            NBQUESTIONS : response[i].nbQuestions,\n                            AVGDURATION : response[i].duration\n                        };\n                        previousAnswersContent+=\"<li>\"+replaceAll(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(previousAnswersStats, mapLineContent)+\"</h5>\"+previousAnswersContent;\n                    addElement(explanationsContent, \"ul\", previousAnswersContent);\n                }\n                else\n                    addElement(explanationsContent, \"ul\", noPreviousAnswer);\n                // dans un cas comme dans l'autre, bouton pour revenir à l'accueil du compte\n                addElement(explanationsContent, \"p\", \"<a href=\\\"/\"+configTemplate.userHomePage+\"\\\" class=\\\"button cardboard\\\">\"+configTemplate.userHomePageTxt+\"</a>\", \"\", [\"btn\"], \"\", false);\n                \n            }\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) || !isEmpty(answer.GroupId)))\n    {\n        saveLocaly(\"lastAnswer\", answer);\n        return true;\n    }\n    else\n        return false;\n}\n\n// Retourne un 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}","// 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\\nLe code de WikiLerni est libre et vous pouvez le trouver à cette adresse :\\nhttps://forge.chapril.org/Fab_Blab/WikiLerni\\n\\nPour les suggestions d'amélioration ou questions : dev@wililerni.com ****\");\n    return true;\n}\n\nexport const updateAccountLink = (status, configTemplate) =>\n{\n    const link=document.getElementById(\"accountHeadLink\");\n    const homePage=status+\"HomePage\";\n    link.href=\"/\"+configTemplate[homePage];\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}","import { matomo } from \"../../../config/matomo.js\";\n\n// Fonction chargeant le code matomo\nexport const loadMatomo = () =>\n{\n    // chargement de matomo\n    var _paq = window._paq = window._paq || [];\n    //ajout cnil  :\n    _paq.push([function()\n    {\n        var self = this;\n        function getOriginalVisitorCookieTimeout()\n        {\n            var now = new Date(),\n            nowTs = Math.round(now.getTime() / 1000),\n            visitorInfo = self.getVisitorInfo();\n            var createTs = parseInt(visitorInfo[2]);\n            var cookieTimeout = 33696000; // 13 mois en secondes\n            var originalTimeout = createTs + cookieTimeout - nowTs;\n            return originalTimeout;\n        }\n        this.setVisitorCookieTimeout( getOriginalVisitorCookieTimeout() );\n    }]);\n    //--fin ajout cnil\n    _paq.push(['trackPageView']);\n    _paq.push(['enableLinkTracking']);\n    (function() {\n        var u=matomo.url;\n        _paq.push(['setTrackerUrl', u+'matomo.php']);\n        _paq.push(['setSiteId', matomo.siteId]);\n        var d=document, g=d.createElement('script'), s=d.getElementsByTagName('script')[0];\n        g.type='text/javascript'; g.async=true; g.src=u+'matomo.js'; s.parentNode.insertBefore(g,s);\n    })();\n}","import { apiUrl, availableLangs, siteUrl, theme } from \"../../../config/instance.js\";\nconst lang=availableLangs[0];\n\nconst configTemplate = require(\"../../../views/\"+theme+\"/config/\"+lang+\".js\");\n\nimport {  checkLoginRoute, timeDifferenceMax, timeDifferenceMin, userRoutes } from \"../../../config/users.js\";\n\nimport { getLocaly, removeLocaly, saveLocaly } from \"./clientstorage.js\";\nimport { isEmpty } from \"../../../tools/main\";\n\nexport const getTimeDifference = () =>\n{\n    // multiplier par -1, car c'est ce qu'il faut \"ajouter\" à l'heure UTC pour revenir en heure locale qui m'intéresse et non l'inverse\n    const timeLocal=new Date().getTimezoneOffset()*-1;\n    if(timeLocal > timeDifferenceMax || timeLocal < timeDifferenceMin)\n        return 0;\n    else\n        return timeLocal;\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 ou d'un groupe de quizs\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.nbQuestions) && (!isEmpty(answer.QuestionnaireId) || !isEmpty(answer.GroupId)))\n        {\n            datas.duration=answer.duration;\n            datas.nbCorrectAnswers=answer.nbCorrectAnswers;\n            datas.nbQuestions=answer.nbQuestions;\n            if(!isEmpty(answer.QuestionnaireId))\n                datas.QuestionnaireId=answer.QuestionnaireId;\n            else\n                datas.GroupId=answer.GroupId;\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 (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\", apiUrl+userRoutes+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;// c'est le token qui sert à vérifier le statut à 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=siteUrl+\"/\"+configTemplate.accountPage;\n                                if(window.location.href.indexOf(urlAccount)===-1)\n                                    window.location.assign(\"/\"+configTemplate.accountPage+\"#subscribe\");// 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":""}