add more filtering possibilities with order helper

This commit is contained in:
Julien Fastré 2021-10-28 00:11:05 +02:00
parent aea5e9b1d7
commit 97dbc4bc16
6 changed files with 127 additions and 79 deletions

View File

@ -32,6 +32,7 @@ final class FilterOrderType extends \Symfony\Component\Form\AbstractType
]); ]);
} }
$checkboxesBuilder = $builder->create('checkboxes', null, [ 'compound' => true ]);
foreach ($helper->getCheckboxes() as $name => $c) { foreach ($helper->getCheckboxes() as $name => $c) {
$choices = \array_combine( $choices = \array_combine(
@ -42,17 +43,21 @@ final class FilterOrderType extends \Symfony\Component\Form\AbstractType
$c['choices'] $c['choices']
); );
$builder->add('c_'.$name, ChoiceType::class, [ $checkboxesBuilder->add($name, ChoiceType::class, [
'choices' => $choices, 'choices' => $choices,
'expanded' => true, 'expanded' => true,
'multiple' => true, 'multiple' => true,
]); ]);
} }
if (0 < count($helper->getCheckboxes())) {
$builder->add($checkboxesBuilder);
}
foreach ($this->requestStack->getCurrentRequest()->query->getIterator() as $key => $value) { foreach ($this->requestStack->getCurrentRequest()->query->getIterator() as $key => $value) {
switch($key) { switch($key) {
case 'q': case 'q':
case 'c_'.$key: case 'checkboxes'.$key:
break; break;
case 'page': case 'page':
$builder->add($key, HiddenType::class, [ $builder->add($key, HiddenType::class, [
@ -75,7 +80,7 @@ final class FilterOrderType extends \Symfony\Component\Form\AbstractType
$view->vars['has_search_box'] = $helper->hasSearchBox(); $view->vars['has_search_box'] = $helper->hasSearchBox();
$view->vars['checkboxes'] = []; $view->vars['checkboxes'] = [];
foreach ($helper->getCheckboxes() as $name => $c) { foreach ($helper->getCheckboxes() as $name => $c) {
$view->vars['checkboxes']['c_'.$name] = []; $view->vars['checkboxes'][$name] = [];
} }
} }

View File

@ -10,17 +10,30 @@
</div> </div>
{% endif %} {% endif %}
</div> </div>
{% for checkbox_name, options in form.vars.checkboxes %} {% if form.checkboxes|length > 0 %}
{% for checkbox_name, options in form.checkboxes %}
<div class="row gx-0">
<div class="col-md-12"> <div class="col-md-12">
<div class="col-md-11"> {% for c in form['checkboxes'][checkbox_name].children %}
{{ form_widget(form[checkbox_name]) }} <div class="form-check form-check-inline">
</div> {{ form_widget(c) }}
{% if loop.last %} {{ form_label(c) }}
<div class="col-md-1">
<button type="submit" class="btn btn-misc"><i class="fa fa-filter"></i></button>
</div>
{% endif %}
</div> </div>
{% endfor %} {% endfor %}
</div> </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>
{% endif %}
{% endfor %}
{% endif %}
</div>
{{ form_end(form) }} {{ form_end(form) }}

View File

@ -13,6 +13,11 @@ class FilterOrderHelper
private RequestStack $requestStack; private RequestStack $requestStack;
private ?array $searchBoxFields = null; private ?array $searchBoxFields = null;
private array $checkboxes = []; private array $checkboxes = [];
private ?array $submitted = null;
private ?string $formName = 'filter';
private string $formType = FilterOrderType::class;
private array $formOptions = [];
public function __construct( public function __construct(
FormFactoryInterface $formFactory, FormFactoryInterface $formFactory,
@ -45,10 +50,9 @@ class FilterOrderHelper
return $this; return $this;
} }
public function getCheckbox(string $name): array public function getCheckboxData(string $name): array
{ {
return $this->requestStack->getCurrentRequest() return $this->getFormData()['checkboxes'][$name];
->query->get('c_'.$name, $this->checkboxes[$name]['default']);
} }
public function getCheckboxes(): array public function getCheckboxes(): array
@ -63,12 +67,23 @@ class FilterOrderHelper
private function getFormData(): array private function getFormData(): array
{ {
$r = [ if (NULL === $this->submitted) {
'q' => $this->getQueryString(), $this->submitted = $this->buildForm()
]; ->getData();
}
return $this->submitted;
}
private function getDefaultData(): array
{
$r = [];
if ($this->hasSearchBox()) {
$r['q'] = '';
}
foreach ($this->checkboxes as $name => $c) { foreach ($this->checkboxes as $name => $c) {
$r[$name] = $this->getCheckbox($name); $r['checkboxes'][$name] = $c['default'];
} }
return $r; return $r;
@ -76,21 +91,17 @@ class FilterOrderHelper
public function getQueryString(): ?string public function getQueryString(): ?string
{ {
$q = $this->requestStack->getCurrentRequest() return $this->getFormData()['q'];
->query->get('q', null);
return empty($q) ? NULL : $q;
} }
public function buildForm($name = null, string $type = FilterOrderType::class, array $options = []): FormInterface public function buildForm(): FormInterface
{ {
$form = $this->formFactory return $this->formFactory
->createNamed($name, $type, $this->getFormData(), \array_merge([ ->createNamed($this->formName, $this->formType, $this->getDefaultData(), \array_merge([
'helper' => $this, 'helper' => $this,
'method' => 'GET', 'method' => 'GET',
'csrf_protection' => false, 'csrf_protection' => false,
], $options)); ], $this->formOptions))
->handleRequest($this->requestStack->getCurrentRequest());
return $form;
} }
} }

View File

@ -532,14 +532,18 @@ final class SingleTaskController extends AbstractController
$this->denyAccessUnlessGranted('ROLE_USER'); $this->denyAccessUnlessGranted('ROLE_USER');
$filterOrder = $this->buildFilterOrder(); $filterOrder = $this->buildFilterOrder();
$flags = \array_merge(
$filterOrder->getCheckboxData('status'),
\array_map(fn ($i) => 'state_'.$i, $filterOrder->getCheckboxData('states'))
);
$nb = $this->singleTaskAclAwareRepository->countByCurrentUsersTasks( $nb = $this->singleTaskAclAwareRepository->countByCurrentUsersTasks(
$filterOrder->getQueryString(), $filterOrder->getQueryString(),
$filterOrder->getCheckbox('status') $flags
); );
$paginator = $this->paginatorFactory->create($nb); $paginator = $this->paginatorFactory->create($nb);
$tasks = $this->singleTaskAclAwareRepository->findByCurrentUsersTasks( $tasks = $this->singleTaskAclAwareRepository->findByCurrentUsersTasks(
$filterOrder->getQueryString(), $filterOrder->getQueryString(),
$filterOrder->getCheckbox('status'), $flags,
$paginator->getCurrentPageFirstItemNumber(), $paginator->getCurrentPageFirstItemNumber(),
$paginator->getItemsPerPage() $paginator->getItemsPerPage()
); );
@ -555,14 +559,19 @@ final class SingleTaskController extends AbstractController
{ {
$statuses = ['no-alert', 'warning', 'alert']; $statuses = ['no-alert', 'warning', 'alert'];
$statusTrans = [ $statusTrans = [
'Tasks without alert',
'Tasks near deadline', 'Tasks near deadline',
'Tasks over deadline', 'Tasks over deadline',
'Tasks without alert', ];
$states = [
// todo: get a list of possible states dynamically
'new', 'in_progress', 'closed', 'canceled'
]; ];
return $this->filterOrderHelperFactory return $this->filterOrderHelperFactory
->create(self::class) ->create(self::class)
->addSearchBox() ->addSearchBox()
->addCheckbox('status', $statuses, $statuses, $statusTrans) ->addCheckbox('status', $statuses, $statuses, $statusTrans)
->addCheckbox('states', $states, ['new', 'in_progress'])
->build() ->build()
; ;
} }

View File

@ -63,16 +63,19 @@ final class SingleTaskAclAwareRepository implements SingleTaskAclAwareRepository
if (!empty($pattern)) { if (!empty($pattern)) {
$qb->andWhere($qb->expr()->like('LOWER(UNACCENT(t.title))', 'LOWER(UNACCENT(:pattern))')) $qb->andWhere($qb->expr()->like('LOWER(UNACCENT(t.title))', 'LOWER(UNACCENT(:pattern))'))
->setParameter('pattern', $pattern) ->setParameter('pattern', '%'.$pattern.'%')
; ;
} }
if (count($flags) > 0) { if (count($flags) > 0) {
$orX = $qb->expr()->orX(); $orXDate = $qb->expr()->orX();
$orXState = $qb->expr()->orX();
$now = new \DateTime(); $now = new \DateTime();
if (\in_array('no-alert', $flags)) { foreach ($flags as $key => $flag) {
$orX switch ($flag) {
case 'no-alert':
$orXDate
->add( ->add(
$qb->expr()->orX( $qb->expr()->orX(
$qb->expr()->isNull('t.endDate'), $qb->expr()->isNull('t.endDate'),
@ -81,45 +84,52 @@ final class SingleTaskAclAwareRepository implements SingleTaskAclAwareRepository
); );
$qb $qb
->setParameter('intervalBlank', new \DateInterval('P0D')) ->setParameter('intervalBlank', new \DateInterval('P0D'))
->setParameter('now', $now) ->setParameter('now', $now);
; break;
} case 'warning':
$orXDate
if (\in_array('warning', $flags)) {
$orX
->add( ->add(
$qb->expr()->andX( $qb->expr()->andX(
$qb->expr()->eq('t.closed', "'FALSE'"),
$qb->expr()->not($qb->expr()->isNull('t.endDate')), $qb->expr()->not($qb->expr()->isNull('t.endDate')),
$qb->expr()->not($qb->expr()->isNull('t.warningInterval')), $qb->expr()->not($qb->expr()->isNull('t.warningInterval')),
$qb->expr()->lte('t.endDate - t.warningInterval', ':now') $qb->expr()->gte('t.endDate - t.warningInterval', ':now'),
$qb->expr()->lt('t.endDate', ':now')
) )
) );
;
$qb $qb
->setParameter('now', $now) ->setParameter('now', $now);
; break;
} case 'alert':
$orXDate
if (\in_array('alert', $flags)) {
$orX
->add( ->add(
$qb->expr()->andX( $qb->expr()->andX(
$qb->expr()->eq('t.closed', "'FALSE'"),
$qb->expr()->not($qb->expr()->isNull('t.endDate')), $qb->expr()->not($qb->expr()->isNull('t.endDate')),
$qb->expr()->lte('t.endDate', ':now') $qb->expr()->lte('t.endDate', ':now')
) )
) );
;
$qb $qb
->setParameter('now', $now) ->setParameter('now', $now);
; break;
case \substr($flag, 0, 6) === 'state_':
$state = \substr($flag, 6);
$orXState
->add(
"JSONB_EXISTS_IN_ARRAY(t.currentStates, :state_$key) = 'TRUE'"
);
$qb->setParameter("state_$key", $state);
break;
default:
throw new \LogicException("this flag is not supported: $flag");
}
} }
$qb->andWhere($orX); if ($orXDate->count() > 0) {
$qb->andWhere($orXDate);
}
if ($orXState->count() > 0) {
$qb->andWhere($orXState);
}
} }
return $qb; return $qb;
} }

View File

@ -99,7 +99,7 @@ Are you sure you want to start this task ?: Êtes-vous sûrs de vouloir démarre
Tasks near deadline: Tâches à échéance proche Tasks near deadline: Tâches à échéance proche
Tasks over deadline: Tâches à échéance dépassée Tasks over deadline: Tâches à échéance dépassée
Tasks without alert: Tâches sans alerte Tasks without alert: Tâches à échéance future ou sans échéance
#title #title
My tasks near deadline: Mes tâches à échéance proche My tasks near deadline: Mes tâches à échéance proche