Add post and species forms
continuous-integration/drone/push Build is passing
Details
continuous-integration/drone/push Build is passing
Details
Squashed commit of the following: Author: Hugo PRADIER <Hugo.PRADIER2@etu.uca.fr> Author: bastien ollier <bastien.ollier@etu.uca.fr> Author: clfreville2 <clement.freville2@etu.uca.fr> Reviewed on #7pull/10/head
parent
82a3f69fa4
commit
49d60871c9
@ -0,0 +1,38 @@
|
||||
<?php
|
||||
|
||||
namespace App\Form;
|
||||
|
||||
use App\Entity\Post;
|
||||
use App\Entity\Species;
|
||||
use Symfony\Bridge\Doctrine\Form\Type\EntityType;
|
||||
use Symfony\Component\Form\AbstractType;
|
||||
use Symfony\Component\Form\FormBuilderInterface;
|
||||
use Symfony\Component\OptionsResolver\OptionsResolver;
|
||||
|
||||
class PostType extends AbstractType
|
||||
{
|
||||
public function buildForm(FormBuilderInterface $builder, array $options): void
|
||||
{
|
||||
$builder
|
||||
->add('foundDate', null, [
|
||||
'widget' => 'single_text',
|
||||
])
|
||||
->add('latitude')
|
||||
->add('longitude')
|
||||
->add('altitude')
|
||||
->add('commentary')
|
||||
->add('species', EntityType::class, [
|
||||
'class' => Species::class,
|
||||
'choice_label' => 'scientific_name',
|
||||
])
|
||||
|
||||
;
|
||||
}
|
||||
|
||||
public function configureOptions(OptionsResolver $resolver): void
|
||||
{
|
||||
$resolver->setDefaults([
|
||||
'data_class' => Post::class,
|
||||
]);
|
||||
}
|
||||
}
|
@ -0,0 +1,27 @@
|
||||
<?php
|
||||
|
||||
namespace App\Form;
|
||||
|
||||
use App\Entity\Species;
|
||||
use Symfony\Component\Form\AbstractType;
|
||||
use Symfony\Component\Form\FormBuilderInterface;
|
||||
use Symfony\Component\OptionsResolver\OptionsResolver;
|
||||
|
||||
class SpeciesType extends AbstractType
|
||||
{
|
||||
public function buildForm(FormBuilderInterface $builder, array $options): void
|
||||
{
|
||||
$builder
|
||||
->add('scientific_name')
|
||||
->add('vernacular_name')
|
||||
->add('region')
|
||||
;
|
||||
}
|
||||
|
||||
public function configureOptions(OptionsResolver $resolver): void
|
||||
{
|
||||
$resolver->setDefaults([
|
||||
'data_class' => Species::class,
|
||||
]);
|
||||
}
|
||||
}
|
@ -0,0 +1,4 @@
|
||||
<form method="post" action="{{ path('app_post_delete', {'id': post.id}) }}" onsubmit="return confirm('Are you sure you want to delete this item?');">
|
||||
<input type="hidden" name="_token" value="{{ csrf_token('delete' ~ post.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 Post{% endblock %}
|
||||
|
||||
{% block body %}
|
||||
<h1>Edit Post</h1>
|
||||
|
||||
{{ include('post/_form.html.twig', {'button_label': 'Update'}) }}
|
||||
|
||||
<a href="{{ path('app_post_index') }}">back to list</a>
|
||||
|
||||
{{ include('post/_delete_form.html.twig') }}
|
||||
{% endblock %}
|
@ -1,11 +0,0 @@
|
||||
{% extends 'base.html.twig' %}
|
||||
|
||||
{% block title %}Posts!{% endblock %}
|
||||
|
||||
{% block body %}
|
||||
{% for post in posts %}
|
||||
<div class="card">
|
||||
#{{ post.id }} trouvé le {{ post.foundDate | date("d/m/Y \\à H \\h") }}
|
||||
</div>
|
||||
{% endfor %}
|
||||
{% endblock %}
|
@ -0,0 +1,11 @@
|
||||
{% extends 'base.html.twig' %}
|
||||
|
||||
{% block title %}New Post{% endblock %}
|
||||
|
||||
{% block body %}
|
||||
<h1>Create new Post</h1>
|
||||
|
||||
{{ include('post/_form.html.twig') }}
|
||||
|
||||
<a href="{{ path('app_post_index') }}">back to list</a>
|
||||
{% endblock %}
|
@ -0,0 +1,46 @@
|
||||
{% extends 'base.html.twig' %}
|
||||
|
||||
{% block title %}Post{% endblock %}
|
||||
|
||||
{% block body %}
|
||||
<h1>Post</h1>
|
||||
|
||||
<table class="table">
|
||||
<tbody>
|
||||
<tr>
|
||||
<th>Id</th>
|
||||
<td>{{ post.id }}</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th>FoundDate</th>
|
||||
<td>{{ post.foundDate ? post.foundDate|date('Y-m-d H:i:s') : '' }}</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th>PublicationDate</th>
|
||||
<td>{{ post.publicationDate ? post.publicationDate|date('Y-m-d H:i:s') : '' }}</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th>Latitude</th>
|
||||
<td>{{ post.latitude }}</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th>Longitude</th>
|
||||
<td>{{ post.longitude }}</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th>Altitude</th>
|
||||
<td>{{ post.altitude }}</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th>Commentary</th>
|
||||
<td>{{ post.commentary }}</td>
|
||||
</tr>
|
||||
</tbody>
|
||||
</table>
|
||||
|
||||
<a href="{{ path('app_post_index') }}">back to list</a>
|
||||
|
||||
<a href="{{ path('app_post_edit', {'id': post.id}) }}">edit</a>
|
||||
|
||||
{{ include('post/_delete_form.html.twig') }}
|
||||
{% endblock %}
|
@ -0,0 +1,45 @@
|
||||
{% extends 'base.html.twig' %}
|
||||
|
||||
{% block title %}Post index{% endblock %}
|
||||
|
||||
{% block body %}
|
||||
<h1>Post index</h1>
|
||||
|
||||
<table class="table">
|
||||
<thead>
|
||||
<tr>
|
||||
<th>Id</th>
|
||||
<th>FoundDate</th>
|
||||
<th>PublicationDate</th>
|
||||
<th>Latitude</th>
|
||||
<th>Longitude</th>
|
||||
<th>Altitude</th>
|
||||
<th>Commentary</th>
|
||||
<th>actions</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
{% for post in posts %}
|
||||
<tr>
|
||||
<td>{{ post.id }}</td>
|
||||
<td>{{ post.foundDate ? post.foundDate|date('Y-m-d H:i:s') : '' }}</td>
|
||||
<td>{{ post.publicationDate ? post.publicationDate|date('Y-m-d H:i:s') : '' }}</td>
|
||||
<td>{{ post.latitude }}</td>
|
||||
<td>{{ post.longitude }}</td>
|
||||
<td>{{ post.altitude }}</td>
|
||||
<td>{{ post.commentary }}</td>
|
||||
<td>
|
||||
<a href="{{ path('app_post_show', {'id': post.id}) }}">show</a>
|
||||
<a href="{{ path('app_post_edit', {'id': post.id}) }}">edit</a>
|
||||
</td>
|
||||
</tr>
|
||||
{% else %}
|
||||
<tr>
|
||||
<td colspan="8">no records found</td>
|
||||
</tr>
|
||||
{% endfor %}
|
||||
</tbody>
|
||||
</table>
|
||||
|
||||
<a href="{{ path('app_post_new') }}">Create new</a>
|
||||
{% endblock %}
|
@ -0,0 +1,4 @@
|
||||
<form method="post" action="{{ path('app_species_delete', {'id': species.id}) }}" onsubmit="return confirm('Are you sure you want to delete this item?');">
|
||||
<input type="hidden" name="_token" value="{{ csrf_token('delete' ~ species.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) }}
|
@ -1,43 +0,0 @@
|
||||
{% extends 'base.html.twig' %}
|
||||
|
||||
{% block title %}Herbarium - Détail de l'espèces{% endblock %}
|
||||
|
||||
{% block body %}
|
||||
<style>
|
||||
.example-wrapper { margin: 1em auto; max-width: 800px; width: 95%; font: 18px/1.5 sans-serif; }
|
||||
.example-wrapper code { background: #F5F5F5; padding: 2px 6px; }
|
||||
</style>
|
||||
|
||||
<div class="example-wrapper">
|
||||
<h1>{{ specie.vernacularName }}</h1>
|
||||
<p>
|
||||
🔬 Nom Scientifique : {{ specie.scientificName }}<br/>
|
||||
📍 Region : {{ specie.region }}
|
||||
</p>
|
||||
<div>
|
||||
<h2>Posts :</h2>
|
||||
|
||||
{% for post in specie.posts %}
|
||||
<div>
|
||||
<dt>
|
||||
<h3>
|
||||
<a href="">
|
||||
📅 {{ post.publicationDate | date }}
|
||||
</a>
|
||||
</h3>
|
||||
</dt>
|
||||
<dd>
|
||||
📍 géolocalisation :<br/>
|
||||
- Longitude : {{ post.longitude }}<br/>
|
||||
- Latitude : {{ post.latitude }}<br/>
|
||||
- Altitude : {{ post.altitude }}<br/><br/>
|
||||
|
||||
💬 Commentaire :<br/>
|
||||
- {{ post.getCommentary }}
|
||||
</dd>
|
||||
</div>
|
||||
{% endfor %}
|
||||
</div>
|
||||
|
||||
</div>
|
||||
{% endblock %}
|
@ -0,0 +1,13 @@
|
||||
{% extends 'base.html.twig' %}
|
||||
|
||||
{% block title %}Edit Species{% endblock %}
|
||||
|
||||
{% block body %}
|
||||
<h1>Edit Species</h1>
|
||||
|
||||
{{ include('species/_form.html.twig', {'button_label': 'Update'}) }}
|
||||
|
||||
<a href="{{ path('app_species_index') }}">back to list</a>
|
||||
|
||||
{{ include('species/_delete_form.html.twig') }}
|
||||
{% endblock %}
|
@ -0,0 +1,11 @@
|
||||
{% extends 'base.html.twig' %}
|
||||
|
||||
{% block title %}New Species{% endblock %}
|
||||
|
||||
{% block body %}
|
||||
<h1>Create new Species</h1>
|
||||
|
||||
{{ include('species/_form.html.twig') }}
|
||||
|
||||
<a href="{{ path('app_species_index') }}">back to list</a>
|
||||
{% endblock %}
|
@ -0,0 +1,36 @@
|
||||
{% extends 'base.html.twig' %}
|
||||
|
||||
{% block title %}Species{% endblock %}
|
||||
|
||||
{% block body %}
|
||||
<h1>{{ species.vernacularName }}</h1>
|
||||
<p>
|
||||
🔬 Nom Scientifique : {{ species.scientificName }}<br/>
|
||||
📍 Region : {{ species.region }}
|
||||
</p>
|
||||
|
||||
<div>
|
||||
<h2>Posts :</h2>
|
||||
|
||||
{% for post in species.posts %}
|
||||
<div>
|
||||
<dt>
|
||||
<h3>{{ post.publicationDate | date }}</h3>
|
||||
</dt>
|
||||
<dl>
|
||||
<dt>📍Géolocalisation</dt>
|
||||
<dd>{{ post.longitude }} - {{ post.latitude }}</dd>
|
||||
|
||||
<dt>📍Commentaire</dt>
|
||||
<dd>{{ post.getCommentary }}</dd>
|
||||
</dl>
|
||||
</div>
|
||||
{% endfor %}
|
||||
|
||||
<a href="{{ path('app_species_index') }}">back to list</a>
|
||||
|
||||
<a href="{{ path('app_species_edit', {'id': species.id}) }}">edit</a>
|
||||
|
||||
{{ include('species/_delete_form.html.twig') }}
|
||||
</div>
|
||||
{% endblock %}
|
@ -0,0 +1,39 @@
|
||||
{% extends 'base.html.twig' %}
|
||||
|
||||
{% block title %}Species index{% endblock %}
|
||||
|
||||
{% block body %}
|
||||
<h1>Species index</h1>
|
||||
|
||||
<table class="table">
|
||||
<thead>
|
||||
<tr>
|
||||
<th>Id</th>
|
||||
<th>Scientific_name</th>
|
||||
<th>Vernacular_name</th>
|
||||
<th>Region</th>
|
||||
<th>actions</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
{% for species in species %}
|
||||
<tr>
|
||||
<td>{{ species.id }}</td>
|
||||
<td>{{ species.scientificName }}</td>
|
||||
<td>{{ species.vernacularName }}</td>
|
||||
<td>{{ species.region }}</td>
|
||||
<td>
|
||||
<a href="{{ path('app_species_show', {'id': species.id}) }}">show</a>
|
||||
<a href="{{ path('app_species_edit', {'id': species.id}) }}">edit</a>
|
||||
</td>
|
||||
</tr>
|
||||
{% else %}
|
||||
<tr>
|
||||
<td colspan="5">no records found</td>
|
||||
</tr>
|
||||
{% endfor %}
|
||||
</tbody>
|
||||
</table>
|
||||
|
||||
<a href="{{ path('app_species_new') }}">Create new</a>
|
||||
{% endblock %}
|
@ -0,0 +1,128 @@
|
||||
<?php
|
||||
|
||||
namespace App\Test\Controller;
|
||||
|
||||
use App\Entity\Post;
|
||||
use App\Entity\User;
|
||||
use App\Repository\PostRepository;
|
||||
use Doctrine\ORM\EntityManagerInterface;
|
||||
use Symfony\Bundle\FrameworkBundle\KernelBrowser;
|
||||
use Symfony\Bundle\FrameworkBundle\Test\WebTestCase;
|
||||
|
||||
class PostControllerTest extends WebTestCase
|
||||
{
|
||||
private KernelBrowser $client;
|
||||
private EntityManagerInterface $manager;
|
||||
private PostRepository $repository;
|
||||
private string $path = '/post/';
|
||||
|
||||
protected function setUp(): void
|
||||
{
|
||||
$this->client = static::createClient();
|
||||
/** @var EntityManagerInterface $manager */
|
||||
$manager = static::getContainer()->get(EntityManagerInterface::class);
|
||||
$this->manager = $manager;
|
||||
$this->repository = $this->manager->getRepository(Post::class);
|
||||
|
||||
foreach ($this->repository->findAll() as $object) {
|
||||
$this->manager->remove($object);
|
||||
}
|
||||
|
||||
$this->manager->flush();
|
||||
|
||||
$userRepository = $this->manager->getRepository(User::class);
|
||||
/** @var User $user */
|
||||
$user = $userRepository->findOneByEmail('test@test.fr');
|
||||
$this->client->loginUser($user);
|
||||
$this->client->request('GET', sprintf('%snew', $this->path));
|
||||
}
|
||||
|
||||
public function testIndex(): void
|
||||
{
|
||||
$crawler = $this->client->request('GET', '/post');
|
||||
|
||||
self::assertResponseStatusCodeSame(200);
|
||||
self::assertPageTitleContains('Post index');
|
||||
|
||||
// Use the $crawler to perform additional assertions e.g.
|
||||
// self::assertSame('Some text on the page', $crawler->filter('.p')->first());
|
||||
}
|
||||
|
||||
public function testNew(): void
|
||||
{
|
||||
self::assertResponseStatusCodeSame(200);
|
||||
|
||||
$this->client->submitForm('Save', [
|
||||
'post[foundDate]' => '2024-01-01 00:00:00',
|
||||
'post[latitude]' => '45.0',
|
||||
'post[longitude]' => '45.0',
|
||||
'post[altitude]' => '500.0',
|
||||
'post[commentary]' => 'Testing',
|
||||
]);
|
||||
|
||||
self::assertResponseRedirects('/');
|
||||
|
||||
self::assertSame(1, $this->repository->count());
|
||||
}
|
||||
|
||||
public function testShow(): void
|
||||
{
|
||||
$fixture = new Post();
|
||||
$fixture->setFoundDate(new \DateTimeImmutable('2024-01-01 00:00:00'));
|
||||
$fixture->setCommentary('Cool stuff');
|
||||
|
||||
$this->manager->persist($fixture);
|
||||
$this->manager->flush();
|
||||
|
||||
$this->client->request('GET', sprintf('%s%s', $this->path, $fixture->getId()));
|
||||
|
||||
self::assertResponseStatusCodeSame(200);
|
||||
self::assertSelectorTextContains('h1', 'Post');
|
||||
}
|
||||
|
||||
public function testEdit(): void
|
||||
{
|
||||
$fixture = new Post();
|
||||
$fixture->setFoundDate(new \DateTimeImmutable('2024-01-01 00:00:00'));
|
||||
$fixture->setCommentary('Cool stuff');
|
||||
|
||||
$this->manager->persist($fixture);
|
||||
$this->manager->flush();
|
||||
|
||||
$this->client->request('GET', sprintf('%s%s/edit', $this->path, $fixture->getId()));
|
||||
|
||||
$this->client->submitForm('Update', [
|
||||
'post[foundDate]' => '2024-03-25 00:00:00',
|
||||
'post[latitude]' => '90',
|
||||
'post[longitude]' => '90',
|
||||
'post[altitude]' => '200',
|
||||
'post[commentary]' => 'Something New',
|
||||
]);
|
||||
|
||||
self::assertResponseRedirects('/');
|
||||
|
||||
$fixture = $this->repository->findAll();
|
||||
|
||||
self::assertEquals(new \DateTimeImmutable('2024-03-25 00:00:00'), $fixture[0]->getFoundDate());
|
||||
self::assertSame(90., $fixture[0]->getLatitude());
|
||||
self::assertSame(90., $fixture[0]->getLongitude());
|
||||
self::assertSame(200., $fixture[0]->getAltitude());
|
||||
self::assertSame('Something New', $fixture[0]->getCommentary());
|
||||
}
|
||||
|
||||
public function testRemove(): void
|
||||
{
|
||||
$fixture = new Post();
|
||||
$fixture->setFoundDate(new \DateTimeImmutable('2024-01-01 00:00:00'));
|
||||
$fixture->setCommentary('Cool stuff');
|
||||
|
||||
$this->manager->persist($fixture);
|
||||
$this->manager->flush();
|
||||
|
||||
$this->client->request('GET', sprintf('%s%s', $this->path, $fixture->getId()));
|
||||
$this->client->submitForm('Delete');
|
||||
|
||||
self::assertResponseRedirects('/');
|
||||
self::assertSame(0, $this->repository->count());
|
||||
}
|
||||
}
|
@ -0,0 +1,129 @@
|
||||
<?php
|
||||
|
||||
namespace App\Test\Controller;
|
||||
|
||||
use App\Entity\Species;
|
||||
use App\Entity\User;
|
||||
use App\Repository\SpeciesRepository;
|
||||
use Doctrine\ORM\EntityManagerInterface;
|
||||
use Symfony\Bundle\FrameworkBundle\KernelBrowser;
|
||||
use Symfony\Bundle\FrameworkBundle\Test\WebTestCase;
|
||||
|
||||
class SpeciesControllerTest extends WebTestCase
|
||||
{
|
||||
private KernelBrowser $client;
|
||||
private EntityManagerInterface $manager;
|
||||
private SpeciesRepository $repository;
|
||||
private string $path = '/species/';
|
||||
|
||||
protected function setUp(): void
|
||||
{
|
||||
$this->client = static::createClient();
|
||||
/** @var EntityManagerInterface $manager */
|
||||
$manager = static::getContainer()->get(EntityManagerInterface::class);
|
||||
$this->manager = $manager;
|
||||
$this->repository = $this->manager->getRepository(Species::class);
|
||||
|
||||
foreach ($this->repository->findAll() as $object) {
|
||||
$this->manager->remove($object);
|
||||
}
|
||||
|
||||
$this->manager->flush();
|
||||
|
||||
$userRepository = $this->manager->getRepository(User::class);
|
||||
/** @var User $user */
|
||||
$user = $userRepository->findOneByEmail('test@test.fr');
|
||||
$this->client->loginUser($user);
|
||||
$this->client->request('GET', sprintf('%snew', $this->path));
|
||||
}
|
||||
|
||||
public function testIndex(): void
|
||||
{
|
||||
$crawler = $this->client->request('GET', $this->path);
|
||||
|
||||
self::assertResponseStatusCodeSame(200);
|
||||
self::assertPageTitleContains('Species index');
|
||||
|
||||
// Use the $crawler to perform additional assertions e.g.
|
||||
// self::assertSame('Some text on the page', $crawler->filter('.p')->first());
|
||||
}
|
||||
|
||||
public function testNew(): void
|
||||
{
|
||||
$this->client->request('GET', sprintf('%snew', $this->path));
|
||||
|
||||
self::assertResponseStatusCodeSame(200);
|
||||
|
||||
$this->client->submitForm('Save', [
|
||||
'species[scientific_name]' => 'Testing',
|
||||
'species[vernacular_name]' => 'Testing',
|
||||
'species[region]' => 'Testing',
|
||||
]);
|
||||
|
||||
self::assertResponseRedirects($this->path);
|
||||
|
||||
self::assertSame(1, $this->repository->count([]));
|
||||
}
|
||||
|
||||
public function testShow(): void
|
||||
{
|
||||
$fixture = new Species();
|
||||
$fixture->setScientificName('My Title');
|
||||
$fixture->setVernacularName('My Title');
|
||||
$fixture->setRegion('My Title');
|
||||
|
||||
$this->manager->persist($fixture);
|
||||
$this->manager->flush();
|
||||
|
||||
$this->client->request('GET', sprintf('%s%s', $this->path, $fixture->getId()));
|
||||
|
||||
self::assertResponseStatusCodeSame(200);
|
||||
self::assertPageTitleContains('Species');
|
||||
|
||||
// Use assertions to check that the properties are properly displayed.
|
||||
}
|
||||
|
||||
public function testEdit(): void
|
||||
{
|
||||
$fixture = new Species();
|
||||
$fixture->setScientificName('Value');
|
||||
$fixture->setVernacularName('Value');
|
||||
$fixture->setRegion('Value');
|
||||
|
||||
$this->manager->persist($fixture);
|
||||
$this->manager->flush();
|
||||
|
||||
$this->client->request('GET', sprintf('%s%s/edit', $this->path, $fixture->getId()));
|
||||
|
||||
$this->client->submitForm('Update', [
|
||||
'species[scientific_name]' => 'Something New',
|
||||
'species[vernacular_name]' => 'Something New',
|
||||
'species[region]' => 'Something New',
|
||||
]);
|
||||
|
||||
self::assertResponseRedirects('/species/');
|
||||
|
||||
$fixture = $this->repository->findAll();
|
||||
|
||||
self::assertSame('Something New', $fixture[0]->getScientificName());
|
||||
self::assertSame('Something New', $fixture[0]->getVernacularName());
|
||||
self::assertSame('Something New', $fixture[0]->getRegion());
|
||||
}
|
||||
|
||||
public function testRemove(): void
|
||||
{
|
||||
$fixture = new Species();
|
||||
$fixture->setScientificName('Value');
|
||||
$fixture->setVernacularName('Value');
|
||||
$fixture->setRegion('Value');
|
||||
|
||||
$this->manager->persist($fixture);
|
||||
$this->manager->flush();
|
||||
|
||||
$this->client->request('GET', sprintf('%s%s', $this->path, $fixture->getId()));
|
||||
$this->client->submitForm('Delete');
|
||||
|
||||
self::assertResponseRedirects('/species/');
|
||||
self::assertSame(0, $this->repository->count([]));
|
||||
}
|
||||
}
|
Loading…
Reference in new issue