Merge pull request #16 in WFCC/cc34 from 15-ajout-admin to master
* commit 'fc5b58fffb47d28f9d83b315371d3cb067c1a437': Mise à jour README Ajout d'un admin dans la fixture Mot de passe quand on créer un utilisateur depuis interface admin Remplacement ROLE_INSTRUCTOR en ROLE_INSTRUCTEUR Ajout CRUD pour gerer utilisateurs, uniquement accessible pour un admin
This commit is contained in:
commit
7e6a8e51f4
|
@ -0,0 +1,5 @@
|
||||||
|
<component name="ProjectCodeStyleConfiguration">
|
||||||
|
<state>
|
||||||
|
<option name="PREFERRED_PROJECT_CODE_STYLE" value="Default" />
|
||||||
|
</state>
|
||||||
|
</component>
|
|
@ -94,3 +94,8 @@ Pas de commandes.
|
||||||
```bash
|
```bash
|
||||||
symfony console doctrine:fixtures:load
|
symfony console doctrine:fixtures:load
|
||||||
```
|
```
|
||||||
|
|
||||||
|
### Question 15
|
||||||
|
```bash
|
||||||
|
symfony console doctrine:fixtures:load
|
||||||
|
```
|
|
@ -10,7 +10,8 @@ security:
|
||||||
class: App\Entity\User
|
class: App\Entity\User
|
||||||
property: email
|
property: email
|
||||||
role_hierarchy:
|
role_hierarchy:
|
||||||
ROLE_INSTRUCTOR: ROLE_USER
|
ROLE_ADMINISTRATEUR: [ ROLE_INSTRUCTEUR, ROLE_APPRENTI, ROLE_USER ]
|
||||||
|
ROLE_INSTRUCTEUR: ROLE_USER
|
||||||
ROLE_APPRENTI: ROLE_USER
|
ROLE_APPRENTI: ROLE_USER
|
||||||
firewalls:
|
firewalls:
|
||||||
dev:
|
dev:
|
||||||
|
@ -34,8 +35,8 @@ security:
|
||||||
# Easy way to control access for large sections of your site
|
# Easy way to control access for large sections of your site
|
||||||
# Note: Only the *first* access control that matches will be used
|
# Note: Only the *first* access control that matches will be used
|
||||||
access_control:
|
access_control:
|
||||||
# - { path: ^/admin, roles: ROLE_ADMIN }
|
# - { path: ^/admin, roles: ROLE_ADMIN }
|
||||||
# - { path: ^/profile, roles: ROLE_USER }
|
# - { path: ^/profile, roles: ROLE_USER }
|
||||||
|
|
||||||
when@test:
|
when@test:
|
||||||
security:
|
security:
|
||||||
|
|
|
@ -24,7 +24,7 @@ class AtelierController extends AbstractController
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[IsGranted('ROLE_INSTRUCTOR')]
|
#[IsGranted('ROLE_INSTRUCTEUR')]
|
||||||
#[Route('/byme', name: 'app_atelier_by_me', methods: ['GET'])]
|
#[Route('/byme', name: 'app_atelier_by_me', methods: ['GET'])]
|
||||||
public function index_created_by_user(AtelierRepository $atelierRepository, MarkdownAtelier $markdown): Response
|
public function index_created_by_user(AtelierRepository $atelierRepository, MarkdownAtelier $markdown): Response
|
||||||
{
|
{
|
||||||
|
@ -44,7 +44,7 @@ class AtelierController extends AbstractController
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[IsGranted('ROLE_INSTRUCTOR')]
|
#[IsGranted('ROLE_INSTRUCTEUR')]
|
||||||
#[Route('/new', name: 'app_atelier_new', methods: ['GET', 'POST'])]
|
#[Route('/new', name: 'app_atelier_new', methods: ['GET', 'POST'])]
|
||||||
public function new(Request $request, AtelierRepository $atelierRepository): Response
|
public function new(Request $request, AtelierRepository $atelierRepository): Response
|
||||||
{
|
{
|
||||||
|
@ -74,7 +74,7 @@ class AtelierController extends AbstractController
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[IsGranted('ROLE_INSTRUCTOR')]
|
#[IsGranted('ROLE_INSTRUCTEUR')]
|
||||||
#[Route('/{id}/edit', name: 'app_atelier_edit', methods: ['GET', 'POST'])]
|
#[Route('/{id}/edit', name: 'app_atelier_edit', methods: ['GET', 'POST'])]
|
||||||
public function edit(Request $request, Atelier $atelier, AtelierRepository $atelierRepository): Response
|
public function edit(Request $request, Atelier $atelier, AtelierRepository $atelierRepository): Response
|
||||||
{
|
{
|
||||||
|
@ -97,7 +97,7 @@ class AtelierController extends AbstractController
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[IsGranted('ROLE_INSTRUCTOR')]
|
#[IsGranted('ROLE_INSTRUCTEUR')]
|
||||||
#[Route('/{id}', name: 'app_atelier_delete', methods: ['POST'])]
|
#[Route('/{id}', name: 'app_atelier_delete', methods: ['POST'])]
|
||||||
public function delete(Request $request, Atelier $atelier, AtelierRepository $atelierRepository): Response
|
public function delete(Request $request, Atelier $atelier, AtelierRepository $atelierRepository): Response
|
||||||
{
|
{
|
||||||
|
|
|
@ -13,7 +13,7 @@ class SecurityController extends AbstractController
|
||||||
public function login(AuthenticationUtils $authenticationUtils): Response
|
public function login(AuthenticationUtils $authenticationUtils): Response
|
||||||
{
|
{
|
||||||
if ($this->getUser()) {
|
if ($this->getUser()) {
|
||||||
return $this->redirectToRoute('app_atelier_index');
|
return $this->redirectToRoute('app_index');
|
||||||
}
|
}
|
||||||
|
|
||||||
// get the login error if there is one
|
// get the login error if there is one
|
||||||
|
|
|
@ -0,0 +1,88 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Controller;
|
||||||
|
|
||||||
|
use App\Entity\User;
|
||||||
|
use App\Form\NewUserType;
|
||||||
|
use App\Form\UserType;
|
||||||
|
use App\Repository\UserRepository;
|
||||||
|
use Sensio\Bundle\FrameworkExtraBundle\Configuration\IsGranted;
|
||||||
|
use Symfony\Bundle\FrameworkBundle\Controller\AbstractController;
|
||||||
|
use Symfony\Component\HttpFoundation\Request;
|
||||||
|
use Symfony\Component\HttpFoundation\Response;
|
||||||
|
use Symfony\Component\PasswordHasher\Hasher\UserPasswordHasherInterface;
|
||||||
|
use Symfony\Component\Routing\Annotation\Route;
|
||||||
|
|
||||||
|
#[IsGranted('ROLE_ADMINISTRATEUR')]
|
||||||
|
#[Route('/user')]
|
||||||
|
class UserController extends AbstractController
|
||||||
|
{
|
||||||
|
#[Route('/', name: 'app_user_index', methods: ['GET'])]
|
||||||
|
public function index(UserRepository $userRepository): Response
|
||||||
|
{
|
||||||
|
return $this->render('user/index.html.twig', [
|
||||||
|
'users' => $userRepository->findAll(),
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Route('/new', name: 'app_user_new', methods: ['GET', 'POST'])]
|
||||||
|
public function new(Request $request, UserRepository $userRepository, UserPasswordHasherInterface $userPasswordHasher): Response
|
||||||
|
{
|
||||||
|
$user = new User();
|
||||||
|
$form = $this->createForm(NewUserType::class, $user);
|
||||||
|
$form->handleRequest($request);
|
||||||
|
|
||||||
|
if ($form->isSubmitted() && $form->isValid()) {
|
||||||
|
$user->setPassword(
|
||||||
|
$userPasswordHasher->hashPassword(
|
||||||
|
$user,
|
||||||
|
$form->get('plainPassword')->getData()
|
||||||
|
)
|
||||||
|
);
|
||||||
|
$userRepository->save($user, true);
|
||||||
|
|
||||||
|
return $this->redirectToRoute('app_user_index', [], Response::HTTP_SEE_OTHER);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->renderForm('user/new.html.twig', [
|
||||||
|
'user' => $user,
|
||||||
|
'form' => $form,
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Route('/{id}', name: 'app_user_show', methods: ['GET'])]
|
||||||
|
public function show(User $user): Response
|
||||||
|
{
|
||||||
|
return $this->render('user/show.html.twig', [
|
||||||
|
'user' => $user,
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Route('/{id}/edit', name: 'app_user_edit', methods: ['GET', 'POST'])]
|
||||||
|
public function edit(Request $request, User $user, UserRepository $userRepository): Response
|
||||||
|
{
|
||||||
|
$form = $this->createForm(UserType::class, $user);
|
||||||
|
$form->handleRequest($request);
|
||||||
|
|
||||||
|
if ($form->isSubmitted() && $form->isValid()) {
|
||||||
|
$userRepository->save($user, true);
|
||||||
|
|
||||||
|
return $this->redirectToRoute('app_user_index', [], Response::HTTP_SEE_OTHER);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->renderForm('user/edit.html.twig', [
|
||||||
|
'user' => $user,
|
||||||
|
'form' => $form,
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Route('/{id}', name: 'app_user_delete', methods: ['POST'])]
|
||||||
|
public function delete(Request $request, User $user, UserRepository $userRepository): Response
|
||||||
|
{
|
||||||
|
if ($this->isCsrfTokenValid('delete' . $user->getId(), $request->request->get('_token'))) {
|
||||||
|
$userRepository->remove($user, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->redirectToRoute('app_user_index', [], Response::HTTP_SEE_OTHER);
|
||||||
|
}
|
||||||
|
}
|
|
@ -7,29 +7,45 @@ use App\Entity\User;
|
||||||
use Doctrine\Bundle\FixturesBundle\Fixture;
|
use Doctrine\Bundle\FixturesBundle\Fixture;
|
||||||
use Doctrine\Persistence\ObjectManager;
|
use Doctrine\Persistence\ObjectManager;
|
||||||
use Faker\Generator;
|
use Faker\Generator;
|
||||||
|
use Symfony\Component\PasswordHasher\Hasher\UserPasswordHasherInterface;
|
||||||
|
|
||||||
class AtelierFixture extends Fixture
|
class AtelierFixture extends Fixture
|
||||||
{
|
{
|
||||||
|
private UserPasswordHasherInterface $userPasswordHasher;
|
||||||
|
|
||||||
|
public function __construct(UserPasswordHasherInterface $userPasswordHasher)
|
||||||
|
{
|
||||||
|
$this->userPasswordHasher = $userPasswordHasher;
|
||||||
|
}
|
||||||
|
|
||||||
public function load(ObjectManager $manager): void
|
public function load(ObjectManager $manager): void
|
||||||
{
|
{
|
||||||
$faker = \Faker\Factory::create("fr_FR");
|
$faker = \Faker\Factory::create("fr_FR");
|
||||||
$user = new User();
|
$user = new User();
|
||||||
$user->setEmail('test@hotmail.com')
|
$user
|
||||||
->setNom("test")
|
->setEmail('admin@admin.fr')
|
||||||
->setPrenom("test")
|
->setNom("admin")
|
||||||
->setPassword("");
|
->setPrenom("admin")
|
||||||
|
->setRoles(['ROLE_ADMINISTRATEUR']);
|
||||||
|
|
||||||
|
$user->setPassword(
|
||||||
|
$this->userPasswordHasher->hashPassword(
|
||||||
|
$user,
|
||||||
|
'admin'
|
||||||
|
)
|
||||||
|
);
|
||||||
|
$manager->persist($user);
|
||||||
|
|
||||||
$users = $this->createUsers($manager, $faker);
|
$users = $this->createUsers($manager, $faker);
|
||||||
|
|
||||||
$manager->persist($user);
|
|
||||||
for ($i = 0; $i <= 20; $i++) {
|
for ($i = 0; $i <= 20; $i++) {
|
||||||
$atelier = new Atelier();
|
$atelier = new Atelier();
|
||||||
$atelier->setNom($faker->word)
|
$atelier->setNom($faker->word)
|
||||||
->setDescription("# " . $faker->sentence(3) . "\n" . $faker->paragraph())
|
->setDescription("# " . $faker->sentence(3) . "\n" . $faker->paragraph())
|
||||||
->setInstructeur($user);
|
->setInstructeur($user);
|
||||||
|
|
||||||
foreach ($faker->randomElements($users, $faker->randomNumber() % sizeof($users)) as $user) {
|
foreach ($faker->randomElements($users, $faker->randomNumber() % sizeof($users)) as $eleve) {
|
||||||
$atelier->addEleve($user);
|
$atelier->addEleve($eleve);
|
||||||
}
|
}
|
||||||
|
|
||||||
$manager->persist($atelier);
|
$manager->persist($atelier);
|
||||||
|
|
|
@ -83,9 +83,9 @@ class User implements UserInterface, PasswordAuthenticatedUserInterface
|
||||||
{
|
{
|
||||||
$roles = $this->roles;
|
$roles = $this->roles;
|
||||||
// guarantee every user at least has ROLE_USER
|
// guarantee every user at least has ROLE_USER
|
||||||
$roles[] = 'ROLE_USER';
|
if (sizeof($roles) == 0) {
|
||||||
$roles[] = 'ROLE_INSTRUCTOR';
|
$roles[] = "ROLE_APPRENTI";
|
||||||
$roles[] = 'ROLE_APPRENTI';
|
}
|
||||||
|
|
||||||
return array_unique($roles);
|
return array_unique($roles);
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,71 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Form;
|
||||||
|
|
||||||
|
use App\Entity\User;
|
||||||
|
use Symfony\Component\Form\AbstractType;
|
||||||
|
use Symfony\Component\Form\CallbackTransformer;
|
||||||
|
use Symfony\Component\Form\Extension\Core\Type\ChoiceType;
|
||||||
|
use Symfony\Component\Form\Extension\Core\Type\PasswordType;
|
||||||
|
use Symfony\Component\Form\FormBuilderInterface;
|
||||||
|
use Symfony\Component\OptionsResolver\OptionsResolver;
|
||||||
|
use Symfony\Component\Validator\Constraints\Length;
|
||||||
|
use Symfony\Component\Validator\Constraints\NotBlank;
|
||||||
|
|
||||||
|
class NewUserType extends AbstractType
|
||||||
|
{
|
||||||
|
public function buildForm(FormBuilderInterface $builder, array $options): void
|
||||||
|
{
|
||||||
|
$builder
|
||||||
|
->add('email')
|
||||||
|
->add('plainPassword', PasswordType::class, [
|
||||||
|
// instead of being set onto the object directly,
|
||||||
|
// this is read and encoded in the controller
|
||||||
|
'mapped' => false,
|
||||||
|
'attr' => ['autocomplete' => 'new-password'],
|
||||||
|
'constraints' => [
|
||||||
|
new NotBlank([
|
||||||
|
'message' => 'Please enter a password',
|
||||||
|
]),
|
||||||
|
new Length([
|
||||||
|
'min' => 6,
|
||||||
|
'minMessage' => 'Your password should be at least {{ limit }} characters',
|
||||||
|
// max length allowed by Symfony for security reasons
|
||||||
|
'max' => 4096,
|
||||||
|
]),
|
||||||
|
],
|
||||||
|
])
|
||||||
|
->add('roles', ChoiceType::class, [
|
||||||
|
'choices' => [
|
||||||
|
'Apprenti' => 'ROLE_APPRENTI',
|
||||||
|
'Instructeur' => 'ROLE_INSTRUCTEUR',
|
||||||
|
'Admin' => 'ROLE_ADMINISTRATEUR'
|
||||||
|
],
|
||||||
|
'required' => true,
|
||||||
|
'multiple' => false,
|
||||||
|
'expanded' => false,
|
||||||
|
])
|
||||||
|
->add('nom')
|
||||||
|
->add('prenom');
|
||||||
|
|
||||||
|
$builder->get('roles')
|
||||||
|
->addModelTransformer(new CallbackTransformer(
|
||||||
|
function ($rolesArray) {
|
||||||
|
// transform the array to a string
|
||||||
|
return count($rolesArray) ? $rolesArray[0] : null;
|
||||||
|
},
|
||||||
|
function ($rolesString) {
|
||||||
|
// transform the string back to an array
|
||||||
|
return [$rolesString];
|
||||||
|
}
|
||||||
|
));
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public function configureOptions(OptionsResolver $resolver): void
|
||||||
|
{
|
||||||
|
$resolver->setDefaults([
|
||||||
|
'data_class' => User::class,
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,52 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Form;
|
||||||
|
|
||||||
|
use App\Entity\User;
|
||||||
|
use PhpParser\Node\Expr\Array_;
|
||||||
|
use Symfony\Component\Form\AbstractType;
|
||||||
|
use Symfony\Component\Form\CallbackTransformer;
|
||||||
|
use Symfony\Component\Form\Extension\Core\Type\ChoiceType;
|
||||||
|
use Symfony\Component\Form\FormBuilderInterface;
|
||||||
|
use Symfony\Component\OptionsResolver\OptionsResolver;
|
||||||
|
|
||||||
|
class UserType extends AbstractType
|
||||||
|
{
|
||||||
|
public function buildForm(FormBuilderInterface $builder, array $options): void
|
||||||
|
{
|
||||||
|
$builder
|
||||||
|
->add('email')
|
||||||
|
->add('roles', ChoiceType::class, [
|
||||||
|
'choices' => [
|
||||||
|
'Apprenti' => 'ROLE_APPRENTI',
|
||||||
|
'Instructeur' => 'ROLE_INSTRUCTEUR',
|
||||||
|
'Admin' => 'ROLE_ADMINISTRATEUR'
|
||||||
|
],
|
||||||
|
'required' => true,
|
||||||
|
'multiple' => false,
|
||||||
|
'expanded' => false,
|
||||||
|
])
|
||||||
|
->add('nom')
|
||||||
|
->add('prenom');
|
||||||
|
|
||||||
|
$builder->get('roles')
|
||||||
|
->addModelTransformer(new CallbackTransformer(
|
||||||
|
function ($rolesArray) {
|
||||||
|
// transform the array to a string
|
||||||
|
return count($rolesArray)? $rolesArray[0]: null;
|
||||||
|
},
|
||||||
|
function ($rolesString) {
|
||||||
|
// transform the string back to an array
|
||||||
|
return [$rolesString];
|
||||||
|
}
|
||||||
|
));
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public function configureOptions(OptionsResolver $resolver): void
|
||||||
|
{
|
||||||
|
$resolver->setDefaults([
|
||||||
|
'data_class' => User::class,
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
}
|
|
@ -46,7 +46,7 @@ class AppAuthentificatorAuthenticator extends AbstractLoginFormAuthenticator
|
||||||
return new RedirectResponse($targetPath);
|
return new RedirectResponse($targetPath);
|
||||||
}
|
}
|
||||||
|
|
||||||
return new RedirectResponse($this->urlGenerator->generate('app_atelier_index'));
|
return new RedirectResponse($this->urlGenerator->generate('app_index'));
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function getLoginUrl(Request $request): string
|
protected function getLoginUrl(Request $request): string
|
||||||
|
|
|
@ -13,16 +13,22 @@
|
||||||
<li class="nav-item">
|
<li class="nav-item">
|
||||||
<a class="nav-link" href="{{ path('app_atelier_index') }}">Atelier</a>
|
<a class="nav-link" href="{{ path('app_atelier_index') }}">Atelier</a>
|
||||||
</li>
|
</li>
|
||||||
{% if app.user %}
|
|
||||||
|
{% if app.user and is_granted('ROLE_INSTRUCTEUR') %}
|
||||||
<li class="nav-item">
|
<li class="nav-item">
|
||||||
<a class="nav-link" href="{{ path('app_atelier_by_me') }}">Mes Ateliers</a>
|
<a class="nav-link" href="{{ path('app_atelier_by_me') }}">Mes Ateliers</a>
|
||||||
</li>
|
</li>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% if app.user %}
|
{% if app.user and is_granted('ROLE_APPRENTI') %}
|
||||||
<li class="nav-item">
|
<li class="nav-item">
|
||||||
<a class="nav-link" href="{{ path('app_atelier_inscrit') }}">Mes Formations</a>
|
<a class="nav-link" href="{{ path('app_atelier_inscrit') }}">Mes Formations</a>
|
||||||
</li>
|
</li>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
{% if app.user and is_granted('ROLE_ADMINISTRATEUR') %}
|
||||||
|
<li class="nav-item">
|
||||||
|
<a class="nav-link" href="{{ path('app_user_index') }}">Liste des Utilisateurs</a>
|
||||||
|
</li>
|
||||||
|
{% endif %}
|
||||||
</ul>
|
</ul>
|
||||||
</div>
|
</div>
|
||||||
{% if app.user %}
|
{% if app.user %}
|
||||||
|
|
|
@ -0,0 +1,4 @@
|
||||||
|
<form method="post" action="{{ path('app_user_delete', {'id': user.id}) }}" onsubmit="return confirm('Are you sure you want to delete this item?');">
|
||||||
|
<input type="hidden" name="_token" value="{{ csrf_token('delete' ~ user.id) }}">
|
||||||
|
<button class="btn">Delete</button>
|
||||||
|
</form>
|
|
@ -0,0 +1,4 @@
|
||||||
|
{{ form_start(form) }}
|
||||||
|
{{ form_widget(form) }}
|
||||||
|
<button class="btn">{{ button_label|default('Save') }}</button>
|
||||||
|
{{ form_end(form) }}
|
|
@ -0,0 +1,13 @@
|
||||||
|
{% extends 'base.html.twig' %}
|
||||||
|
|
||||||
|
{% block title %}Edit User{% endblock %}
|
||||||
|
|
||||||
|
{% block body %}
|
||||||
|
<h1>Edit User</h1>
|
||||||
|
|
||||||
|
{{ include('user/_form.html.twig', {'button_label': 'Update'}) }}
|
||||||
|
|
||||||
|
<a href="{{ path('app_user_index') }}">back to list</a>
|
||||||
|
|
||||||
|
{{ include('user/_delete_form.html.twig') }}
|
||||||
|
{% endblock %}
|
|
@ -0,0 +1,41 @@
|
||||||
|
{% extends 'base.html.twig' %}
|
||||||
|
|
||||||
|
{% block title %}User index{% endblock %}
|
||||||
|
|
||||||
|
{% block body %}
|
||||||
|
<h1>User index</h1>
|
||||||
|
|
||||||
|
<table class="table">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>Id</th>
|
||||||
|
<th>Email</th>
|
||||||
|
<th>Roles</th>
|
||||||
|
<th>Nom</th>
|
||||||
|
<th>Prenom</th>
|
||||||
|
<th>actions</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
{% for user in users %}
|
||||||
|
<tr>
|
||||||
|
<td>{{ user.id }}</td>
|
||||||
|
<td>{{ user.email }}</td>
|
||||||
|
<td>{{ user.roles ? user.roles[0] : '' }}</td>
|
||||||
|
<td>{{ user.nom }}</td>
|
||||||
|
<td>{{ user.prenom }}</td>
|
||||||
|
<td>
|
||||||
|
<a href="{{ path('app_user_show', {'id': user.id}) }}">show</a>
|
||||||
|
<a href="{{ path('app_user_edit', {'id': user.id}) }}">edit</a>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% else %}
|
||||||
|
<tr>
|
||||||
|
<td colspan="7">no records found</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
|
||||||
|
<a href="{{ path('app_user_new') }}">Create new</a>
|
||||||
|
{% endblock %}
|
|
@ -0,0 +1,11 @@
|
||||||
|
{% extends 'base.html.twig' %}
|
||||||
|
|
||||||
|
{% block title %}New User{% endblock %}
|
||||||
|
|
||||||
|
{% block body %}
|
||||||
|
<h1>Create new User</h1>
|
||||||
|
|
||||||
|
{{ include('user/_form.html.twig') }}
|
||||||
|
|
||||||
|
<a href="{{ path('app_user_index') }}">back to list</a>
|
||||||
|
{% endblock %}
|
|
@ -0,0 +1,38 @@
|
||||||
|
{% extends 'base.html.twig' %}
|
||||||
|
|
||||||
|
{% block title %}User{% endblock %}
|
||||||
|
|
||||||
|
{% block body %}
|
||||||
|
<h1>User</h1>
|
||||||
|
|
||||||
|
<table class="table">
|
||||||
|
<tbody>
|
||||||
|
<tr>
|
||||||
|
<th>Id</th>
|
||||||
|
<td>{{ user.id }}</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<th>Email</th>
|
||||||
|
<td>{{ user.email }}</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<th>Roles</th>
|
||||||
|
<td>{{ user.roles ? user.roles[0] : '' }}</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<th>Nom</th>
|
||||||
|
<td>{{ user.nom }}</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<th>Prenom</th>
|
||||||
|
<td>{{ user.prenom }}</td>
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
|
||||||
|
<a href="{{ path('app_user_index') }}">back to list</a>
|
||||||
|
|
||||||
|
<a href="{{ path('app_user_edit', {'id': user.id}) }}">edit</a>
|
||||||
|
|
||||||
|
{{ include('user/_delete_form.html.twig') }}
|
||||||
|
{% endblock %}
|
Loading…
Reference in New Issue