frontend et backend: Crée une nouvelle fonction
This commit is contained in:
parent
c157f7e922
commit
42193066a8
@ -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 {
|
||||||
|
@ -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);
|
||||||
});
|
});
|
||||||
}
|
}
|
@ -9,8 +9,13 @@ const Categories = require('../models/categories');
|
|||||||
|
|
||||||
const AdminRouter = Router();
|
const AdminRouter = Router();
|
||||||
|
|
||||||
|
AdminRouter.route('/functions')
|
||||||
|
|
||||||
|
// Récupère les fonctions
|
||||||
|
.get(isAuth, isAdmin, adminController.getFunctions)
|
||||||
|
|
||||||
// Permet de créé une fonction
|
// Permet de créé une fonction
|
||||||
AdminRouter.post('/functions', isAuth, isAdmin,
|
.post(isAuth, isAdmin,
|
||||||
fileUpload({
|
fileUpload({
|
||||||
useTempFiles: true,
|
useTempFiles: true,
|
||||||
safeFileNames: true,
|
safeFileNames: true,
|
||||||
@ -24,13 +29,25 @@ fileUpload({
|
|||||||
.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!");
|
||||||
}
|
}
|
||||||
@ -72,7 +95,9 @@ fileUpload({
|
|||||||
})
|
})
|
||||||
], adminController.postFunction);
|
], adminController.postFunction);
|
||||||
|
|
||||||
|
AdminRouter.route('/functions/:id')
|
||||||
|
|
||||||
// Supprime une fonction avec son id
|
// Supprime une fonction avec son id
|
||||||
AdminRouter.delete('/functions/:id', isAuth, isAdmin, adminController.deleteFunction);
|
.delete(isAuth, isAdmin, adminController.deleteFunction);
|
||||||
|
|
||||||
module.exports = AdminRouter;
|
module.exports = AdminRouter;
|
@ -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);
|
||||||
});
|
});
|
||||||
|
@ -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;
|
|
@ -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>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -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>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
22
website/public/css/pages/admin.css
Normal file
22
website/public/css/pages/admin.css
Normal 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);
|
||||||
|
}
|
Reference in New Issue
Block a user