Modification page et script de création de compte pour prendre en compte simplification (uniquement e-mail + validation CGU durant première étape)
This commit is contained in:
parent
e55f9c02bc
commit
f497a05144
@ -62,7 +62,7 @@ exports.getGodfatherId = async (req, res, next) =>
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Contrôleur traitant les données envoyées pour une inscription
|
// Contrôleur traitant les données envoyées pour une inscription
|
||||||
// Il peut n'y avoir qu'une adresse e-mail ou des données plus complètes (pseudo, parrain, etc.) dès l'inscription
|
// Les CGU doivent être acceptées et une adresse e-mail envoyée. Le reste peut être adapté sur la page de validation.
|
||||||
exports.signup = async (req, res, next) =>
|
exports.signup = async (req, res, next) =>
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
@ -70,34 +70,22 @@ exports.signup = async (req, res, next) =>
|
|||||||
const db = require("../models/index");
|
const db = require("../models/index");
|
||||||
if(req.body.cguOk !== "true")
|
if(req.body.cguOk !== "true")
|
||||||
res.status(400).json({ errors: [txt.needUGCOk] });
|
res.status(400).json({ errors: [txt.needUGCOk] });
|
||||||
else if((req.body.password !== undefined) && (req.body.password.length < config.passwordMinLength))
|
|
||||||
res.status(400).json({ errors: [txt.needLongPassWord.replace("MIN_LENGTH", config.passwordMinLength)] });
|
|
||||||
else if(tool.isEmpty(req.body.email))
|
else if(tool.isEmpty(req.body.email))
|
||||||
res.status(400).json({ errors: [txt.needEmail] });
|
res.status(400).json({ errors: [txt.needEmail] });
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
// Dans le cas d'une inscription simplifiée, seule l'adresse e-mail est demandée :
|
// Un mot de passe temporaire et non communiqué est généré :
|
||||||
if(tool.isEmpty(req.body.password))
|
req.body.passwordTemp=tool.getPassword(8, 12);
|
||||||
req.body.password=tool.getPassword(8, 12);
|
console.log(req.body.passwordTemp);
|
||||||
req.body.password=await bcrypt.hash(req.body.password, config.bcryptSaltRounds);
|
req.body.password=await bcrypt.hash(req.body.passwordTemp, config.bcryptSaltRounds);
|
||||||
// Si pas de pseudo envoyé, on utilise la partie de l'e-mail précédant le "@"
|
// Un pseudo temporaire est créé en utilisant la partie de l'e-mail précédant le "@" :
|
||||||
if(tool.isEmpty(req.body.name))
|
|
||||||
{
|
|
||||||
const lastIndex=req.body.email.indexOf("@");
|
const lastIndex=req.body.email.indexOf("@");
|
||||||
if(lastIndex === -1)
|
if(lastIndex === -1)// possible car validité de l'e-mail testé par le modèle lors de l'enregistrement
|
||||||
lastIndex=1;
|
lastIndex=1;
|
||||||
req.body.name=req.body.email.substring(0, lastIndex);
|
req.body.name=req.body.email.substring(0, lastIndex);
|
||||||
}
|
const user=await db["User"].create({ ...req.body }, { fields: ["name", "email", "password", "timeDifference"] });
|
||||||
req.body.GodfatherId=null;
|
req.body.UserId=user.id;
|
||||||
if(req.body.codeGodfather!=="")
|
// si l'utilisateur a répondu à un quiz avant de créer son compte, on enregistre son résultat :
|
||||||
{
|
|
||||||
const godfather=await searchIdGodfather(req.body.codeGodfather);
|
|
||||||
if(godfather)
|
|
||||||
req.body.GodfatherId=godfather.id;
|
|
||||||
}
|
|
||||||
const user=await db["User"].create({ ...req.body }, { fields: ["name", "email", "password", "newsletterOk", "GodfatherId", "timeDifference"] });
|
|
||||||
req.body.UserId=user.id;/// revoir si noticeOk et newsletterOk toujours utiles ?
|
|
||||||
// si l'utilisateur a répondu à un quiz avant de créer son compte, on enregistre son résultats.
|
|
||||||
if(req.body.QuestionnaireId)
|
if(req.body.QuestionnaireId)
|
||||||
{
|
{
|
||||||
await Promise.all([
|
await Promise.all([
|
||||||
@ -123,6 +111,7 @@ exports.signup = async (req, res, next) =>
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Contrôleur testant le lien de validation du compte et envoyant un message de bienvenue si tout est ok
|
||||||
exports.signupValidation = async (req, res, next) =>
|
exports.signupValidation = async (req, res, next) =>
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
@ -137,7 +126,7 @@ exports.signupValidation = async (req, res, next) =>
|
|||||||
{
|
{
|
||||||
const now=new Date();
|
const now=new Date();
|
||||||
await Promise.all([
|
await Promise.all([
|
||||||
db["Subscription"].create({ numberOfDays: config.freeAccountTimingInDays, numberOfDays: config.defaultReceiptDays, noticeOk: datas.User.newsletterOk, UserId: datas.User.id }),/// revoir si noticeOk et newsletterOk toujours utiles ?
|
db["Subscription"].create({ numberOfDays: config.freeAccountTimingInDays, numberOfDays: config.defaultReceiptDays, UserId: datas.User.id }),
|
||||||
db["User"].update({ connectedAt: now }, { where: { id : datas.User.id }, limit:1 })
|
db["User"].update({ connectedAt: now }, { where: { id : datas.User.id }, limit:1 })
|
||||||
]);
|
]);
|
||||||
const newUser=await creaUserJson(datas.User.id);
|
const newUser=await creaUserJson(datas.User.id);
|
||||||
@ -189,7 +178,7 @@ exports.signUpCompletion = async (req, res, next) =>
|
|||||||
res.status(400).json({ errors: txt.needLongPassWord.replace("MIN_LENGTH", config.passwordMinLength) });
|
res.status(400).json({ errors: txt.needLongPassWord.replace("MIN_LENGTH", config.passwordMinLength) });
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if(!tool.isEmpty(req.body.newPassword))
|
if(!tool.isEmpty(req.body.newPassword))// dans ce cas, l'utilisateur n'a pas choisi de mot de passe, mais pourra se connecter via les tokens de connexion
|
||||||
req.body.password=await bcrypt.hash(req.body.newPassword, config.bcryptSaltRounds);
|
req.body.password=await bcrypt.hash(req.body.newPassword, config.bcryptSaltRounds);
|
||||||
req.body.GodfatherId=null;
|
req.body.GodfatherId=null;
|
||||||
if(req.body.codeGodfather !== "")
|
if(req.body.codeGodfather !== "")
|
||||||
@ -203,7 +192,7 @@ exports.signUpCompletion = async (req, res, next) =>
|
|||||||
db["Subscription"].update({ ...req.body }, { where: { UserId : req.connectedUser.User.id }, fields: ["receiptDays"], limit:1 })
|
db["Subscription"].update({ ...req.body }, { where: { UserId : req.connectedUser.User.id }, fields: ["receiptDays"], limit:1 })
|
||||||
]);
|
]);
|
||||||
const user=await creaUserJson(req.connectedUser.User.id);
|
const user=await creaUserJson(req.connectedUser.User.id);
|
||||||
// si un parrain a été désigné, on prévient l'heureux élu :) :
|
// Si un parrain a été désigné, on prévient l'heureux élu :) :
|
||||||
if(user!==false && req.body.GodfatherId !== null)
|
if(user!==false && req.body.GodfatherId !== null)
|
||||||
{
|
{
|
||||||
const godfather=await searchUserById(req.body.GodfatherId);
|
const godfather=await searchUserById(req.body.GodfatherId);
|
||||||
|
@ -4,7 +4,7 @@
|
|||||||
<meta charset="utf-8">
|
<meta charset="utf-8">
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
<meta name="description" content="Formulaire d'inscription à WikiLerni.">
|
<meta name="description" content="Formulaire d'inscription à WikiLerni.">
|
||||||
<title>S'inscrire à WikiLerni</title>
|
<title>Créer son compte WikiLerni</title>
|
||||||
<!-- Version lisible des scripts : https://gitlab.com/lefablab/wikilerni/-/tree/master/front/src -->
|
<!-- Version lisible des scripts : https://gitlab.com/lefablab/wikilerni/-/tree/master/front/src -->
|
||||||
<script src="/JS/polyfill.app.js" defer></script>
|
<script src="/JS/polyfill.app.js" defer></script>
|
||||||
<script src="/JS/subscribe.app.js" defer></script>
|
<script src="/JS/subscribe.app.js" defer></script>
|
||||||
@ -37,46 +37,28 @@
|
|||||||
|
|
||||||
<noscript>Désolé, mais pour l'instant, l'utilisation de WikiLerni nécessite l'activation du JavaScript.</noscript>
|
<noscript>Désolé, mais pour l'instant, l'utilisation de WikiLerni nécessite l'activation du JavaScript.</noscript>
|
||||||
<form class="needJS" id="subscription" method="POST">
|
<form class="needJS" id="subscription" method="POST">
|
||||||
<fieldset>
|
|
||||||
<label for="name">Nom ou pseudonyme : </label><input id="name" type="text" name="name" placeholder="Nom de votre choix" class="cardboard" >
|
|
||||||
</fieldset>
|
|
||||||
|
|
||||||
<fieldset>
|
<fieldset>
|
||||||
<label for="email">E-mail : </label><input id="email" type="email" name="email" placeholder="Adresse e-mail" class="cardboard">
|
<label for="email">E-mail : </label><input id="email" type="email" name="email" placeholder="Adresse e-mail" class="cardboard">
|
||||||
<div id="emailMessage"></div>
|
<div id="emailMessage"></div>
|
||||||
</fieldset>
|
</fieldset>
|
||||||
|
|
||||||
<fieldset>
|
|
||||||
<label for="password">Mot de passe : </label><input id="password" type="password" name="password" placeholder="Mot de passe de votre choix" class="cardboard">
|
|
||||||
<div id="passwordMessage"><span class="info">Au moins 8 caractères. <a href="#password" id="getPassword">Générer un mot de passe</a>.</span></div>
|
|
||||||
</fieldset>
|
|
||||||
|
|
||||||
<fieldset>
|
|
||||||
<label for="codeGodfather">Code ou e-mail parrain : </label><input id="codeGodfather" type="text" name="codeGodfather" placeholder="Code ou email de votre parrain." class="cardboard">
|
|
||||||
<div id="codeGodfatherMessage"><span class="info">Facultatif.</span></div>
|
|
||||||
</fieldset>
|
|
||||||
|
|
||||||
<ul class="checkbox_li">
|
<ul class="checkbox_li">
|
||||||
<li class="checkbox_li">
|
<li class="checkbox_li">
|
||||||
<label for="cguOk" class="check"><input type="checkbox" id="cguOk" name="cguOk" value="true" /><div class="checkbox_override"></div> J'accepte <a href="/CGV-CGU.html" target="_blank" rel="noopener" title="À lire :)">les Conditions Générale d'Utilisation</a> du site (requis).</label>
|
<label for="cguOk" class="check"><input type="checkbox" id="cguOk" name="cguOk" value="true" /><div class="checkbox_override"></div> J'accepte <a href="/CGV-CGU.html" target="_blank" rel="noopener" title="À lire :)">les Conditions Générale d'Utilisation</a> du site.</label>
|
||||||
</li>
|
|
||||||
<li class="checkbox_li">
|
|
||||||
<label for="newsletterOk" class="check"><input type="checkbox" id="newsletterOk" name="newsletterOk" value="true" /><div class="checkbox_override"></div> Je souhaite recevoir des suggestions de quiz par e-mail (facultatif et vous pourrez stopper facilement).</label>
|
|
||||||
</li>
|
</li>
|
||||||
</ul>
|
</ul>
|
||||||
<div class="input_wrapper">
|
<div class="input_wrapper">
|
||||||
<input id="submitDatas" type="submit" value="Valider." class="cardboard" />
|
<input id="submitDatas" type="submit" value="Valider" class="cardboard" />
|
||||||
</div>
|
</div>
|
||||||
</form>
|
</form>
|
||||||
<div id="response"></div>
|
<div id="response"></div>
|
||||||
|
|
||||||
<div id="explanations" class="framed engraved">
|
<div id="explanations" class="framed engraved">
|
||||||
<h2>Besoin d'aide ?</h2>
|
<h2>Besoin d'aide ?</h2>
|
||||||
<p>La saisie d'un pseudonyme, d'une adresse e-mail et d'un mot de passe <b>est obligatoire</b> et <b>vous devez accepter les Conditions Générales d'Utilisation</b>.</p>
|
<p>Votre compte vous permettra de <b>recevoir régulièrement de nouvelles "graines de culture" directement sur votre adresse e-mail</b>. Vous pourrez aussi sauvegarder vos résultats aux quizs.</p>
|
||||||
<p>Votre pseudonyme est complétement libre, mais servira à personnaliser certains affichages.</p>
|
<p>La saisie d'une adresse e-mail <b>est obligatoire</b> et <b>vous devez accepter les Conditions Générales d'Utilisation</b>.</p>
|
||||||
<p><b>Vous recevrez un lien sur l'adresse e-mail saisie</b> sur lequel vous devrez cliquer pour valider la création de votre compte.</p>
|
<p><b>Vous recevrez un lien sur l'adresse e-mail saisie</b> sur lequel vous devrez cliquer pour valider la création de votre compte.</p>
|
||||||
<p><b>Votre mot de passe doit compter au moins 8 caractères.</b> Si vous manquez d'inspiration, cliquez sur le lien "Générer un mot de passe" pour en obtenir un.</p>
|
<p>Une fois cliqué sur ce lien, <b>vous serez invité à compléter vos informations (pseudo, mot de passe...) ou encore à désigner votre "parrain"</b> (si c'est le cas).</p>
|
||||||
<p>Si vous connaissez une personne déjà inscrite à WikiLerni, <b>vous pouvez fournir son code parrain ou encore son adresse e-mail</b>. Dans le cas où vous optez ensuite pour un abonnement prémium, il sera alors allongé de 30 jours grâce à ce parrainage.</p>
|
<p>La création de votre compte est <b>gratuite et sans engagement</b>. Elle vous permet de tester WikiLerni durant une période de 15 jours. Libre à vous ensuite de vous abonner ou pas.</p>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
@ -123,7 +123,7 @@ myForm.addEventListener("submit", function(e)
|
|||||||
if(datas)
|
if(datas)
|
||||||
{
|
{
|
||||||
datas.timeDifference=getTimeDifference(configUsers);
|
datas.timeDifference=getTimeDifference(configUsers);
|
||||||
// si l'utilisateur a précédement répondu à un quiz, on ajoute les données de son résultat :
|
// Si l'utilisateur a précédement répondu à un quiz, on ajoute les données de son résultat :
|
||||||
datas=checkAnswerDatas(datas);
|
datas=checkAnswerDatas(datas);
|
||||||
xhr.send(JSON.stringify(datas));
|
xhr.send(JSON.stringify(datas));
|
||||||
}
|
}
|
||||||
|
@ -1,7 +1,8 @@
|
|||||||
// -- GESTION DU FORMULAIRE PERMETTANT DE CRÉER SON COMPTE
|
// -- GESTION DU FORMULAIRE PERMETTANT DE CRÉER SON COMPTE
|
||||||
|
|
||||||
/// L'utilisateur peut avoir répondu à un quiz avant d'arriver sur la page d'inscription
|
/// L'utilisateur peut avoir répondu à un quiz avant d'arriver sur la page d'inscription
|
||||||
/// Des ce cas il faut enregistrer son résultat en même temps que les informations de son compte
|
/// Des ce cas il faut enregistrer son résultat en même temps que les premières informations de son compte (email, ok CGU)
|
||||||
|
/// Les infos du compte sont complétées (mot de passe, code parrain...) au moment de la validation.
|
||||||
|
|
||||||
// Fichier de configuration tirés du backend :
|
// Fichier de configuration tirés du backend :
|
||||||
import { apiUrl, availableLangs, theme } from "../../config/instance.js";
|
import { apiUrl, availableLangs, theme } from "../../config/instance.js";
|
||||||
@ -15,25 +16,18 @@ import { getLocaly, removeLocaly, saveLocaly } from "./tools/clientstorage.js";
|
|||||||
import { addElement } from "./tools/dom.js";
|
import { addElement } from "./tools/dom.js";
|
||||||
import { helloDev } from "./tools/everywhere.js";
|
import { helloDev } from "./tools/everywhere.js";
|
||||||
import { getDatasFromInputs, setAttributesToInputs } from "./tools/forms.js";
|
import { getDatasFromInputs, setAttributesToInputs } from "./tools/forms.js";
|
||||||
import { getPassword } from "../../tools/main";
|
|
||||||
import { loadMatomo } from "./tools/matomo.js";
|
import { loadMatomo } from "./tools/matomo.js";
|
||||||
import { checkAnswerDatas, checkSession, getTimeDifference } from "./tools/users.js";
|
import { checkAnswerDatas, checkSession, getTimeDifference } from "./tools/users.js";
|
||||||
|
|
||||||
// Dictionnaires :
|
// Dictionnaires :
|
||||||
const { notRequired, serverError } = require("../../lang/"+lang+"/general");
|
const { serverError } = require("../../lang/"+lang+"/general");
|
||||||
const { alreadyConnected, godfatherFound, godfatherNotFound, needUniqueEmail, passwordCopied } = require("../../lang/"+lang+"/user");
|
const { alreadyConnected, needUniqueEmail } = require("../../lang/"+lang+"/user");
|
||||||
|
|
||||||
// Principaux éléments du DOM manipulés :
|
// Principaux éléments du DOM manipulés :
|
||||||
const myForm=document.getElementById("subscription");
|
|
||||||
const divResponse=document.getElementById("response");
|
|
||||||
const passwordInput=document.getElementById("password");
|
|
||||||
const passwordLink=document.getElementById("getPassword");
|
|
||||||
const passwordHelp=document.getElementById("passwordMessage");
|
|
||||||
const emailInput=document.getElementById("email");
|
|
||||||
const btnSubmit=document.getElementById("submitDatas");
|
const btnSubmit=document.getElementById("submitDatas");
|
||||||
const codeGodfatherInput=document.getElementById("codeGodfather");
|
const divResponse=document.getElementById("response");
|
||||||
|
const emailInput=document.getElementById("email");
|
||||||
helloDev();
|
const myForm=document.getElementById("subscription");
|
||||||
|
|
||||||
// Test de connexion de l'utilisateur + affichage formulaire d'inscription.
|
// Test de connexion de l'utilisateur + affichage formulaire d'inscription.
|
||||||
const initialise = async () =>
|
const initialise = async () =>
|
||||||
@ -43,7 +37,7 @@ const initialise = async () =>
|
|||||||
const isConnected=await checkSession();
|
const isConnected=await checkSession();
|
||||||
if(isConnected)
|
if(isConnected)
|
||||||
{
|
{
|
||||||
saveLocaly("message", { message: alreadyConnected, color:"info" });// pour l'afficher sur la page suivante
|
saveLocaly("message", { message: alreadyConnected, color:"info" });
|
||||||
const user=getLocaly("user", true);
|
const user=getLocaly("user", true);
|
||||||
const homePage=user.status+"HomePage";
|
const homePage=user.status+"HomePage";
|
||||||
window.location.assign("/"+configTemplate[homePage]);
|
window.location.assign("/"+configTemplate[homePage]);
|
||||||
@ -62,25 +56,10 @@ const initialise = async () =>
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
initialise();
|
initialise();
|
||||||
|
helloDev();
|
||||||
|
|
||||||
// Générateur de mot de passe "aléatoire"
|
// Teste si l'e-mail saisi est déjà utilisé par un autre compte.
|
||||||
passwordLink.addEventListener("click", function(e)
|
|
||||||
{
|
|
||||||
e.preventDefault();
|
|
||||||
passwordInput.type="text";
|
|
||||||
passwordInput.value=getPassword(8, 12);
|
|
||||||
// Copie du mot de passe généré dans le "presse-papier" de l'ordinateur :
|
|
||||||
passwordInput.select();
|
|
||||||
document.execCommand("copy");
|
|
||||||
addElement(passwordHelp, "div", passwordCopied, "", ["success"]);
|
|
||||||
});
|
|
||||||
|
|
||||||
// Test si l'e-mail saisi est déjà utilisé par un autre compte.
|
|
||||||
// Si c'est le cas, la validation du formulaire est bloquée.
|
// Si c'est le cas, la validation du formulaire est bloquée.
|
||||||
emailInput.addEventListener("focus", function(e)
|
|
||||||
{
|
|
||||||
document.getElementById("emailMessage").innerHTML="";// pour supprimer l'éventuel message d'erreur déjà affiché
|
|
||||||
});
|
|
||||||
emailInput.addEventListener("blur", function(e)
|
emailInput.addEventListener("blur", function(e)
|
||||||
{
|
{
|
||||||
const emailValue=emailInput.value.trim();
|
const emailValue=emailInput.value.trim();
|
||||||
@ -107,33 +86,10 @@ emailInput.addEventListener("blur", function(e)
|
|||||||
xhr.send(JSON.stringify(datas));
|
xhr.send(JSON.stringify(datas));
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
// Supprime l'éventuel message d'erreur déjà injecté si l'utilisateur revient dans le champ :
|
||||||
// Vérification que le code/e-mail de parrainage saisi est valide.
|
emailInput.addEventListener("focus", function(e)
|
||||||
codeGodfatherInput.addEventListener("focus", function(e)
|
|
||||||
{ // on efface l'éventuel message d'erreur si on revient sur le champ pour tester un autre code.
|
|
||||||
addElement(document.getElementById("codeGodfatherMessage"), "i", notRequired);
|
|
||||||
});
|
|
||||||
codeGodfatherInput.addEventListener("blur", function(e)
|
|
||||||
{
|
{
|
||||||
const codeValue=codeGodfatherInput.value.trim();
|
document.getElementById("emailMessage").innerHTML="";
|
||||||
if(codeValue!=="")
|
|
||||||
{
|
|
||||||
const xhr = new XMLHttpRequest();
|
|
||||||
xhr.open("POST", apiUrl+configUsers.userRoutes+configUsers.getGodfatherRoute);
|
|
||||||
xhr.onreadystatechange = function()
|
|
||||||
{
|
|
||||||
if (this.readyState == XMLHttpRequest.DONE)
|
|
||||||
{
|
|
||||||
if (this.status === 204)
|
|
||||||
addElement(document.getElementById("codeGodfatherMessage"), "div", godfatherNotFound, "", ["error"]);
|
|
||||||
else
|
|
||||||
addElement(document.getElementById("codeGodfatherMessage"), "div", godfatherFound, "", ["success"]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
xhr.setRequestHeader("Content-Type", "application/json");
|
|
||||||
const datas={ codeTest:codeValue };
|
|
||||||
xhr.send(JSON.stringify(datas));
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
|
||||||
// Traitement de l'envoi des données d'inscription :
|
// Traitement de l'envoi des données d'inscription :
|
||||||
@ -169,7 +125,7 @@ myForm.addEventListener("submit", function(e)
|
|||||||
if(datas)
|
if(datas)
|
||||||
{
|
{
|
||||||
datas.timeDifference=getTimeDifference(configUsers);
|
datas.timeDifference=getTimeDifference(configUsers);
|
||||||
// si l'utilisateur a précédement répondu à un quiz, j'ajoute les infos de son résultat :
|
// Si l'utilisateur a précédement répondu à un quiz, j'ajoute les infos de son résultat :
|
||||||
datas=checkAnswerDatas(datas);
|
datas=checkAnswerDatas(datas);
|
||||||
xhr.send(JSON.stringify(datas));
|
xhr.send(JSON.stringify(datas));
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user