mirror of
https://gitlab.com/Chill-Projet/chill-bundles.git
synced 2025-06-07 18:44:08 +00:00
accompanying period validation - on social issues suppression
This commit is contained in:
parent
6cc0f1f98b
commit
b683630136
@ -11,6 +11,8 @@ and this project adheres to
|
|||||||
## Unreleased
|
## Unreleased
|
||||||
|
|
||||||
<!-- write down unreleased development here -->
|
<!-- write down unreleased development here -->
|
||||||
|
* [person] add validator for accompanying period with a test on social issues (https://gitlab.com/champs-libres/departement-de-la-vendee/accent-suivi-developpement/-/issues/76)
|
||||||
|
* [activity] fix visibility for location
|
||||||
|
|
||||||
## Test releases
|
## Test releases
|
||||||
|
|
||||||
|
@ -163,16 +163,6 @@ class ActivityType
|
|||||||
*/
|
*/
|
||||||
private int $personVisible = self::FIELD_REQUIRED;
|
private int $personVisible = self::FIELD_REQUIRED;
|
||||||
|
|
||||||
/**
|
|
||||||
* @ORM\Column(type="string", nullable=false, options={"default": ""})
|
|
||||||
*/
|
|
||||||
private string $placeLabel = '';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ORM\Column(type="smallint", nullable=false, options={"default": 1})
|
|
||||||
*/
|
|
||||||
private int $placeVisible = self::FIELD_OPTIONAL;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @ORM\Column(type="string", nullable=false, options={"default": ""})
|
* @ORM\Column(type="string", nullable=false, options={"default": ""})
|
||||||
*/
|
*/
|
||||||
@ -406,16 +396,6 @@ class ActivityType
|
|||||||
return $this->personVisible;
|
return $this->personVisible;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getPlaceLabel(): string
|
|
||||||
{
|
|
||||||
return $this->placeLabel;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getPlaceVisible(): int
|
|
||||||
{
|
|
||||||
return $this->placeVisible;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getReasonsLabel(): string
|
public function getReasonsLabel(): string
|
||||||
{
|
{
|
||||||
return $this->reasonsLabel;
|
return $this->reasonsLabel;
|
||||||
@ -688,20 +668,6 @@ class ActivityType
|
|||||||
return $this;
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function setPlaceLabel(string $placeLabel): self
|
|
||||||
{
|
|
||||||
$this->placeLabel = $placeLabel;
|
|
||||||
|
|
||||||
return $this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function setPlaceVisible(int $placeVisible): self
|
|
||||||
{
|
|
||||||
$this->placeVisible = $placeVisible;
|
|
||||||
|
|
||||||
return $this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function setReasonsLabel(string $reasonsLabel): self
|
public function setReasonsLabel(string $reasonsLabel): self
|
||||||
{
|
{
|
||||||
$this->reasonsLabel = $reasonsLabel;
|
$this->reasonsLabel = $reasonsLabel;
|
||||||
|
@ -143,7 +143,7 @@ class ActivityType extends AbstractType
|
|||||||
|
|
||||||
return array_map(
|
return array_map(
|
||||||
fn (string $id): ?SocialIssue => $this->om->getRepository(SocialIssue::class)->findOneBy(['id' => (int) $id]),
|
fn (string $id): ?SocialIssue => $this->om->getRepository(SocialIssue::class)->findOneBy(['id' => (int) $id]),
|
||||||
explode(',', (string) $socialIssuesAsString)
|
explode(',', $socialIssuesAsString)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
));
|
));
|
||||||
@ -169,7 +169,7 @@ class ActivityType extends AbstractType
|
|||||||
|
|
||||||
return array_map(
|
return array_map(
|
||||||
fn (string $id): ?SocialAction => $this->om->getRepository(SocialAction::class)->findOneBy(['id' => (int) $id]),
|
fn (string $id): ?SocialAction => $this->om->getRepository(SocialAction::class)->findOneBy(['id' => (int) $id]),
|
||||||
explode(',', (string) $socialActionsAsString)
|
explode(',', $socialActionsAsString)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
));
|
));
|
||||||
@ -260,6 +260,10 @@ class ActivityType extends AbstractType
|
|||||||
return implode(',', $personIds);
|
return implode(',', $personIds);
|
||||||
},
|
},
|
||||||
function (?string $personsAsString): array {
|
function (?string $personsAsString): array {
|
||||||
|
if (null === $personsAsString) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
return array_map(
|
return array_map(
|
||||||
fn (string $id): ?Person => $this->om->getRepository(Person::class)->findOneBy(['id' => (int) $id]),
|
fn (string $id): ?Person => $this->om->getRepository(Person::class)->findOneBy(['id' => (int) $id]),
|
||||||
explode(',', (string) $personsAsString)
|
explode(',', (string) $personsAsString)
|
||||||
@ -282,6 +286,10 @@ class ActivityType extends AbstractType
|
|||||||
return implode(',', $thirdpartyIds);
|
return implode(',', $thirdpartyIds);
|
||||||
},
|
},
|
||||||
function (?string $thirdpartyAsString): array {
|
function (?string $thirdpartyAsString): array {
|
||||||
|
if (null === $thirdpartyAsString) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
return array_map(
|
return array_map(
|
||||||
fn (string $id): ?ThirdParty => $this->om->getRepository(ThirdParty::class)->findOneBy(['id' => (int) $id]),
|
fn (string $id): ?ThirdParty => $this->om->getRepository(ThirdParty::class)->findOneBy(['id' => (int) $id]),
|
||||||
explode(',', (string) $thirdpartyAsString)
|
explode(',', (string) $thirdpartyAsString)
|
||||||
@ -315,6 +323,10 @@ class ActivityType extends AbstractType
|
|||||||
return implode(',', $userIds);
|
return implode(',', $userIds);
|
||||||
},
|
},
|
||||||
function (?string $usersAsString): array {
|
function (?string $usersAsString): array {
|
||||||
|
if (null === $usersAsString) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
return array_map(
|
return array_map(
|
||||||
fn (string $id): ?User => $this->om->getRepository(User::class)->findOneBy(['id' => (int) $id]),
|
fn (string $id): ?User => $this->om->getRepository(User::class)->findOneBy(['id' => (int) $id]),
|
||||||
explode(',', (string) $usersAsString)
|
explode(',', (string) $usersAsString)
|
||||||
|
@ -55,7 +55,7 @@ class ActivityTypeType extends AbstractType
|
|||||||
]);
|
]);
|
||||||
|
|
||||||
$fields = [
|
$fields = [
|
||||||
'persons', 'user', 'date', 'place', 'persons',
|
'persons', 'user', 'date', 'location', 'persons',
|
||||||
'thirdParties', 'durationTime', 'travelTime', 'attendee',
|
'thirdParties', 'durationTime', 'travelTime', 'attendee',
|
||||||
'reasons', 'comment', 'sentReceived', 'documents',
|
'reasons', 'comment', 'sentReceived', 'documents',
|
||||||
'emergency', 'socialIssues', 'socialActions', 'users',
|
'emergency', 'socialIssues', 'socialActions', 'users',
|
||||||
|
@ -31,6 +31,7 @@ class ActivityRepository extends ServiceEntityRepository
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
* @deprecated use @link{ActivityACLAwareRepositoryInterface::findByAccompanyingPeriod}
|
||||||
* @return Activity[]
|
* @return Activity[]
|
||||||
*/
|
*/
|
||||||
public function findByAccompanyingPeriod(AccompanyingPeriod $period, array $scopes, ?bool $allowNullScope = false, ?int $limit = 100, ?int $offset = 0, array $orderBy = ['date' => 'desc']): array
|
public function findByAccompanyingPeriod(AccompanyingPeriod $period, array $scopes, ?bool $allowNullScope = false, ?int $limit = 100, ?int $offset = 0, array $orderBy = ['date' => 'desc']): array
|
||||||
|
@ -0,0 +1,34 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Chill is a software for social workers
|
||||||
|
*
|
||||||
|
* For the full copyright and license information, please view
|
||||||
|
* the LICENSE file that was distributed with this source code.
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Chill\Migrations\Activity;
|
||||||
|
|
||||||
|
use Doctrine\DBAL\Schema\Schema;
|
||||||
|
use Doctrine\Migrations\AbstractMigration;
|
||||||
|
|
||||||
|
final class Version20211207152023 extends AbstractMigration
|
||||||
|
{
|
||||||
|
public function down(Schema $schema): void
|
||||||
|
{
|
||||||
|
$this->throwIrreversibleMigrationException('placevisible and placelabel could not be created');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getDescription(): string
|
||||||
|
{
|
||||||
|
return 'DROP place visible and place label, which are replaced by location';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function up(Schema $schema): void
|
||||||
|
{
|
||||||
|
$this->addSql('ALTER TABLE activitytype DROP placevisible');
|
||||||
|
$this->addSql('ALTER TABLE activitytype DROP placelabel');
|
||||||
|
}
|
||||||
|
}
|
@ -146,8 +146,8 @@ User visible: Visibilité du champ Utilisateur
|
|||||||
User label: Libellé du champ Utilisateur
|
User label: Libellé du champ Utilisateur
|
||||||
Date visible: Visibilité du champ Date
|
Date visible: Visibilité du champ Date
|
||||||
Date label: Libellé du champ Date
|
Date label: Libellé du champ Date
|
||||||
Place visible: Visibilité du champ Lieu
|
Location visible: Visibilité du champ Lieu
|
||||||
Place label: Libellé du champ Lieu
|
Location label: Libellé du champ Lieu
|
||||||
Third parties visible: Visibilité du champ Tiers
|
Third parties visible: Visibilité du champ Tiers
|
||||||
Third parties label: Libellé du champ Tiers
|
Third parties label: Libellé du champ Tiers
|
||||||
Duration time visible: Visibilité du champ Durée
|
Duration time visible: Visibilité du champ Durée
|
||||||
|
@ -25,6 +25,7 @@ use Chill\PersonBundle\Entity\AccompanyingPeriod\Comment;
|
|||||||
use Chill\PersonBundle\Entity\AccompanyingPeriod\Origin;
|
use Chill\PersonBundle\Entity\AccompanyingPeriod\Origin;
|
||||||
use Chill\PersonBundle\Entity\AccompanyingPeriod\Resource;
|
use Chill\PersonBundle\Entity\AccompanyingPeriod\Resource;
|
||||||
use Chill\PersonBundle\Entity\SocialWork\SocialIssue;
|
use Chill\PersonBundle\Entity\SocialWork\SocialIssue;
|
||||||
|
use Chill\PersonBundle\Validator\Constraints\AccompanyingPeriod\AccompanyingPeriodValidity;
|
||||||
use Chill\PersonBundle\Validator\Constraints\AccompanyingPeriod\ParticipationOverlap;
|
use Chill\PersonBundle\Validator\Constraints\AccompanyingPeriod\ParticipationOverlap;
|
||||||
use Chill\PersonBundle\Validator\Constraints\AccompanyingPeriod\ResourceDuplicateCheck;
|
use Chill\PersonBundle\Validator\Constraints\AccompanyingPeriod\ResourceDuplicateCheck;
|
||||||
use Chill\ThirdPartyBundle\Entity\ThirdParty;
|
use Chill\ThirdPartyBundle\Entity\ThirdParty;
|
||||||
@ -58,6 +59,8 @@ use const SORT_REGULAR;
|
|||||||
* "this.isConfidential and this.getUser === NULL",
|
* "this.isConfidential and this.getUser === NULL",
|
||||||
* message="If the accompanying course is confirmed and confidential, a referrer must remain assigned."
|
* message="If the accompanying course is confirmed and confidential, a referrer must remain assigned."
|
||||||
* )
|
* )
|
||||||
|
*
|
||||||
|
* @AccompanyingPeriodValidity(groups={AccompanyingPeriod::STEP_DRAFT, AccompanyingPeriod::STEP_CONFIRMED})
|
||||||
*/
|
*/
|
||||||
class AccompanyingPeriod implements
|
class AccompanyingPeriod implements
|
||||||
GroupSequenceProviderInterface,
|
GroupSequenceProviderInterface,
|
||||||
|
@ -121,6 +121,27 @@ class SocialIssue
|
|||||||
return $ancestors;
|
return $ancestors;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* get all the ancestors of the social issue
|
||||||
|
*
|
||||||
|
* @param bool $includeThis if the array in the result must include the present SocialIssue
|
||||||
|
*/
|
||||||
|
public function getAncestors(bool $includeThis = true): array
|
||||||
|
{
|
||||||
|
$ancestors = [];
|
||||||
|
|
||||||
|
if ($includeThis) {
|
||||||
|
$ancestors[] = $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
$current = $this;
|
||||||
|
while ($current->hasParent()) {
|
||||||
|
$ancestors[] = $current = $current->getParent();
|
||||||
|
}
|
||||||
|
|
||||||
|
return $ancestors;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return Collection|self[]
|
* @return Collection|self[]
|
||||||
*/
|
*/
|
||||||
|
@ -61,4 +61,20 @@ final class SocialIssueTest extends TestCase
|
|||||||
|
|
||||||
$this->assertFalse($child->isDescendantOf($grandChild));
|
$this->assertFalse($child->isDescendantOf($grandChild));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testGetAncestors()
|
||||||
|
{
|
||||||
|
$parent = new SocialIssue();
|
||||||
|
$child = (new SocialIssue())->setParent($parent);
|
||||||
|
$grandChild = (new SocialIssue())->setParent($child);
|
||||||
|
$grandGrandChild = (new SocialIssue())->setParent($grandChild);
|
||||||
|
$unrelated = new SocialIssue();
|
||||||
|
|
||||||
|
$this->assertContains($parent, $grandGrandChild->getAncestors(true));
|
||||||
|
$this->assertContains($child, $grandGrandChild->getAncestors(true));
|
||||||
|
$this->assertContains($grandChild, $grandGrandChild->getAncestors(true));
|
||||||
|
$this->assertContains($grandGrandChild, $grandGrandChild->getAncestors(true));
|
||||||
|
$this->assertNotContains($grandGrandChild, $grandGrandChild->getAncestors(false));
|
||||||
|
$this->assertCount(0, $unrelated->getAncestors(false));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,27 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Chill is a software for social workers
|
||||||
|
*
|
||||||
|
* For the full copyright and license information, please view
|
||||||
|
* the LICENSE file that was distributed with this source code.
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Chill\PersonBundle\Validator\Constraints\AccompanyingPeriod;
|
||||||
|
|
||||||
|
use Symfony\Component\Validator\Constraint;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @Annotation
|
||||||
|
*/
|
||||||
|
class AccompanyingPeriodValidity extends Constraint
|
||||||
|
{
|
||||||
|
public $messageSocialIssueCannotBeDeleted = 'The social %name% issue cannot be deleted because it is associated with an activity or an action';
|
||||||
|
|
||||||
|
public function getTargets()
|
||||||
|
{
|
||||||
|
return self::CLASS_CONSTRAINT;
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,84 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Chill is a software for social workers
|
||||||
|
*
|
||||||
|
* For the full copyright and license information, please view
|
||||||
|
* the LICENSE file that was distributed with this source code.
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Chill\PersonBundle\Validator\Constraints\AccompanyingPeriod;
|
||||||
|
|
||||||
|
use Chill\ActivityBundle\Repository\ActivityACLAwareRepository;
|
||||||
|
use Chill\ActivityBundle\Repository\ActivityRepository;
|
||||||
|
use Chill\PersonBundle\Entity\AccompanyingPeriod;
|
||||||
|
use Chill\PersonBundle\Entity\SocialWork\SocialIssue;
|
||||||
|
use Chill\PersonBundle\Templating\Entity\SocialIssueRender;
|
||||||
|
use Symfony\Component\Validator\Constraint;
|
||||||
|
use Symfony\Component\Validator\ConstraintValidator;
|
||||||
|
use Symfony\Component\Validator\Exception\UnexpectedTypeException;
|
||||||
|
use Symfony\Component\Validator\Exception\UnexpectedValueException;
|
||||||
|
use function in_array;
|
||||||
|
|
||||||
|
class AccompanyingPeriodValidityValidator extends ConstraintValidator
|
||||||
|
{
|
||||||
|
private ActivityRepository $activityRepository;
|
||||||
|
|
||||||
|
private SocialIssueRender $socialIssueRender;
|
||||||
|
|
||||||
|
public function __construct(ActivityRepository $activityRepository, SocialIssueRender $socialIssueRender)
|
||||||
|
{
|
||||||
|
$this->activityRepository = $activityRepository;
|
||||||
|
$this->socialIssueRender = $socialIssueRender;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function validate($period, Constraint $constraint)
|
||||||
|
{
|
||||||
|
if (!$constraint instanceof AccompanyingPeriodValidity) {
|
||||||
|
throw new UnexpectedTypeException($constraint, AccompanyingPeriodValidity::class);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!$period instanceof AccompanyingPeriod) {
|
||||||
|
throw new UnexpectedValueException($period, AccompanyingPeriod::class);
|
||||||
|
}
|
||||||
|
|
||||||
|
$socialIssues = [];
|
||||||
|
|
||||||
|
$activities = $this->activityRepository->findBy(['accompanyingPeriod' => $period]);
|
||||||
|
|
||||||
|
foreach ($activities as $activity) {
|
||||||
|
$socialIssues = $activity->getSocialIssues()->toArray();
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($period->getWorks() as $work) {
|
||||||
|
$socialIssues[] = $work->getSocialAction()->getIssue();
|
||||||
|
}
|
||||||
|
|
||||||
|
$socialIssuesByKey = [];
|
||||||
|
foreach ($socialIssues as $si) {
|
||||||
|
$socialIssuesByKey[$si->getId()] = $si;
|
||||||
|
}
|
||||||
|
|
||||||
|
$periodIssuesWithAncestors = [];
|
||||||
|
foreach ($period->getSocialIssues() as $si) {
|
||||||
|
/** @var SocialIssue $si */
|
||||||
|
$periodIssuesWithAncestors = array_merge($periodIssuesWithAncestors, \array_map(
|
||||||
|
function (SocialIssue $si) { return $si->getId(); },
|
||||||
|
$si->getAncestors(true))
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($socialIssuesByKey as $key => $si) {
|
||||||
|
if (!in_array($key, $periodIssuesWithAncestors)) {
|
||||||
|
$this->context
|
||||||
|
->buildViolation(
|
||||||
|
$constraint->messageSocialIssueCannotBeDeleted
|
||||||
|
)
|
||||||
|
->setParameter('%name%', $this->socialIssueRender->renderString($si, []))
|
||||||
|
->addViolation();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -46,3 +46,4 @@ household_membership:
|
|||||||
'{{ name }} is already associated to this accompanying course.': '{{ name }} est déjà associé à ce parcours.'
|
'{{ name }} is already associated to this accompanying course.': '{{ name }} est déjà associé à ce parcours.'
|
||||||
A course must contains at least one social issue: 'Un parcours doit être associé à au moins une problématique sociale'
|
A course must contains at least one social issue: 'Un parcours doit être associé à au moins une problématique sociale'
|
||||||
A course must be associated to at least one scope: 'Un parcours doit être associé à au moins un service'
|
A course must be associated to at least one scope: 'Un parcours doit être associé à au moins un service'
|
||||||
|
The social %name% issue cannot be deleted because it is associated with an activity or an action: 'La problématique sociale "%name%" ne peut pas être supprimée car elle est associée à une activité ou une action'
|
||||||
|
Loading…
x
Reference in New Issue
Block a user