News.vue 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222
  1. <template>
  2. <div>
  3. <page-metadata title="Admin | News" />
  4. <div class="container">
  5. <div class="button-row">
  6. <button class="is-primary button" @click="edit()">
  7. Create News Item
  8. </button>
  9. </div>
  10. <advanced-table
  11. :column-default="columnDefault"
  12. :columns="columns"
  13. :filters="filters"
  14. data-action="news.getData"
  15. name="admin-news"
  16. max-width="1200"
  17. :events="events"
  18. >
  19. <template #column-options="slotProps">
  20. <div class="row-options">
  21. <button
  22. class="
  23. button
  24. is-primary
  25. icon-with-button
  26. material-icons
  27. "
  28. @click="edit(slotProps.item._id)"
  29. content="Edit News"
  30. v-tippy
  31. >
  32. edit
  33. </button>
  34. <quick-confirm
  35. @confirm="remove(slotProps.item._id)"
  36. :disabled="slotProps.item.removed"
  37. >
  38. <button
  39. class="
  40. button
  41. is-danger
  42. icon-with-button
  43. material-icons
  44. "
  45. content="Remove News"
  46. v-tippy
  47. >
  48. delete_forever
  49. </button>
  50. </quick-confirm>
  51. </div>
  52. </template>
  53. <template #column-status="slotProps">
  54. <span :title="slotProps.item.status">{{
  55. slotProps.item.status
  56. }}</span>
  57. </template>
  58. <template #column-title="slotProps">
  59. <span :title="slotProps.item.title">{{
  60. slotProps.item.title
  61. }}</span>
  62. </template>
  63. <template #column-createdBy="slotProps">
  64. <user-id-to-username
  65. :user-id="slotProps.item.createdBy"
  66. :alt="slotProps.item.createdBy"
  67. :link="true"
  68. />
  69. </template>
  70. <template #column-markdown="slotProps">
  71. <span :title="slotProps.item.markdown">{{
  72. slotProps.item.markdown
  73. }}</span>
  74. </template>
  75. </advanced-table>
  76. </div>
  77. <edit-news
  78. v-if="modals.editNews"
  79. :news-id="editingNewsId"
  80. sector="admin"
  81. />
  82. </div>
  83. </template>
  84. <script>
  85. import { mapActions, mapState, mapGetters } from "vuex";
  86. import { defineAsyncComponent } from "vue";
  87. import Toast from "toasters";
  88. import AdvancedTable from "@/components/AdvancedTable.vue";
  89. import QuickConfirm from "@/components/QuickConfirm.vue";
  90. import UserIdToUsername from "@/components/UserIdToUsername.vue";
  91. export default {
  92. components: {
  93. AdvancedTable,
  94. QuickConfirm,
  95. UserIdToUsername,
  96. EditNews: defineAsyncComponent(() =>
  97. import("@/components/modals/EditNews.vue")
  98. )
  99. },
  100. data() {
  101. return {
  102. editingNewsId: "",
  103. columnDefault: {
  104. sortable: true,
  105. hidable: true,
  106. defaultVisibility: "shown",
  107. draggable: true,
  108. resizable: true,
  109. minWidth: 150,
  110. maxWidth: 600
  111. },
  112. columns: [
  113. {
  114. name: "options",
  115. displayName: "Options",
  116. properties: ["_id"],
  117. sortable: false,
  118. hidable: false,
  119. resizable: false,
  120. minWidth: 85,
  121. defaultWidth: 85
  122. },
  123. {
  124. name: "status",
  125. displayName: "Status",
  126. properties: ["status"],
  127. sortProperty: "status",
  128. defaultWidth: 150
  129. },
  130. {
  131. name: "title",
  132. displayName: "Title",
  133. properties: ["title"],
  134. sortProperty: "title"
  135. },
  136. {
  137. name: "createdBy",
  138. displayName: "Created By",
  139. properties: ["createdBy"],
  140. sortProperty: "createdBy",
  141. defaultWidth: 150
  142. },
  143. {
  144. name: "markdown",
  145. displayName: "Markdown",
  146. properties: ["markdown"],
  147. sortProperty: "markdown"
  148. }
  149. ],
  150. filters: [
  151. {
  152. name: "status",
  153. displayName: "Status",
  154. property: "status",
  155. filterTypes: ["contains", "exact", "regex"],
  156. defaultFilterType: "contains"
  157. },
  158. {
  159. name: "title",
  160. displayName: "Title",
  161. property: "title",
  162. filterTypes: ["contains", "exact", "regex"],
  163. defaultFilterType: "contains"
  164. },
  165. {
  166. name: "createdBy",
  167. displayName: "Created By",
  168. property: "createdBy",
  169. filterTypes: ["contains", "exact", "regex"],
  170. defaultFilterType: "contains"
  171. },
  172. {
  173. name: "markdown",
  174. displayName: "Markdown",
  175. property: "markdown",
  176. filterTypes: ["contains", "exact", "regex"],
  177. defaultFilterType: "contains"
  178. }
  179. ],
  180. events: {
  181. adminRoom: "news",
  182. updated: {
  183. event: "admin.news.updated",
  184. id: "news._id",
  185. item: "news"
  186. },
  187. removed: {
  188. event: "admin.news.deleted",
  189. id: "newsId"
  190. }
  191. }
  192. };
  193. },
  194. computed: {
  195. ...mapState("modalVisibility", {
  196. modals: state => state.modals
  197. }),
  198. ...mapGetters({
  199. socket: "websockets/getSocket"
  200. })
  201. },
  202. methods: {
  203. edit(id) {
  204. if (id) this.editingNewsId = id;
  205. else this.editingNewsId = "";
  206. this.openModal("editNews");
  207. },
  208. remove(id) {
  209. this.socket.dispatch(
  210. "news.remove",
  211. id,
  212. res => new Toast(res.message)
  213. );
  214. },
  215. ...mapActions("modalVisibility", ["openModal"])
  216. }
  217. };
  218. </script>