News.vue 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413
  1. <template>
  2. <div>
  3. <metadata title="Admin | News" />
  4. <div class="container">
  5. <table class="table is-striped">
  6. <thead>
  7. <tr>
  8. <td>Title</td>
  9. <td>Description</td>
  10. <td>Bugs</td>
  11. <td>Features</td>
  12. <td>Improvements</td>
  13. <td>Upcoming</td>
  14. <td>Options</td>
  15. </tr>
  16. </thead>
  17. <tbody>
  18. <tr v-for="(news, index) in news" :key="index">
  19. <td>
  20. <strong>{{ news.title }}</strong>
  21. </td>
  22. <td>{{ news.description }}</td>
  23. <td>{{ news.bugs.join(", ") }}</td>
  24. <td>{{ news.features.join(", ") }}</td>
  25. <td>{{ news.improvements.join(", ") }}</td>
  26. <td>{{ news.upcoming.join(", ") }}</td>
  27. <td>
  28. <button
  29. class="button is-primary"
  30. @click="editNewsClick(news)"
  31. >
  32. Edit
  33. </button>
  34. <button
  35. class="button is-danger"
  36. @click="remove(news)"
  37. >
  38. Remove
  39. </button>
  40. </td>
  41. </tr>
  42. </tbody>
  43. </table>
  44. <div class="card is-fullwidth">
  45. <header class="card-header">
  46. <p class="card-header-title">Create News</p>
  47. </header>
  48. <div class="card-content">
  49. <div class="content">
  50. <label class="label">Title & Description</label>
  51. <div class="control is-horizontal">
  52. <div class="control is-grouped">
  53. <p class="control is-expanded">
  54. <input
  55. v-model="creating.title"
  56. class="input"
  57. type="text"
  58. placeholder="Title"
  59. />
  60. </p>
  61. <p class="control is-expanded">
  62. <input
  63. v-model="creating.description"
  64. class="input"
  65. type="text"
  66. placeholder="Short description"
  67. />
  68. </p>
  69. </div>
  70. </div>
  71. <div class="columns">
  72. <div class="column">
  73. <label class="label">Bugs</label>
  74. <p class="control has-addons">
  75. <input
  76. id="new-bugs"
  77. class="input"
  78. type="text"
  79. placeholder="Bug"
  80. @keyup.enter="addChange('bugs')"
  81. />
  82. <a
  83. class="button is-info"
  84. href="#"
  85. @click="addChange('bugs')"
  86. >Add</a
  87. >
  88. </p>
  89. <span
  90. v-for="(bug, index) in creating.bugs"
  91. :key="index"
  92. class="tag is-info"
  93. >
  94. {{ bug }}
  95. <button
  96. class="delete is-info"
  97. @click="removeChange('bugs', index)"
  98. />
  99. </span>
  100. </div>
  101. <div class="column">
  102. <label class="label">Features</label>
  103. <p class="control has-addons">
  104. <input
  105. id="new-features"
  106. class="input"
  107. type="text"
  108. placeholder="Feature"
  109. @keyup.enter="addChange('features')"
  110. />
  111. <a
  112. class="button is-info"
  113. href="#"
  114. @click="addChange('features')"
  115. >Add</a
  116. >
  117. </p>
  118. <span
  119. v-for="(feature,
  120. index) in creating.features"
  121. :key="index"
  122. class="tag is-info"
  123. >
  124. {{ feature }}
  125. <button
  126. class="delete is-info"
  127. @click="removeChange('features', index)"
  128. />
  129. </span>
  130. </div>
  131. </div>
  132. <div class="columns">
  133. <div class="column">
  134. <label class="label">Improvements</label>
  135. <p class="control has-addons">
  136. <input
  137. id="new-improvements"
  138. class="input"
  139. type="text"
  140. placeholder="Improvement"
  141. @keyup.enter="addChange('improvements')"
  142. />
  143. <a
  144. class="button is-info"
  145. href="#"
  146. @click="addChange('improvements')"
  147. >Add</a
  148. >
  149. </p>
  150. <span
  151. v-for="(improvement,
  152. index) in creating.improvements"
  153. :key="index"
  154. class="tag is-info"
  155. >
  156. {{ improvement }}
  157. <button
  158. class="delete is-info"
  159. @click="
  160. removeChange('improvements', index)
  161. "
  162. />
  163. </span>
  164. </div>
  165. <div class="column">
  166. <label class="label">Upcoming</label>
  167. <p class="control has-addons">
  168. <input
  169. id="new-upcoming"
  170. class="input"
  171. type="text"
  172. placeholder="Upcoming"
  173. @keyup.enter="addChange('upcoming')"
  174. />
  175. <a
  176. class="button is-info"
  177. href="#"
  178. @click="addChange('upcoming')"
  179. >Add</a
  180. >
  181. </p>
  182. <span
  183. v-for="(upcoming,
  184. index) in creating.upcoming"
  185. :key="index"
  186. class="tag is-info"
  187. >
  188. {{ upcoming }}
  189. <button
  190. class="delete is-info"
  191. @click="removeChange('upcoming', index)"
  192. />
  193. </span>
  194. </div>
  195. </div>
  196. </div>
  197. </div>
  198. <footer class="card-footer">
  199. <a class="card-footer-item" @click="createNews()" href="#"
  200. >Create</a
  201. >
  202. </footer>
  203. </div>
  204. </div>
  205. <edit-news
  206. v-if="modals.editNews"
  207. :news-id="editingNewsId"
  208. sector="admin"
  209. />
  210. </div>
  211. </template>
  212. <script>
  213. import { mapActions, mapState } from "vuex";
  214. import Toast from "toasters";
  215. import io from "../../../io";
  216. import EditNews from "../EditNews.vue";
  217. export default {
  218. components: { EditNews },
  219. data() {
  220. return {
  221. editingNewsId: "",
  222. creating: {
  223. title: "",
  224. description: "",
  225. bugs: [],
  226. features: [],
  227. improvements: [],
  228. upcoming: []
  229. }
  230. };
  231. },
  232. computed: {
  233. ...mapState("modals", {
  234. modals: state => state.modals.admin
  235. }),
  236. ...mapState("admin/news", {
  237. news: state => state.news
  238. })
  239. },
  240. mounted() {
  241. io.getSocket(socket => {
  242. this.socket = socket;
  243. this.socket.emit("news.index", res => {
  244. res.data.forEach(news => {
  245. this.addNews(news);
  246. });
  247. });
  248. this.socket.on("event:admin.news.created", news => {
  249. this.addNews(news);
  250. });
  251. this.socket.on("event:admin.news.updated", updatedNews => {
  252. this.updateNews(updatedNews);
  253. });
  254. this.socket.on("event:admin.news.removed", news => {
  255. this.removeNews(news._id);
  256. });
  257. if (this.socket.connected) this.init();
  258. io.onConnect(() => this.init());
  259. });
  260. },
  261. methods: {
  262. createNews() {
  263. const {
  264. creating: { bugs, features, improvements, upcoming }
  265. } = this;
  266. if (this.creating.title === "")
  267. return new Toast({
  268. content: "Field (Title) cannot be empty",
  269. timeout: 3000
  270. });
  271. if (this.creating.description === "")
  272. return new Toast({
  273. content: "Field (Description) cannot be empty",
  274. timeout: 3000
  275. });
  276. if (
  277. bugs.length <= 0 &&
  278. features.length <= 0 &&
  279. improvements.length <= 0 &&
  280. upcoming.length <= 0
  281. )
  282. return new Toast({
  283. content: "You must have at least one News Item",
  284. timeout: 3000
  285. });
  286. return this.socket.emit("news.create", this.creating, result => {
  287. new Toast(result.message, 4000);
  288. if (result.status === "success")
  289. this.creating = {
  290. title: "",
  291. description: "",
  292. bugs: [],
  293. features: [],
  294. improvements: [],
  295. upcoming: []
  296. };
  297. });
  298. },
  299. remove(news) {
  300. this.socket.emit(
  301. "news.remove",
  302. news,
  303. res => new Toast({ content: res.message, timeout: 8000 })
  304. );
  305. },
  306. editNewsClick(news) {
  307. this.editingNewsId = news._id;
  308. this.openModal({ sector: "admin", modal: "editNews" });
  309. },
  310. addChange(type) {
  311. const change = document.getElementById(`new-${type}`).value.trim();
  312. if (this.creating[type].indexOf(change) !== -1)
  313. return new Toast({
  314. content: `Tag already exists`,
  315. timeout: 3000
  316. });
  317. if (change) {
  318. document.getElementById(`new-${type}`).value = "";
  319. this.creating[type].push(change);
  320. return true;
  321. }
  322. return new Toast({
  323. content: `${type} cannot be empty`,
  324. timeout: 3000
  325. });
  326. },
  327. removeChange(type, index) {
  328. this.creating[type].splice(index, 1);
  329. },
  330. init() {
  331. this.socket.emit("apis.joinAdminRoom", "news", () => {});
  332. },
  333. ...mapActions("modals", ["openModal", "closeModal"]),
  334. ...mapActions("admin/news", [
  335. "editNews",
  336. "addNews",
  337. "removeNews",
  338. "updateNews"
  339. ])
  340. }
  341. };
  342. </script>
  343. <style lang="scss" scoped>
  344. @import "../../../styles/global.scss";
  345. .night-mode {
  346. .table {
  347. color: $night-mode-text;
  348. background-color: $night-mode-bg-secondary;
  349. thead tr {
  350. background: $night-mode-bg-secondary;
  351. td {
  352. color: #fff;
  353. }
  354. }
  355. tbody tr:hover {
  356. background-color: #111 !important;
  357. }
  358. tbody tr:nth-child(even) {
  359. background-color: #444;
  360. }
  361. strong {
  362. color: $night-mode-text;
  363. }
  364. }
  365. .card {
  366. background: $night-mode-bg-secondary;
  367. .card-header {
  368. box-shadow: 0 1px 2px rgba(10, 10, 10, 0.8);
  369. }
  370. p,
  371. .label {
  372. color: $night-mode-text;
  373. }
  374. }
  375. }
  376. .tag:not(:last-child) {
  377. margin-right: 5px;
  378. }
  379. td {
  380. vertical-align: middle;
  381. }
  382. .is-info:focus {
  383. background-color: $primary-color;
  384. }
  385. .card-footer-item {
  386. color: $primary-color;
  387. }
  388. </style>