tagController.php 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216
  1. <?php
  2. declare(strict_types=1);
  3. /**
  4. * Controller to handle every tag actions.
  5. */
  6. class FreshRSS_tag_Controller extends FreshRSS_ActionController {
  7. /**
  8. * JavaScript request or not.
  9. */
  10. private bool $ajax = false;
  11. /**
  12. * This action is called before every other action in that class. It is
  13. * the common boilerplate for every action. It is triggered by the
  14. * underlying framework.
  15. */
  16. #[\Override]
  17. public function firstAction(): void {
  18. // If ajax request, we do not print layout
  19. $this->ajax = Minz_Request::paramBoolean('ajax');
  20. if ($this->ajax) {
  21. $this->view->_layout(null);
  22. }
  23. }
  24. /**
  25. * This action adds (checked=true) or removes (checked=false) a tag to an entry.
  26. */
  27. public function tagEntryAction(): void {
  28. if (!FreshRSS_Auth::hasAccess()) {
  29. Minz_Error::error(403);
  30. }
  31. if (Minz_Request::isPost()) {
  32. $id_tag = Minz_Request::paramInt('id_tag');
  33. $name_tag = Minz_Request::paramString('name_tag');
  34. $id_entry = Minz_Request::paramString('id_entry');
  35. $checked = Minz_Request::paramBoolean('checked');
  36. if ($id_entry != '') {
  37. $tagDAO = FreshRSS_Factory::createTagDao();
  38. if ($id_tag == 0 && $name_tag !== '' && $checked) {
  39. if (($existing_tag = $tagDAO->searchByName($name_tag)) !== null) {
  40. // Use existing tag
  41. $tagDAO->tagEntry($existing_tag->id(), $id_entry, $checked);
  42. } else {
  43. //Create new tag
  44. $id_tag = $tagDAO->addTag(['name' => $name_tag]);
  45. }
  46. }
  47. if ($id_tag != false) {
  48. $tagDAO->tagEntry($id_tag, $id_entry, $checked);
  49. }
  50. }
  51. } else {
  52. Minz_Error::error(405);
  53. }
  54. if (!$this->ajax) {
  55. Minz_Request::forward([
  56. 'c' => 'index',
  57. 'a' => 'index',
  58. ], true);
  59. }
  60. }
  61. public function deleteAction(): void {
  62. if (!FreshRSS_Auth::hasAccess()) {
  63. Minz_Error::error(403);
  64. }
  65. if (Minz_Request::isPost()) {
  66. $id_tag = Minz_Request::paramInt('id_tag');
  67. if ($id_tag !== 0) {
  68. $tagDAO = FreshRSS_Factory::createTagDao();
  69. $tagDAO->deleteTag($id_tag);
  70. }
  71. } else {
  72. Minz_Error::error(405);
  73. }
  74. if (!$this->ajax) {
  75. Minz_Request::forward([
  76. 'c' => 'tag',
  77. 'a' => 'index',
  78. ], true);
  79. }
  80. }
  81. /**
  82. * This action updates the given tag.
  83. */
  84. public function updateAction(): void {
  85. if (!FreshRSS_Auth::hasAccess()) {
  86. Minz_Error::error(403);
  87. return;
  88. }
  89. if (Minz_Request::paramBoolean('ajax')) {
  90. $this->view->_layout(null);
  91. }
  92. $tagDAO = FreshRSS_Factory::createTagDao();
  93. $id = Minz_Request::paramInt('id');
  94. $tag = $tagDAO->searchById($id);
  95. if ($id === 0 || $tag === null) {
  96. Minz_Error::error(404);
  97. return;
  98. }
  99. $this->view->tag = $tag;
  100. FreshRSS_View::prependTitle($tag->name() . ' · ' . _t('sub.title') . ' · ');
  101. if (Minz_Request::isPost()) {
  102. invalidateHttpCache();
  103. $ok = true;
  104. if ($tag->name() !== Minz_Request::paramString('name')) {
  105. $ok = $tagDAO->updateTagName($tag->id(), Minz_Request::paramString('name')) !== false;
  106. }
  107. if ($ok) {
  108. $tag->_filtersAction('label', Minz_Request::paramTextToArray('filteractions_label'));
  109. $ok = $tagDAO->updateTagAttributes($tag->id(), $tag->attributes()) !== false;
  110. }
  111. invalidateHttpCache();
  112. $url_redirect = ['c' => 'tag', 'a' => 'update', 'params' => ['id' => $id]];
  113. if ($ok) {
  114. Minz_Request::good(_t('feedback.tag.updated'), $url_redirect);
  115. } else {
  116. Minz_Request::bad(_t('feedback.tag.error'), $url_redirect);
  117. }
  118. }
  119. }
  120. public function getTagsForEntryAction(): void {
  121. if (!FreshRSS_Auth::hasAccess() && !FreshRSS_Context::systemConf()->allow_anonymous) {
  122. Minz_Error::error(403);
  123. }
  124. $this->view->_layout(null);
  125. header('Content-Type: application/json; charset=UTF-8');
  126. header('Cache-Control: private, no-cache, no-store, must-revalidate');
  127. $id_entry = Minz_Request::paramString('id_entry');
  128. $tagDAO = FreshRSS_Factory::createTagDao();
  129. $this->view->tagsForEntry = $tagDAO->getTagsForEntry($id_entry);
  130. }
  131. public function addAction(): void {
  132. if (!FreshRSS_Auth::hasAccess()) {
  133. Minz_Error::error(403);
  134. }
  135. if (!Minz_Request::isPost()) {
  136. Minz_Error::error(405);
  137. }
  138. $name = Minz_Request::paramString('name');
  139. $tagDAO = FreshRSS_Factory::createTagDao();
  140. if (strlen($name) > 0 && null === $tagDAO->searchByName($name)) {
  141. $tagDAO->addTag(['name' => $name]);
  142. Minz_Request::good(_t('feedback.tag.created', $name), ['c' => 'tag', 'a' => 'index']);
  143. }
  144. Minz_Request::bad(_t('feedback.tag.name_exists', $name), ['c' => 'tag', 'a' => 'index']);
  145. }
  146. /**
  147. * @throws Minz_ConfigurationNamespaceException
  148. * @throws Minz_PDOConnectionException
  149. */
  150. public function renameAction(): void {
  151. if (!FreshRSS_Auth::hasAccess()) {
  152. Minz_Error::error(403);
  153. }
  154. if (!Minz_Request::isPost()) {
  155. Minz_Error::error(405);
  156. }
  157. $targetName = Minz_Request::paramString('name');
  158. $sourceId = Minz_Request::paramInt('id_tag');
  159. if ($targetName == '' || $sourceId == 0) {
  160. Minz_Error::error(400);
  161. return;
  162. }
  163. $tagDAO = FreshRSS_Factory::createTagDao();
  164. $sourceTag = $tagDAO->searchById($sourceId);
  165. $sourceName = $sourceTag === null ? '' : $sourceTag->name();
  166. $targetTag = $tagDAO->searchByName($targetName);
  167. if ($targetTag === null) {
  168. // There is no existing tag with the same target name
  169. $tagDAO->updateTagName($sourceId, $targetName);
  170. } else {
  171. // There is an existing tag with the same target name
  172. $tagDAO->updateEntryTag($sourceId, $targetTag->id());
  173. $tagDAO->deleteTag($sourceId);
  174. }
  175. Minz_Request::good(_t('feedback.tag.renamed', $sourceName, $targetName), ['c' => 'tag', 'a' => 'index']);
  176. }
  177. public function indexAction(): void {
  178. FreshRSS_View::prependTitle(_t('sub.menu.label_management') . ' · ');
  179. if (!FreshRSS_Auth::hasAccess()) {
  180. Minz_Error::error(403);
  181. }
  182. $tagDAO = FreshRSS_Factory::createTagDao();
  183. $this->view->tags = $tagDAO->listTags(precounts: true);
  184. }
  185. public static function escapeForSearch(string $tag): string {
  186. $tag = htmlspecialchars_decode($tag, ENT_QUOTES);
  187. return str_replace([' ', '(', ')'], ['+', '\\(', '\\)'], $tag);
  188. }
  189. }