mirror of
https://gitlab.com/Chill-Projet/chill-bundles.git
synced 2025-09-10 00:34:58 +00:00
Compare commits
46 Commits
Author | SHA1 | Date | |
---|---|---|---|
801f799e45
|
|||
98b8f3dcff
|
|||
a333a0312a
|
|||
0288fd22cf
|
|||
1dd4398c43
|
|||
6065680e1e
|
|||
88114e3ba6
|
|||
bf93c1ddb2 | |||
0d365e16e5
|
|||
802ff20b5c | |||
cdfe201574 | |||
43419f9f15
|
|||
39896ea6e2
|
|||
ca62c3fd0b
|
|||
b3b84c5dc0
|
|||
6bdb3e9695
|
|||
20e64e8768 | |||
4f4b3dbb44
|
|||
1c3e6e0dba
|
|||
e7ca81e057
|
|||
63f9bd5548
|
|||
c8146ded17
|
|||
17d2b795b4 | |||
7f30742fc3
|
|||
56d9072abe
|
|||
7ccff61c25
|
|||
c04fd66163
|
|||
145c1df313 | |||
7f9738975c | |||
3e63b4abf3 | |||
1485d1ce7a | |||
9687debb57
|
|||
769504c497
|
|||
811364e139
|
|||
0e5f1b4ab9
|
|||
f7c11d3567
|
|||
51544cfc48
|
|||
659dff3d2c
|
|||
deffc5e4db
|
|||
40ecaab5b4
|
|||
f7be53f790
|
|||
6fb01b19ec
|
|||
b8ecff4f08
|
|||
4c340dd086
|
|||
8f1955c536
|
|||
c9c15cdd56
|
5
.changes/unreleased/Feature-20230707-123609.yaml
Normal file
5
.changes/unreleased/Feature-20230707-123609.yaml
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
kind: Feature
|
||||||
|
body: '[export] Add a list for people with their associated course'
|
||||||
|
time: 2023-07-07T12:36:09.596469063+02:00
|
||||||
|
custom:
|
||||||
|
Issue: "125"
|
6
.changes/unreleased/Feature-20230707-124132.yaml
Normal file
6
.changes/unreleased/Feature-20230707-124132.yaml
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
kind: Feature
|
||||||
|
body: '[export] Add ordering by person''s lastname or course opening date in list
|
||||||
|
which concerns accompanying course or peoples'
|
||||||
|
time: 2023-07-07T12:41:32.112725962+02:00
|
||||||
|
custom:
|
||||||
|
Issue: ""
|
5
.changes/unreleased/Feature-20230711-150055.yaml
Normal file
5
.changes/unreleased/Feature-20230711-150055.yaml
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
kind: Feature
|
||||||
|
body: '[Export] allow to group activities by localisation'
|
||||||
|
time: 2023-07-11T15:00:55.770070399+02:00
|
||||||
|
custom:
|
||||||
|
Issue: "128"
|
@@ -30,6 +30,8 @@ kinds:
|
|||||||
auto: patch
|
auto: patch
|
||||||
- label: DX
|
- label: DX
|
||||||
auto: patch
|
auto: patch
|
||||||
|
- label: UX
|
||||||
|
auto: patch
|
||||||
newlines:
|
newlines:
|
||||||
afterChangelogHeader: 1
|
afterChangelogHeader: 1
|
||||||
beforeChangelogVersion: 1
|
beforeChangelogVersion: 1
|
||||||
|
@@ -18,11 +18,17 @@ use Chill\ActivityBundle\Repository\ActivityACLAwareRepositoryInterface;
|
|||||||
use Chill\ActivityBundle\Repository\ActivityRepository;
|
use Chill\ActivityBundle\Repository\ActivityRepository;
|
||||||
use Chill\ActivityBundle\Repository\ActivityTypeCategoryRepository;
|
use Chill\ActivityBundle\Repository\ActivityTypeCategoryRepository;
|
||||||
use Chill\ActivityBundle\Repository\ActivityTypeRepositoryInterface;
|
use Chill\ActivityBundle\Repository\ActivityTypeRepositoryInterface;
|
||||||
|
use Chill\ActivityBundle\Repository\ActivityUserJobRepository;
|
||||||
use Chill\ActivityBundle\Security\Authorization\ActivityVoter;
|
use Chill\ActivityBundle\Security\Authorization\ActivityVoter;
|
||||||
use Chill\MainBundle\Entity\Embeddable\CommentEmbeddable;
|
use Chill\MainBundle\Entity\Embeddable\CommentEmbeddable;
|
||||||
|
use Chill\MainBundle\Entity\UserJob;
|
||||||
|
use Chill\MainBundle\Pagination\PaginatorFactory;
|
||||||
use Chill\MainBundle\Repository\LocationRepository;
|
use Chill\MainBundle\Repository\LocationRepository;
|
||||||
use Chill\MainBundle\Repository\UserRepositoryInterface;
|
use Chill\MainBundle\Repository\UserRepositoryInterface;
|
||||||
use Chill\MainBundle\Security\Resolver\CenterResolverManagerInterface;
|
use Chill\MainBundle\Security\Resolver\CenterResolverManagerInterface;
|
||||||
|
use Chill\MainBundle\Templating\Listing\FilterOrderHelper;
|
||||||
|
use Chill\MainBundle\Templating\Listing\FilterOrderHelperFactoryInterface;
|
||||||
|
use Chill\MainBundle\Templating\TranslatableStringHelperInterface;
|
||||||
use Chill\PersonBundle\Entity\AccompanyingPeriod;
|
use Chill\PersonBundle\Entity\AccompanyingPeriod;
|
||||||
use Chill\PersonBundle\Entity\Person;
|
use Chill\PersonBundle\Entity\Person;
|
||||||
use Chill\PersonBundle\Privacy\PrivacyEvent;
|
use Chill\PersonBundle\Privacy\PrivacyEvent;
|
||||||
@@ -47,68 +53,26 @@ use function array_key_exists;
|
|||||||
|
|
||||||
final class ActivityController extends AbstractController
|
final class ActivityController extends AbstractController
|
||||||
{
|
{
|
||||||
private AccompanyingPeriodRepository $accompanyingPeriodRepository;
|
|
||||||
|
|
||||||
private ActivityACLAwareRepositoryInterface $activityACLAwareRepository;
|
|
||||||
|
|
||||||
private ActivityRepository $activityRepository;
|
|
||||||
|
|
||||||
private ActivityTypeCategoryRepository $activityTypeCategoryRepository;
|
|
||||||
|
|
||||||
private ActivityTypeRepositoryInterface $activityTypeRepository;
|
|
||||||
|
|
||||||
private CenterResolverManagerInterface $centerResolver;
|
|
||||||
|
|
||||||
private EntityManagerInterface $entityManager;
|
|
||||||
|
|
||||||
private EventDispatcherInterface $eventDispatcher;
|
|
||||||
|
|
||||||
private LocationRepository $locationRepository;
|
|
||||||
|
|
||||||
private LoggerInterface $logger;
|
|
||||||
|
|
||||||
private PersonRepository $personRepository;
|
|
||||||
|
|
||||||
private SerializerInterface $serializer;
|
|
||||||
|
|
||||||
private ThirdPartyRepository $thirdPartyRepository;
|
|
||||||
|
|
||||||
private TranslatorInterface $translator;
|
|
||||||
|
|
||||||
private UserRepositoryInterface $userRepository;
|
|
||||||
|
|
||||||
public function __construct(
|
public function __construct(
|
||||||
ActivityACLAwareRepositoryInterface $activityACLAwareRepository,
|
private readonly ActivityACLAwareRepositoryInterface $activityACLAwareRepository,
|
||||||
ActivityTypeRepositoryInterface $activityTypeRepository,
|
private readonly ActivityTypeRepositoryInterface $activityTypeRepository,
|
||||||
ActivityTypeCategoryRepository $activityTypeCategoryRepository,
|
private readonly ActivityTypeCategoryRepository $activityTypeCategoryRepository,
|
||||||
PersonRepository $personRepository,
|
private readonly PersonRepository $personRepository,
|
||||||
ThirdPartyRepository $thirdPartyRepository,
|
private readonly ThirdPartyRepository $thirdPartyRepository,
|
||||||
LocationRepository $locationRepository,
|
private readonly LocationRepository $locationRepository,
|
||||||
ActivityRepository $activityRepository,
|
private readonly ActivityRepository $activityRepository,
|
||||||
AccompanyingPeriodRepository $accompanyingPeriodRepository,
|
private readonly AccompanyingPeriodRepository $accompanyingPeriodRepository,
|
||||||
EntityManagerInterface $entityManager,
|
private readonly EntityManagerInterface $entityManager,
|
||||||
EventDispatcherInterface $eventDispatcher,
|
private readonly EventDispatcherInterface $eventDispatcher,
|
||||||
LoggerInterface $logger,
|
private readonly LoggerInterface $logger,
|
||||||
SerializerInterface $serializer,
|
private readonly SerializerInterface $serializer,
|
||||||
UserRepositoryInterface $userRepository,
|
private readonly UserRepositoryInterface $userRepository,
|
||||||
CenterResolverManagerInterface $centerResolver,
|
private readonly CenterResolverManagerInterface $centerResolver,
|
||||||
TranslatorInterface $translator
|
private readonly TranslatorInterface $translator,
|
||||||
|
private readonly FilterOrderHelperFactoryInterface $filterOrderHelperFactory,
|
||||||
|
private readonly TranslatableStringHelperInterface $translatableStringHelper,
|
||||||
|
private readonly PaginatorFactory $paginatorFactory,
|
||||||
) {
|
) {
|
||||||
$this->activityACLAwareRepository = $activityACLAwareRepository;
|
|
||||||
$this->activityTypeRepository = $activityTypeRepository;
|
|
||||||
$this->activityTypeCategoryRepository = $activityTypeCategoryRepository;
|
|
||||||
$this->personRepository = $personRepository;
|
|
||||||
$this->thirdPartyRepository = $thirdPartyRepository;
|
|
||||||
$this->locationRepository = $locationRepository;
|
|
||||||
$this->activityRepository = $activityRepository;
|
|
||||||
$this->accompanyingPeriodRepository = $accompanyingPeriodRepository;
|
|
||||||
$this->entityManager = $entityManager;
|
|
||||||
$this->eventDispatcher = $eventDispatcher;
|
|
||||||
$this->logger = $logger;
|
|
||||||
$this->serializer = $serializer;
|
|
||||||
$this->userRepository = $userRepository;
|
|
||||||
$this->centerResolver = $centerResolver;
|
|
||||||
$this->translator = $translator;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -289,14 +253,31 @@ final class ActivityController extends AbstractController
|
|||||||
{
|
{
|
||||||
$view = null;
|
$view = null;
|
||||||
$activities = [];
|
$activities = [];
|
||||||
// TODO: add pagination
|
|
||||||
|
|
||||||
[$person, $accompanyingPeriod] = $this->getEntity($request);
|
[$person, $accompanyingPeriod] = $this->getEntity($request);
|
||||||
|
$filter = $this->buildFilterOrder($person ?? $accompanyingPeriod);
|
||||||
|
|
||||||
|
$filterArgs = [
|
||||||
|
'my_activities' => $filter->getSingleCheckboxData('my_activities'),
|
||||||
|
'types' => $filter->hasEntityChoice('activity_types') ? $filter->getEntityChoiceData('activity_types') : [],
|
||||||
|
'jobs' => $filter->hasEntityChoice('jobs') ? $filter->getEntityChoiceData('jobs') : [],
|
||||||
|
'before' => $filter->getDateRangeData('activity_date')['to'],
|
||||||
|
'after' => $filter->getDateRangeData('activity_date')['from'],
|
||||||
|
];
|
||||||
|
|
||||||
if ($person instanceof Person) {
|
if ($person instanceof Person) {
|
||||||
$this->denyAccessUnlessGranted(ActivityVoter::SEE, $person);
|
$this->denyAccessUnlessGranted(ActivityVoter::SEE, $person);
|
||||||
|
$count = $this->activityACLAwareRepository->countByPerson($person, ActivityVoter::SEE, $filterArgs);
|
||||||
|
$paginator = $this->paginatorFactory->create($count);
|
||||||
$activities = $this->activityACLAwareRepository
|
$activities = $this->activityACLAwareRepository
|
||||||
->findByPerson($person, ActivityVoter::SEE, 0, null, ['date' => 'DESC', 'id' => 'DESC']);
|
->findByPerson(
|
||||||
|
$person,
|
||||||
|
ActivityVoter::SEE,
|
||||||
|
$paginator->getCurrentPageFirstItemNumber(),
|
||||||
|
$paginator->getItemsPerPage(),
|
||||||
|
['date' => 'DESC', 'id' => 'DESC'],
|
||||||
|
$filterArgs
|
||||||
|
);
|
||||||
|
|
||||||
$event = new PrivacyEvent($person, [
|
$event = new PrivacyEvent($person, [
|
||||||
'element_class' => Activity::class,
|
'element_class' => Activity::class,
|
||||||
@@ -308,10 +289,21 @@ final class ActivityController extends AbstractController
|
|||||||
} elseif ($accompanyingPeriod instanceof AccompanyingPeriod) {
|
} elseif ($accompanyingPeriod instanceof AccompanyingPeriod) {
|
||||||
$this->denyAccessUnlessGranted(ActivityVoter::SEE, $accompanyingPeriod);
|
$this->denyAccessUnlessGranted(ActivityVoter::SEE, $accompanyingPeriod);
|
||||||
|
|
||||||
|
$count = $this->activityACLAwareRepository->countByAccompanyingPeriod($accompanyingPeriod, ActivityVoter::SEE, $filterArgs);
|
||||||
|
$paginator = $this->paginatorFactory->create($count);
|
||||||
$activities = $this->activityACLAwareRepository
|
$activities = $this->activityACLAwareRepository
|
||||||
->findByAccompanyingPeriod($accompanyingPeriod, ActivityVoter::SEE, 0, null, ['date' => 'DESC', 'id' => 'DESC']);
|
->findByAccompanyingPeriod(
|
||||||
|
$accompanyingPeriod,
|
||||||
|
ActivityVoter::SEE,
|
||||||
|
$paginator->getCurrentPageFirstItemNumber(),
|
||||||
|
$paginator->getItemsPerPage(),
|
||||||
|
['date' => 'DESC', 'id' => 'DESC'],
|
||||||
|
$filterArgs
|
||||||
|
);
|
||||||
|
|
||||||
$view = 'ChillActivityBundle:Activity:listAccompanyingCourse.html.twig';
|
$view = 'ChillActivityBundle:Activity:listAccompanyingCourse.html.twig';
|
||||||
|
} else {
|
||||||
|
throw new \LogicException("Unsupported");
|
||||||
}
|
}
|
||||||
|
|
||||||
return $this->render(
|
return $this->render(
|
||||||
@@ -320,10 +312,47 @@ final class ActivityController extends AbstractController
|
|||||||
'activities' => $activities,
|
'activities' => $activities,
|
||||||
'person' => $person,
|
'person' => $person,
|
||||||
'accompanyingCourse' => $accompanyingPeriod,
|
'accompanyingCourse' => $accompanyingPeriod,
|
||||||
|
'filter' => $filter,
|
||||||
|
'paginator' => $paginator,
|
||||||
]
|
]
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function buildFilterOrder(AccompanyingPeriod|Person $associated): FilterOrderHelper
|
||||||
|
{
|
||||||
|
|
||||||
|
$filterBuilder = $this->filterOrderHelperFactory->create(self::class);
|
||||||
|
$types = $this->activityACLAwareRepository->findActivityTypeByAssociated($associated);
|
||||||
|
$jobs = $this->activityACLAwareRepository->findUserJobByAssociated($associated);
|
||||||
|
|
||||||
|
$filterBuilder
|
||||||
|
->addDateRange('activity_date', 'activity.date')
|
||||||
|
->addSingleCheckbox('my_activities', 'activity_filter.My activities');
|
||||||
|
|
||||||
|
if (1 < count($types)) {
|
||||||
|
$filterBuilder
|
||||||
|
->addEntityChoice('activity_types', 'activity_filter.Types', \Chill\ActivityBundle\Entity\ActivityType::class, $types, [
|
||||||
|
'choice_label' => function (\Chill\ActivityBundle\Entity\ActivityType $activityType) {
|
||||||
|
$text = match ($activityType->hasCategory()) {
|
||||||
|
true => $this->translatableStringHelper->localize($activityType->getCategory()->getName()) . ' > ',
|
||||||
|
false => '',
|
||||||
|
};
|
||||||
|
|
||||||
|
return $text . $this->translatableStringHelper->localize($activityType->getName());
|
||||||
|
}
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (1 < count($jobs)) {
|
||||||
|
$filterBuilder
|
||||||
|
->addEntityChoice('jobs', 'activity_filter.Jobs', UserJob::class, $jobs, [
|
||||||
|
'choice_label' => fn (UserJob $u) => $this->translatableStringHelper->localize($u->getLabel())
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $filterBuilder->build();
|
||||||
|
}
|
||||||
|
|
||||||
public function newAction(Request $request): Response
|
public function newAction(Request $request): Response
|
||||||
{
|
{
|
||||||
$view = null;
|
$view = null;
|
||||||
|
@@ -0,0 +1,80 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace Chill\ActivityBundle\Export\Aggregator;
|
||||||
|
|
||||||
|
use Chill\ActivityBundle\Export\Declarations;
|
||||||
|
use Chill\ActivityBundle\Repository\ActivityTypeRepositoryInterface;
|
||||||
|
use Chill\MainBundle\Export\AggregatorInterface;
|
||||||
|
use Chill\MainBundle\Repository\LocationRepository;
|
||||||
|
use Chill\MainBundle\Templating\TranslatableStringHelperInterface;
|
||||||
|
use Closure;
|
||||||
|
use Doctrine\ORM\QueryBuilder;
|
||||||
|
use Symfony\Component\Form\FormBuilderInterface;
|
||||||
|
use function in_array;
|
||||||
|
|
||||||
|
final readonly class ActivityLocationAggregator implements AggregatorInterface
|
||||||
|
{
|
||||||
|
public const KEY = 'activity_location_aggregator';
|
||||||
|
|
||||||
|
public function addRole(): ?string
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function alterQuery(QueryBuilder $qb, $data)
|
||||||
|
{
|
||||||
|
if (!in_array('actloc', $qb->getAllAliases(), true)) {
|
||||||
|
$qb->leftJoin('activity.location', 'actloc');
|
||||||
|
}
|
||||||
|
$qb->addSelect(sprintf('actloc.name AS %s', self::KEY));
|
||||||
|
$qb->addGroupBy(self::KEY);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function applyOn(): string
|
||||||
|
{
|
||||||
|
return Declarations::ACTIVITY;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function buildForm(FormBuilderInterface $builder)
|
||||||
|
{
|
||||||
|
// no form required for this aggregator
|
||||||
|
}
|
||||||
|
public function getFormDefaultData(): array
|
||||||
|
{
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getLabels($key, array $values, $data): Closure
|
||||||
|
{
|
||||||
|
return function ($value): string {
|
||||||
|
if ('_header' === $value) {
|
||||||
|
return 'export.aggregator.activity.by_location.Activity Location';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (null === $value || '' === $value) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
return $value;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getQueryKeys($data): array
|
||||||
|
{
|
||||||
|
return [self::KEY];
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getTitle()
|
||||||
|
{
|
||||||
|
return 'export.aggregator.activity.by_location.Title';
|
||||||
|
}
|
||||||
|
}
|
@@ -18,67 +18,193 @@ use Chill\ActivityBundle\Security\Authorization\ActivityVoter;
|
|||||||
use Chill\MainBundle\Entity\Location;
|
use Chill\MainBundle\Entity\Location;
|
||||||
use Chill\MainBundle\Entity\LocationType;
|
use Chill\MainBundle\Entity\LocationType;
|
||||||
use Chill\MainBundle\Entity\Scope;
|
use Chill\MainBundle\Entity\Scope;
|
||||||
use Chill\MainBundle\Security\Authorization\AuthorizationHelper;
|
use Chill\MainBundle\Entity\User;
|
||||||
use Chill\MainBundle\Security\Resolver\CenterResolverDispatcherInterface;
|
use Chill\MainBundle\Entity\UserJob;
|
||||||
|
use Chill\MainBundle\Security\Authorization\AuthorizationHelperForCurrentUserInterface;
|
||||||
|
use Chill\MainBundle\Security\Resolver\CenterResolverManagerInterface;
|
||||||
use Chill\PersonBundle\Entity\AccompanyingPeriod;
|
use Chill\PersonBundle\Entity\AccompanyingPeriod;
|
||||||
use Chill\PersonBundle\Entity\Person;
|
use Chill\PersonBundle\Entity\Person;
|
||||||
use Doctrine\DBAL\Types\Types;
|
use Doctrine\DBAL\Types\Types;
|
||||||
use Doctrine\ORM\AbstractQuery;
|
use Doctrine\ORM\AbstractQuery;
|
||||||
use Doctrine\ORM\EntityManagerInterface;
|
use Doctrine\ORM\EntityManagerInterface;
|
||||||
|
use Doctrine\ORM\NonUniqueResultException;
|
||||||
|
use Doctrine\ORM\NoResultException;
|
||||||
|
use Doctrine\ORM\Query\Expr\Join;
|
||||||
use Doctrine\ORM\Query\ResultSetMappingBuilder;
|
use Doctrine\ORM\Query\ResultSetMappingBuilder;
|
||||||
use Symfony\Component\Security\Core\Authentication\Token\Storage\TokenStorageInterface;
|
use Doctrine\ORM\QueryBuilder;
|
||||||
use Symfony\Component\Security\Core\Role\Role;
|
use Symfony\Component\HttpFoundation\RequestStack;
|
||||||
use Symfony\Component\Security\Core\Security;
|
use Symfony\Component\Security\Core\Security;
|
||||||
|
|
||||||
use function count;
|
use function count;
|
||||||
use function in_array;
|
use function in_array;
|
||||||
|
|
||||||
final class ActivityACLAwareRepository implements ActivityACLAwareRepositoryInterface
|
final readonly class ActivityACLAwareRepository implements ActivityACLAwareRepositoryInterface
|
||||||
{
|
{
|
||||||
private AuthorizationHelper $authorizationHelper;
|
|
||||||
|
|
||||||
private CenterResolverDispatcherInterface $centerResolverDispatcher;
|
|
||||||
|
|
||||||
private EntityManagerInterface $em;
|
|
||||||
|
|
||||||
private ActivityRepository $repository;
|
|
||||||
|
|
||||||
private Security $security;
|
|
||||||
|
|
||||||
private TokenStorageInterface $tokenStorage;
|
|
||||||
|
|
||||||
public function __construct(
|
public function __construct(
|
||||||
AuthorizationHelper $authorizationHelper,
|
private AuthorizationHelperForCurrentUserInterface $authorizationHelper,
|
||||||
CenterResolverDispatcherInterface $centerResolverDispatcher,
|
private CenterResolverManagerInterface $centerResolverManager,
|
||||||
TokenStorageInterface $tokenStorage,
|
private ActivityRepository $repository,
|
||||||
ActivityRepository $repository,
|
private EntityManagerInterface $em,
|
||||||
EntityManagerInterface $em,
|
private Security $security,
|
||||||
Security $security
|
private RequestStack $requestStack,
|
||||||
) {
|
) {
|
||||||
$this->authorizationHelper = $authorizationHelper;
|
|
||||||
$this->centerResolverDispatcher = $centerResolverDispatcher;
|
|
||||||
$this->tokenStorage = $tokenStorage;
|
|
||||||
$this->repository = $repository;
|
|
||||||
$this->em = $em;
|
|
||||||
$this->security = $security;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function findByAccompanyingPeriod(AccompanyingPeriod $period, string $role, ?int $start = 0, ?int $limit = 1000, ?array $orderBy = []): array
|
/**
|
||||||
|
* @throws NonUniqueResultException
|
||||||
|
* @throws NoResultException
|
||||||
|
*/
|
||||||
|
public function countByAccompanyingPeriod(AccompanyingPeriod $period, string $role, array $filters = []): int
|
||||||
{
|
{
|
||||||
$user = $this->security->getUser();
|
$qb = $this->buildBaseQuery($filters);
|
||||||
$center = $this->centerResolverDispatcher->resolveCenter($period);
|
|
||||||
|
|
||||||
if (0 === count($orderBy)) {
|
$qb
|
||||||
$orderBy = ['date' => 'DESC'];
|
->select('COUNT(a)')
|
||||||
|
->andWhere('a.accompanyingPeriod = :period')->setParameter('period', $period);
|
||||||
|
|
||||||
|
return $qb->getQuery()->getSingleScalarResult();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function countByPerson(Person $person, string $role, array $filters = []): int
|
||||||
|
{
|
||||||
|
$qb = $this->buildBaseQuery($filters);
|
||||||
|
|
||||||
|
$qb = $this->filterBaseQueryByPerson($qb, $person, $role);
|
||||||
|
|
||||||
|
$qb->select('COUNT(a)');
|
||||||
|
|
||||||
|
return $qb->getQuery()->getSingleScalarResult();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public function findByAccompanyingPeriod(AccompanyingPeriod $period, string $role, ?int $start = 0, ?int $limit = 1000, array $orderBy = ['date' => 'DESC'], array $filters = []): array
|
||||||
|
{
|
||||||
|
$qb = $this->buildBaseQuery($filters);
|
||||||
|
|
||||||
|
$qb->andWhere('a.accompanyingPeriod = :period')->setParameter('period', $period);
|
||||||
|
|
||||||
|
foreach ($orderBy as $field => $order) {
|
||||||
|
$qb->addOrderBy('a.' . $field, $order);
|
||||||
}
|
}
|
||||||
|
|
||||||
$scopes = $this->authorizationHelper
|
if (null !== $start) {
|
||||||
->getReachableCircles($user, $role, $center);
|
$qb->setFirstResult($start);
|
||||||
|
}
|
||||||
|
if (null !== $limit) {
|
||||||
|
$qb->setMaxResults($limit);
|
||||||
|
}
|
||||||
|
|
||||||
return $this->em->getRepository(Activity::class)
|
return $qb->getQuery()->getResult();
|
||||||
->findByAccompanyingPeriod($period, $scopes, true, $limit, $start, $orderBy);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function buildBaseQuery(array $filters): QueryBuilder
|
||||||
|
{
|
||||||
|
$qb = $this->repository
|
||||||
|
->createQueryBuilder('a')
|
||||||
|
;
|
||||||
|
|
||||||
|
if (($filters['my_activities'] ?? false) and ($user = $this->security->getUser()) instanceof User) {
|
||||||
|
$qb->andWhere(
|
||||||
|
$qb->expr()->orX(
|
||||||
|
'a.createdBy = :user',
|
||||||
|
'a.user = :user',
|
||||||
|
':user MEMBER OF a.users'
|
||||||
|
)
|
||||||
|
)->setParameter('user', $user);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ([] !== ($types = $filters['types'] ?? [])) {
|
||||||
|
$qb->andWhere('a.activityType IN (:types)')->setParameter('types', $types);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ([] !== ($jobs = $filters['jobs'] ?? [])) {
|
||||||
|
$qb
|
||||||
|
->leftJoin('a.createdBy', 'creator')
|
||||||
|
->leftJoin('a.user', 'activity_u')
|
||||||
|
->andWhere(
|
||||||
|
$qb->expr()->orX(
|
||||||
|
'creator.userJob IN (:jobs)',
|
||||||
|
'activity_u.userJob IN (:jobs)',
|
||||||
|
'EXISTS (SELECT 1 FROM ' . User::class . ' activity_user WHERE activity_user MEMBER OF a.users AND activity_user.userJob IN (:jobs))'
|
||||||
|
)
|
||||||
|
)
|
||||||
|
->setParameter('jobs', $jobs);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (null !== ($after = $filters['after'] ?? null)) {
|
||||||
|
$qb->andWhere('a.date >= :after')->setParameter('after', $after);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (null !== ($before = $filters['before'] ?? null)) {
|
||||||
|
$qb->andWhere('a.date <= :before')->setParameter('before', $before);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $qb;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param AccompanyingPeriod|Person $associated
|
||||||
|
* @return array<ActivityType>
|
||||||
|
*/
|
||||||
|
public function findActivityTypeByAssociated(AccompanyingPeriod|Person $associated): array
|
||||||
|
{
|
||||||
|
$in = $this->em->createQueryBuilder();
|
||||||
|
$in
|
||||||
|
->select('1')
|
||||||
|
->from(Activity::class, 'a');
|
||||||
|
|
||||||
|
if ($associated instanceof Person) {
|
||||||
|
$in = $this->filterBaseQueryByPerson($in, $associated, ActivityVoter::SEE);
|
||||||
|
} else {
|
||||||
|
$in->andWhere('a.accompanyingPeriod = :period')->setParameter('period', $associated);
|
||||||
|
}
|
||||||
|
|
||||||
|
// join between the embedded exist query and the main query
|
||||||
|
$in->andWhere('a.activityType = t');
|
||||||
|
|
||||||
|
$qb = $this->em->createQueryBuilder()->setParameters($in->getParameters());
|
||||||
|
$qb
|
||||||
|
->select('t')
|
||||||
|
->from(ActivityType::class, 't')
|
||||||
|
->where(
|
||||||
|
$qb->expr()->exists($in->getDQL())
|
||||||
|
);
|
||||||
|
|
||||||
|
return $qb->getQuery()->getResult();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function findUserJobByAssociated(Person|AccompanyingPeriod $associated): array
|
||||||
|
{
|
||||||
|
$in = $this->em->createQueryBuilder();
|
||||||
|
$in->select('IDENTITY(u.userJob)')
|
||||||
|
->from(User::class, 'u')
|
||||||
|
->join(
|
||||||
|
Activity::class,
|
||||||
|
'a',
|
||||||
|
Join::WITH,
|
||||||
|
'a.createdBy = u OR a.user = u OR u MEMBER OF a.users'
|
||||||
|
);
|
||||||
|
|
||||||
|
if ($associated instanceof Person) {
|
||||||
|
$in = $this->filterBaseQueryByPerson($in, $associated, ActivityVoter::SEE);
|
||||||
|
} else {
|
||||||
|
$in->andWhere('a.accompanyingPeriod = :associated');
|
||||||
|
$in->setParameter('associated', $associated);
|
||||||
|
}
|
||||||
|
|
||||||
|
$qb = $this->em->createQueryBuilder()->setParameters($in->getParameters());
|
||||||
|
|
||||||
|
$qb->select('ub', 'JSON_EXTRACT(ub.label, :lang) AS HIDDEN lang')
|
||||||
|
->from(UserJob::class, 'ub')
|
||||||
|
->where($qb->expr()->in('ub.id', $in->getDQL()))
|
||||||
|
->setParameter('lang', $this->requestStack->getCurrentRequest()->getLocale())
|
||||||
|
->orderBy('lang')
|
||||||
|
;
|
||||||
|
|
||||||
|
return $qb->getQuery()->getResult();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
public function findByAccompanyingPeriodSimplified(AccompanyingPeriod $period, ?int $limit = 1000): array
|
public function findByAccompanyingPeriodSimplified(AccompanyingPeriod $period, ?int $limit = 1000): array
|
||||||
{
|
{
|
||||||
$rsm = new ResultSetMappingBuilder($this->em);
|
$rsm = new ResultSetMappingBuilder($this->em);
|
||||||
@@ -159,25 +285,73 @@ final class ActivityACLAwareRepository implements ActivityACLAwareRepositoryInte
|
|||||||
return $nq->getResult(AbstractQuery::HYDRATE_ARRAY);
|
return $nq->getResult(AbstractQuery::HYDRATE_ARRAY);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
public function findByPerson(Person $person, string $role, ?int $start = 0, ?int $limit = 1000, ?array $orderBy = [], array $filters = []): array
|
||||||
* @param array $orderBy
|
|
||||||
*
|
|
||||||
* @return Activity[]|array
|
|
||||||
*/
|
|
||||||
public function findByPerson(Person $person, string $role, ?int $start = 0, ?int $limit = 1000, ?array $orderBy = []): array
|
|
||||||
{
|
{
|
||||||
$user = $this->security->getUser();
|
$qb = $this->buildBaseQuery($filters);
|
||||||
$center = $this->centerResolverDispatcher->resolveCenter($person);
|
|
||||||
|
|
||||||
if (0 === count($orderBy)) {
|
$qb = $this->filterBaseQueryByPerson($qb, $person, $role);
|
||||||
$orderBy = ['date' => 'DESC'];
|
|
||||||
|
foreach ($orderBy as $field => $direction) {
|
||||||
|
$qb->addOrderBy('a.' . $field, $direction);
|
||||||
}
|
}
|
||||||
|
|
||||||
$reachableScopes = $this->authorizationHelper
|
if (null !== $start) {
|
||||||
->getReachableCircles($user, $role, $center);
|
$qb->setFirstResult($start);
|
||||||
|
}
|
||||||
|
if (null !== $limit) {
|
||||||
|
$qb->setMaxResults($limit);
|
||||||
|
}
|
||||||
|
|
||||||
return $this->em->getRepository(Activity::class)
|
return $qb->getQuery()->getResult();
|
||||||
->findByPersonImplied($person, $reachableScopes, $orderBy, $limit, $start);
|
}
|
||||||
|
|
||||||
|
private function filterBaseQueryByPerson(QueryBuilder $qb, Person $person, string $role): QueryBuilder
|
||||||
|
{
|
||||||
|
$orX = $qb->expr()->orX();
|
||||||
|
$counter = 0;
|
||||||
|
foreach ($this->centerResolverManager->resolveCenters($person) as $center) {
|
||||||
|
$scopes = $this->authorizationHelper->getReachableScopes($role, $center);
|
||||||
|
|
||||||
|
if ([] === $scopes) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$orX->add(sprintf('a.person = :person AND a.scope IN (:scopes_%d)', $counter));
|
||||||
|
$qb->setParameter(sprintf('scopes_%d', $counter), $scopes);
|
||||||
|
$qb->setParameter('person', $person);
|
||||||
|
$counter++;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($person->getAccompanyingPeriodParticipations() as $participation) {
|
||||||
|
if (!$this->security->isGranted(ActivityVoter::SEE, $participation->getAccompanyingPeriod())) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$and = $qb->expr()->andX(
|
||||||
|
sprintf('a.accompanyingPeriod = :period_%d', $counter),
|
||||||
|
sprintf('a.date >= :participation_start_%d', $counter)
|
||||||
|
);
|
||||||
|
|
||||||
|
$qb
|
||||||
|
->setParameter(sprintf('period_%d', $counter), $participation->getAccompanyingPeriod())
|
||||||
|
->setParameter(sprintf('participation_start_%d', $counter), $participation->getStartDate());
|
||||||
|
|
||||||
|
if (null !== $participation->getEndDate()) {
|
||||||
|
$and->add(sprintf('a.date < :participation_end_%d', $counter));
|
||||||
|
$qb
|
||||||
|
->setParameter(sprintf('participation_end_%d', $counter), $participation->getEndDate());
|
||||||
|
}
|
||||||
|
$orX->add($and);
|
||||||
|
$counter++;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (0 === $orX->count()) {
|
||||||
|
$qb->andWhere('FALSE = TRUE');
|
||||||
|
} else {
|
||||||
|
$qb->andWhere($orX);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $qb;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function queryTimelineIndexer(string $context, array $args = []): array
|
public function queryTimelineIndexer(string $context, array $args = []): array
|
||||||
@@ -226,7 +400,6 @@ final class ActivityACLAwareRepository implements ActivityACLAwareRepositoryInte
|
|||||||
|
|
||||||
// acls:
|
// acls:
|
||||||
$reachableCenters = $this->authorizationHelper->getReachableCenters(
|
$reachableCenters = $this->authorizationHelper->getReachableCenters(
|
||||||
$this->tokenStorage->getToken()->getUser(),
|
|
||||||
ActivityVoter::SEE
|
ActivityVoter::SEE
|
||||||
);
|
);
|
||||||
|
|
||||||
@@ -251,7 +424,7 @@ final class ActivityACLAwareRepository implements ActivityACLAwareRepositoryInte
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
// we get all the reachable scopes for this center
|
// we get all the reachable scopes for this center
|
||||||
$reachableScopes = $this->authorizationHelper->getReachableScopes($this->tokenStorage->getToken()->getUser(), ActivityVoter::SEE, $center);
|
$reachableScopes = $this->authorizationHelper->getReachableScopes(ActivityVoter::SEE, $center);
|
||||||
// we get the ids for those scopes
|
// we get the ids for those scopes
|
||||||
$reachablesScopesId = array_map(
|
$reachablesScopesId = array_map(
|
||||||
static fn (Scope $scope) => $scope->getId(),
|
static fn (Scope $scope) => $scope->getId(),
|
||||||
|
@@ -11,15 +11,32 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
namespace Chill\ActivityBundle\Repository;
|
namespace Chill\ActivityBundle\Repository;
|
||||||
|
|
||||||
|
use Chill\ActivityBundle\Entity\Activity;
|
||||||
|
use Chill\ActivityBundle\Entity\ActivityType;
|
||||||
|
use Chill\MainBundle\Entity\UserJob;
|
||||||
use Chill\PersonBundle\Entity\AccompanyingPeriod;
|
use Chill\PersonBundle\Entity\AccompanyingPeriod;
|
||||||
use Chill\PersonBundle\Entity\Person;
|
use Chill\PersonBundle\Entity\Person;
|
||||||
|
|
||||||
interface ActivityACLAwareRepositoryInterface
|
interface ActivityACLAwareRepositoryInterface
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
* @return Activity[]|array
|
* Return all the activities associated to an accompanying period and that the user is allowed to apply the given role.
|
||||||
|
*
|
||||||
|
*
|
||||||
|
* @param array{my_activities?: bool, types?: array<ActivityType>, jobs?: array<UserJob>, after?: \DateTimeImmutable|null, before?: \DateTimeImmutable|null} $filters
|
||||||
|
* @return array<Activity>
|
||||||
*/
|
*/
|
||||||
public function findByAccompanyingPeriod(AccompanyingPeriod $period, string $role, ?int $start = 0, ?int $limit = 1000, ?array $orderBy = []): array;
|
public function findByAccompanyingPeriod(AccompanyingPeriod $period, string $role, ?int $start = 0, ?int $limit = 1000, array $orderBy = ['date' => 'DESC'], array $filters = []): array;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array{my_activities?: bool, types?: array<ActivityType>, jobs?: array<UserJob>, after?: \DateTimeImmutable|null, before?: \DateTimeImmutable|null} $filters
|
||||||
|
*/
|
||||||
|
public function countByAccompanyingPeriod(AccompanyingPeriod $period, string $role, array $filters = []): int;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array{my_activities?: bool, types?: array<ActivityType>, jobs?: array<UserJob>, after?: \DateTimeImmutable|null, before?: \DateTimeImmutable|null} $filters
|
||||||
|
*/
|
||||||
|
public function countByPerson(Person $person, string $role, array $filters = []): int;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return a list of activities, simplified as array (not object).
|
* Return a list of activities, simplified as array (not object).
|
||||||
@@ -31,7 +48,28 @@ interface ActivityACLAwareRepositoryInterface
|
|||||||
public function findByAccompanyingPeriodSimplified(AccompanyingPeriod $period, ?int $limit = 1000): array;
|
public function findByAccompanyingPeriodSimplified(AccompanyingPeriod $period, ?int $limit = 1000): array;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return Activity[]|array
|
* @param array{my_activities?: bool, types?: array<ActivityType>, jobs?: array<UserJob>, after?: \DateTimeImmutable|null, before?: \DateTimeImmutable|null} $filters
|
||||||
|
* @return array<Activity>
|
||||||
*/
|
*/
|
||||||
public function findByPerson(Person $person, string $role, ?int $start = 0, ?int $limit = 1000, ?array $orderBy = []): array;
|
public function findByPerson(Person $person, string $role, ?int $start = 0, ?int $limit = 1000, array $orderBy = ['date' => 'DESC'], array $filters = []): array;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return a list of the type for the activities associated to person or accompanying period
|
||||||
|
*
|
||||||
|
* @return array<ActivityType>
|
||||||
|
*/
|
||||||
|
public function findActivityTypeByAssociated(AccompanyingPeriod|Person $associated): array;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return a list of the user job for the activities associated to person or accompanying period
|
||||||
|
*
|
||||||
|
* Associated mean the job:
|
||||||
|
* - of the creator;
|
||||||
|
* - of the user (activity.user)
|
||||||
|
* - of all the users
|
||||||
|
*
|
||||||
|
* @return array<UserJob>
|
||||||
|
*/
|
||||||
|
public function findUserJobByAssociated(AccompanyingPeriod|Person $associated): array;
|
||||||
}
|
}
|
||||||
|
@@ -11,9 +11,13 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
namespace Chill\ActivityBundle\Repository;
|
namespace Chill\ActivityBundle\Repository;
|
||||||
|
|
||||||
|
use Chill\ActivityBundle\Entity\Activity;
|
||||||
use Chill\ActivityBundle\Entity\ActivityType;
|
use Chill\ActivityBundle\Entity\ActivityType;
|
||||||
|
use Chill\PersonBundle\Entity\AccompanyingPeriod;
|
||||||
|
use Chill\PersonBundle\Entity\Person;
|
||||||
use Doctrine\ORM\EntityManagerInterface;
|
use Doctrine\ORM\EntityManagerInterface;
|
||||||
use Doctrine\ORM\EntityRepository;
|
use Doctrine\ORM\EntityRepository;
|
||||||
|
use Doctrine\ORM\Query\Expr\Join;
|
||||||
|
|
||||||
final class ActivityTypeRepository implements ActivityTypeRepositoryInterface
|
final class ActivityTypeRepository implements ActivityTypeRepositoryInterface
|
||||||
{
|
{
|
||||||
|
@@ -12,12 +12,14 @@ declare(strict_types=1);
|
|||||||
namespace Chill\ActivityBundle\Repository;
|
namespace Chill\ActivityBundle\Repository;
|
||||||
|
|
||||||
use Chill\ActivityBundle\Entity\ActivityType;
|
use Chill\ActivityBundle\Entity\ActivityType;
|
||||||
|
use Chill\PersonBundle\Entity\AccompanyingPeriod;
|
||||||
|
use Chill\PersonBundle\Entity\Person;
|
||||||
use Doctrine\Persistence\ObjectRepository;
|
use Doctrine\Persistence\ObjectRepository;
|
||||||
|
|
||||||
interface ActivityTypeRepositoryInterface extends ObjectRepository
|
interface ActivityTypeRepositoryInterface extends ObjectRepository
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
* @return array|ActivityType[]
|
* @return array<ActivityType>
|
||||||
*/
|
*/
|
||||||
public function findAllActive(): array;
|
public function findAllActive(): array;
|
||||||
}
|
}
|
||||||
|
@@ -80,12 +80,15 @@
|
|||||||
|
|
||||||
<div class="context-{{ context }}">
|
<div class="context-{{ context }}">
|
||||||
|
|
||||||
|
{{ filter|chill_render_filter_order_helper }}
|
||||||
|
|
||||||
{% if activities|length == 0 %}
|
{% if activities|length == 0 %}
|
||||||
<p class="chill-no-data-statement">
|
<p class="chill-no-data-statement">
|
||||||
{{ "There isn't any activities."|trans }}
|
{{ "There isn't any activities."|trans }}
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
{% else %}
|
{% else %}
|
||||||
|
|
||||||
<div class="flex-table activity-list">
|
<div class="flex-table activity-list">
|
||||||
{% for activity in activities %}
|
{% for activity in activities %}
|
||||||
{% include 'ChillActivityBundle:Activity:_list_item.html.twig' with {
|
{% include 'ChillActivityBundle:Activity:_list_item.html.twig' with {
|
||||||
@@ -96,4 +99,6 @@
|
|||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
|
{{ chill_pagination(paginator) }}
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
@@ -0,0 +1,325 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace Chill\ActivityBundle\Tests\Repository;
|
||||||
|
|
||||||
|
use Chill\ActivityBundle\Entity\ActivityType;
|
||||||
|
use Chill\ActivityBundle\Repository\ActivityACLAwareRepository;
|
||||||
|
use Chill\ActivityBundle\Repository\ActivityRepository;
|
||||||
|
use Chill\ActivityBundle\Security\Authorization\ActivityVoter;
|
||||||
|
use Chill\MainBundle\Entity\Center;
|
||||||
|
use Chill\MainBundle\Entity\Scope;
|
||||||
|
use Chill\MainBundle\Entity\User;
|
||||||
|
use Chill\MainBundle\Entity\UserJob;
|
||||||
|
use Chill\MainBundle\Security\Authorization\AuthorizationHelperForCurrentUserInterface;
|
||||||
|
use Chill\MainBundle\Security\Resolver\CenterResolverManagerInterface;
|
||||||
|
use Chill\PersonBundle\Entity\AccompanyingPeriod;
|
||||||
|
use Chill\PersonBundle\Entity\Person;
|
||||||
|
use Doctrine\ORM\EntityManagerInterface;
|
||||||
|
use Prophecy\Argument;
|
||||||
|
use Prophecy\PhpUnit\ProphecyTrait;
|
||||||
|
use Symfony\Bundle\FrameworkBundle\Test\KernelTestCase;
|
||||||
|
use Symfony\Component\HttpFoundation\Request;
|
||||||
|
use Symfony\Component\HttpFoundation\RequestStack;
|
||||||
|
use Symfony\Component\Security\Core\Security;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
* @coversNothing
|
||||||
|
*/
|
||||||
|
class ActivityACLAwareRepositoryTest extends KernelTestCase
|
||||||
|
{
|
||||||
|
use ProphecyTrait;
|
||||||
|
private AuthorizationHelperForCurrentUserInterface $authorizationHelperForCurrentUser;
|
||||||
|
|
||||||
|
private CenterResolverManagerInterface $centerResolverManager;
|
||||||
|
|
||||||
|
private ActivityRepository $activityRepository;
|
||||||
|
|
||||||
|
private EntityManagerInterface $entityManager;
|
||||||
|
|
||||||
|
private Security $security;
|
||||||
|
|
||||||
|
private RequestStack $requestStack;
|
||||||
|
|
||||||
|
protected function setUp(): void
|
||||||
|
{
|
||||||
|
self::bootKernel();
|
||||||
|
|
||||||
|
$this->authorizationHelperForCurrentUser = self::$container->get(AuthorizationHelperForCurrentUserInterface::class);
|
||||||
|
$this->centerResolverManager = self::$container->get(CenterResolverManagerInterface::class);
|
||||||
|
$this->activityRepository = self::$container->get(ActivityRepository::class);
|
||||||
|
$this->entityManager = self::$container->get(EntityManagerInterface::class);
|
||||||
|
$this->security = self::$container->get(Security::class);
|
||||||
|
|
||||||
|
$this->requestStack = $requestStack = new RequestStack();
|
||||||
|
$request = $this->prophesize(Request::class);
|
||||||
|
$request->getLocale()->willReturn('fr');
|
||||||
|
$request->getDefaultLocale()->willReturn('fr');
|
||||||
|
$requestStack->push($request->reveal());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @dataProvider provideDataFindByAccompanyingPeriod
|
||||||
|
*/
|
||||||
|
public function testFindByAccompanyingPeriod(AccompanyingPeriod $period, User $user, string $role, ?int $start = 0, ?int $limit = 1000, array $orderBy = ['date' => 'DESC'], array $filters = []): void
|
||||||
|
{
|
||||||
|
$security = $this->prophesize(Security::class);
|
||||||
|
$security->isGranted($role, $period)->willReturn(true);
|
||||||
|
$security->getUser()->willReturn($user);
|
||||||
|
|
||||||
|
$repository = new ActivityACLAwareRepository(
|
||||||
|
$this->authorizationHelperForCurrentUser,
|
||||||
|
$this->centerResolverManager,
|
||||||
|
$this->activityRepository,
|
||||||
|
$this->entityManager,
|
||||||
|
$security->reveal(),
|
||||||
|
$this->requestStack
|
||||||
|
);
|
||||||
|
|
||||||
|
$actual = $repository->findByAccompanyingPeriod($period, $role, $start, $limit, $orderBy, $filters);
|
||||||
|
|
||||||
|
self::assertIsArray($actual);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @dataProvider provideDataFindByAccompanyingPeriod
|
||||||
|
*/
|
||||||
|
public function testFindActivityTypeByAccompanyingPeriod(AccompanyingPeriod $period, User $user, string $role, ?int $start = 0, ?int $limit = 1000, array $orderBy = ['date' => 'DESC'], array $filters = []): void
|
||||||
|
{
|
||||||
|
$security = $this->prophesize(Security::class);
|
||||||
|
$security->isGranted($role, $period)->willReturn(true);
|
||||||
|
$security->getUser()->willReturn($user);
|
||||||
|
|
||||||
|
$repository = new ActivityACLAwareRepository(
|
||||||
|
$this->authorizationHelperForCurrentUser,
|
||||||
|
$this->centerResolverManager,
|
||||||
|
$this->activityRepository,
|
||||||
|
$this->entityManager,
|
||||||
|
$security->reveal(),
|
||||||
|
$this->requestStack
|
||||||
|
);
|
||||||
|
|
||||||
|
$actual = $repository->findActivityTypeByAssociated($period);
|
||||||
|
|
||||||
|
self::assertIsArray($actual);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @dataProvider provideDataFindByPerson
|
||||||
|
*/
|
||||||
|
public function testFindActivityTypeByPerson(Person $person, User $user, array $centers, array $scopes, string $role, ?int $start = 0, ?int $limit = 1000, ?array $orderBy = [], array $filters = []): void
|
||||||
|
{
|
||||||
|
$role = ActivityVoter::SEE;
|
||||||
|
$centerResolver = $this->prophesize(CenterResolverManagerInterface::class);
|
||||||
|
$centerResolver->resolveCenters($person)->willReturn($centers);
|
||||||
|
|
||||||
|
$authorizationHelper = $this->prophesize(AuthorizationHelperForCurrentUserInterface::class);
|
||||||
|
$authorizationHelper->getReachableScopes($role, Argument::type(Center::class))
|
||||||
|
->willReturn($scopes);
|
||||||
|
|
||||||
|
$security = $this->prophesize(Security::class);
|
||||||
|
$security->isGranted($role, Argument::type(AccompanyingPeriod::class))->willReturn(true);
|
||||||
|
$security->getUser()->willReturn($user);
|
||||||
|
|
||||||
|
$repository = new ActivityACLAwareRepository(
|
||||||
|
$authorizationHelper->reveal(),
|
||||||
|
$centerResolver->reveal(),
|
||||||
|
$this->activityRepository,
|
||||||
|
$this->entityManager,
|
||||||
|
$security->reveal(),
|
||||||
|
$this->requestStack
|
||||||
|
);
|
||||||
|
|
||||||
|
$actual = $repository->findByPerson($person, $role, $start, $limit, $orderBy, $filters);
|
||||||
|
|
||||||
|
self::assertIsArray($actual);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @dataProvider provideDataFindByPerson
|
||||||
|
*/
|
||||||
|
public function testFindByPerson(Person $person, User $user, array $centers, array $scopes, string $role, ?int $start = 0, ?int $limit = 1000, ?array $orderBy = [], array $filters = []): void
|
||||||
|
{
|
||||||
|
$centerResolver = $this->prophesize(CenterResolverManagerInterface::class);
|
||||||
|
$centerResolver->resolveCenters($person)->willReturn($centers);
|
||||||
|
|
||||||
|
$authorizationHelper = $this->prophesize(AuthorizationHelperForCurrentUserInterface::class);
|
||||||
|
$authorizationHelper->getReachableScopes($role, Argument::type(Center::class))
|
||||||
|
->willReturn($scopes);
|
||||||
|
|
||||||
|
$security = $this->prophesize(Security::class);
|
||||||
|
$security->isGranted($role, Argument::type(AccompanyingPeriod::class))->willReturn(true);
|
||||||
|
$security->getUser()->willReturn($user);
|
||||||
|
|
||||||
|
$repository = new ActivityACLAwareRepository(
|
||||||
|
$authorizationHelper->reveal(),
|
||||||
|
$centerResolver->reveal(),
|
||||||
|
$this->activityRepository,
|
||||||
|
$this->entityManager,
|
||||||
|
$security->reveal(),
|
||||||
|
$this->requestStack
|
||||||
|
);
|
||||||
|
|
||||||
|
$actual = $repository->findByPerson($person, $role, $start, $limit, $orderBy, $filters);
|
||||||
|
|
||||||
|
self::assertIsArray($actual);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function provideDataFindByPerson(): iterable
|
||||||
|
{
|
||||||
|
$this->setUp();
|
||||||
|
|
||||||
|
/** @var Person $person */
|
||||||
|
if (null === $person = $this->entityManager->createQueryBuilder()
|
||||||
|
->select('p')->from(Person::class, 'p')->setMaxResults(1)
|
||||||
|
->getQuery()->getSingleResult()) {
|
||||||
|
throw new \RuntimeException("person not found");
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @var AccompanyingPeriod $period1 */
|
||||||
|
if (null === $period1 = $this->entityManager
|
||||||
|
->createQueryBuilder()
|
||||||
|
->select('a')
|
||||||
|
->from(AccompanyingPeriod::class, 'a')
|
||||||
|
->setMaxResults(1)
|
||||||
|
->getQuery()
|
||||||
|
->getSingleResult()) {
|
||||||
|
throw new \RuntimeException("no period found");
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @var AccompanyingPeriod $period2 */
|
||||||
|
if (null === $period2 = $this->entityManager
|
||||||
|
->createQueryBuilder()
|
||||||
|
->select('a')
|
||||||
|
->from(AccompanyingPeriod::class, 'a')
|
||||||
|
->where('a.id > :pid')
|
||||||
|
->setParameter('pid', $period1->getId())
|
||||||
|
->setMaxResults(1)
|
||||||
|
->getQuery()
|
||||||
|
->getSingleResult()) {
|
||||||
|
throw new \RuntimeException("no second period found");
|
||||||
|
}
|
||||||
|
// add a period
|
||||||
|
$period1->addPerson($person);
|
||||||
|
$period2->addPerson($person);
|
||||||
|
$period1->getParticipationsContainsPerson($person)->first()->setEndDate(
|
||||||
|
(new \DateTime('now'))->add(new \DateInterval('P1M'))
|
||||||
|
);
|
||||||
|
|
||||||
|
if ([] === $types = $this->entityManager
|
||||||
|
->createQueryBuilder()
|
||||||
|
->select('t')
|
||||||
|
->from(ActivityType::class, 't')
|
||||||
|
->setMaxResults(2)
|
||||||
|
->getQuery()
|
||||||
|
->getResult()) {
|
||||||
|
throw new \RuntimeException("no types");
|
||||||
|
}
|
||||||
|
|
||||||
|
if ([] === $jobs = $this->entityManager
|
||||||
|
->createQueryBuilder()
|
||||||
|
->select('j')
|
||||||
|
->from(UserJob::class, 'j')
|
||||||
|
->setMaxResults(2)
|
||||||
|
->getQuery()
|
||||||
|
->getResult()
|
||||||
|
) {
|
||||||
|
throw new \RuntimeException("no jobs found");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (null === $user = $this->entityManager
|
||||||
|
->createQueryBuilder()
|
||||||
|
->select('u')
|
||||||
|
->from(User::class, 'u')
|
||||||
|
->setMaxResults(1)
|
||||||
|
->getQuery()
|
||||||
|
->getSingleResult()
|
||||||
|
) {
|
||||||
|
throw new \RuntimeException("no user found");
|
||||||
|
}
|
||||||
|
|
||||||
|
if ([] === $centers = $this->entityManager->createQueryBuilder()
|
||||||
|
->select('c')->from(Center::class, 'c')->setMaxResults(2)->getQuery()
|
||||||
|
->getResult()) {
|
||||||
|
throw new \RuntimeException("no centers found");
|
||||||
|
}
|
||||||
|
|
||||||
|
if ([] === $scopes = $this->entityManager->createQueryBuilder()
|
||||||
|
->select('s')->from(Scope::class, 's')->setMaxResults(2)->getQuery()
|
||||||
|
->getResult()) {
|
||||||
|
throw new \RuntimeException("no scopes found");
|
||||||
|
}
|
||||||
|
|
||||||
|
yield [$person, $user, $centers, $scopes, ActivityVoter::SEE, 0, 5, ['date' => 'DESC'], []];
|
||||||
|
yield [$person, $user, $centers, $scopes, ActivityVoter::SEE, 0, 5, ['date' => 'DESC'], ['my_activities' => true]];
|
||||||
|
yield [$person, $user, $centers, $scopes, ActivityVoter::SEE, 0, 5, ['date' => 'DESC'], ['types' => $types]];
|
||||||
|
yield [$person, $user, $centers, $scopes, ActivityVoter::SEE, 0, 5, ['date' => 'DESC'], ['jobs' => $jobs]];
|
||||||
|
yield [$person, $user, $centers, $scopes, ActivityVoter::SEE, 0, 5, ['date' => 'DESC'], ['after' => new \DateTimeImmutable('1 year ago')]];
|
||||||
|
yield [$person, $user, $centers, $scopes, ActivityVoter::SEE, 0, 5, ['date' => 'DESC'], ['before' => new \DateTimeImmutable('1 year ago')]];
|
||||||
|
yield [$person, $user, $centers, $scopes, ActivityVoter::SEE, 0, 5, ['date' => 'DESC'], ['after' => new \DateTimeImmutable('1 year ago'), 'before' => new \DateTimeImmutable('1 month ago')]];
|
||||||
|
}
|
||||||
|
|
||||||
|
public function provideDataFindByAccompanyingPeriod(): iterable
|
||||||
|
{
|
||||||
|
$this->setUp();
|
||||||
|
|
||||||
|
if (null === $period = $this->entityManager
|
||||||
|
->createQueryBuilder()
|
||||||
|
->select('a')
|
||||||
|
->from(AccompanyingPeriod::class, 'a')
|
||||||
|
->setMaxResults(1)
|
||||||
|
->getQuery()
|
||||||
|
->getSingleResult()) {
|
||||||
|
throw new \RuntimeException("no period found");
|
||||||
|
}
|
||||||
|
|
||||||
|
if ([] === $types = $this->entityManager
|
||||||
|
->createQueryBuilder()
|
||||||
|
->select('t')
|
||||||
|
->from(ActivityType::class, 't')
|
||||||
|
->setMaxResults(2)
|
||||||
|
->getQuery()
|
||||||
|
->getResult()) {
|
||||||
|
throw new \RuntimeException("no types");
|
||||||
|
}
|
||||||
|
|
||||||
|
if ([] === $jobs = $this->entityManager
|
||||||
|
->createQueryBuilder()
|
||||||
|
->select('j')
|
||||||
|
->from(UserJob::class, 'j')
|
||||||
|
->setMaxResults(2)
|
||||||
|
->getQuery()
|
||||||
|
->getResult()
|
||||||
|
) {
|
||||||
|
throw new \RuntimeException("no jobs found");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (null === $user = $this->entityManager
|
||||||
|
->createQueryBuilder()
|
||||||
|
->select('u')
|
||||||
|
->from(User::class, 'u')
|
||||||
|
->setMaxResults(1)
|
||||||
|
->getQuery()
|
||||||
|
->getSingleResult()
|
||||||
|
) {
|
||||||
|
throw new \RuntimeException("no user found");
|
||||||
|
}
|
||||||
|
|
||||||
|
yield [$period, $user, ActivityVoter::SEE, 0, 10, ['date' => 'DESC'], []];
|
||||||
|
yield [$period, $user, ActivityVoter::SEE, 0, 10, ['date' => 'DESC'], ['my_activities' => true]];
|
||||||
|
yield [$period, $user, ActivityVoter::SEE, 0, 10, ['date' => 'DESC'], ['types' => $types]];
|
||||||
|
yield [$period, $user, ActivityVoter::SEE, 0, 10, ['date' => 'DESC'], ['jobs' => $jobs]];
|
||||||
|
yield [$period, $user, ActivityVoter::SEE, 0, 10, ['date' => 'DESC'], ['after' => new \DateTimeImmutable('1 year ago')]];
|
||||||
|
yield [$period, $user, ActivityVoter::SEE, 0, 10, ['date' => 'DESC'], ['before' => new \DateTimeImmutable('1 year ago')]];
|
||||||
|
yield [$period, $user, ActivityVoter::SEE, 0, 10, ['date' => 'DESC'], ['after' => new \DateTimeImmutable('1 year ago'), 'before' => new \DateTimeImmutable('1 month ago')]];
|
||||||
|
}
|
||||||
|
}
|
@@ -148,6 +148,10 @@ services:
|
|||||||
tags:
|
tags:
|
||||||
- { name: chill.export_aggregator, alias: activity_common_type_aggregator }
|
- { name: chill.export_aggregator, alias: activity_common_type_aggregator }
|
||||||
|
|
||||||
|
Chill\ActivityBundle\Export\Aggregator\ActivityLocationAggregator:
|
||||||
|
tags:
|
||||||
|
- { name: chill.export_aggregator, alias: activity_common_location_aggregator }
|
||||||
|
|
||||||
chill.activity.export.user_aggregator:
|
chill.activity.export.user_aggregator:
|
||||||
class: Chill\ActivityBundle\Export\Aggregator\ActivityUserAggregator
|
class: Chill\ActivityBundle\Export\Aggregator\ActivityUserAggregator
|
||||||
tags:
|
tags:
|
||||||
|
@@ -83,12 +83,20 @@ Third persons: Tiers non-pro.
|
|||||||
Others persons: Usagers
|
Others persons: Usagers
|
||||||
Third parties: Tiers professionnels
|
Third parties: Tiers professionnels
|
||||||
Users concerned: T(M)S
|
Users concerned: T(M)S
|
||||||
|
|
||||||
activity:
|
activity:
|
||||||
|
date: Date de l'échange
|
||||||
Insert a document: Insérer un document
|
Insert a document: Insérer un document
|
||||||
Remove a document: Supprimer le document
|
Remove a document: Supprimer le document
|
||||||
comment: Commentaire
|
comment: Commentaire
|
||||||
No documents: Aucun document
|
No documents: Aucun document
|
||||||
|
|
||||||
|
# activity filter in list page
|
||||||
|
activity_filter:
|
||||||
|
My activities: Mes échanges (où j'interviens)
|
||||||
|
Types: Par type d'échange
|
||||||
|
Jobs: Par métier impliqué
|
||||||
|
|
||||||
#timeline
|
#timeline
|
||||||
'%user% has done an %activity_type%': '%user% a effectué un échange de type "%activity_type%"'
|
'%user% has done an %activity_type%': '%user% a effectué un échange de type "%activity_type%"'
|
||||||
|
|
||||||
@@ -378,6 +386,9 @@ export:
|
|||||||
is sent: envoyé
|
is sent: envoyé
|
||||||
is received: reçu
|
is received: reçu
|
||||||
Group activity by sentreceived: Grouper les échanges par envoyé / reçu
|
Group activity by sentreceived: Grouper les échanges par envoyé / reçu
|
||||||
|
by_location:
|
||||||
|
Activity Location: Localisation de l'échange
|
||||||
|
Title: Grouper les échanges par localisation de l'échange
|
||||||
|
|
||||||
generic_doc:
|
generic_doc:
|
||||||
filter:
|
filter:
|
||||||
|
@@ -37,7 +37,7 @@ class UserJob
|
|||||||
protected ?int $id = null;
|
protected ?int $id = null;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @var array|string[]A
|
* @var array<string, string>
|
||||||
* @ORM\Column(name="label", type="json")
|
* @ORM\Column(name="label", type="json")
|
||||||
* @Serializer\Groups({"read", "docgen:read"})
|
* @Serializer\Groups({"read", "docgen:read"})
|
||||||
* @Serializer\Context({"is-translatable": true}, groups={"docgen:read"})
|
* @Serializer\Context({"is-translatable": true}, groups={"docgen:read"})
|
||||||
|
@@ -97,7 +97,7 @@ interface ExportInterface extends ExportElementInterface
|
|||||||
* @param mixed[] $values The values from the result. if there are duplicates, those might be given twice. Example: array('FR', 'BE', 'CZ', 'FR', 'BE', 'FR')
|
* @param mixed[] $values The values from the result. if there are duplicates, those might be given twice. Example: array('FR', 'BE', 'CZ', 'FR', 'BE', 'FR')
|
||||||
* @param mixed $data The data from the export's form (as defined in `buildForm`)
|
* @param mixed $data The data from the export's form (as defined in `buildForm`)
|
||||||
*
|
*
|
||||||
* @return callable(null|string|int|float|'_header' $value): string|int|\DateTimeInterface where the first argument is the value, and the function should return the label to show in the formatted file. Example : `function($countryCode) use ($countries) { return $countries[$countryCode]->getName(); }`
|
* @return (callable(null|string|int|float|'_header' $value): string|int|\DateTimeInterface) where the first argument is the value, and the function should return the label to show in the formatted file. Example : `function($countryCode) use ($countries) { return $countries[$countryCode]->getName(); }`
|
||||||
*/
|
*/
|
||||||
public function getLabels($key, array $values, $data);
|
public function getLabels($key, array $values, $data);
|
||||||
|
|
||||||
|
@@ -13,6 +13,8 @@ namespace Chill\MainBundle\Form\Type\Listing;
|
|||||||
|
|
||||||
use Chill\MainBundle\Form\Type\ChillDateType;
|
use Chill\MainBundle\Form\Type\ChillDateType;
|
||||||
use Chill\MainBundle\Templating\Listing\FilterOrderHelper;
|
use Chill\MainBundle\Templating\Listing\FilterOrderHelper;
|
||||||
|
use Symfony\Bridge\Doctrine\Form\Type\EntityType;
|
||||||
|
use Symfony\Component\Form\Extension\Core\Type\CheckboxType;
|
||||||
use Symfony\Component\Form\Extension\Core\Type\ChoiceType;
|
use Symfony\Component\Form\Extension\Core\Type\ChoiceType;
|
||||||
use Symfony\Component\Form\Extension\Core\Type\HiddenType;
|
use Symfony\Component\Form\Extension\Core\Type\HiddenType;
|
||||||
use Symfony\Component\Form\Extension\Core\Type\SearchType;
|
use Symfony\Component\Form\Extension\Core\Type\SearchType;
|
||||||
@@ -27,13 +29,6 @@ use function count;
|
|||||||
|
|
||||||
final class FilterOrderType extends \Symfony\Component\Form\AbstractType
|
final class FilterOrderType extends \Symfony\Component\Form\AbstractType
|
||||||
{
|
{
|
||||||
private RequestStack $requestStack;
|
|
||||||
|
|
||||||
public function __construct(RequestStack $requestStack)
|
|
||||||
{
|
|
||||||
$this->requestStack = $requestStack;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function buildForm(FormBuilderInterface $builder, array $options)
|
public function buildForm(FormBuilderInterface $builder, array $options)
|
||||||
{
|
{
|
||||||
/** @var FilterOrderHelper $helper */
|
/** @var FilterOrderHelper $helper */
|
||||||
@@ -43,22 +38,16 @@ final class FilterOrderType extends \Symfony\Component\Form\AbstractType
|
|||||||
$builder->add('q', SearchType::class, [
|
$builder->add('q', SearchType::class, [
|
||||||
'label' => false,
|
'label' => false,
|
||||||
'required' => false,
|
'required' => false,
|
||||||
|
'attr' => [
|
||||||
|
'placeholder' => 'filter_order.Search',
|
||||||
|
]
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
$checkboxesBuilder = $builder->create('checkboxes', null, ['compound' => true]);
|
$checkboxesBuilder = $builder->create('checkboxes', null, ['compound' => true]);
|
||||||
|
|
||||||
foreach ($helper->getCheckboxes() as $name => $c) {
|
foreach ($helper->getCheckboxes() as $name => $c) {
|
||||||
$choices = array_combine(
|
$choices = self::buildCheckboxChoices($c['choices'], $c['trans']);
|
||||||
array_map(static function ($c, $t) {
|
|
||||||
if (null !== $t) {
|
|
||||||
return $t;
|
|
||||||
}
|
|
||||||
|
|
||||||
return $c;
|
|
||||||
}, $c['choices'], $c['trans']),
|
|
||||||
$c['choices']
|
|
||||||
);
|
|
||||||
|
|
||||||
$checkboxesBuilder->add($name, ChoiceType::class, [
|
$checkboxesBuilder->add($name, ChoiceType::class, [
|
||||||
'choices' => $choices,
|
'choices' => $choices,
|
||||||
@@ -71,6 +60,25 @@ final class FilterOrderType extends \Symfony\Component\Form\AbstractType
|
|||||||
$builder->add($checkboxesBuilder);
|
$builder->add($checkboxesBuilder);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ([] !== $helper->getEntityChoices()) {
|
||||||
|
$entityChoicesBuilder = $builder->create('entity_choices', null, ['compound' => true]);
|
||||||
|
|
||||||
|
foreach ($helper->getEntityChoices() as $key => [
|
||||||
|
'label' => $label, 'choices' => $choices, 'options' => $opts, 'class' => $class
|
||||||
|
]) {
|
||||||
|
$entityChoicesBuilder->add($key, EntityType::class, [
|
||||||
|
'label' => $label,
|
||||||
|
'choices' => $choices,
|
||||||
|
'class' => $class,
|
||||||
|
'multiple' => true,
|
||||||
|
'expanded' => true,
|
||||||
|
...$opts,
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
$builder->add($entityChoicesBuilder);
|
||||||
|
}
|
||||||
|
|
||||||
if (0 < count($helper->getDateRanges())) {
|
if (0 < count($helper->getDateRanges())) {
|
||||||
$dateRangesBuilder = $builder->create('dateRanges', null, ['compound' => true]);
|
$dateRangesBuilder = $builder->create('dateRanges', null, ['compound' => true]);
|
||||||
|
|
||||||
@@ -97,31 +105,31 @@ final class FilterOrderType extends \Symfony\Component\Form\AbstractType
|
|||||||
$builder->add($dateRangesBuilder);
|
$builder->add($dateRangesBuilder);
|
||||||
}
|
}
|
||||||
|
|
||||||
foreach ($this->requestStack->getCurrentRequest()->query->getIterator() as $key => $value) {
|
if ([] !== $helper->getSingleCheckbox()) {
|
||||||
switch ($key) {
|
$singleCheckBoxBuilder = $builder->create('single_checkboxes', null, ['compound' => true]);
|
||||||
case 'q':
|
|
||||||
case 'checkboxes' . $key:
|
|
||||||
case $key . '_from':
|
|
||||||
case $key . '_to':
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 'page':
|
foreach ($helper->getSingleCheckbox() as $name => ['label' => $label]) {
|
||||||
$builder->add($key, HiddenType::class, [
|
$singleCheckBoxBuilder->add($name, CheckboxType::class, ['label' => $label, 'required' => false]);
|
||||||
'data' => 1,
|
|
||||||
]);
|
|
||||||
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
$builder->add($key, HiddenType::class, [
|
|
||||||
'data' => $value,
|
|
||||||
]);
|
|
||||||
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$builder->add($singleCheckBoxBuilder);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static function buildCheckboxChoices(array $choices, array $trans = []): array
|
||||||
|
{
|
||||||
|
return array_combine(
|
||||||
|
array_map(static function ($c, $t) {
|
||||||
|
if (null !== $t) {
|
||||||
|
return $t;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $c;
|
||||||
|
}, $choices, $trans),
|
||||||
|
$choices
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
public function buildView(FormView $view, FormInterface $form, array $options)
|
public function buildView(FormView $view, FormInterface $form, array $options)
|
||||||
{
|
{
|
||||||
/** @var FilterOrderHelper $helper */
|
/** @var FilterOrderHelper $helper */
|
||||||
|
@@ -42,3 +42,7 @@ form {
|
|||||||
font-weight: 700;
|
font-weight: 700;
|
||||||
margin-bottom: .375em;
|
margin-bottom: .375em;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.chill_filter_order {
|
||||||
|
background: $gray-100;
|
||||||
|
}
|
@@ -1,65 +1,120 @@
|
|||||||
{{ form_start(form) }}
|
{{ form_start(form) }}
|
||||||
<div class="chill_filter_order container my-4">
|
<div class="accordion my-3" id="filterOrderAccordion">
|
||||||
<div class="row">
|
<h2 class="accordion-header" id="filterOrderHeading">
|
||||||
{% if form.vars.has_search_box %}
|
<button class="accordion-button" type="button" data-bs-toggle="collapse" data-bs-target="#filterOrderCollapse" aria-expanded="true" aria-controls="filterOrderCollapse">
|
||||||
<div class="col-md-12">
|
<strong><i class="fa fa-fw fa-filter"></i>Filtrer la liste</strong>
|
||||||
<div class="input-group mb-3">
|
</button>
|
||||||
{{ form_widget(form.q)}}
|
</h2>
|
||||||
<button type="submit" class="btn btn-chill-l-gray"><i class="fa fa-search"></i></button>
|
<div class="accordion-collapse collapse" id="filterOrderCollapse" aria-labelledby="filterOrderHeading" data-bs-parent="#filterOrderAccordion">
|
||||||
</div>
|
{% set btnSubmit = 0 %}
|
||||||
</div>
|
<div class="accordion-body chill_filter_order container-xxl p-5 py-2">
|
||||||
{% endif %}
|
<div class="row my-2">
|
||||||
</div>
|
{% if form.vars.has_search_box %}
|
||||||
{% if form.dateRanges is defined %}
|
<div class="col-sm-12">
|
||||||
{% if form.dateRanges|length > 0 %}
|
<div class="input-group">
|
||||||
{% for dateRangeName, _o in form.dateRanges %}
|
{{ form_widget(form.q) }}
|
||||||
<div class="row gx-2 justify-content-center">
|
<button type="submit" class="btn btn-misc"><i class="fa fa-search"></i></button>
|
||||||
{% if form.dateRanges[dateRangeName].vars.label is not same as(false) %}
|
|
||||||
<div class="col-md-5">
|
|
||||||
{{ form_label(form.dateRanges[dateRangeName])}}
|
|
||||||
</div>
|
|
||||||
{% endif %}
|
|
||||||
<div class="col-md-6">
|
|
||||||
<div class="input-group mb-3">
|
|
||||||
<span class="input-group-text">{{ 'chill_calendar.From'|trans }}</span>
|
|
||||||
{{ form_widget(form.dateRanges[dateRangeName]['from']) }}
|
|
||||||
<span class="input-group-text">{{ 'chill_calendar.To'|trans }}</span>
|
|
||||||
{{ form_widget(form.dateRanges[dateRangeName]['to']) }}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="col-md-1">
|
|
||||||
<button type="submit" class="btn btn-misc"><i class="fa fa-filter"></i></button>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{% endfor %}
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
{% if form.dateRanges is defined %}
|
||||||
|
{% set btnSubmit = 1 %}
|
||||||
|
{% if form.dateRanges|length > 0 %}
|
||||||
|
{% for dateRangeName, _o in form.dateRanges %}
|
||||||
|
<div class="row my-2">
|
||||||
|
{% if form.dateRanges[dateRangeName].vars.label is not same as(false) %}
|
||||||
|
{{ form_label(form.dateRanges[dateRangeName])}}
|
||||||
|
{% else %}
|
||||||
|
<div class="col-sm-4 col-form-label">{{ 'filter_order.By date'|trans }}</div>
|
||||||
|
{% endif %}
|
||||||
|
<div class="col-sm-8 pt-1">
|
||||||
|
<div class="input-group">
|
||||||
|
<span class="input-group-text">{{ 'chill_calendar.From'|trans }}</span>
|
||||||
|
{{ form_widget(form.dateRanges[dateRangeName]['from']) }}
|
||||||
|
<span class="input-group-text">{{ 'chill_calendar.To'|trans }}</span>
|
||||||
|
{{ form_widget(form.dateRanges[dateRangeName]['to']) }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endfor %}
|
||||||
|
{% endif %}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% endif %}
|
{% if form.checkboxes is defined %}
|
||||||
{% if form.checkboxes is defined %}
|
{% set btnSubmit = 1 %}
|
||||||
{% if form.checkboxes|length > 0 %}
|
{% if form.checkboxes|length > 0 %}
|
||||||
{% for checkbox_name, options in form.checkboxes %}
|
{% for checkbox_name, options in form.checkboxes %}
|
||||||
<div class="row gx-0">
|
<div class="row my-2">
|
||||||
<div class="col-md-12">
|
<div class="col-sm-4 col-form-label">{{ 'filter_order.By'|trans }}</div>
|
||||||
{% for c in form['checkboxes'][checkbox_name].children %}
|
<div class="col-sm-8 pt-2">
|
||||||
<div class="form-check form-check-inline">
|
{% for c in form['checkboxes'][checkbox_name].children %}
|
||||||
{{ form_widget(c) }}
|
{{ form_widget(c) }}
|
||||||
{{ form_label(c) }}
|
{{ form_label(c) }}
|
||||||
</div>
|
{% endfor %}
|
||||||
{% endfor %}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{% if loop.last %}
|
|
||||||
<div class="row gx-0">
|
|
||||||
<div class="col-md-12">
|
|
||||||
<ul class="record_actions">
|
|
||||||
<li>
|
|
||||||
<button type="submit" class="btn btn-misc"><i class="fa fa-filter"></i></button>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endfor %}
|
||||||
|
{% endif %}
|
||||||
|
{% endif %}
|
||||||
|
{% if form.entity_choices is defined %}
|
||||||
|
{% set btnSubmit = 1 %}
|
||||||
|
{% if form.entity_choices |length > 0 %}
|
||||||
|
{% for checkbox_name, options in form.entity_choices %}
|
||||||
|
<div class="row my-2">
|
||||||
|
{% if form.entity_choices[checkbox_name].vars.label is not same as(false) %}
|
||||||
|
{{ form_label(form.entity_choices[checkbox_name])}}
|
||||||
|
{% endif %}
|
||||||
|
<div class="col-sm-8 pt-2">
|
||||||
|
{% for c in form['entity_choices'][checkbox_name].children %}
|
||||||
|
{{ form_widget(c) }}
|
||||||
|
{{ form_label(c) }}
|
||||||
|
{% endfor %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endfor %}
|
||||||
|
{% endif %}
|
||||||
|
{% endif %}
|
||||||
|
{% if form.single_checkboxes is defined %}
|
||||||
|
{% set btnSubmit = 1 %}
|
||||||
|
{% for name, _o in form.single_checkboxes %}
|
||||||
|
<div class="row my-2">
|
||||||
|
<div class="col-sm-4 col-form-label">{{ 'filter_order.By'|trans }}</div>
|
||||||
|
<div class="col-sm-8 pt-2">
|
||||||
|
{{ form_widget(form.single_checkboxes[name]) }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% endif %}
|
|
||||||
|
{% if btnSubmit == 1 %}
|
||||||
|
<div class="row my-2">
|
||||||
|
<button type="submit" class="btn btn-sm btn-misc"><i class="fa fa-fw fa-filter"></i>{{ 'Filter'|trans }}</button>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
{% if active|length > 0 %}
|
||||||
|
<div class="activeFilters mt-3">
|
||||||
|
{% for f in active %}
|
||||||
|
<span class="badge rounded-pill bg-secondary ms-1 {{ f.position }} {{ f.name }}">
|
||||||
|
{%- if f.label != '' %}
|
||||||
|
<span class="text-dark">{{ f.label|trans }} : </span>
|
||||||
|
{% endif -%}
|
||||||
|
{%- if f.position == 'search_box' and f.value is not null %}
|
||||||
|
<span class="text-dark">{{ 'filter_order.search_box'|trans ~ ' :' }}</span>
|
||||||
|
{% endif -%}
|
||||||
|
{{ f.value}}{#
|
||||||
|
#}</span>
|
||||||
|
{% endfor %}
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
<div>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{% for k,v in otherParameters %}
|
||||||
|
<input type="hidden" name="{{ k }}" value="{{ v }}" />
|
||||||
|
{% endfor %}
|
||||||
{{ form_end(form) }}
|
{{ form_end(form) }}
|
||||||
|
|
||||||
|
@@ -0,0 +1,84 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace Chill\MainBundle\Templating\Listing;
|
||||||
|
|
||||||
|
use Symfony\Component\PropertyAccess\PropertyAccessorInterface;
|
||||||
|
use Symfony\Component\PropertyAccess\PropertyPathInterface;
|
||||||
|
use Symfony\Contracts\Translation\TranslatorInterface;
|
||||||
|
|
||||||
|
final readonly class FilterOrderGetActiveFilterHelper
|
||||||
|
{
|
||||||
|
public function __construct(
|
||||||
|
private TranslatorInterface $translator,
|
||||||
|
private PropertyAccessorInterface $propertyAccessor,
|
||||||
|
) {
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return all the data required to display the active filters
|
||||||
|
*
|
||||||
|
* @param FilterOrderHelper $filterOrderHelper
|
||||||
|
* @return array<array{label: string, value: string, position: string, name: string}>
|
||||||
|
*/
|
||||||
|
public function getActiveFilters(FilterOrderHelper $filterOrderHelper): array
|
||||||
|
{
|
||||||
|
$result = [];
|
||||||
|
|
||||||
|
if ($filterOrderHelper->hasSearchBox() && '' !== $filterOrderHelper->getQueryString()) {
|
||||||
|
$result[] = ['label' => '', 'value' => $filterOrderHelper->getQueryString(), 'position' => FilterOrderPositionEnum::SearchBox->value, 'name' => 'q'];
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($filterOrderHelper->getDateRanges() as $name => ['label' => $label]) {
|
||||||
|
$base = ['position' => FilterOrderPositionEnum::DateRange->value, 'name' => $name, 'label' => (string)$label];
|
||||||
|
|
||||||
|
if (null !== ($from = $filterOrderHelper->getDateRangeData($name)['from'] ?? null)) {
|
||||||
|
$result[] = ['value' => $this->translator->trans('filter_order.by_date.From', ['from_date' => $from]), ...$base];
|
||||||
|
}
|
||||||
|
if (null !== ($to = $filterOrderHelper->getDateRangeData($name)['to'] ?? null)) {
|
||||||
|
$result[] = ['value' => $this->translator->trans('filter_order.by_date.To', ['to_date' => $to]), ...$base];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($filterOrderHelper->getCheckboxes() as $name => ['choices' => $choices, 'trans' => $trans]) {
|
||||||
|
$translatedChoice = array_combine($choices, [...$trans]);
|
||||||
|
foreach ($filterOrderHelper->getCheckboxData($name) as $keyChoice) {
|
||||||
|
$result[] = ['value' => $this->translator->trans($translatedChoice[$keyChoice]), 'label' => '', 'position' => FilterOrderPositionEnum::Checkboxes->value, 'name' => $name];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($filterOrderHelper->getEntityChoices() as $name => ['label' => $label, 'class' => $class, 'choices' => $choices, 'options' => $options]) {
|
||||||
|
foreach ($filterOrderHelper->getEntityChoiceData($name) as $selected) {
|
||||||
|
if (is_callable($options['choice_label'])) {
|
||||||
|
$value = call_user_func($options['choice_label'], $selected);
|
||||||
|
} elseif ($options['choice_label'] instanceof PropertyPathInterface || is_string($options['choice_label'])) {
|
||||||
|
$value = $this->propertyAccessor->getValue($selected, $options['choice_label']);
|
||||||
|
} else {
|
||||||
|
if (!$selected instanceof \Stringable) {
|
||||||
|
throw new \UnexpectedValueException(sprintf("we are not able to transform the value of %s to a string. Implements \\Stringable or add a 'choice_label' option to the filterFormBuilder", get_class($selected)));
|
||||||
|
}
|
||||||
|
|
||||||
|
$value = (string)$selected;
|
||||||
|
}
|
||||||
|
|
||||||
|
$result[] = ['value' => $this->translator->trans($value), 'label' => $label, 'position' => FilterOrderPositionEnum::EntityChoice->value, 'name' => $name];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($filterOrderHelper->getSingleCheckbox() as $name => ['label' => $label]) {
|
||||||
|
if (true === $filterOrderHelper->getSingleCheckboxData($name)) {
|
||||||
|
$result[] = ['label' => '', 'value' => $this->translator->trans($label), 'position' => FilterOrderPositionEnum::SingleCheckbox->value, 'name' => $name];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $result;
|
||||||
|
}
|
||||||
|
}
|
@@ -17,47 +17,80 @@ use Symfony\Component\Form\FormFactoryInterface;
|
|||||||
use Symfony\Component\Form\FormInterface;
|
use Symfony\Component\Form\FormInterface;
|
||||||
use Symfony\Component\HttpFoundation\RequestStack;
|
use Symfony\Component\HttpFoundation\RequestStack;
|
||||||
|
|
||||||
|
use Symfony\Component\PropertyAccess\PropertyAccessor;
|
||||||
|
use Symfony\Component\PropertyAccess\PropertyAccessorInterface;
|
||||||
|
use Symfony\Component\PropertyAccess\PropertyPath;
|
||||||
|
use Symfony\Component\PropertyAccess\PropertyPathInterface;
|
||||||
|
use Symfony\Contracts\Translation\TranslatorInterface;
|
||||||
use function array_merge;
|
use function array_merge;
|
||||||
use function count;
|
use function count;
|
||||||
|
|
||||||
class FilterOrderHelper
|
final class FilterOrderHelper
|
||||||
{
|
{
|
||||||
private array $checkboxes = [];
|
private array $checkboxes = [];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var array<string, array{label: string}>
|
||||||
|
*/
|
||||||
|
private array $singleCheckbox = [];
|
||||||
|
|
||||||
private array $dateRanges = [];
|
private array $dateRanges = [];
|
||||||
|
|
||||||
private FormFactoryInterface $formFactory;
|
public const FORM_NAME = 'f';
|
||||||
|
|
||||||
private ?string $formName = 'f';
|
|
||||||
|
|
||||||
private array $formOptions = [];
|
private array $formOptions = [];
|
||||||
|
|
||||||
private string $formType = FilterOrderType::class;
|
private string $formType = FilterOrderType::class;
|
||||||
|
|
||||||
private RequestStack $requestStack;
|
|
||||||
|
|
||||||
private ?array $searchBoxFields = null;
|
private ?array $searchBoxFields = null;
|
||||||
|
|
||||||
private ?array $submitted = null;
|
private ?array $submitted = null;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var array<string, array{label: string, choices: array, options: array}>
|
||||||
|
*/
|
||||||
|
private array $entityChoices = [];
|
||||||
|
|
||||||
|
|
||||||
public function __construct(
|
public function __construct(
|
||||||
FormFactoryInterface $formFactory,
|
private readonly FormFactoryInterface $formFactory,
|
||||||
RequestStack $requestStack
|
private readonly RequestStack $requestStack,
|
||||||
) {
|
) {
|
||||||
$this->formFactory = $formFactory;
|
|
||||||
$this->requestStack = $requestStack;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function addCheckbox(string $name, array $choices, ?array $default = [], ?array $trans = []): self
|
public function addSingleCheckbox(string $name, string $label): self
|
||||||
{
|
{
|
||||||
$missing = count($choices) - count($trans) - 1;
|
$this->singleCheckbox[$name] = ['label' => $label];
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param class-string $class
|
||||||
|
*/
|
||||||
|
public function addEntityChoice(string $name, string $class, string $label, array $choices, array $options = []): self
|
||||||
|
{
|
||||||
|
$this->entityChoices[$name] = ['label' => $label, 'class' => $class, 'choices' => $choices, 'options' => $options];
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getEntityChoices(): array
|
||||||
|
{
|
||||||
|
return $this->entityChoices;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function addCheckbox(string $name, array $choices, ?array $default = [], ?array $trans = [], array $options = []): self
|
||||||
|
{
|
||||||
|
if ([] === $trans) {
|
||||||
|
$trans = $choices;
|
||||||
|
}
|
||||||
|
|
||||||
$this->checkboxes[$name] = [
|
$this->checkboxes[$name] = [
|
||||||
'choices' => $choices, 'default' => $default,
|
'choices' => $choices,
|
||||||
'trans' => array_merge(
|
'default' => $default,
|
||||||
$trans,
|
'trans' => $trans,
|
||||||
0 < $missing ?
|
...$options,
|
||||||
array_fill(0, $missing, null) : []
|
|
||||||
),
|
|
||||||
];
|
];
|
||||||
|
|
||||||
return $this;
|
return $this;
|
||||||
@@ -73,7 +106,7 @@ class FilterOrderHelper
|
|||||||
public function buildForm(): FormInterface
|
public function buildForm(): FormInterface
|
||||||
{
|
{
|
||||||
return $this->formFactory
|
return $this->formFactory
|
||||||
->createNamed($this->formName, $this->formType, $this->getDefaultData(), array_merge([
|
->createNamed(self::FORM_NAME, $this->formType, $this->getDefaultData(), array_merge([
|
||||||
'helper' => $this,
|
'helper' => $this,
|
||||||
'method' => 'GET',
|
'method' => 'GET',
|
||||||
'csrf_protection' => false,
|
'csrf_protection' => false,
|
||||||
@@ -81,11 +114,36 @@ class FilterOrderHelper
|
|||||||
->handleRequest($this->requestStack->getCurrentRequest());
|
->handleRequest($this->requestStack->getCurrentRequest());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function hasCheckboxData(string $name): bool
|
||||||
|
{
|
||||||
|
return array_key_exists($name, $this->checkboxes);
|
||||||
|
}
|
||||||
|
|
||||||
public function getCheckboxData(string $name): array
|
public function getCheckboxData(string $name): array
|
||||||
{
|
{
|
||||||
return $this->getFormData()['checkboxes'][$name];
|
return $this->getFormData()['checkboxes'][$name];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function hasSingleCheckboxData(string $name): bool
|
||||||
|
{
|
||||||
|
return array_key_exists($name, $this->singleCheckbox);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getSingleCheckboxData(string $name): ?bool
|
||||||
|
{
|
||||||
|
return $this->getFormData()['single_checkboxes'][$name];
|
||||||
|
}
|
||||||
|
|
||||||
|
public function hasEntityChoice(string $name): bool
|
||||||
|
{
|
||||||
|
return array_key_exists($name, $this->entityChoices);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getEntityChoiceData($name): mixed
|
||||||
|
{
|
||||||
|
return $this->getFormData()['entity_choices'][$name];
|
||||||
|
}
|
||||||
|
|
||||||
public function getCheckboxes(): array
|
public function getCheckboxes(): array
|
||||||
{
|
{
|
||||||
return $this->checkboxes;
|
return $this->checkboxes;
|
||||||
@@ -97,7 +155,20 @@ class FilterOrderHelper
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return array<'to': DateTimeImmutable, 'from': DateTimeImmutable>
|
* @return array<string, array{label: string}>
|
||||||
|
*/
|
||||||
|
public function getSingleCheckbox(): array
|
||||||
|
{
|
||||||
|
return $this->singleCheckbox;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function hasDateRangeData(string $name): bool
|
||||||
|
{
|
||||||
|
return array_key_exists($name, $this->dateRanges);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array{to: ?DateTimeImmutable, from: ?DateTimeImmutable}
|
||||||
*/
|
*/
|
||||||
public function getDateRangeData(string $name): array
|
public function getDateRangeData(string $name): array
|
||||||
{
|
{
|
||||||
@@ -128,7 +199,12 @@ class FilterOrderHelper
|
|||||||
|
|
||||||
private function getDefaultData(): array
|
private function getDefaultData(): array
|
||||||
{
|
{
|
||||||
$r = [];
|
$r = [
|
||||||
|
'checkboxes' => [],
|
||||||
|
'dateRanges' => [],
|
||||||
|
'single_checkboxes' => [],
|
||||||
|
'entity_choices' => []
|
||||||
|
];
|
||||||
|
|
||||||
if ($this->hasSearchBox()) {
|
if ($this->hasSearchBox()) {
|
||||||
$r['q'] = '';
|
$r['q'] = '';
|
||||||
@@ -143,6 +219,14 @@ class FilterOrderHelper
|
|||||||
$r['dateRanges'][$name]['to'] = $defaults['to'];
|
$r['dateRanges'][$name]['to'] = $defaults['to'];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
foreach ($this->singleCheckbox as $name => $c) {
|
||||||
|
$r['single_checkboxes'][$name] = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($this->entityChoices as $name => $c) {
|
||||||
|
$r['entity_choices'][$name] = ($c['options']['multiple'] ?? true) ? [] : null;
|
||||||
|
}
|
||||||
|
|
||||||
return $r;
|
return $r;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -14,6 +14,8 @@ namespace Chill\MainBundle\Templating\Listing;
|
|||||||
use DateTimeImmutable;
|
use DateTimeImmutable;
|
||||||
use Symfony\Component\Form\FormFactoryInterface;
|
use Symfony\Component\Form\FormFactoryInterface;
|
||||||
use Symfony\Component\HttpFoundation\RequestStack;
|
use Symfony\Component\HttpFoundation\RequestStack;
|
||||||
|
use Symfony\Component\PropertyAccess\PropertyAccessorInterface;
|
||||||
|
use Symfony\Contracts\Translation\TranslatorInterface;
|
||||||
|
|
||||||
class FilterOrderHelperBuilder
|
class FilterOrderHelperBuilder
|
||||||
{
|
{
|
||||||
@@ -27,14 +29,31 @@ class FilterOrderHelperBuilder
|
|||||||
|
|
||||||
private ?array $searchBoxFields = null;
|
private ?array $searchBoxFields = null;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var array<string, array{label: string}>
|
||||||
|
*/
|
||||||
|
private array $singleCheckboxes = [];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var array<string, array{label: string, class: class-string, choices: array, options: array}>
|
||||||
|
*/
|
||||||
|
private array $entityChoices = [];
|
||||||
|
|
||||||
public function __construct(
|
public function __construct(
|
||||||
FormFactoryInterface $formFactory,
|
FormFactoryInterface $formFactory,
|
||||||
RequestStack $requestStack
|
RequestStack $requestStack,
|
||||||
) {
|
) {
|
||||||
$this->formFactory = $formFactory;
|
$this->formFactory = $formFactory;
|
||||||
$this->requestStack = $requestStack;
|
$this->requestStack = $requestStack;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function addSingleCheckbox(string $name, string $label): self
|
||||||
|
{
|
||||||
|
$this->singleCheckboxes[$name] = ['label' => $label];
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
public function addCheckbox(string $name, array $choices, ?array $default = [], ?array $trans = []): self
|
public function addCheckbox(string $name, array $choices, ?array $default = [], ?array $trans = []): self
|
||||||
{
|
{
|
||||||
$this->checkboxes[$name] = ['choices' => $choices, 'default' => $default, 'trans' => $trans];
|
$this->checkboxes[$name] = ['choices' => $choices, 'default' => $default, 'trans' => $trans];
|
||||||
@@ -42,6 +61,16 @@ class FilterOrderHelperBuilder
|
|||||||
return $this;
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param class-string $class
|
||||||
|
*/
|
||||||
|
public function addEntityChoice(string $name, string $label, string $class, array $choices, ?array $options = []): self
|
||||||
|
{
|
||||||
|
$this->entityChoices[$name] = ['label' => $label, 'class' => $class, 'choices' => $choices, 'options' => $options];
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
public function addDateRange(string $name, ?string $label = null, ?DateTimeImmutable $from = null, ?DateTimeImmutable $to = null): self
|
public function addDateRange(string $name, ?string $label = null, ?DateTimeImmutable $from = null, ?DateTimeImmutable $to = null): self
|
||||||
{
|
{
|
||||||
$this->dateRanges[$name] = ['from' => $from, 'to' => $to, 'label' => $label];
|
$this->dateRanges[$name] = ['from' => $from, 'to' => $to, 'label' => $label];
|
||||||
@@ -60,7 +89,7 @@ class FilterOrderHelperBuilder
|
|||||||
{
|
{
|
||||||
$helper = new FilterOrderHelper(
|
$helper = new FilterOrderHelper(
|
||||||
$this->formFactory,
|
$this->formFactory,
|
||||||
$this->requestStack
|
$this->requestStack,
|
||||||
);
|
);
|
||||||
|
|
||||||
$helper->setSearchBox($this->searchBoxFields);
|
$helper->setSearchBox($this->searchBoxFields);
|
||||||
@@ -75,6 +104,18 @@ class FilterOrderHelperBuilder
|
|||||||
$helper->addCheckbox($name, $choices, $default, $trans);
|
$helper->addCheckbox($name, $choices, $default, $trans);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
foreach (
|
||||||
|
$this->singleCheckboxes as $name => ['label' => $label]
|
||||||
|
) {
|
||||||
|
$helper->addSingleCheckbox($name, $label);
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (
|
||||||
|
$this->entityChoices as $name => ['label' => $label, 'class' => $class, 'choices' => $choices, 'options' => $options]
|
||||||
|
) {
|
||||||
|
$helper->addEntityChoice($name, $class, $label, $choices, $options);
|
||||||
|
}
|
||||||
|
|
||||||
foreach (
|
foreach (
|
||||||
$this->dateRanges as $name => [
|
$this->dateRanges as $name => [
|
||||||
'from' => $from,
|
'from' => $from,
|
||||||
|
@@ -13,6 +13,8 @@ namespace Chill\MainBundle\Templating\Listing;
|
|||||||
|
|
||||||
use Symfony\Component\Form\FormFactoryInterface;
|
use Symfony\Component\Form\FormFactoryInterface;
|
||||||
use Symfony\Component\HttpFoundation\RequestStack;
|
use Symfony\Component\HttpFoundation\RequestStack;
|
||||||
|
use Symfony\Component\PropertyAccess\PropertyAccessorInterface;
|
||||||
|
use Symfony\Contracts\Translation\TranslatorInterface;
|
||||||
|
|
||||||
class FilterOrderHelperFactory implements FilterOrderHelperFactoryInterface
|
class FilterOrderHelperFactory implements FilterOrderHelperFactoryInterface
|
||||||
{
|
{
|
||||||
@@ -22,7 +24,7 @@ class FilterOrderHelperFactory implements FilterOrderHelperFactoryInterface
|
|||||||
|
|
||||||
public function __construct(
|
public function __construct(
|
||||||
FormFactoryInterface $formFactory,
|
FormFactoryInterface $formFactory,
|
||||||
RequestStack $requestStack
|
RequestStack $requestStack,
|
||||||
) {
|
) {
|
||||||
$this->formFactory = $formFactory;
|
$this->formFactory = $formFactory;
|
||||||
$this->requestStack = $requestStack;
|
$this->requestStack = $requestStack;
|
||||||
|
@@ -0,0 +1,21 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace Chill\MainBundle\Templating\Listing;
|
||||||
|
|
||||||
|
enum FilterOrderPositionEnum: string
|
||||||
|
{
|
||||||
|
case SearchBox = 'search_box';
|
||||||
|
case Checkboxes = 'checkboxes';
|
||||||
|
case DateRange = 'date_range';
|
||||||
|
case EntityChoice = 'entity_choice';
|
||||||
|
case SingleCheckbox = 'single_checkbox';
|
||||||
|
}
|
@@ -11,13 +11,24 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
namespace Chill\MainBundle\Templating\Listing;
|
namespace Chill\MainBundle\Templating\Listing;
|
||||||
|
|
||||||
|
use Chill\MainBundle\Pagination\PaginatorFactory;
|
||||||
|
use Symfony\Component\HttpFoundation\RequestStack;
|
||||||
use Twig\Environment;
|
use Twig\Environment;
|
||||||
|
use Twig\Error\LoaderError;
|
||||||
|
use Twig\Error\RuntimeError;
|
||||||
|
use Twig\Error\SyntaxError;
|
||||||
use Twig\Extension\AbstractExtension;
|
use Twig\Extension\AbstractExtension;
|
||||||
use Twig\TwigFilter;
|
use Twig\TwigFilter;
|
||||||
|
|
||||||
class Templating extends AbstractExtension
|
class Templating extends AbstractExtension
|
||||||
{
|
{
|
||||||
public function getFilters()
|
public function __construct(
|
||||||
|
private readonly RequestStack $requestStack,
|
||||||
|
private readonly FilterOrderGetActiveFilterHelper $filterOrderGetActiveFilterHelper,
|
||||||
|
) {
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getFilters(): array
|
||||||
{
|
{
|
||||||
return [
|
return [
|
||||||
new TwigFilter('chill_render_filter_order_helper', [$this, 'renderFilterOrderHelper'], [
|
new TwigFilter('chill_render_filter_order_helper', [$this, 'renderFilterOrderHelper'], [
|
||||||
@@ -26,16 +37,42 @@ class Templating extends AbstractExtension
|
|||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @throws SyntaxError
|
||||||
|
* @throws RuntimeError
|
||||||
|
* @throws LoaderError
|
||||||
|
*/
|
||||||
public function renderFilterOrderHelper(
|
public function renderFilterOrderHelper(
|
||||||
Environment $environment,
|
Environment $environment,
|
||||||
FilterOrderHelper $helper,
|
FilterOrderHelper $helper,
|
||||||
?string $template = '@ChillMain/FilterOrder/base.html.twig',
|
?string $template = '@ChillMain/FilterOrder/base.html.twig',
|
||||||
?array $options = []
|
?array $options = []
|
||||||
) {
|
): string {
|
||||||
|
$otherParameters = [];
|
||||||
|
|
||||||
|
foreach ($this->requestStack->getCurrentRequest()->query->getIterator() as $key => $value) {
|
||||||
|
switch ($key) {
|
||||||
|
case FilterOrderHelper::FORM_NAME:
|
||||||
|
break;
|
||||||
|
|
||||||
|
case PaginatorFactory::DEFAULT_CURRENT_PAGE_KEY:
|
||||||
|
// when filtering, go back to page 1
|
||||||
|
$otherParameters[PaginatorFactory::DEFAULT_CURRENT_PAGE_KEY] = 1;
|
||||||
|
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
$otherParameters[$key] = $value;
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return $environment->render($template, [
|
return $environment->render($template, [
|
||||||
'helper' => $helper,
|
'helper' => $helper,
|
||||||
|
'active' => $this->filterOrderGetActiveFilterHelper->getActiveFilters($helper),
|
||||||
'form' => $helper->buildForm()->createView(),
|
'form' => $helper->buildForm()->createView(),
|
||||||
'options' => $options,
|
'options' => $options,
|
||||||
|
'otherParameters' => $otherParameters,
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -54,3 +54,12 @@ duration:
|
|||||||
few {# minutes}
|
few {# minutes}
|
||||||
other {# minutes}
|
other {# minutes}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
filter_order:
|
||||||
|
by_date:
|
||||||
|
From: Depuis le {from_date, date, long}
|
||||||
|
To: Jusqu'au {to_date, date, long}
|
||||||
|
By: Filtrer par
|
||||||
|
Search: Chercher dans la liste
|
||||||
|
By date: Filtrer par date
|
||||||
|
search_box: Filtrer par contenu
|
||||||
|
@@ -29,6 +29,7 @@ use Chill\PersonBundle\Entity\Household\PersonHouseholdAddress;
|
|||||||
use Chill\PersonBundle\Entity\Person\PersonCenterHistory;
|
use Chill\PersonBundle\Entity\Person\PersonCenterHistory;
|
||||||
use Chill\PersonBundle\Entity\SocialWork\SocialIssue;
|
use Chill\PersonBundle\Entity\SocialWork\SocialIssue;
|
||||||
use Chill\PersonBundle\Export\Declarations;
|
use Chill\PersonBundle\Export\Declarations;
|
||||||
|
use Chill\PersonBundle\Export\Helper\ListAccompanyingPeriodHelper;
|
||||||
use Chill\PersonBundle\Repository\PersonRepository;
|
use Chill\PersonBundle\Repository\PersonRepository;
|
||||||
use Chill\PersonBundle\Repository\SocialWork\SocialIssueRepository;
|
use Chill\PersonBundle\Repository\SocialWork\SocialIssueRepository;
|
||||||
use Chill\PersonBundle\Security\Authorization\PersonVoter;
|
use Chill\PersonBundle\Security\Authorization\PersonVoter;
|
||||||
@@ -45,95 +46,13 @@ use Symfony\Component\Form\FormBuilderInterface;
|
|||||||
use Symfony\Contracts\Translation\TranslatorInterface;
|
use Symfony\Contracts\Translation\TranslatorInterface;
|
||||||
use function strlen;
|
use function strlen;
|
||||||
|
|
||||||
class ListAccompanyingPeriod implements ListInterface, GroupedExportInterface
|
final readonly class ListAccompanyingPeriod implements ListInterface, GroupedExportInterface
|
||||||
{
|
{
|
||||||
private const FIELDS = [
|
|
||||||
'id',
|
|
||||||
'step',
|
|
||||||
'stepSince',
|
|
||||||
'openingDate',
|
|
||||||
'closingDate',
|
|
||||||
'referrer',
|
|
||||||
'referrerSince',
|
|
||||||
'administrativeLocation',
|
|
||||||
'locationIsPerson',
|
|
||||||
'locationIsTemp',
|
|
||||||
'locationPersonName',
|
|
||||||
'locationPersonId',
|
|
||||||
'origin',
|
|
||||||
'closingMotive',
|
|
||||||
'confidential',
|
|
||||||
'emergency',
|
|
||||||
'intensity',
|
|
||||||
'job',
|
|
||||||
'isRequestorPerson',
|
|
||||||
'isRequestorThirdParty',
|
|
||||||
'requestorPerson',
|
|
||||||
'requestorPersonId',
|
|
||||||
'requestorThirdParty',
|
|
||||||
'requestorThirdPartyId',
|
|
||||||
'scopes',
|
|
||||||
'socialIssues',
|
|
||||||
'createdAt',
|
|
||||||
'createdBy',
|
|
||||||
'updatedAt',
|
|
||||||
'updatedBy',
|
|
||||||
];
|
|
||||||
|
|
||||||
private ExportAddressHelper $addressHelper;
|
|
||||||
|
|
||||||
private DateTimeHelper $dateTimeHelper;
|
|
||||||
|
|
||||||
private EntityManagerInterface $entityManager;
|
|
||||||
|
|
||||||
private PersonRenderInterface $personRender;
|
|
||||||
|
|
||||||
private PersonRepository $personRepository;
|
|
||||||
|
|
||||||
private RollingDateConverterInterface $rollingDateConverter;
|
|
||||||
|
|
||||||
private SocialIssueRender $socialIssueRender;
|
|
||||||
|
|
||||||
private SocialIssueRepository $socialIssueRepository;
|
|
||||||
|
|
||||||
private ThirdPartyRender $thirdPartyRender;
|
|
||||||
|
|
||||||
private ThirdPartyRepository $thirdPartyRepository;
|
|
||||||
|
|
||||||
private TranslatableStringHelperInterface $translatableStringHelper;
|
|
||||||
|
|
||||||
private TranslatorInterface $translator;
|
|
||||||
|
|
||||||
private UserHelper $userHelper;
|
|
||||||
|
|
||||||
public function __construct(
|
public function __construct(
|
||||||
ExportAddressHelper $addressHelper,
|
private EntityManagerInterface $entityManager,
|
||||||
DateTimeHelper $dateTimeHelper,
|
private RollingDateConverterInterface $rollingDateConverter,
|
||||||
EntityManagerInterface $entityManager,
|
private ListAccompanyingPeriodHelper $listAccompanyingPeriodHelper,
|
||||||
PersonRenderInterface $personRender,
|
|
||||||
PersonRepository $personRepository,
|
|
||||||
ThirdPartyRepository $thirdPartyRepository,
|
|
||||||
ThirdPartyRender $thirdPartyRender,
|
|
||||||
SocialIssueRepository $socialIssueRepository,
|
|
||||||
SocialIssueRender $socialIssueRender,
|
|
||||||
TranslatableStringHelperInterface $translatableStringHelper,
|
|
||||||
TranslatorInterface $translator,
|
|
||||||
RollingDateConverterInterface $rollingDateConverter,
|
|
||||||
UserHelper $userHelper
|
|
||||||
) {
|
) {
|
||||||
$this->addressHelper = $addressHelper;
|
|
||||||
$this->dateTimeHelper = $dateTimeHelper;
|
|
||||||
$this->entityManager = $entityManager;
|
|
||||||
$this->personRender = $personRender;
|
|
||||||
$this->personRepository = $personRepository;
|
|
||||||
$this->socialIssueRender = $socialIssueRender;
|
|
||||||
$this->socialIssueRepository = $socialIssueRepository;
|
|
||||||
$this->thirdPartyRender = $thirdPartyRender;
|
|
||||||
$this->thirdPartyRepository = $thirdPartyRepository;
|
|
||||||
$this->translatableStringHelper = $translatableStringHelper;
|
|
||||||
$this->translator = $translator;
|
|
||||||
$this->rollingDateConverter = $rollingDateConverter;
|
|
||||||
$this->userHelper = $userHelper;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function buildForm(FormBuilderInterface $builder)
|
public function buildForm(FormBuilderInterface $builder)
|
||||||
@@ -169,141 +88,12 @@ class ListAccompanyingPeriod implements ListInterface, GroupedExportInterface
|
|||||||
|
|
||||||
public function getLabels($key, array $values, $data)
|
public function getLabels($key, array $values, $data)
|
||||||
{
|
{
|
||||||
if (substr($key, 0, strlen('address_fields')) === 'address_fields') {
|
return $this->listAccompanyingPeriodHelper->getLabels($key, $values, $data);
|
||||||
return $this->addressHelper->getLabel($key, $values, $data, 'address_fields');
|
|
||||||
}
|
|
||||||
|
|
||||||
switch ($key) {
|
|
||||||
case 'stepSince':
|
|
||||||
case 'openingDate':
|
|
||||||
case 'closingDate':
|
|
||||||
case 'referrerSince':
|
|
||||||
case 'createdAt':
|
|
||||||
case 'updatedAt':
|
|
||||||
return $this->dateTimeHelper->getLabel('export.list.acp.' . $key);
|
|
||||||
|
|
||||||
case 'origin':
|
|
||||||
case 'closingMotive':
|
|
||||||
case 'job':
|
|
||||||
return function ($value) use ($key) {
|
|
||||||
if ('_header' === $value) {
|
|
||||||
return 'export.list.acp.' . $key;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (null === $value) {
|
|
||||||
return '';
|
|
||||||
}
|
|
||||||
|
|
||||||
return $this->translatableStringHelper->localize(json_decode($value, true, 512, JSON_THROW_ON_ERROR));
|
|
||||||
};
|
|
||||||
|
|
||||||
case 'locationPersonName':
|
|
||||||
case 'requestorPerson':
|
|
||||||
return function ($value) use ($key) {
|
|
||||||
if ('_header' === $value) {
|
|
||||||
return 'export.list.acp.' . $key;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (null === $value || null === $person = $this->personRepository->find($value)) {
|
|
||||||
return '';
|
|
||||||
}
|
|
||||||
|
|
||||||
return $this->personRender->renderString($person, []);
|
|
||||||
};
|
|
||||||
|
|
||||||
case 'requestorThirdParty':
|
|
||||||
return function ($value) use ($key) {
|
|
||||||
if ('_header' === $value) {
|
|
||||||
return 'export.list.acp.' . $key;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (null === $value || null === $thirdparty = $this->thirdPartyRepository->find($value)) {
|
|
||||||
return '';
|
|
||||||
}
|
|
||||||
|
|
||||||
return $this->thirdPartyRender->renderString($thirdparty, []);
|
|
||||||
};
|
|
||||||
|
|
||||||
case 'scopes':
|
|
||||||
return function ($value) use ($key) {
|
|
||||||
if ('_header' === $value) {
|
|
||||||
return 'export.list.acp.' . $key;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (null === $value) {
|
|
||||||
return '';
|
|
||||||
}
|
|
||||||
|
|
||||||
return implode(
|
|
||||||
'|',
|
|
||||||
array_map(
|
|
||||||
fn ($s) => $this->translatableStringHelper->localize($s),
|
|
||||||
json_decode($value, true, 512, JSON_THROW_ON_ERROR)
|
|
||||||
)
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
case 'socialIssues':
|
|
||||||
return function ($value) use ($key) {
|
|
||||||
if ('_header' === $value) {
|
|
||||||
return 'export.list.acp.' . $key;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (null === $value) {
|
|
||||||
return '';
|
|
||||||
}
|
|
||||||
|
|
||||||
return implode(
|
|
||||||
'|',
|
|
||||||
array_map(
|
|
||||||
fn ($s) => $this->socialIssueRender->renderString($this->socialIssueRepository->find($s), []),
|
|
||||||
json_decode($value, true, 512, JSON_THROW_ON_ERROR)
|
|
||||||
)
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
case 'step':
|
|
||||||
return fn ($value) => match ($value) {
|
|
||||||
'_header' => 'export.list.acp.step',
|
|
||||||
null => '',
|
|
||||||
AccompanyingPeriod::STEP_DRAFT => $this->translator->trans('course.draft'),
|
|
||||||
AccompanyingPeriod::STEP_CONFIRMED => $this->translator->trans('course.confirmed'),
|
|
||||||
AccompanyingPeriod::STEP_CLOSED => $this->translator->trans('course.closed'),
|
|
||||||
AccompanyingPeriod::STEP_CONFIRMED_INACTIVE_SHORT => $this->translator->trans('course.inactive_short'),
|
|
||||||
AccompanyingPeriod::STEP_CONFIRMED_INACTIVE_LONG => $this->translator->trans('course.inactive_long'),
|
|
||||||
default => $value,
|
|
||||||
};
|
|
||||||
|
|
||||||
case 'intensity':
|
|
||||||
return fn ($value) => match ($value) {
|
|
||||||
'_header' => 'export.list.acp.intensity',
|
|
||||||
null => '',
|
|
||||||
AccompanyingPeriod::INTENSITY_OCCASIONAL => $this->translator->trans('occasional'),
|
|
||||||
AccompanyingPeriod::INTENSITY_REGULAR => $this->translator->trans('regular'),
|
|
||||||
default => $value,
|
|
||||||
};
|
|
||||||
|
|
||||||
default:
|
|
||||||
return static function ($value) use ($key) {
|
|
||||||
if ('_header' === $value) {
|
|
||||||
return 'export.list.acp.' . $key;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (null === $value) {
|
|
||||||
return '';
|
|
||||||
}
|
|
||||||
|
|
||||||
return $value;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getQueryKeys($data)
|
public function getQueryKeys($data)
|
||||||
{
|
{
|
||||||
return array_merge(
|
return $this->listAccompanyingPeriodHelper->getQueryKeys($data);
|
||||||
self::FIELDS,
|
|
||||||
$this->addressHelper->getKeys(ExportAddressHelper::F_ALL, 'address_fields')
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getResult($query, $data)
|
public function getResult($query, $data)
|
||||||
@@ -341,7 +131,12 @@ class ListAccompanyingPeriod implements ListInterface, GroupedExportInterface
|
|||||||
->setParameter('list_acp_step', AccompanyingPeriod::STEP_DRAFT)
|
->setParameter('list_acp_step', AccompanyingPeriod::STEP_DRAFT)
|
||||||
->setParameter('authorized_centers', $centers);
|
->setParameter('authorized_centers', $centers);
|
||||||
|
|
||||||
$this->addSelectClauses($qb, $this->rollingDateConverter->convert($data['calc_date']));
|
$this->listAccompanyingPeriodHelper->addSelectClauses($qb, $this->rollingDateConverter->convert($data['calc_date']));
|
||||||
|
|
||||||
|
$qb
|
||||||
|
->addOrderBy('acp.openingDate')
|
||||||
|
->addOrderBy('acp.closingDate')
|
||||||
|
->addOrderBy('acp.id');
|
||||||
|
|
||||||
return $qb;
|
return $qb;
|
||||||
}
|
}
|
||||||
@@ -357,91 +152,4 @@ class ListAccompanyingPeriod implements ListInterface, GroupedExportInterface
|
|||||||
Declarations::ACP_TYPE,
|
Declarations::ACP_TYPE,
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
private function addSelectClauses(QueryBuilder $qb, DateTimeImmutable $calcDate): void
|
|
||||||
{
|
|
||||||
// add the regular fields
|
|
||||||
foreach (['id', 'openingDate', 'closingDate', 'confidential', 'emergency', 'intensity', 'createdAt', 'updatedAt'] as $field) {
|
|
||||||
$qb->addSelect(sprintf('acp.%s AS %s', $field, $field));
|
|
||||||
}
|
|
||||||
|
|
||||||
// add the field which are simple association
|
|
||||||
foreach (['origin' => 'label', 'closingMotive' => 'name', 'job' => 'label', 'createdBy' => 'label', 'updatedBy' => 'label', 'administrativeLocation' => 'name'] as $entity => $field) {
|
|
||||||
$qb
|
|
||||||
->leftJoin(sprintf('acp.%s', $entity), "{$entity}_t")
|
|
||||||
->addSelect(sprintf('%s_t.%s AS %s', $entity, $field, $entity));
|
|
||||||
}
|
|
||||||
|
|
||||||
// step at date
|
|
||||||
$qb
|
|
||||||
->addSelect('stepHistory.step AS step')
|
|
||||||
->addSelect('stepHistory.startDate AS stepSince')
|
|
||||||
->leftJoin('acp.stepHistories', 'stepHistory')
|
|
||||||
->andWhere(
|
|
||||||
$qb->expr()->andX(
|
|
||||||
$qb->expr()->lte('stepHistory.startDate', ':calcDate'),
|
|
||||||
$qb->expr()->orX($qb->expr()->isNull('stepHistory.endDate'), $qb->expr()->gt('stepHistory.endDate', ':calcDate'))
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
// referree at date
|
|
||||||
$qb
|
|
||||||
->addSelect('referrer_t.label AS referrer')
|
|
||||||
->addSelect('userHistory.startDate AS referrerSince')
|
|
||||||
->leftJoin('acp.userHistories', 'userHistory')
|
|
||||||
->leftJoin('userHistory.user', 'referrer_t')
|
|
||||||
->andWhere(
|
|
||||||
$qb->expr()->orX(
|
|
||||||
$qb->expr()->isNull('userHistory'),
|
|
||||||
$qb->expr()->andX(
|
|
||||||
$qb->expr()->lte('userHistory.startDate', ':calcDate'),
|
|
||||||
$qb->expr()->orX($qb->expr()->isNull('userHistory.endDate'), $qb->expr()->gt('userHistory.endDate', ':calcDate'))
|
|
||||||
)
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
// location of the acp
|
|
||||||
$qb
|
|
||||||
->addSelect('CASE WHEN locationHistory.personLocation IS NOT NULL THEN 1 ELSE 0 END AS locationIsPerson')
|
|
||||||
->addSelect('CASE WHEN locationHistory.personLocation IS NOT NULL THEN 0 ELSE 1 END AS locationIsTemp')
|
|
||||||
->addSelect('IDENTITY(locationHistory.personLocation) AS locationPersonName')
|
|
||||||
->addSelect('IDENTITY(locationHistory.personLocation) AS locationPersonId')
|
|
||||||
->leftJoin('acp.locationHistories', 'locationHistory')
|
|
||||||
->andWhere(
|
|
||||||
$qb->expr()->orX(
|
|
||||||
$qb->expr()->isNull('locationHistory'),
|
|
||||||
$qb->expr()->andX(
|
|
||||||
$qb->expr()->lte('locationHistory.startDate', ':calcDate'),
|
|
||||||
$qb->expr()->orX($qb->expr()->isNull('locationHistory.endDate'), $qb->expr()->gt('locationHistory.endDate', ':calcDate'))
|
|
||||||
)
|
|
||||||
)
|
|
||||||
)
|
|
||||||
->leftJoin(
|
|
||||||
PersonHouseholdAddress::class,
|
|
||||||
'personAddress',
|
|
||||||
Join::WITH,
|
|
||||||
'locationHistory.personLocation = personAddress.person AND (personAddress.validFrom <= :calcDate AND (personAddress.validTo IS NULL OR personAddress.validTo > :calcDate))'
|
|
||||||
)
|
|
||||||
->leftJoin(Address::class, 'acp_address', Join::WITH, 'COALESCE(IDENTITY(locationHistory.addressLocation), IDENTITY(personAddress.address)) = acp_address.id');
|
|
||||||
|
|
||||||
$this->addressHelper->addSelectClauses(ExportAddressHelper::F_ALL, $qb, 'acp_address', 'address_fields');
|
|
||||||
|
|
||||||
// requestor
|
|
||||||
$qb
|
|
||||||
->addSelect('CASE WHEN acp.requestorPerson IS NULL THEN 1 ELSE 0 END AS isRequestorPerson')
|
|
||||||
->addSelect('CASE WHEN acp.requestorPerson IS NULL THEN 0 ELSE 1 END AS isRequestorThirdParty')
|
|
||||||
->addSelect('IDENTITY(acp.requestorPerson) AS requestorPersonId')
|
|
||||||
->addSelect('IDENTITY(acp.requestorThirdParty) AS requestorThirdPartyId')
|
|
||||||
->addSelect('IDENTITY(acp.requestorPerson) AS requestorPerson')
|
|
||||||
->addSelect('IDENTITY(acp.requestorThirdParty) AS requestorThirdParty');
|
|
||||||
|
|
||||||
$qb
|
|
||||||
// scopes
|
|
||||||
->addSelect('(SELECT AGGREGATE(scope.name) FROM ' . Scope::class . ' scope WHERE scope MEMBER OF acp.scopes) AS scopes')
|
|
||||||
// social issues
|
|
||||||
->addSelect('(SELECT AGGREGATE(socialIssue.id) FROM ' . SocialIssue::class . ' socialIssue WHERE socialIssue MEMBER OF acp.socialIssues) AS socialIssues');
|
|
||||||
|
|
||||||
// add parameter
|
|
||||||
$qb->setParameter('calcDate', $calcDate);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@@ -35,7 +35,12 @@ use function count;
|
|||||||
use function in_array;
|
use function in_array;
|
||||||
use function strlen;
|
use function strlen;
|
||||||
|
|
||||||
class ListPersonWithAccompanyingPeriod implements ExportElementValidatedInterface, ListInterface, GroupedExportInterface
|
/**
|
||||||
|
* List the persons, having an accompanying period.
|
||||||
|
*
|
||||||
|
* Details of the accompanying period are not included
|
||||||
|
*/
|
||||||
|
class ListPersonHavingAccompanyingPeriod implements ExportElementValidatedInterface, ListInterface, GroupedExportInterface
|
||||||
{
|
{
|
||||||
private ExportAddressHelper $addressHelper;
|
private ExportAddressHelper $addressHelper;
|
||||||
|
|
||||||
@@ -185,6 +190,11 @@ class ListPersonWithAccompanyingPeriod implements ExportElementValidatedInterfac
|
|||||||
|
|
||||||
$this->listPersonHelper->addSelect($qb, $fields, $data['address_date']);
|
$this->listPersonHelper->addSelect($qb, $fields, $data['address_date']);
|
||||||
|
|
||||||
|
$qb
|
||||||
|
->addOrderBy('person.lastName')
|
||||||
|
->addOrderBy('person.firstName')
|
||||||
|
->addOrderBy('person.id');
|
||||||
|
|
||||||
return $qb;
|
return $qb;
|
||||||
}
|
}
|
||||||
|
|
@@ -0,0 +1,155 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace Chill\PersonBundle\Export\Export;
|
||||||
|
|
||||||
|
use Chill\MainBundle\Export\ExportElementValidatedInterface;
|
||||||
|
use Chill\MainBundle\Export\FormatterInterface;
|
||||||
|
use Chill\MainBundle\Export\GroupedExportInterface;
|
||||||
|
use Chill\MainBundle\Export\Helper\ExportAddressHelper;
|
||||||
|
use Chill\MainBundle\Export\ListInterface;
|
||||||
|
use Chill\MainBundle\Form\Type\ChillDateType;
|
||||||
|
use Chill\MainBundle\Form\Type\PickRollingDateType;
|
||||||
|
use Chill\MainBundle\Service\RollingDate\RollingDate;
|
||||||
|
use Chill\MainBundle\Service\RollingDate\RollingDateConverterInterface;
|
||||||
|
use Chill\PersonBundle\Entity\AccompanyingPeriod;
|
||||||
|
use Chill\PersonBundle\Entity\Person;
|
||||||
|
use Chill\PersonBundle\Entity\Person\PersonCenterHistory;
|
||||||
|
use Chill\PersonBundle\Export\Declarations;
|
||||||
|
use Chill\PersonBundle\Export\Helper\ListAccompanyingPeriodHelper;
|
||||||
|
use Chill\PersonBundle\Export\Helper\ListPersonHelper;
|
||||||
|
use Chill\PersonBundle\Security\Authorization\PersonVoter;
|
||||||
|
use DateTimeImmutable;
|
||||||
|
use Doctrine\ORM\AbstractQuery;
|
||||||
|
use Doctrine\ORM\EntityManagerInterface;
|
||||||
|
use Symfony\Component\Form\Extension\Core\Type\ChoiceType;
|
||||||
|
use Symfony\Component\Form\FormBuilderInterface;
|
||||||
|
use Symfony\Component\Validator\Constraints\Callback;
|
||||||
|
use Symfony\Component\Validator\Context\ExecutionContextInterface;
|
||||||
|
use function array_key_exists;
|
||||||
|
use function count;
|
||||||
|
use function in_array;
|
||||||
|
use function strlen;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* List the persons having an accompanying period, with the accompanying period details
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
final readonly class ListPersonWithAccompanyingPeriodDetails implements ListInterface, GroupedExportInterface
|
||||||
|
{
|
||||||
|
public function __construct(
|
||||||
|
private ListPersonHelper $listPersonHelper,
|
||||||
|
private ListAccompanyingPeriodHelper $listAccompanyingPeriodHelper,
|
||||||
|
private EntityManagerInterface $entityManager,
|
||||||
|
private RollingDateConverterInterface $rollingDateConverter,
|
||||||
|
) {
|
||||||
|
}
|
||||||
|
|
||||||
|
public function buildForm(FormBuilderInterface $builder)
|
||||||
|
{
|
||||||
|
$builder->add('address_date', PickRollingDateType::class, [
|
||||||
|
'label' => 'Data valid at this date',
|
||||||
|
'help' => 'Data regarding center, addresses, and so on will be computed at this date',
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
public function getFormDefaultData(): array
|
||||||
|
{
|
||||||
|
return ['address_date' => new RollingDate(RollingDate::T_TODAY)];
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getAllowedFormattersTypes()
|
||||||
|
{
|
||||||
|
return [FormatterInterface::TYPE_LIST];
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getDescription()
|
||||||
|
{
|
||||||
|
return 'export.list.person_with_acp.Create a list of people having an accompaying periods with details of period, according to various filters.';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getGroup(): string
|
||||||
|
{
|
||||||
|
return 'Exports of persons';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getLabels($key, array $values, $data)
|
||||||
|
{
|
||||||
|
if (in_array($key, $this->listPersonHelper->getAllKeys(), true)) {
|
||||||
|
return $this->listPersonHelper->getLabels($key, $values, $data);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->listAccompanyingPeriodHelper->getLabels($key, $values, $data);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getQueryKeys($data)
|
||||||
|
{
|
||||||
|
return array_merge(
|
||||||
|
$this->listPersonHelper->getAllKeys(),
|
||||||
|
$this->listAccompanyingPeriodHelper->getQueryKeys($data),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getResult($query, $data)
|
||||||
|
{
|
||||||
|
return $query->getQuery()->getResult(AbstractQuery::HYDRATE_SCALAR);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getTitle()
|
||||||
|
{
|
||||||
|
return 'export.list.person_with_acp.List peoples having an accompanying period with period details';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getType()
|
||||||
|
{
|
||||||
|
return Declarations::PERSON_TYPE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* param array{fields: string[], address_date: DateTimeImmutable} $data.
|
||||||
|
*/
|
||||||
|
public function initiateQuery(array $requiredModifiers, array $acl, array $data = [])
|
||||||
|
{
|
||||||
|
$centers = array_map(static fn ($el) => $el['center'], $acl);
|
||||||
|
|
||||||
|
$qb = $this->entityManager->createQueryBuilder();
|
||||||
|
|
||||||
|
$qb->from(Person::class, 'person')
|
||||||
|
->join('person.accompanyingPeriodParticipations', 'acppart')
|
||||||
|
->join('acppart.accompanyingPeriod', 'acp')
|
||||||
|
->andWhere($qb->expr()->neq('acp.step', "'" . AccompanyingPeriod::STEP_DRAFT . "'"))
|
||||||
|
->andWhere(
|
||||||
|
$qb->expr()->exists(
|
||||||
|
'SELECT 1 FROM ' . PersonCenterHistory::class . ' pch WHERE pch.person = person.id AND pch.center IN (:authorized_centers)'
|
||||||
|
)
|
||||||
|
)->setParameter('authorized_centers', $centers);
|
||||||
|
|
||||||
|
$this->listPersonHelper->addSelect($qb, ListPersonHelper::FIELDS, $this->rollingDateConverter->convert($data['address_date']));
|
||||||
|
$this->listAccompanyingPeriodHelper->addSelectClauses($qb, $this->rollingDateConverter->convert($data['address_date']));
|
||||||
|
|
||||||
|
$qb
|
||||||
|
->addOrderBy('person.lastName')
|
||||||
|
->addOrderBy('person.firstName')
|
||||||
|
->addOrderBy('person.id')
|
||||||
|
->addOrderBy('acp.id');
|
||||||
|
|
||||||
|
return $qb;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function requiredRole(): string
|
||||||
|
{
|
||||||
|
return PersonVoter::LISTS;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function supportsModifiers()
|
||||||
|
{
|
||||||
|
return [Declarations::PERSON_TYPE, Declarations::PERSON_IMPLIED_IN, Declarations::ACP_TYPE];
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,317 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace Chill\PersonBundle\Export\Helper;
|
||||||
|
|
||||||
|
use Chill\MainBundle\Entity\Address;
|
||||||
|
use Chill\MainBundle\Entity\Scope;
|
||||||
|
use Chill\MainBundle\Export\Helper\DateTimeHelper;
|
||||||
|
use Chill\MainBundle\Export\Helper\ExportAddressHelper;
|
||||||
|
use Chill\MainBundle\Templating\TranslatableStringHelperInterface;
|
||||||
|
use Chill\PersonBundle\Entity\AccompanyingPeriod;
|
||||||
|
use Chill\PersonBundle\Entity\Household\PersonHouseholdAddress;
|
||||||
|
use Chill\PersonBundle\Entity\SocialWork\SocialIssue;
|
||||||
|
use Chill\PersonBundle\Repository\PersonRepository;
|
||||||
|
use Chill\PersonBundle\Repository\SocialWork\SocialIssueRepository;
|
||||||
|
use Chill\PersonBundle\Templating\Entity\PersonRenderInterface;
|
||||||
|
use Chill\PersonBundle\Templating\Entity\SocialIssueRender;
|
||||||
|
use Chill\ThirdPartyBundle\Repository\ThirdPartyRepository;
|
||||||
|
use Chill\ThirdPartyBundle\Templating\Entity\ThirdPartyRender;
|
||||||
|
use Doctrine\ORM\Query\Expr\Join;
|
||||||
|
use Doctrine\ORM\QueryBuilder;
|
||||||
|
use Symfony\Contracts\Translation\TranslatorInterface;
|
||||||
|
|
||||||
|
final readonly class ListAccompanyingPeriodHelper
|
||||||
|
{
|
||||||
|
public const FIELDS = [
|
||||||
|
'acpId',
|
||||||
|
'step',
|
||||||
|
'stepSince',
|
||||||
|
'openingDate',
|
||||||
|
'closingDate',
|
||||||
|
'referrer',
|
||||||
|
'referrerSince',
|
||||||
|
'administrativeLocation',
|
||||||
|
'locationIsPerson',
|
||||||
|
'locationIsTemp',
|
||||||
|
'locationPersonName',
|
||||||
|
'locationPersonId',
|
||||||
|
'origin',
|
||||||
|
'closingMotive',
|
||||||
|
'confidential',
|
||||||
|
'emergency',
|
||||||
|
'intensity',
|
||||||
|
'job',
|
||||||
|
'isRequestorPerson',
|
||||||
|
'isRequestorThirdParty',
|
||||||
|
'requestorPerson',
|
||||||
|
'requestorPersonId',
|
||||||
|
'requestorThirdParty',
|
||||||
|
'requestorThirdPartyId',
|
||||||
|
'scopes',
|
||||||
|
'socialIssues',
|
||||||
|
'acpCreatedAt',
|
||||||
|
'acpCreatedBy',
|
||||||
|
'acpUpdatedAt',
|
||||||
|
'acpUpdatedBy',
|
||||||
|
];
|
||||||
|
|
||||||
|
public function __construct(
|
||||||
|
private ExportAddressHelper $addressHelper,
|
||||||
|
private DateTimeHelper $dateTimeHelper,
|
||||||
|
private PersonRenderInterface $personRender,
|
||||||
|
private PersonRepository $personRepository,
|
||||||
|
private ThirdPartyRepository $thirdPartyRepository,
|
||||||
|
private ThirdPartyRender $thirdPartyRender,
|
||||||
|
private SocialIssueRepository $socialIssueRepository,
|
||||||
|
private SocialIssueRender $socialIssueRender,
|
||||||
|
private TranslatableStringHelperInterface $translatableStringHelper,
|
||||||
|
private TranslatorInterface $translator,
|
||||||
|
) {
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getQueryKeys($data)
|
||||||
|
{
|
||||||
|
return array_merge(
|
||||||
|
ListAccompanyingPeriodHelper::FIELDS,
|
||||||
|
$this->addressHelper->getKeys(ExportAddressHelper::F_ALL, 'acp_address_fields')
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getLabels($key, array $values, $data)
|
||||||
|
{
|
||||||
|
if (str_starts_with($key, 'acp_address_fields')) {
|
||||||
|
return $this->addressHelper->getLabel($key, $values, $data, 'acp_address_fields');
|
||||||
|
}
|
||||||
|
|
||||||
|
switch ($key) {
|
||||||
|
case 'stepSince':
|
||||||
|
case 'openingDate':
|
||||||
|
case 'closingDate':
|
||||||
|
case 'referrerSince':
|
||||||
|
case 'acpCreatedAt':
|
||||||
|
case 'acpUpdatedAt':
|
||||||
|
return $this->dateTimeHelper->getLabel('export.list.acp.' . $key);
|
||||||
|
|
||||||
|
case 'origin':
|
||||||
|
case 'closingMotive':
|
||||||
|
case 'job':
|
||||||
|
return function ($value) use ($key) {
|
||||||
|
if ('_header' === $value) {
|
||||||
|
return 'export.list.acp.' . $key;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (null === $value) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->translatableStringHelper->localize(json_decode($value, true, 512, JSON_THROW_ON_ERROR));
|
||||||
|
};
|
||||||
|
|
||||||
|
case 'locationPersonName':
|
||||||
|
case 'requestorPerson':
|
||||||
|
return function ($value) use ($key) {
|
||||||
|
if ('_header' === $value) {
|
||||||
|
return 'export.list.acp.' . $key;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (null === $value || null === $person = $this->personRepository->find($value)) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->personRender->renderString($person, []);
|
||||||
|
};
|
||||||
|
|
||||||
|
case 'requestorThirdParty':
|
||||||
|
return function ($value) use ($key) {
|
||||||
|
if ('_header' === $value) {
|
||||||
|
return 'export.list.acp.' . $key;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (null === $value || null === $thirdparty = $this->thirdPartyRepository->find($value)) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->thirdPartyRender->renderString($thirdparty, []);
|
||||||
|
};
|
||||||
|
|
||||||
|
case 'scopes':
|
||||||
|
return function ($value) use ($key) {
|
||||||
|
if ('_header' === $value) {
|
||||||
|
return 'export.list.acp.' . $key;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (null === $value) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
return implode(
|
||||||
|
'|',
|
||||||
|
array_map(
|
||||||
|
fn ($s) => $this->translatableStringHelper->localize($s),
|
||||||
|
json_decode($value, true, 512, JSON_THROW_ON_ERROR)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
case 'socialIssues':
|
||||||
|
return function ($value) use ($key) {
|
||||||
|
if ('_header' === $value) {
|
||||||
|
return 'export.list.acp.' . $key;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (null === $value) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
return implode(
|
||||||
|
'|',
|
||||||
|
array_map(
|
||||||
|
fn ($s) => $this->socialIssueRender->renderString($this->socialIssueRepository->find($s), []),
|
||||||
|
json_decode($value, true, 512, JSON_THROW_ON_ERROR)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
case 'step':
|
||||||
|
return fn ($value) => match ($value) {
|
||||||
|
'_header' => 'export.list.acp.step',
|
||||||
|
null => '',
|
||||||
|
AccompanyingPeriod::STEP_DRAFT => $this->translator->trans('course.draft'),
|
||||||
|
AccompanyingPeriod::STEP_CONFIRMED => $this->translator->trans('course.confirmed'),
|
||||||
|
AccompanyingPeriod::STEP_CLOSED => $this->translator->trans('course.closed'),
|
||||||
|
AccompanyingPeriod::STEP_CONFIRMED_INACTIVE_SHORT => $this->translator->trans('course.inactive_short'),
|
||||||
|
AccompanyingPeriod::STEP_CONFIRMED_INACTIVE_LONG => $this->translator->trans('course.inactive_long'),
|
||||||
|
default => $value,
|
||||||
|
};
|
||||||
|
|
||||||
|
case 'intensity':
|
||||||
|
return fn ($value) => match ($value) {
|
||||||
|
'_header' => 'export.list.acp.intensity',
|
||||||
|
null => '',
|
||||||
|
AccompanyingPeriod::INTENSITY_OCCASIONAL => $this->translator->trans('occasional'),
|
||||||
|
AccompanyingPeriod::INTENSITY_REGULAR => $this->translator->trans('regular'),
|
||||||
|
default => $value,
|
||||||
|
};
|
||||||
|
|
||||||
|
default:
|
||||||
|
return static function ($value) use ($key) {
|
||||||
|
if ('_header' === $value) {
|
||||||
|
return 'export.list.acp.' . $key;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (null === $value) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
return $value;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function addSelectClauses(QueryBuilder $qb, \DateTimeImmutable $calcDate): void
|
||||||
|
{
|
||||||
|
$qb->addSelect('acp.id AS acpId');
|
||||||
|
$qb->addSelect('acp.createdAt AS acpCreatedAt');
|
||||||
|
$qb->addSelect('acp.updatedAt AS acpUpdatedAt');
|
||||||
|
|
||||||
|
// add the regular fields
|
||||||
|
foreach (['openingDate', 'closingDate', 'confidential', 'emergency', 'intensity'] as $field) {
|
||||||
|
$qb->addSelect(sprintf('acp.%s AS %s', $field, $field));
|
||||||
|
}
|
||||||
|
|
||||||
|
// add the field which are simple association
|
||||||
|
$qb
|
||||||
|
->leftJoin('acp.createdBy', "acp_created_by_t")
|
||||||
|
->addSelect('acp_created_by_t.label AS acpCreatedBy');
|
||||||
|
$qb
|
||||||
|
->leftJoin('acp.updatedBy', "acp_updated_by_t")
|
||||||
|
->addSelect('acp_updated_by_t.label AS acpUpdatedBy');
|
||||||
|
|
||||||
|
foreach (['origin' => 'label', 'closingMotive' => 'name', 'job' => 'label', 'administrativeLocation' => 'name'] as $entity => $field) {
|
||||||
|
$qb
|
||||||
|
->leftJoin(sprintf('acp.%s', $entity), "{$entity}_t")
|
||||||
|
->addSelect(sprintf('%s_t.%s AS %s', $entity, $field, $entity));
|
||||||
|
}
|
||||||
|
|
||||||
|
// step at date
|
||||||
|
$qb
|
||||||
|
->addSelect('stepHistory.step AS step')
|
||||||
|
->addSelect('stepHistory.startDate AS stepSince')
|
||||||
|
->leftJoin('acp.stepHistories', 'stepHistory')
|
||||||
|
->andWhere(
|
||||||
|
$qb->expr()->andX(
|
||||||
|
$qb->expr()->lte('stepHistory.startDate', ':calcDate'),
|
||||||
|
$qb->expr()->orX($qb->expr()->isNull('stepHistory.endDate'), $qb->expr()->gt('stepHistory.endDate', ':calcDate'))
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
// referree at date
|
||||||
|
$qb
|
||||||
|
->addSelect('referrer_t.label AS referrer')
|
||||||
|
->addSelect('userHistory.startDate AS referrerSince')
|
||||||
|
->leftJoin('acp.userHistories', 'userHistory')
|
||||||
|
->leftJoin('userHistory.user', 'referrer_t')
|
||||||
|
->andWhere(
|
||||||
|
$qb->expr()->orX(
|
||||||
|
$qb->expr()->isNull('userHistory'),
|
||||||
|
$qb->expr()->andX(
|
||||||
|
$qb->expr()->lte('userHistory.startDate', ':calcDate'),
|
||||||
|
$qb->expr()->orX($qb->expr()->isNull('userHistory.endDate'), $qb->expr()->gt('userHistory.endDate', ':calcDate'))
|
||||||
|
)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
// location of the acp
|
||||||
|
$qb
|
||||||
|
->addSelect('CASE WHEN locationHistory.personLocation IS NOT NULL THEN 1 ELSE 0 END AS locationIsPerson')
|
||||||
|
->addSelect('CASE WHEN locationHistory.personLocation IS NOT NULL THEN 0 ELSE 1 END AS locationIsTemp')
|
||||||
|
->addSelect('IDENTITY(locationHistory.personLocation) AS locationPersonName')
|
||||||
|
->addSelect('IDENTITY(locationHistory.personLocation) AS locationPersonId')
|
||||||
|
->leftJoin('acp.locationHistories', 'locationHistory')
|
||||||
|
->andWhere(
|
||||||
|
$qb->expr()->orX(
|
||||||
|
$qb->expr()->isNull('locationHistory'),
|
||||||
|
$qb->expr()->andX(
|
||||||
|
$qb->expr()->lte('locationHistory.startDate', ':calcDate'),
|
||||||
|
$qb->expr()->orX($qb->expr()->isNull('locationHistory.endDate'), $qb->expr()->gt('locationHistory.endDate', ':calcDate'))
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
->leftJoin(
|
||||||
|
PersonHouseholdAddress::class,
|
||||||
|
'acpPersonAddress',
|
||||||
|
Join::WITH,
|
||||||
|
'locationHistory.personLocation = acpPersonAddress.person AND (acpPersonAddress.validFrom <= :calcDate AND (acpPersonAddress.validTo IS NULL OR acpPersonAddress.validTo > :calcDate))'
|
||||||
|
)
|
||||||
|
->leftJoin(Address::class, 'acp_address', Join::WITH, 'COALESCE(IDENTITY(locationHistory.addressLocation), IDENTITY(acpPersonAddress.address)) = acp_address.id');
|
||||||
|
|
||||||
|
$this->addressHelper->addSelectClauses(ExportAddressHelper::F_ALL, $qb, 'acp_address', 'acp_address_fields');
|
||||||
|
|
||||||
|
// requestor
|
||||||
|
$qb
|
||||||
|
->addSelect('CASE WHEN acp.requestorPerson IS NULL THEN 1 ELSE 0 END AS isRequestorPerson')
|
||||||
|
->addSelect('CASE WHEN acp.requestorPerson IS NULL THEN 0 ELSE 1 END AS isRequestorThirdParty')
|
||||||
|
->addSelect('IDENTITY(acp.requestorPerson) AS requestorPersonId')
|
||||||
|
->addSelect('IDENTITY(acp.requestorThirdParty) AS requestorThirdPartyId')
|
||||||
|
->addSelect('IDENTITY(acp.requestorPerson) AS requestorPerson')
|
||||||
|
->addSelect('IDENTITY(acp.requestorThirdParty) AS requestorThirdParty');
|
||||||
|
|
||||||
|
$qb
|
||||||
|
// scopes
|
||||||
|
->addSelect('(SELECT AGGREGATE(scope.name) FROM ' . Scope::class . ' scope WHERE scope MEMBER OF acp.scopes) AS scopes')
|
||||||
|
// social issues
|
||||||
|
->addSelect('(SELECT AGGREGATE(socialIssue.id) FROM ' . SocialIssue::class . ' socialIssue WHERE socialIssue MEMBER OF acp.socialIssues) AS socialIssues');
|
||||||
|
|
||||||
|
// add parameter
|
||||||
|
$qb->setParameter('calcDate', $calcDate);
|
||||||
|
}
|
||||||
|
}
|
@@ -11,6 +11,7 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
namespace Chill\PersonBundle\Export\Helper;
|
namespace Chill\PersonBundle\Export\Helper;
|
||||||
|
|
||||||
|
use Chill\MainBundle\Entity\Language;
|
||||||
use Chill\MainBundle\Export\Helper\ExportAddressHelper;
|
use Chill\MainBundle\Export\Helper\ExportAddressHelper;
|
||||||
use Chill\MainBundle\Repository\CenterRepositoryInterface;
|
use Chill\MainBundle\Repository\CenterRepositoryInterface;
|
||||||
use Chill\MainBundle\Repository\CivilityRepositoryInterface;
|
use Chill\MainBundle\Repository\CivilityRepositoryInterface;
|
||||||
@@ -42,7 +43,7 @@ use function strlen;
|
|||||||
class ListPersonHelper
|
class ListPersonHelper
|
||||||
{
|
{
|
||||||
public const FIELDS = [
|
public const FIELDS = [
|
||||||
'id',
|
'personId',
|
||||||
'civility',
|
'civility',
|
||||||
'firstName',
|
'firstName',
|
||||||
'lastName',
|
'lastName',
|
||||||
@@ -114,7 +115,26 @@ class ListPersonHelper
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param array|value-of<self::FIELDS>[] $fields
|
* Those keys are the "direct" keys, which are created when we decide to use to list all the keys.
|
||||||
|
*
|
||||||
|
* This method must be used in `getKeys` instead of the `self::FIELDS`
|
||||||
|
*
|
||||||
|
* @return array<string>
|
||||||
|
*/
|
||||||
|
public function getAllKeys(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
...array_filter(
|
||||||
|
ListPersonHelper::FIELDS,
|
||||||
|
fn (string $key) => !in_array($key, ['address_fields', 'lifecycleUpdate'], true)
|
||||||
|
),
|
||||||
|
...$this->addressHelper->getKeys(ExportAddressHelper::F_ALL, 'address_fields'),
|
||||||
|
...['createdAt', 'createdBy', 'updatedAt', 'updatedBy'],
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<value-of<self::FIELDS>> $fields
|
||||||
*/
|
*/
|
||||||
public function addSelect(QueryBuilder $qb, array $fields, DateTimeImmutable $computedDate): void
|
public function addSelect(QueryBuilder $qb, array $fields, DateTimeImmutable $computedDate): void
|
||||||
{
|
{
|
||||||
@@ -124,6 +144,11 @@ class ListPersonHelper
|
|||||||
}
|
}
|
||||||
|
|
||||||
switch ($f) {
|
switch ($f) {
|
||||||
|
case 'personId':
|
||||||
|
$qb->addSelect('person.id AS personId');
|
||||||
|
|
||||||
|
break;
|
||||||
|
|
||||||
case 'countryOfBirth':
|
case 'countryOfBirth':
|
||||||
case 'nationality':
|
case 'nationality':
|
||||||
$qb->addSelect(sprintf('IDENTITY(person.%s) as %s', $f, $f));
|
$qb->addSelect(sprintf('IDENTITY(person.%s) as %s', $f, $f));
|
||||||
@@ -138,25 +163,7 @@ class ListPersonHelper
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case 'spokenLanguages':
|
case 'spokenLanguages':
|
||||||
$qb
|
$qb->addSelect('(SELECT AGGREGATE(language.id) FROM ' . Language::class . ' language WHERE language MEMBER OF person.spokenLanguages) AS spokenLanguages');
|
||||||
->leftJoin('person.spokenLanguages', 'spokenLanguage')
|
|
||||||
->addSelect('AGGREGATE(spokenLanguage.id) AS spokenLanguages')
|
|
||||||
->addGroupBy('person');
|
|
||||||
|
|
||||||
if (in_array('center', $fields, true)) {
|
|
||||||
$qb->addGroupBy('center');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (in_array('address_fields', $fields, true)) {
|
|
||||||
$qb
|
|
||||||
->addGroupBy('address_fieldsid')
|
|
||||||
->addGroupBy('address_fieldscountry_t.id')
|
|
||||||
->addGroupBy('address_fieldspostcode_t.id');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (in_array('household_id', $fields, true)) {
|
|
||||||
$qb->addGroupBy('household_id');
|
|
||||||
}
|
|
||||||
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
@@ -4,35 +4,27 @@ services:
|
|||||||
autowire: true
|
autowire: true
|
||||||
|
|
||||||
## Indicators
|
## Indicators
|
||||||
chill.person.export.count_person:
|
Chill\PersonBundle\Export\Export\CountPerson:
|
||||||
class: Chill\PersonBundle\Export\Export\CountPerson
|
|
||||||
autowire: true
|
|
||||||
autoconfigure: true
|
|
||||||
tags:
|
tags:
|
||||||
- { name: chill.export, alias: count_person }
|
- { name: chill.export, alias: count_person }
|
||||||
|
|
||||||
chill.person.export.count_person_with_accompanying_course:
|
Chill\PersonBundle\Export\Export\CountPersonWithAccompanyingCourse:
|
||||||
class: Chill\PersonBundle\Export\Export\CountPersonWithAccompanyingCourse
|
|
||||||
autowire: true
|
|
||||||
autoconfigure: true
|
|
||||||
tags:
|
tags:
|
||||||
- { name: chill.export, alias: count_person_with_accompanying_course }
|
- { name: chill.export, alias: count_person_with_accompanying_course }
|
||||||
|
|
||||||
Chill\PersonBundle\Export\Export\ListPerson:
|
Chill\PersonBundle\Export\Export\ListPerson:
|
||||||
autowire: true
|
|
||||||
autoconfigure: true
|
|
||||||
tags:
|
tags:
|
||||||
- { name: chill.export, alias: list_person }
|
- { name: chill.export, alias: list_person }
|
||||||
|
|
||||||
Chill\PersonBundle\Export\Export\ListPersonWithAccompanyingPeriod:
|
Chill\PersonBundle\Export\Export\ListPersonHavingAccompanyingPeriod:
|
||||||
autowire: true
|
|
||||||
autoconfigure: true
|
|
||||||
tags:
|
tags:
|
||||||
- { name: chill.export, alias: list_person_with_acp }
|
- { name: chill.export, alias: list_person_with_acp }
|
||||||
|
|
||||||
|
Chill\PersonBundle\Export\Export\ListPersonWithAccompanyingPeriodDetails:
|
||||||
|
tags:
|
||||||
|
- { name: chill.export, alias: list_person_with_acp_details }
|
||||||
|
|
||||||
Chill\PersonBundle\Export\Export\ListAccompanyingPeriod:
|
Chill\PersonBundle\Export\Export\ListAccompanyingPeriod:
|
||||||
autowire: true
|
|
||||||
autoconfigure: true
|
|
||||||
tags:
|
tags:
|
||||||
- { name: chill.export, alias: list_acp }
|
- { name: chill.export, alias: list_acp }
|
||||||
|
|
||||||
|
@@ -998,6 +998,8 @@ notification:
|
|||||||
Notify referrer: Notifier le référent
|
Notify referrer: Notifier le référent
|
||||||
Notify any: Notifier d'autres utilisateurs
|
Notify any: Notifier d'autres utilisateurs
|
||||||
|
|
||||||
|
personId: Identifiant de l'usager
|
||||||
|
|
||||||
export:
|
export:
|
||||||
export:
|
export:
|
||||||
acp_stats:
|
acp_stats:
|
||||||
@@ -1147,13 +1149,15 @@ export:
|
|||||||
list:
|
list:
|
||||||
person_with_acp:
|
person_with_acp:
|
||||||
List peoples having an accompanying period: Liste des usagers ayant un parcours d'accompagnement
|
List peoples having an accompanying period: Liste des usagers ayant un parcours d'accompagnement
|
||||||
|
List peoples having an accompanying period with period details: Liste des usagers concernés avec détail de chaque parcours
|
||||||
Create a list of people having an accompaying periods, according to various filters.: Génère une liste des usagers ayant un parcours d'accompagnement, selon différents critères liés au parcours ou à l'usager
|
Create a list of people having an accompaying periods, according to various filters.: Génère une liste des usagers ayant un parcours d'accompagnement, selon différents critères liés au parcours ou à l'usager
|
||||||
|
Create a list of people having an accompaying periods with details of period, according to various filters.: Génère une liste des usagers ayant un parcours d'accompagnement, selon différents critères liés au parcours ou à l'usager. Ajoute les détails du parcours à la liste.
|
||||||
acp:
|
acp:
|
||||||
List of accompanying periods: Liste des parcours d'accompagnements
|
List of accompanying periods: Liste des parcours d'accompagnements
|
||||||
Generate a list of accompanying periods, filtered on different parameters.: Génère une liste des parcours d'accompagnement, filtrée sur différents paramètres.
|
Generate a list of accompanying periods, filtered on different parameters.: Génère une liste des parcours d'accompagnement, filtrée sur différents paramètres.
|
||||||
Date of calculation for associated elements: Date de calcul des éléments associés
|
Date of calculation for associated elements: Date de calcul des éléments associés
|
||||||
The associated referree, localisation, and other elements will be valid at this date: Les éléments associés, comme la localisation, le référent et d'autres éléments seront valides à cette date
|
The associated referree, localisation, and other elements will be valid at this date: Les éléments associés, comme la localisation, le référent et d'autres éléments seront valides à cette date
|
||||||
id: Identifiant du parcours
|
acpId: Identifiant du parcours
|
||||||
openingDate: Date d'ouverture du parcours
|
openingDate: Date d'ouverture du parcours
|
||||||
closingDate: Date de fermeture du parcours
|
closingDate: Date de fermeture du parcours
|
||||||
closingMotive: Motif de cloture
|
closingMotive: Motif de cloture
|
||||||
@@ -1161,14 +1165,14 @@ export:
|
|||||||
confidential: Confidentiel
|
confidential: Confidentiel
|
||||||
emergency: Urgent
|
emergency: Urgent
|
||||||
intensity: Intensité
|
intensity: Intensité
|
||||||
createdAt: Créé le
|
acpCreatedAt: Créé le
|
||||||
updatedAt: Dernière mise à jour le
|
acpUpdatedAt: Dernière mise à jour le
|
||||||
acpOrigin: Origine du parcours
|
acpOrigin: Origine du parcours
|
||||||
origin: Origine du parcours
|
origin: Origine du parcours
|
||||||
acpClosingMotive: Motif de fermeture
|
acpClosingMotive: Motif de fermeture
|
||||||
acpJob: Métier du parcours
|
acpJob: Métier du parcours
|
||||||
createdBy: Créé par
|
acpCreatedBy: Créé par
|
||||||
updatedBy: Dernière modification par
|
acpUpdatedBy: Dernière modification par
|
||||||
administrativeLocation: Location administrative
|
administrativeLocation: Location administrative
|
||||||
step: Etape
|
step: Etape
|
||||||
stepSince: Dernière modification de l'étape
|
stepSince: Dernière modification de l'étape
|
||||||
@@ -1176,7 +1180,7 @@ export:
|
|||||||
referrerSince: Référent depuis le
|
referrerSince: Référent depuis le
|
||||||
locationIsPerson: Parcours localisé auprès d'un usager concerné
|
locationIsPerson: Parcours localisé auprès d'un usager concerné
|
||||||
locationIsTemp: Parcours avec une localisation temporaire
|
locationIsTemp: Parcours avec une localisation temporaire
|
||||||
acpLocationPersonName: Usager auprès duquel le parcours est localisé
|
locationPersonName: Usager auprès duquel le parcours est localisé
|
||||||
locationPersonId: Identifiant de l'usager auprès duquel le parcours est localisé
|
locationPersonId: Identifiant de l'usager auprès duquel le parcours est localisé
|
||||||
acpaddress_fieldscountry: Pays de l'adresse
|
acpaddress_fieldscountry: Pays de l'adresse
|
||||||
isRequestorPerson: Le demandeur est-il un usager ?
|
isRequestorPerson: Le demandeur est-il un usager ?
|
||||||
|
Reference in New Issue
Block a user