wallabag/src/Wallabag/CoreBundle/Form/Type/EntryFilterType.php

204 lines
8.7 KiB
PHP
Raw Normal View History

<?php
namespace Wallabag\CoreBundle\Form\Type;
2017-05-03 09:08:56 +00:00
use Lexik\Bundle\FormFilterBundle\Filter\FilterOperands;
use Lexik\Bundle\FormFilterBundle\Filter\Form\Type\CheckboxFilterType;
use Lexik\Bundle\FormFilterBundle\Filter\Form\Type\ChoiceFilterType;
2017-07-01 07:52:38 +00:00
use Lexik\Bundle\FormFilterBundle\Filter\Form\Type\DateRangeFilterType;
use Lexik\Bundle\FormFilterBundle\Filter\Form\Type\NumberRangeFilterType;
use Lexik\Bundle\FormFilterBundle\Filter\Form\Type\TextFilterType;
use Lexik\Bundle\FormFilterBundle\Filter\Query\QueryInterface;
use Symfony\Component\Form\AbstractType;
use Symfony\Component\Form\FormBuilderInterface;
2016-11-18 22:05:02 +00:00
use Symfony\Component\HttpFoundation\Response;
use Symfony\Component\OptionsResolver\OptionsResolver;
use Symfony\Component\Security\Core\Authentication\Token\Storage\TokenStorageInterface;
2022-08-27 18:22:48 +00:00
use Wallabag\CoreBundle\Repository\EntryRepository;
2022-11-23 14:51:33 +00:00
use Wallabag\UserBundle\Entity\User;
class EntryFilterType extends AbstractType
{
private $repository;
2022-11-23 14:51:33 +00:00
private $tokenStorage;
/**
2015-10-01 07:26:52 +00:00
* Repository & user are used to get a list of language entries for this user.
*/
2022-08-27 18:22:48 +00:00
public function __construct(EntryRepository $entryRepository, TokenStorageInterface $tokenStorage)
{
$this->repository = $entryRepository;
2022-11-23 14:51:33 +00:00
$this->tokenStorage = $tokenStorage;
}
public function buildForm(FormBuilderInterface $builder, array $options)
{
2022-11-23 14:51:33 +00:00
$user = $this->tokenStorage->getToken() ? $this->tokenStorage->getToken()->getUser() : null;
\assert($user instanceof User);
2015-08-14 20:57:16 +00:00
$builder
->add('readingTime', NumberRangeFilterType::class, [
2017-05-09 10:12:23 +00:00
'left_number_options' => [
'condition_operator' => FilterOperands::OPERATOR_GREATER_THAN_EQUAL,
'attr' => ['min' => 0],
],
'right_number_options' => [
'condition_operator' => FilterOperands::OPERATOR_LOWER_THAN_EQUAL,
'attr' => ['min' => 0],
],
2022-11-23 14:51:33 +00:00
'apply_filter' => function (QueryInterface $filterQuery, $field, $values) use ($user) {
$lower = $values['value']['left_number'][0];
$upper = $values['value']['right_number'][0];
if (null === $lower && null === $upper) {
// no value? no filter
2016-04-24 18:46:25 +00:00
return;
2022-11-23 14:51:33 +00:00
}
$min = (int) ($lower * $user->getConfig()->getReadingSpeed() / 200);
$max = (int) ($upper * $user->getConfig()->getReadingSpeed() / 200);
if (null === $lower && null !== $upper) {
// only lower value is defined: query all entries with reading LOWER THAN this value
$expression = $filterQuery->getExpr()->lte($field, $max);
} elseif (null !== $lower && null === $upper) {
// only upper value is defined: query all entries with reading GREATER THAN this value
$expression = $filterQuery->getExpr()->gte($field, $min);
} else {
// both value are defined, perform a between
$expression = $filterQuery->getExpr()->between($field, $min, $max);
2016-04-24 18:46:25 +00:00
}
return $filterQuery->createCondition($expression);
},
'label' => 'entry.filters.reading_time.label',
])
->add('createdAt', DateRangeFilterType::class, [
'left_date_options' => [
'attr' => [
'placeholder' => 'yyyy-mm-dd',
],
'format' => 'yyyy-MM-dd',
'widget' => 'single_text',
],
'right_date_options' => [
'attr' => [
'placeholder' => 'yyyy-mm-dd',
],
'format' => 'yyyy-MM-dd',
'widget' => 'single_text',
],
'label' => 'entry.filters.created_at.label',
])
->add('domainName', TextFilterType::class, [
'apply_filter' => function (QueryInterface $filterQuery, $field, $values) {
$value = $values['value'];
if (\strlen($value) <= 2 || empty($value)) {
2021-08-02 14:57:42 +00:00
return false;
}
2017-07-01 07:52:38 +00:00
$expression = $filterQuery->getExpr()->like($field, $filterQuery->getExpr()->lower($filterQuery->getExpr()->literal('%' . $value . '%')));
return $filterQuery->createCondition($expression);
2015-09-12 15:08:12 +00:00
},
'label' => 'entry.filters.domain_label',
])
2016-11-18 14:09:21 +00:00
->add('httpStatus', TextFilterType::class, [
2016-11-18 22:05:02 +00:00
'apply_filter' => function (QueryInterface $filterQuery, $field, $values) {
$value = $values['value'];
2019-02-11 10:50:24 +00:00
if (false === \array_key_exists($value, Response::$statusTexts)) {
2021-08-02 14:57:42 +00:00
return false;
2016-11-18 22:05:02 +00:00
}
$paramName = sprintf('%s', str_replace('.', '_', $field));
2017-07-01 07:52:38 +00:00
$expression = $filterQuery->getExpr()->eq($field, ':' . $paramName);
$parameters = [$paramName => $value];
2016-11-18 22:05:02 +00:00
return $filterQuery->createCondition($expression, $parameters);
},
2016-11-18 14:09:21 +00:00
'label' => 'entry.filters.http_status_label',
])
->add('isArchived', CheckboxFilterType::class, [
'label' => 'entry.filters.archived_label',
'data' => $options['filter_archived'],
])
->add('isStarred', CheckboxFilterType::class, [
'label' => 'entry.filters.starred_label',
'data' => $options['filter_starred'],
])
->add('isUnread', CheckboxFilterType::class, [
'label' => 'entry.filters.unread_label',
'data' => $options['filter_unread'],
'apply_filter' => function (QueryInterface $filterQuery, $field, $values) {
if (false === $values['value']) {
2021-08-02 14:57:42 +00:00
return false;
}
$expression = $filterQuery->getExpr()->eq('e.isArchived', 'false');
return $filterQuery->createCondition($expression);
},
])
2021-08-02 14:57:42 +00:00
->add('isAnnotated', CheckboxFilterType::class, [
'label' => 'entry.filters.annotated_label',
'data' => $options['filter_annotated'],
'apply_filter' => function (QueryInterface $filterQuery, $field, $values) {
if (false === $values['value']) {
return false;
}
$qb = $filterQuery->getQueryBuilder();
$qb->innerJoin('e.annotations', 'a');
},
])
->add('previewPicture', CheckboxFilterType::class, [
2015-09-12 15:08:12 +00:00
'apply_filter' => function (QueryInterface $filterQuery, $field, $values) {
2015-09-13 07:57:35 +00:00
if (false === $values['value']) {
2021-08-02 14:57:42 +00:00
return false;
2015-09-13 06:43:15 +00:00
}
2015-09-12 15:08:12 +00:00
$expression = $filterQuery->getExpr()->isNotNull($field);
return $filterQuery->createCondition($expression);
},
'label' => 'entry.filters.preview_picture_label',
])
2017-06-10 13:00:52 +00:00
->add('isPublic', CheckboxFilterType::class, [
'apply_filter' => function (QueryInterface $filterQuery, $field, $values) {
if (false === $values['value']) {
2021-08-02 14:57:42 +00:00
return false;
2017-06-10 13:00:52 +00:00
}
// is_public isn't a real field
// we should use the "uid" field to determine if the entry has been made public
2017-07-01 07:52:38 +00:00
$expression = $filterQuery->getExpr()->isNotNull($values['alias'] . '.uid');
2017-06-10 13:00:52 +00:00
return $filterQuery->createCondition($expression);
},
'label' => 'entry.filters.is_public_label',
])
->add('language', ChoiceFilterType::class, [
2022-11-23 14:51:33 +00:00
'choices' => array_flip($this->repository->findDistinctLanguageByUser($user->getId())),
'label' => 'entry.filters.language_label',
])
;
}
public function getBlockPrefix()
{
return 'entry_filter';
}
public function configureOptions(OptionsResolver $resolver)
{
$resolver->setDefaults([
'csrf_protection' => false,
'validation_groups' => ['filtering'],
'filter_archived' => false,
'filter_starred' => false,
'filter_unread' => false,
2021-08-02 14:57:42 +00:00
'filter_annotated' => false,
]);
}
}