]> git.immae.eu Git - github/wallabag/wallabag.git/blame - src/Wallabag/CoreBundle/Form/Type/EntryFilterType.php
Merge pull request #4151 from ldidry/fix-4060
[github/wallabag/wallabag.git] / src / Wallabag / CoreBundle / Form / Type / EntryFilterType.php
CommitLineData
26864574
NL
1<?php
2
b0b893ea 3namespace Wallabag\CoreBundle\Form\Type;
26864574 4
619cc453 5use Doctrine\ORM\EntityRepository;
1267905d 6use Lexik\Bundle\FormFilterBundle\Filter\FilterOperands;
5c895a7f
JB
7use Lexik\Bundle\FormFilterBundle\Filter\Form\Type\CheckboxFilterType;
8use Lexik\Bundle\FormFilterBundle\Filter\Form\Type\ChoiceFilterType;
f808b016
JB
9use Lexik\Bundle\FormFilterBundle\Filter\Form\Type\DateRangeFilterType;
10use Lexik\Bundle\FormFilterBundle\Filter\Form\Type\NumberRangeFilterType;
11use Lexik\Bundle\FormFilterBundle\Filter\Form\Type\TextFilterType;
12use Lexik\Bundle\FormFilterBundle\Filter\Query\QueryInterface;
26864574
NL
13use Symfony\Component\Form\AbstractType;
14use Symfony\Component\Form\FormBuilderInterface;
d215273c 15use Symfony\Component\HttpFoundation\Response;
26864574 16use Symfony\Component\OptionsResolver\OptionsResolver;
2fe2e411 17use Symfony\Component\Security\Core\Authentication\Token\Storage\TokenStorageInterface;
26864574
NL
18
19class EntryFilterType extends AbstractType
20{
d4ebe5c5
JB
21 private $user;
22 private $repository;
23
24 /**
cfb28c9d 25 * Repository & user are used to get a list of language entries for this user.
d4ebe5c5 26 */
2fe2e411 27 public function __construct(EntityRepository $entryRepository, TokenStorageInterface $tokenStorage)
d4ebe5c5
JB
28 {
29 $this->repository = $entryRepository;
2fe2e411
NL
30
31 $this->user = $tokenStorage->getToken() ? $tokenStorage->getToken()->getUser() : null;
32
2a1ceb67 33 if (null === $this->user || !\is_object($this->user)) {
1b164717 34 return;
2fe2e411 35 }
d4ebe5c5
JB
36 }
37
26864574
NL
38 public function buildForm(FormBuilderInterface $builder, array $options)
39 {
3c5b025a 40 $builder
4094ea47 41 ->add('readingTime', NumberRangeFilterType::class, [
1b164717
JB
42 'left_number_options' => [
43 'condition_operator' => FilterOperands::OPERATOR_GREATER_THAN_EQUAL,
44 'attr' => ['min' => 0],
45 ],
46 'right_number_options' => [
47 'condition_operator' => FilterOperands::OPERATOR_LOWER_THAN_EQUAL,
48 'attr' => ['min' => 0],
49 ],
d8f8a590 50 'apply_filter' => function (QueryInterface $filterQuery, $field, $values) {
1bdbc39f
JB
51 $lower = $values['value']['left_number'][0];
52 $upper = $values['value']['right_number'][0];
d8f8a590 53
41022cb2
JB
54 $min = (int) ($lower * $this->user->getConfig()->getReadingSpeed() / 200);
55 $max = (int) ($upper * $this->user->getConfig()->getReadingSpeed() / 200);
1bdbc39f
JB
56
57 if (null === $lower && null === $upper) {
58 // no value? no filter
d6a9e139 59 return;
2c00dddf 60 } elseif (null === $lower && null !== $upper) {
1bdbc39f
JB
61 // only lower value is defined: query all entries with reading LOWER THAN this value
62 $expression = $filterQuery->getExpr()->lte($field, $max);
2c00dddf 63 } elseif (null !== $lower && null === $upper) {
1bdbc39f
JB
64 // only upper value is defined: query all entries with reading GREATER THAN this value
65 $expression = $filterQuery->getExpr()->gte($field, $min);
66 } else {
67 // both value are defined, perform a between
68 $expression = $filterQuery->getExpr()->between($field, $min, $max);
d6a9e139
NL
69 }
70
d8f8a590
NL
71 return $filterQuery->createCondition($expression);
72 },
0d42217e 73 'label' => 'entry.filters.reading_time.label',
4094ea47
JB
74 ])
75 ->add('createdAt', DateRangeFilterType::class, [
76 'left_date_options' => [
77 'attr' => [
8ce32af6 78 'placeholder' => 'dd/mm/yyyy',
4094ea47 79 ],
3c5b025a 80 'format' => 'dd/MM/yyyy',
8ce32af6 81 'widget' => 'single_text',
4094ea47
JB
82 ],
83 'right_date_options' => [
84 'attr' => [
8ce32af6 85 'placeholder' => 'dd/mm/yyyy',
4094ea47 86 ],
3c5b025a 87 'format' => 'dd/MM/yyyy',
8ce32af6 88 'widget' => 'single_text',
4094ea47 89 ],
0d42217e 90 'label' => 'entry.filters.created_at.label',
4094ea47 91 ]
7d6c3edc 92 )
4094ea47 93 ->add('domainName', TextFilterType::class, [
8ce32af6 94 'apply_filter' => function (QueryInterface $filterQuery, $field, $values) {
1d76102a 95 $value = $values['value'];
2a1ceb67 96 if (\strlen($value) <= 2 || empty($value)) {
1d76102a
JB
97 return;
98 }
f808b016 99 $expression = $filterQuery->getExpr()->like($field, $filterQuery->getExpr()->lower($filterQuery->getExpr()->literal('%' . $value . '%')));
8ce32af6 100
1d76102a 101 return $filterQuery->createCondition($expression);
a3bcd60a 102 },
0d42217e 103 'label' => 'entry.filters.domain_label',
4094ea47 104 ])
10b35097 105 ->add('httpStatus', TextFilterType::class, [
d215273c
NL
106 'apply_filter' => function (QueryInterface $filterQuery, $field, $values) {
107 $value = $values['value'];
0182cdae 108 if (false === \array_key_exists($value, Response::$statusTexts)) {
d215273c
NL
109 return;
110 }
111
112 $paramName = sprintf('%s', str_replace('.', '_', $field));
f808b016
JB
113 $expression = $filterQuery->getExpr()->eq($field, ':' . $paramName);
114 $parameters = [$paramName => $value];
d215273c
NL
115
116 return $filterQuery->createCondition($expression, $parameters);
117 },
10b35097
NL
118 'label' => 'entry.filters.http_status_label',
119 ])
4094ea47 120 ->add('isArchived', CheckboxFilterType::class, [
0d42217e 121 'label' => 'entry.filters.archived_label',
4094ea47
JB
122 ])
123 ->add('isStarred', CheckboxFilterType::class, [
0d42217e 124 'label' => 'entry.filters.starred_label',
4094ea47 125 ])
733b2cf1
DB
126 ->add('isUnread', CheckboxFilterType::class, [
127 'label' => 'entry.filters.unread_label',
128 'apply_filter' => function (QueryInterface $filterQuery, $field, $values) {
129 if (false === $values['value']) {
130 return;
131 }
132
133 $expression = $filterQuery->getExpr()->eq('e.isArchived', 'false');
134
135 return $filterQuery->createCondition($expression);
136 },
137 ])
4094ea47 138 ->add('previewPicture', CheckboxFilterType::class, [
a3bcd60a 139 'apply_filter' => function (QueryInterface $filterQuery, $field, $values) {
497e0cad 140 if (false === $values['value']) {
616f9fea
NL
141 return;
142 }
143
a3bcd60a
NL
144 $expression = $filterQuery->getExpr()->isNotNull($field);
145
146 return $filterQuery->createCondition($expression);
147 },
0d42217e 148 'label' => 'entry.filters.preview_picture_label',
4094ea47 149 ])
e8911f7c
JB
150 ->add('isPublic', CheckboxFilterType::class, [
151 'apply_filter' => function (QueryInterface $filterQuery, $field, $values) {
152 if (false === $values['value']) {
153 return;
154 }
155
156 // is_public isn't a real field
157 // we should use the "uid" field to determine if the entry has been made public
f808b016 158 $expression = $filterQuery->getExpr()->isNotNull($values['alias'] . '.uid');
e8911f7c
JB
159
160 return $filterQuery->createCondition($expression);
161 },
162 'label' => 'entry.filters.is_public_label',
163 ])
4094ea47 164 ->add('language', ChoiceFilterType::class, [
5c895a7f 165 'choices' => array_flip($this->repository->findDistinctLanguageByUser($this->user->getId())),
0d42217e 166 'label' => 'entry.filters.language_label',
4094ea47 167 ])
d4ebe5c5 168 ;
26864574
NL
169 }
170
619cc453 171 public function getBlockPrefix()
26864574
NL
172 {
173 return 'entry_filter';
174 }
175
176 public function configureOptions(OptionsResolver $resolver)
177 {
4094ea47 178 $resolver->setDefaults([
8ce32af6 179 'csrf_protection' => false,
4094ea47
JB
180 'validation_groups' => ['filtering'],
181 ]);
26864574
NL
182 }
183}