frontend et backend: Crée une nouvelle fonction

This commit is contained in:
Divlo 2020-04-11 23:29:22 +02:00
parent c157f7e922
commit 42193066a8
8 changed files with 422 additions and 210 deletions

View File

@ -4,6 +4,50 @@ const { validationResult } = require('express-validator');
const errorHandling = require('../assets/utils/errorHandling'); const errorHandling = require('../assets/utils/errorHandling');
const { serverError } = require('../assets/config/errors'); const { serverError } = require('../assets/config/errors');
const Functions = require('../models/functions'); const Functions = require('../models/functions');
const Categories = require('../models/categories');
const helperQueryNumber = require('../assets/utils/helperQueryNumber');
const Sequelize = require('sequelize');
exports.getFunctions = (req, res, next) => {
const page = helperQueryNumber(req.query.page, 1);
const limit = helperQueryNumber(req.query.limit, 10);
const categoryId = helperQueryNumber(req.query.categoryId, 0);
let search = req.query.search;
try { search = search.toLowerCase(); } catch {}
const offset = (page - 1) * limit;
Functions.findAndCountAll({
limit,
offset,
where: {
// Trie par catégorie
... (categoryId !== 0) && { categorieId: categoryId },
// Recherche
... (search != undefined) && {
[Sequelize.Op.or]: [
{ title: Sequelize.where(Sequelize.fn('LOWER', Sequelize.col('title')), 'LIKE', `%${search}%`) },
{ slug: Sequelize.where(Sequelize.fn('LOWER', Sequelize.col('slug')), 'LIKE', `%${search}%`) },
{ description: Sequelize.where(Sequelize.fn('LOWER', Sequelize.col('description')), 'LIKE', `%${search}%`) }
]
}
},
include: [
{ model: Categories, attributes: ["name", "color"] }
],
attributes: {
exclude: ["updatedAt", "utilizationForm", "article", "isOnline"]
},
order: [['createdAt', 'DESC']]
})
.then((result) => {
const { count, rows } = result;
const hasMore = (page * limit) < count;
return res.status(200).json({ totalItems: count, hasMore, rows });
})
.catch((error) => {
console.log(error);
return errorHandling(next, serverError);
});
}
exports.postFunction = (req, res, next) => { exports.postFunction = (req, res, next) => {
const { title, slug, description, type, categorieId } = req.body; const { title, slug, description, type, categorieId } = req.body;
@ -19,7 +63,9 @@ exports.postFunction = (req, res, next) => {
)) { )) {
return errorHandling(next, { message:"La fonction doit avoir une image valide.", statusCode: 400 }); return errorHandling(next, { message:"La fonction doit avoir une image valide.", statusCode: 400 });
} }
const imageName = slug + image.name; const splitedImageName = image.name.split('.');
if (splitedImageName.length !== 2) return errorHandling(next, serverError);
const imageName = slug + '.' + splitedImageName[1];
image.mv(path.join(__dirname, '..', 'assets', 'images', 'functions') + '/' + imageName, async (error) => { image.mv(path.join(__dirname, '..', 'assets', 'images', 'functions') + '/' + imageName, async (error) => {
if (error) return errorHandling(next, serverError); if (error) return errorHandling(next, serverError);
try { try {

View File

@ -18,6 +18,6 @@ module.exports = (req, _res, next) => {
}) })
.catch((error) => { .catch((error) => {
console.log(error); console.log(error);
errorHandling(next, serverError); return errorHandling(next, serverError);
}); });
} }

View File

@ -9,28 +9,45 @@ const Categories = require('../models/categories');
const AdminRouter = Router(); const AdminRouter = Router();
// Permet de créé une fonction AdminRouter.route('/functions')
AdminRouter.post('/functions', isAuth, isAdmin,
fileUpload({ // Récupère les fonctions
.get(isAuth, isAdmin, adminController.getFunctions)
// Permet de créé une fonction
.post(isAuth, isAdmin,
fileUpload({
useTempFiles: true, useTempFiles: true,
safeFileNames: true, safeFileNames: true,
preserveExtension: Number, preserveExtension: Number,
limits: { fileSize: 5 * 1024 * 1024 }, // 5mb, limits: { fileSize: 5 * 1024 * 1024 }, // 5mb,
parseNested: true parseNested: true
}), }),
[ [
body('title') body('title')
.not() .not()
.isEmpty() .isEmpty()
.withMessage("La fonction doit avoir un titre.") .withMessage("La fonction doit avoir un titre.")
.isLength({ max: 100 }) .isLength({ max: 100 })
.withMessage("Le titre est trop long."), .withMessage("Le titre est trop long.")
.custom(((title) => {
if (title === 'undefined') {
return Promise.reject("La fonction doit avoir un titre.");
}
return true;
})),
body('slug') body('slug')
.not() .not()
.isEmpty() .isEmpty()
.withMessage("La fonction doit avoir un slug.") .withMessage("La fonction doit avoir un slug.")
.isLength({ max: 100 }) .isLength({ max: 100 })
.withMessage("Le slug est trop long.") .withMessage("Le slug est trop long.")
.custom(((slug) => {
if (slug === 'undefined') {
return Promise.reject("La fonction doit avoir un slug.");
}
return true;
}))
.custom((async (slug) => { .custom((async (slug) => {
try { try {
const FunctionSlug = await Functions.findOne({ where: { slug } }); const FunctionSlug = await Functions.findOne({ where: { slug } });
@ -46,15 +63,21 @@ fileUpload({
.not() .not()
.isEmpty() .isEmpty()
.withMessage("La fonction doit avoir une description.") .withMessage("La fonction doit avoir une description.")
.isLength({ max: 255 }) .isLength({ max: 255, min: 1 })
.withMessage("La description est trop longue."), .withMessage("La description est trop longue.")
.custom(((description) => {
if (description === 'undefined') {
return Promise.reject("La fonction doit avoir une description.");
}
return true;
})),
body('categorieId') body('categorieId')
.not() .not()
.isEmpty() .isEmpty()
.withMessage("La fonction doit avoir une catégorie.") .withMessage("La fonction doit avoir une catégorie.")
.custom(async (categorieId) => { .custom(async (categorieId) => {
try { try {
const categorieFound = await Categories.findOne({ where: { id: categorieId } }); const categorieFound = await Categories.findOne({ where: { id: parseInt(categorieId) } });
if (!categorieFound) { if (!categorieFound) {
return Promise.reject("La catégorie n'existe pas!"); return Promise.reject("La catégorie n'existe pas!");
} }
@ -70,9 +93,11 @@ fileUpload({
} }
return true; return true;
}) })
], adminController.postFunction); ], adminController.postFunction);
// Supprime une fonction avec son id AdminRouter.route('/functions/:id')
AdminRouter.delete('/functions/:id', isAuth, isAdmin, adminController.deleteFunction);
// Supprime une fonction avec son id
.delete(isAuth, isAdmin, adminController.deleteFunction);
module.exports = AdminRouter; module.exports = AdminRouter;

View File

@ -55,7 +55,12 @@ const FunctionsList = (props) => {
const getFunctionsData = () => { const getFunctionsData = () => {
setLoadingFunctions(true); setLoadingFunctions(true);
return new Promise(async (next) => { return new Promise(async (next) => {
const result = await api.get(`/functions?page=${pageFunctions}&limit=10&categoryId=${inputSearch.selectedCategory}&search=${inputSearch.search}`); const URL = `${(props.isAdmin) ? "/admin/functions" : "/functions"}?page=${pageFunctions}&limit=10&categoryId=${inputSearch.selectedCategory}&search=${inputSearch.search}`;
const result = await api.get(URL, {
headers: {
...(props.isAdmin && props.token != undefined) && { 'Authorization': props.token }
}
});
setLoadingFunctions(false); setLoadingFunctions(false);
next(result.data); next(result.data);
}); });

View File

@ -1,18 +0,0 @@
import Cookies from "universal-cookie";
const addFunction = (props) => {
return (
<p>Crée une nouvelle fonction</p>
);
}
export async function getServerSideProps({ req }) {
const cookies = new Cookies(req.headers.cookie);
return {
props: {
user: { ...cookies.get('user') }
}
};
}
export default addFunction;

View File

@ -1,12 +1,66 @@
import { Fragment } from 'react'; import { Fragment, useState, useEffect } from 'react';
import Link from 'next/link';
import Cookies from "universal-cookie"; import Cookies from "universal-cookie";
import HeadTag from '../../components/HeadTag'; import HeadTag from '../../components/HeadTag';
import FunctionsList from '../../components/FunctionsList/FunctionsList'; import FunctionsList from '../../components/FunctionsList/FunctionsList';
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
import { faTimes } from '@fortawesome/free-solid-svg-icons';
import Modal from '../../components/Modal';
import redirect from '../../utils/redirect'; import redirect from '../../utils/redirect';
import htmlParser from 'html-react-parser';
import Loader from '../../components/Loader';
import useAPI from '../../hooks/useAPI';
import '../../public/css/pages/admin.css';
import api from '../../utils/api';
const Admin = (props) => { const Admin = (props) => {
const [, categories] = useAPI('/categories');
const [isOpen, setIsOpen] = useState(false);
const [inputState, setInputState] = useState({ type: 'form' });
const [message, setMessage] = useState("");
const [isLoading, setIsLoading] = useState(false);
useEffect(() => {
if (categories.length > 0) {
handleChange({
target: {
name: "categorieId",
value: categories[0].id
}
});
}
}, [categories]);
const toggleModal = () => setIsOpen(!isOpen);
const handleChange = (event, isTypeCheck = false) => {
const inputStateNew = { ...inputState };
inputStateNew[event.target.name] = (event.target.files != undefined) ? event.target.files[0] : (isTypeCheck) ? event.target.checked : event.target.value;
setInputState(inputStateNew);
}
const handleSubmit = (event) => {
event.preventDefault();
setIsLoading(true);
const formData = new FormData();
formData.append('type', inputState.type);
formData.append('categorieId', inputState.categorieId);
formData.append('title', inputState.title);
formData.append('slug', inputState.slug);
formData.append('description', inputState.description);
formData.append('image', inputState.image);
api.post('/admin/functions', formData, { headers: { 'Authorization': props.user.token } })
.then((response) => {
setMessage(`<p class="form-success"><b>Succès:</b> ${response.data.message}</p>`);
setIsLoading(false);
})
.catch((error) => {
setMessage(`<p class="form-error"><b>Erreur:</b> ${error.response.data.message}</p>`);
setIsLoading(false);
});
}
if (!props.user.isAdmin && typeof window != 'undefined') { if (!props.user.isAdmin && typeof window != 'undefined') {
return redirect({}, '/404'); return redirect({}, '/404');
} }
@ -15,14 +69,89 @@ const Admin = (props) => {
<Fragment> <Fragment>
<HeadTag title="Admin - FunctionProject" description="Page d'administration de FunctionProject." /> <HeadTag title="Admin - FunctionProject" description="Page d'administration de FunctionProject." />
<FunctionsList isAdmin> {/* Création d'une fonction */}
{(isOpen) ?
<Modal toggleModal={toggleModal}>
<div className="Admin__Modal__container container-fluid">
<div className="Admin__Modal__row row">
<div className="col-24">
<div className="Admin__Modal-top-container row">
<div className="col-24">
<span onClick={toggleModal} style={{ cursor: 'pointer', position: 'absolute', left: 0 }}>
<FontAwesomeIcon icon={faTimes} style={{ width: '1.5rem', color: 'red' }} />
</span>
<h2 className="text-center">Crée une nouvelle fonction</h2>
</div>
</div>
</div>
<div className="col-24">
<form onSubmit={handleSubmit}>
<div className="form-group">
<label className="form-label" htmlFor="title">Titre :</label>
<input value={inputState.name} onChange={handleChange} type="text" name="title" id="title" className="form-control" placeholder="(e.g : Nombre aléatoire)" />
</div>
<div className="form-group">
<label className="form-label" htmlFor="slug">Slug :</label>
<input value={inputState.name} onChange={handleChange} type="text" name="slug" id="slug" className="form-control" placeholder="(e.g : randomNumber)" />
</div>
<div className="form-group">
<label className="form-label" htmlFor="description">Description :</label>
<textarea style={{ height: 'auto' }} value={inputState.biography} onChange={handleChange} name="description" id="description" className="form-control" rows="5"></textarea>
</div>
<div className="form-group">
<label className="form-label" htmlFor="type">Type :</label>
<select onChange={handleChange} name="type" id="type" className="form-control">
<option value="form">Formulaire</option>
<option value="article">Article</option>
<option value="page">Page</option>
</select>
</div>
<div className="form-group">
<label className="form-label" htmlFor="categorieId">Catégorie :</label>
<select onChange={handleChange} name="categorieId" id="categorieId" className="form-control">
{categories.map((category) => (
<option key={category.id} value={category.id} className="Admin__Modal-select-option" style={{ backgroundColor: category.color }}>{category.name}</option>
))}
</select>
</div>
<div className="form-group">
<label className="form-label" htmlFor="image">Image <em>(150x150 recommandé)</em> :</label>
<br/>
<input onChange={handleChange} accept="image/jpeg,image/jpg,image/png" type="file" name="image" id="image" />
</div>
<div className="form-group text-center">
<button type="submit" className="btn btn-dark">Envoyer</button>
</div>
</form>
<div className="form-result text-center">
{
(isLoading) ?
<Loader />
:
htmlParser(message)
}
</div>
</div>
</div>
</div>
</Modal>
:
<FunctionsList isAdmin token={props.user.token}>
<div className="col-24"> <div className="col-24">
<h1 className="Functions__title">Administration</h1> <h1 className="Functions__title">Administration</h1>
<Link href={"/admin/addFunction"}> <button onClick={toggleModal} style={{ margin: '0 0 40px 0' }} className="btn btn-dark">Crée une nouvelle fonction</button>
<button style={{ margin: '0 0 40px 0' }} className="btn btn-dark">Crée une nouvelle fonction</button>
</Link>
</div> </div>
</FunctionsList> </FunctionsList>
}
</Fragment> </Fragment>
); );
} }

View File

@ -70,10 +70,10 @@ const Profile = (props) => {
<HeadTag title={`${props.name} - FunctionProject`} description={`Profil utilisateur de ${props.name}. ${(props.biography != undefined) ? props.biography : ""}`} /> <HeadTag title={`${props.name} - FunctionProject`} description={`Profil utilisateur de ${props.name}. ${(props.biography != undefined) ? props.biography : ""}`} />
{/* Édition du profil */} {/* Édition du profil */}
{(isOpen) && {(isOpen) ?
<Modal toggleModal={toggleModal}> <Modal toggleModal={toggleModal}>
<div className="container-fluid Profile__container"> <div className="Profile__container container-fluid">
<div className="row Profile__row"> <div className="Profile__row row">
<div className="col-24"> <div className="col-24">
<div className="Profile__Modal-top-container row"> <div className="Profile__Modal-top-container row">
<div className="col-24"> <div className="col-24">
@ -130,9 +130,10 @@ const Profile = (props) => {
</div> </div>
</div> </div>
</Modal> </Modal>
}
<div className={`container-fluid Profile__container ${(isOpen) ? "d-none" : ""}`}> :
<div className="container-fluid Profile__container">
<div className="row Profile__row"> <div className="row Profile__row">
<div className="col-20"> <div className="col-20">
<div className="text-center"> <div className="text-center">
@ -201,6 +202,8 @@ const Profile = (props) => {
</div> </div>
} }
</div> </div>
}
</Fragment> </Fragment>
); );
} }

View File

@ -0,0 +1,22 @@
.Admin__Modal__container {
display: flex;
flex-direction: column;
justify-content: center;
margin: 30px 0 0 0;
}
.Admin__Modal__row {
display: flex;
align-items: center;
flex-direction: column;
word-wrap: break-word;
box-shadow: 0px 0px 6px 6px rgba(0, 0, 0, .25);
border: 1px solid black;
border-radius: 1rem;
margin-bottom: 50px;
}
.Admin__Modal-top-container {
margin: 20px 0;
}
.Admin__Modal-select-option {
color: rgb(221, 220, 220);
}