tagController.php 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234
  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', plaintext: true));
  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(
  115. _t('feedback.tag.updated'),
  116. $url_redirect,
  117. showNotification: FreshRSS_Context::userConf()->good_notification_timeout > 0
  118. );
  119. } else {
  120. Minz_Request::bad(_t('feedback.tag.error'), $url_redirect);
  121. }
  122. }
  123. }
  124. public function getTagsForEntryAction(): void {
  125. if (!FreshRSS_Auth::hasAccess() && !FreshRSS_Context::systemConf()->allow_anonymous) {
  126. Minz_Error::error(403);
  127. }
  128. $this->view->_layout(null);
  129. header('Content-Type: application/json; charset=UTF-8');
  130. header('Cache-Control: private, no-cache, no-store, must-revalidate');
  131. $id_entry = Minz_Request::paramString('id_entry');
  132. $tagDAO = FreshRSS_Factory::createTagDao();
  133. $this->view->tagsForEntry = $tagDAO->getTagsForEntry($id_entry);
  134. }
  135. public function addAction(): void {
  136. if (!FreshRSS_Auth::hasAccess()) {
  137. Minz_Error::error(403);
  138. }
  139. if (!Minz_Request::isPost()) {
  140. Minz_Error::error(405);
  141. }
  142. $url_redirect = ['c' => 'tag', 'a' => 'index'];
  143. $name = Minz_Request::paramString('name');
  144. $catDAO = FreshRSS_Factory::createCategoryDao();
  145. if ($catDAO->searchByName($name) !== null) {
  146. Minz_Request::bad(_t('feedback.sub.category.name_exists'), $url_redirect);
  147. }
  148. $tagDAO = FreshRSS_Factory::createTagDao();
  149. if ($tagDAO->searchByName($name) !== null) {
  150. Minz_Request::bad(_t('feedback.tag.name_exists', $name), $url_redirect);
  151. }
  152. $tagDAO->addTag(['name' => $name]);
  153. Minz_Request::good(
  154. _t('feedback.tag.created', $name),
  155. $url_redirect,
  156. showNotification: FreshRSS_Context::userConf()->good_notification_timeout > 0
  157. );
  158. }
  159. /**
  160. * @throws Minz_ConfigurationNamespaceException
  161. * @throws Minz_PDOConnectionException
  162. */
  163. public function renameAction(): void {
  164. if (!FreshRSS_Auth::hasAccess()) {
  165. Minz_Error::error(403);
  166. }
  167. if (!Minz_Request::isPost()) {
  168. Minz_Error::error(405);
  169. }
  170. $targetName = Minz_Request::paramString('name');
  171. $sourceId = Minz_Request::paramInt('id_tag');
  172. if ($targetName == '' || $sourceId == 0) {
  173. Minz_Error::error(400);
  174. return;
  175. }
  176. $tagDAO = FreshRSS_Factory::createTagDao();
  177. $sourceTag = $tagDAO->searchById($sourceId);
  178. $sourceName = $sourceTag === null ? '' : $sourceTag->name();
  179. $targetTag = $tagDAO->searchByName($targetName);
  180. if ($targetTag === null) {
  181. // There is no existing tag with the same target name
  182. $tagDAO->updateTagName($sourceId, $targetName);
  183. } else {
  184. // There is an existing tag with the same target name
  185. $tagDAO->updateEntryTag($sourceId, $targetTag->id());
  186. $tagDAO->deleteTag($sourceId);
  187. }
  188. Minz_Request::good(
  189. _t('feedback.tag.renamed', $sourceName, $targetName),
  190. ['c' => 'tag', 'a' => 'index'],
  191. showNotification: FreshRSS_Context::userConf()->good_notification_timeout > 0);
  192. }
  193. public function indexAction(): void {
  194. FreshRSS_View::prependTitle(_t('sub.menu.label_management') . ' · ');
  195. if (!FreshRSS_Auth::hasAccess()) {
  196. Minz_Error::error(403);
  197. }
  198. $tagDAO = FreshRSS_Factory::createTagDao();
  199. $this->view->tags = $tagDAO->listTags(precounts: true);
  200. }
  201. public static function escapeForSearch(string $tag): string {
  202. $tag = htmlspecialchars_decode($tag, ENT_QUOTES);
  203. return str_replace([' ', '(', ')'], ['+', '\\(', '\\)'], $tag);
  204. }
  205. }