]> git.immae.eu Git - github/wallabag/wallabag.git/blame - src/Wallabag/CoreBundle/Controller/EntryController.php
Add translations about latest Tag changes.
[github/wallabag/wallabag.git] / src / Wallabag / CoreBundle / Controller / EntryController.php
CommitLineData
9d50517c
NL
1<?php
2
ad4d1caa 3namespace Wallabag\CoreBundle\Controller;
9d50517c 4
619cc453 5use Pagerfanta\Adapter\DoctrineORMAdapter;
671a2b88 6use Pagerfanta\Exception\OutOfRangeCurrentPageException;
f808b016 7use Sensio\Bundle\FrameworkExtraBundle\Configuration\Cache;
9d50517c
NL
8use Sensio\Bundle\FrameworkExtraBundle\Configuration\Route;
9use Symfony\Bundle\FrameworkBundle\Controller\Controller;
163eae0b 10use Symfony\Component\HttpFoundation\Request;
2863bf2a 11use Symfony\Component\Routing\Generator\UrlGeneratorInterface;
be463487 12use Wallabag\CoreBundle\Entity\Entry;
f808b016
JB
13use Wallabag\CoreBundle\Event\EntryDeletedEvent;
14use Wallabag\CoreBundle\Event\EntrySavedEvent;
619cc453 15use Wallabag\CoreBundle\Form\Type\EditEntryType;
f808b016 16use Wallabag\CoreBundle\Form\Type\EntryFilterType;
619cc453 17use Wallabag\CoreBundle\Form\Type\NewEntryType;
ee122a75 18use Wallabag\CoreBundle\Form\Type\SearchEntryType;
9d50517c
NL
19
20class EntryController extends Controller
21{
ee122a75
NL
22 /**
23 * @param Request $request
24 * @param int $page
25 *
21e7ccef
JB
26 * @Route("/search/{page}", name="search", defaults={"page" = 1})
27 *
28 * Default parameter for page is hardcoded (in duplication of the defaults from the Route)
29 * because this controller is also called inside the layout template without any page as argument
ee122a75
NL
30 *
31 * @return \Symfony\Component\HttpFoundation\Response
32 */
21e7ccef 33 public function searchFormAction(Request $request, $page = 1, $currentRoute = null)
ee122a75 34 {
21e7ccef
JB
35 // fallback to retrieve currentRoute from query parameter instead of injected one (when using inside a template)
36 if (null === $currentRoute && $request->query->has('currentRoute')) {
37 $currentRoute = $request->query->get('currentRoute');
38 }
39
ee122a75
NL
40 $form = $this->createForm(SearchEntryType::class);
41
42 $form->handleRequest($request);
43
21e7ccef 44 if ($form->isSubmitted() && $form->isValid()) {
ee122a75
NL
45 return $this->showEntries('search', $request, $page);
46 }
47
48 return $this->render('WallabagCoreBundle:Entry:search_form.html.twig', [
49 'form' => $form->createView(),
49b042df 50 'currentRoute' => $currentRoute,
ee122a75
NL
51 ]);
52 }
53
b84a8055 54 /**
3d2b2d62
J
55 * @param Request $request
56 *
053b9568 57 * @Route("/new-entry", name="new_entry")
3d2b2d62 58 *
b84a8055
NL
59 * @return \Symfony\Component\HttpFoundation\Response
60 */
053b9568 61 public function addEntryFormAction(Request $request)
b84a8055 62 {
3b815d2d 63 $entry = new Entry($this->getUser());
b84a8055 64
5c895a7f 65 $form = $this->createForm(NewEntryType::class, $entry);
b84a8055
NL
66
67 $form->handleRequest($request);
68
21e7ccef 69 if ($form->isSubmitted() && $form->isValid()) {
b00a89e0 70 $existingEntry = $this->checkIfEntryAlreadyExists($entry);
dda57bb9 71
5a4bbcc9 72 if (false !== $existingEntry) {
dda57bb9
NL
73 $this->get('session')->getFlashBag()->add(
74 'notice',
4094ea47 75 $this->get('translator')->trans('flashes.entry.notice.entry_already_saved', ['%date%' => $existingEntry->getCreatedAt()->format('d-m-Y')])
dda57bb9
NL
76 );
77
4094ea47 78 return $this->redirect($this->generateUrl('view', ['id' => $existingEntry->getId()]));
dda57bb9
NL
79 }
80
59b97fae 81 $this->updateEntry($entry);
39ba51ca 82
59b97fae
JB
83 $em = $this->getDoctrine()->getManager();
84 $em->persist($entry);
85 $em->flush();
b84a8055 86
e0597476
JB
87 // entry saved, dispatch event about it!
88 $this->get('event_dispatcher')->dispatch(EntrySavedEvent::NAME, new EntrySavedEvent($entry));
89
b84a8055
NL
90 return $this->redirect($this->generateUrl('homepage'));
91 }
92
4094ea47 93 return $this->render('WallabagCoreBundle:Entry:new_form.html.twig', [
b84a8055 94 'form' => $form->createView(),
4094ea47 95 ]);
b84a8055
NL
96 }
97
880a0e1c
NL
98 /**
99 * @param Request $request
100 *
101 * @Route("/bookmarklet", name="bookmarklet")
102 *
103 * @return \Symfony\Component\HttpFoundation\Response
104 */
5f8a7857 105 public function addEntryViaBookmarkletAction(Request $request)
880a0e1c
NL
106 {
107 $entry = new Entry($this->getUser());
108 $entry->setUrl($request->get('url'));
f652f41d 109
b00a89e0 110 if (false === $this->checkIfEntryAlreadyExists($entry)) {
f652f41d 111 $this->updateEntry($entry);
59b97fae
JB
112
113 $em = $this->getDoctrine()->getManager();
114 $em->persist($entry);
115 $em->flush();
e0597476
JB
116
117 // entry saved, dispatch event about it!
118 $this->get('event_dispatcher')->dispatch(EntrySavedEvent::NAME, new EntrySavedEvent($entry));
f652f41d 119 }
880a0e1c
NL
120
121 return $this->redirect($this->generateUrl('homepage'));
122 }
123
053b9568 124 /**
053b9568
NL
125 * @Route("/new", name="new")
126 *
127 * @return \Symfony\Component\HttpFoundation\Response
128 */
4d0ec0e7 129 public function addEntryAction()
053b9568
NL
130 {
131 return $this->render('WallabagCoreBundle:Entry:new.html.twig');
132 }
133
82d6d9cb
JB
134 /**
135 * Edit an entry content.
136 *
137 * @param Request $request
138 * @param Entry $entry
139 *
140 * @Route("/edit/{id}", requirements={"id" = "\d+"}, name="edit")
141 *
142 * @return \Symfony\Component\HttpFoundation\Response
143 */
144 public function editEntryAction(Request $request, Entry $entry)
145 {
146 $this->checkUserAction($entry);
147
5c895a7f 148 $form = $this->createForm(EditEntryType::class, $entry);
82d6d9cb
JB
149
150 $form->handleRequest($request);
151
21e7ccef 152 if ($form->isSubmitted() && $form->isValid()) {
82d6d9cb
JB
153 $em = $this->getDoctrine()->getManager();
154 $em->persist($entry);
155 $em->flush();
156
157 $this->get('session')->getFlashBag()->add(
158 'notice',
4204a06b 159 'flashes.entry.notice.entry_updated'
82d6d9cb
JB
160 );
161
4094ea47 162 return $this->redirect($this->generateUrl('view', ['id' => $entry->getId()]));
82d6d9cb
JB
163 }
164
4094ea47 165 return $this->render('WallabagCoreBundle:Entry:edit.html.twig', [
82d6d9cb 166 'form' => $form->createView(),
4094ea47 167 ]);
82d6d9cb
JB
168 }
169
89659c9e
NL
170 /**
171 * Shows all entries for current user.
172 *
173 * @param Request $request
174 * @param int $page
175 *
176 * @Route("/all/list/{page}", name="all", defaults={"page" = "1"})
177 *
178 * @return \Symfony\Component\HttpFoundation\Response
179 */
180 public function showAllAction(Request $request, $page)
181 {
2b7a4889 182 return $this->showEntries('all', $request, $page);
89659c9e
NL
183 }
184
9d50517c 185 /**
4346a860 186 * Shows unread entries for current user.
163eae0b 187 *
26864574
NL
188 * @param Request $request
189 * @param int $page
190 *
9fb6ac83 191 * @Route("/unread/list/{page}", name="unread", defaults={"page" = "1"})
3d2b2d62 192 *
163eae0b 193 * @return \Symfony\Component\HttpFoundation\Response
9d50517c 194 */
26864574 195 public function showUnreadAction(Request $request, $page)
9d50517c 196 {
5c072d2b 197 // load the quickstart if no entry in database
3ef055ce 198 if (1 === (int) $page && 0 === $this->get('wallabag_core.entry_repository')->countAllEntriesByUser($this->getUser()->getId())) {
5c072d2b
NL
199 return $this->redirect($this->generateUrl('quickstart'));
200 }
201
0ab7404f 202 return $this->showEntries('unread', $request, $page);
9d50517c 203 }
bd9f0815
NL
204
205 /**
4346a860 206 * Shows read entries for current user.
163eae0b 207 *
26864574
NL
208 * @param Request $request
209 * @param int $page
210 *
9fb6ac83 211 * @Route("/archive/list/{page}", name="archive", defaults={"page" = "1"})
3d2b2d62 212 *
163eae0b 213 * @return \Symfony\Component\HttpFoundation\Response
bd9f0815 214 */
26864574 215 public function showArchiveAction(Request $request, $page)
bd9f0815 216 {
0ab7404f
JB
217 return $this->showEntries('archive', $request, $page);
218 }
219
220 /**
221 * Shows starred entries for current user.
222 *
223 * @param Request $request
224 * @param int $page
225 *
226 * @Route("/starred/list/{page}", name="starred", defaults={"page" = "1"})
227 *
228 * @return \Symfony\Component\HttpFoundation\Response
229 */
230 public function showStarredAction(Request $request, $page)
231 {
232 return $this->showEntries('starred', $request, $page);
233 }
26864574 234
bd9f0815 235 /**
4346a860 236 * Shows entry content.
163eae0b 237 *
3d2b2d62
J
238 * @param Entry $entry
239 *
bd9f0815 240 * @Route("/view/{id}", requirements={"id" = "\d+"}, name="view")
3d2b2d62 241 *
163eae0b 242 * @return \Symfony\Component\HttpFoundation\Response
bd9f0815 243 */
be463487 244 public function viewAction(Entry $entry)
bd9f0815 245 {
3d2b2d62
J
246 $this->checkUserAction($entry);
247
bd9f0815 248 return $this->render(
ad4d1caa 249 'WallabagCoreBundle:Entry:entry.html.twig',
4094ea47 250 ['entry' => $entry]
bd9f0815 251 );
163eae0b
NL
252 }
253
831b02aa
JB
254 /**
255 * Reload an entry.
256 * Refetch content from the website and make it readable again.
257 *
258 * @param Entry $entry
259 *
260 * @Route("/reload/{id}", requirements={"id" = "\d+"}, name="reload_entry")
261 *
262 * @return \Symfony\Component\HttpFoundation\RedirectResponse
263 */
264 public function reloadAction(Entry $entry)
265 {
266 $this->checkUserAction($entry);
267
59b97fae 268 $this->updateEntry($entry, 'entry_reloaded');
831b02aa 269
2297d60f
JB
270 // if refreshing entry failed, don't save it
271 if ($this->getParameter('wallabag_core.fetching_error_message') === $entry->getContent()) {
272 $bag = $this->get('session')->getFlashBag();
273 $bag->clear();
274 $bag->add('notice', 'flashes.entry.notice.entry_reloaded_failed');
275
276 return $this->redirect($this->generateUrl('view', ['id' => $entry->getId()]));
277 }
278
59b97fae
JB
279 $em = $this->getDoctrine()->getManager();
280 $em->persist($entry);
281 $em->flush();
831b02aa 282
e0597476
JB
283 // entry saved, dispatch event about it!
284 $this->get('event_dispatcher')->dispatch(EntrySavedEvent::NAME, new EntrySavedEvent($entry));
285
4094ea47 286 return $this->redirect($this->generateUrl('view', ['id' => $entry->getId()]));
831b02aa
JB
287 }
288
163eae0b 289 /**
4346a860 290 * Changes read status for an entry.
163eae0b 291 *
3d2b2d62
J
292 * @param Request $request
293 * @param Entry $entry
294 *
163eae0b 295 * @Route("/archive/{id}", requirements={"id" = "\d+"}, name="archive_entry")
3d2b2d62 296 *
163eae0b
NL
297 * @return \Symfony\Component\HttpFoundation\RedirectResponse
298 */
be463487 299 public function toggleArchiveAction(Request $request, Entry $entry)
163eae0b 300 {
3d2b2d62
J
301 $this->checkUserAction($entry);
302
163eae0b
NL
303 $entry->toggleArchive();
304 $this->getDoctrine()->getManager()->flush();
305
4204a06b
JB
306 $message = 'flashes.entry.notice.entry_unarchived';
307 if ($entry->isArchived()) {
308 $message = 'flashes.entry.notice.entry_archived';
309 }
310
163eae0b
NL
311 $this->get('session')->getFlashBag()->add(
312 'notice',
4204a06b 313 $message
163eae0b
NL
314 );
315
af497a64
NL
316 $redirectUrl = $this->get('wallabag_core.helper.redirect')->to($request->headers->get('referer'));
317
318 return $this->redirect($redirectUrl);
163eae0b
NL
319 }
320
321 /**
eecd7e40 322 * Changes starred status for an entry.
163eae0b 323 *
3d2b2d62
J
324 * @param Request $request
325 * @param Entry $entry
326 *
163eae0b 327 * @Route("/star/{id}", requirements={"id" = "\d+"}, name="star_entry")
3d2b2d62 328 *
163eae0b
NL
329 * @return \Symfony\Component\HttpFoundation\RedirectResponse
330 */
be463487 331 public function toggleStarAction(Request $request, Entry $entry)
163eae0b 332 {
3d2b2d62
J
333 $this->checkUserAction($entry);
334
163eae0b 335 $entry->toggleStar();
a991c46e 336 $entry->updateStar($entry->isStarred());
163eae0b
NL
337 $this->getDoctrine()->getManager()->flush();
338
4204a06b
JB
339 $message = 'flashes.entry.notice.entry_unstarred';
340 if ($entry->isStarred()) {
341 $message = 'flashes.entry.notice.entry_starred';
342 }
343
163eae0b
NL
344 $this->get('session')->getFlashBag()->add(
345 'notice',
4204a06b 346 $message
163eae0b
NL
347 );
348
af497a64
NL
349 $redirectUrl = $this->get('wallabag_core.helper.redirect')->to($request->headers->get('referer'));
350
351 return $this->redirect($redirectUrl);
163eae0b
NL
352 }
353
354 /**
2863bf2a 355 * Deletes entry and redirect to the homepage or the last viewed page.
163eae0b 356 *
16a3d04c 357 * @param Entry $entry
3d2b2d62 358 *
163eae0b 359 * @Route("/delete/{id}", requirements={"id" = "\d+"}, name="delete_entry")
3d2b2d62 360 *
163eae0b
NL
361 * @return \Symfony\Component\HttpFoundation\RedirectResponse
362 */
18d5f454 363 public function deleteEntryAction(Request $request, Entry $entry)
163eae0b 364 {
3d2b2d62
J
365 $this->checkUserAction($entry);
366
2863bf2a
JB
367 // generates the view url for this entry to check for redirection later
368 // to avoid redirecting to the deleted entry. Ugh.
369 $url = $this->generateUrl(
370 'view',
4094ea47 371 ['id' => $entry->getId()],
ce0e9ec3 372 UrlGeneratorInterface::ABSOLUTE_PATH
2863bf2a
JB
373 );
374
e0597476
JB
375 // entry deleted, dispatch event about it!
376 $this->get('event_dispatcher')->dispatch(EntryDeletedEvent::NAME, new EntryDeletedEvent($entry));
377
1d147791
NL
378 $em = $this->getDoctrine()->getManager();
379 $em->remove($entry);
380 $em->flush();
163eae0b
NL
381
382 $this->get('session')->getFlashBag()->add(
383 'notice',
4204a06b 384 'flashes.entry.notice.entry_deleted'
163eae0b 385 );
bd9f0815 386
ce0e9ec3
JB
387 // don't redirect user to the deleted entry (check that the referer doesn't end with the same url)
388 $referer = $request->headers->get('referer');
f808b016 389 $to = (1 !== preg_match('#' . $url . '$#i', $referer) ? $referer : null);
af497a64
NL
390
391 $redirectUrl = $this->get('wallabag_core.helper.redirect')->to($to);
392
393 return $this->redirect($redirectUrl);
bd9f0815 394 }
3d2b2d62 395
f1be7af4
NL
396 /**
397 * Get public URL for entry (and generate it if necessary).
398 *
399 * @param Entry $entry
400 *
401 * @Route("/share/{id}", requirements={"id" = "\d+"}, name="share")
402 *
403 * @return \Symfony\Component\HttpFoundation\Response
404 */
405 public function shareAction(Entry $entry)
406 {
407 $this->checkUserAction($entry);
408
7239082a
NL
409 if (null === $entry->getUid()) {
410 $entry->generateUid();
eddda878
JB
411
412 $em = $this->getDoctrine()->getManager();
413 $em->persist($entry);
414 $em->flush();
f1be7af4
NL
415 }
416
417 return $this->redirect($this->generateUrl('share_entry', [
7239082a 418 'uid' => $entry->getUid(),
f1be7af4
NL
419 ]));
420 }
421
422 /**
423 * Disable public sharing for an entry.
424 *
425 * @param Entry $entry
426 *
427 * @Route("/share/delete/{id}", requirements={"id" = "\d+"}, name="delete_share")
428 *
429 * @return \Symfony\Component\HttpFoundation\Response
430 */
431 public function deleteShareAction(Entry $entry)
432 {
433 $this->checkUserAction($entry);
434
7239082a 435 $entry->cleanUid();
eddda878 436
f1be7af4
NL
437 $em = $this->getDoctrine()->getManager();
438 $em->persist($entry);
439 $em->flush();
440
441 return $this->redirect($this->generateUrl('view', [
442 'id' => $entry->getId(),
443 ]));
444 }
445
d0545b6b 446 /**
eddda878 447 * Ability to view a content publicly.
f3d0cb91
NL
448 *
449 * @param Entry $entry
450 *
7239082a 451 * @Route("/share/{uid}", requirements={"uid" = ".+"}, name="share_entry")
eddda878 452 * @Cache(maxage="25200", smaxage="25200", public=true)
f3d0cb91
NL
453 *
454 * @return \Symfony\Component\HttpFoundation\Response
455 */
d0545b6b 456 public function shareEntryAction(Entry $entry)
f3d0cb91 457 {
eddda878
JB
458 if (!$this->get('craue_config')->get('share_public')) {
459 throw $this->createAccessDeniedException('Sharing an entry is disabled for this user.');
460 }
461
f3d0cb91 462 return $this->render(
2ff9991a 463 '@WallabagCore/themes/common/Entry/share.html.twig',
eddda878 464 ['entry' => $entry]
f3d0cb91
NL
465 );
466 }
b6520f0b
NL
467
468 /**
469 * Shows untagged articles for current user.
470 *
471 * @param Request $request
472 * @param int $page
473 *
474 * @Route("/untagged/list/{page}", name="untagged", defaults={"page" = "1"})
475 *
476 * @return \Symfony\Component\HttpFoundation\Response
477 */
478 public function showUntaggedEntriesAction(Request $request, $page)
479 {
480 return $this->showEntries('untagged', $request, $page);
481 }
f808b016
JB
482
483 /**
484 * Fetch content and update entry.
485 * In case it fails, $entry->getContent will return an error message.
486 *
487 * @param Entry $entry
488 * @param string $prefixMessage Should be the translation key: entry_saved or entry_reloaded
489 */
490 private function updateEntry(Entry $entry, $prefixMessage = 'entry_saved')
491 {
492 $message = 'flashes.entry.notice.' . $prefixMessage;
493
494 try {
495 $this->get('wallabag_core.content_proxy')->updateEntry($entry, $entry->getUrl());
496 } catch (\Exception $e) {
497 $this->get('logger')->error('Error while saving an entry', [
498 'exception' => $e,
499 'entry' => $entry,
500 ]);
501
502 $message = 'flashes.entry.notice.' . $prefixMessage . '_failed';
503 }
504
af29e1bf
KD
505 if (empty($entry->getDomainName())) {
506 $this->get('wallabag_core.content_proxy')->setEntryDomainName($entry);
507 }
508
509 if (empty($entry->getTitle())) {
510 $this->get('wallabag_core.content_proxy')->setDefaultEntryTitle($entry);
511 }
512
f808b016
JB
513 $this->get('session')->getFlashBag()->add('notice', $message);
514 }
515
516 /**
517 * Global method to retrieve entries depending on the given type
518 * It returns the response to be send.
519 *
520 * @param string $type Entries type: unread, starred or archive
521 * @param Request $request
522 * @param int $page
523 *
524 * @return \Symfony\Component\HttpFoundation\Response
525 */
526 private function showEntries($type, Request $request, $page)
527 {
528 $repository = $this->get('wallabag_core.entry_repository');
529 $searchTerm = (isset($request->get('search_entry')['term']) ? $request->get('search_entry')['term'] : '');
530 $currentRoute = (null !== $request->query->get('currentRoute') ? $request->query->get('currentRoute') : '');
531
532 switch ($type) {
533 case 'search':
534 $qb = $repository->getBuilderForSearchByUser($this->getUser()->getId(), $searchTerm, $currentRoute);
f808b016
JB
535 break;
536 case 'untagged':
537 $qb = $repository->getBuilderForUntaggedByUser($this->getUser()->getId());
f808b016
JB
538 break;
539 case 'starred':
540 $qb = $repository->getBuilderForStarredByUser($this->getUser()->getId());
541 break;
542 case 'archive':
543 $qb = $repository->getBuilderForArchiveByUser($this->getUser()->getId());
544 break;
545 case 'unread':
546 $qb = $repository->getBuilderForUnreadByUser($this->getUser()->getId());
547 break;
548 case 'all':
549 $qb = $repository->getBuilderForAllByUser($this->getUser()->getId());
550 break;
551 default:
552 throw new \InvalidArgumentException(sprintf('Type "%s" is not implemented.', $type));
553 }
554
555 $form = $this->createForm(EntryFilterType::class);
556
557 if ($request->query->has($form->getName())) {
558 // manually bind values from the request
559 $form->submit($request->query->get($form->getName()));
560
561 // build the query from the given form object
562 $this->get('lexik_form_filter.query_builder_updater')->addFilterConditions($form, $qb);
563 }
564
565 $pagerAdapter = new DoctrineORMAdapter($qb->getQuery(), true, false);
566
567 $entries = $this->get('wallabag_core.helper.prepare_pager_for_entries')->prepare($pagerAdapter);
568
569 try {
570 $entries->setCurrentPage($page);
571 } catch (OutOfRangeCurrentPageException $e) {
572 if ($page > 1) {
573 return $this->redirect($this->generateUrl($type, ['page' => $entries->getNbPages()]), 302);
574 }
575 }
576
577 return $this->render(
578 'WallabagCoreBundle:Entry:entries.html.twig', [
579 'form' => $form->createView(),
580 'entries' => $entries,
581 'currentPage' => $page,
582 'searchTerm' => $searchTerm,
38321586 583 'isFiltered' => $form->isSubmitted(),
f808b016
JB
584 ]
585 );
586 }
587
588 /**
589 * Check if the logged user can manage the given entry.
590 *
591 * @param Entry $entry
592 */
593 private function checkUserAction(Entry $entry)
594 {
595 if (null === $this->getUser() || $this->getUser()->getId() !== $entry->getUser()->getId()) {
596 throw $this->createAccessDeniedException('You can not access this entry.');
597 }
598 }
599
600 /**
601 * Check for existing entry, if it exists, redirect to it with a message.
602 *
603 * @param Entry $entry
604 *
605 * @return Entry|bool
606 */
607 private function checkIfEntryAlreadyExists(Entry $entry)
608 {
609 return $this->get('wallabag_core.entry_repository')->findByUrlAndUserId($entry->getUrl(), $this->getUser()->getId());
610 }
9d50517c 611}