LongJobs.vue 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260
  1. <script setup lang="ts">
  2. import { defineAsyncComponent, ref, onMounted } from "vue";
  3. import { storeToRefs } from "pinia";
  4. import { useWebsocketsStore } from "@/stores/websockets";
  5. import { useLongJobsStore } from "@/stores/longJobs";
  6. import { useUserAuthStore } from "@/stores/userAuth";
  7. const FloatingBox = defineAsyncComponent(
  8. () => import("@/components/FloatingBox.vue")
  9. );
  10. const body = ref(document.body);
  11. const userAuthStore = useUserAuthStore();
  12. const { loggedIn } = storeToRefs(userAuthStore);
  13. const { socket } = useWebsocketsStore();
  14. const longJobsStore = useLongJobsStore();
  15. const { activeJobs } = storeToRefs(longJobsStore);
  16. const { setJob, setJobs, removeJob } = longJobsStore;
  17. const remove = job => {
  18. if (job.status === "success" || job.status === "error") {
  19. socket.dispatch("users.removeLongJob", job.id, res => {
  20. if (res.status === "success") {
  21. removeJob(job.id);
  22. } else console.log(res.message);
  23. });
  24. }
  25. };
  26. onMounted(() => {
  27. socket.onConnect(() => {
  28. if (loggedIn.value) {
  29. socket.dispatch("users.getLongJobs", {
  30. cb: res => {
  31. if (res.status === "success") {
  32. setJobs(res.data.longJobs);
  33. } else console.log(res.message);
  34. },
  35. onProgress: res => {
  36. setJob(res);
  37. }
  38. });
  39. socket.on("keep.event:longJob.removed", ({ data }) => {
  40. removeJob(data.jobId);
  41. });
  42. socket.on("keep.event:longJob.added", ({ data }) => {
  43. if (
  44. !activeJobs.value.find(
  45. activeJob => activeJob.id === data.jobId
  46. )
  47. )
  48. socket.dispatch("users.getLongJob", data.jobId, {
  49. cb: res => {
  50. if (res.status === "success") {
  51. setJob(res.data.longJob);
  52. } else console.log(res.message);
  53. },
  54. onProgress: res => {
  55. setJob(res);
  56. }
  57. });
  58. });
  59. }
  60. });
  61. });
  62. </script>
  63. <template>
  64. <floating-box
  65. v-if="activeJobs.length > 0"
  66. title="Jobs"
  67. id="longJobs"
  68. ref="longJobs"
  69. :persist="true"
  70. initial="align-bottom"
  71. :min-width="200"
  72. :max-width="400"
  73. :min-height="200"
  74. >
  75. <template #body>
  76. <div class="active-jobs">
  77. <div
  78. v-for="job in activeJobs"
  79. :key="`activeJob-${job.id}`"
  80. class="active-job"
  81. >
  82. <i
  83. v-if="
  84. job.status === 'started' || job.status === 'update'
  85. "
  86. class="material-icons"
  87. content="In Progress"
  88. v-tippy="{ theme: 'info', placement: 'right' }"
  89. >
  90. pending
  91. </i>
  92. <i
  93. v-else-if="job.status === 'success'"
  94. class="material-icons success"
  95. content="Complete"
  96. v-tippy="{ theme: 'info', placement: 'right' }"
  97. >
  98. check_circle
  99. </i>
  100. <i
  101. v-else
  102. class="material-icons error"
  103. content="Failed"
  104. v-tippy="{ theme: 'info', placement: 'right' }"
  105. >
  106. error
  107. </i>
  108. <div class="name" :title="job.name">{{ job.name }}</div>
  109. <div class="actions">
  110. <i
  111. class="material-icons clear"
  112. :class="{
  113. disabled: !(
  114. job.status === 'success' ||
  115. job.status === 'error'
  116. )
  117. }"
  118. content="Clear"
  119. v-tippy="{ placement: 'left' }"
  120. @click="remove(job)"
  121. >
  122. remove_circle
  123. </i>
  124. <tippy
  125. :touch="true"
  126. :interactive="true"
  127. placement="left"
  128. ref="longJobMessage"
  129. :append-to="body"
  130. >
  131. <i class="material-icons message">chat</i>
  132. <template #content>
  133. <div class="long-job-message">
  134. <strong>Latest Update:</strong>
  135. <span :title="job.message">{{
  136. job.message
  137. }}</span>
  138. </div>
  139. </template>
  140. </tippy>
  141. </div>
  142. </div>
  143. </div>
  144. </template>
  145. </floating-box>
  146. </template>
  147. <style lang="less" scoped>
  148. .night-mode {
  149. #longJobs {
  150. .active-jobs {
  151. .active-job {
  152. background-color: var(--dark-grey);
  153. border: 0;
  154. }
  155. }
  156. }
  157. .long-job-message {
  158. color: var(--black);
  159. }
  160. }
  161. #longJobs {
  162. z-index: 5000 !important;
  163. .active-jobs {
  164. .active-job {
  165. display: flex;
  166. padding: 5px;
  167. margin: 5px 0;
  168. border: 1px solid var(--light-grey-3);
  169. border-radius: @border-radius;
  170. &:first-child {
  171. margin-top: 0;
  172. }
  173. &:last-child {
  174. margin-bottom: 0;
  175. }
  176. .name {
  177. line-height: 24px;
  178. font-weight: 600;
  179. text-transform: capitalize;
  180. text-overflow: ellipsis;
  181. white-space: nowrap;
  182. overflow: hidden;
  183. margin-right: auto;
  184. }
  185. .material-icons {
  186. font-size: 20px;
  187. color: var(--primary-color);
  188. margin: auto 5px auto 0;
  189. cursor: pointer;
  190. &.success {
  191. color: var(--green);
  192. }
  193. &.error,
  194. &.clear {
  195. color: var(--red);
  196. }
  197. &.disabled {
  198. color: var(--light-grey-3);
  199. cursor: not-allowed;
  200. }
  201. }
  202. .actions {
  203. display: flex;
  204. .material-icons {
  205. margin: auto 0 auto 5px;
  206. }
  207. & > span {
  208. display: flex;
  209. padding: 0;
  210. }
  211. }
  212. }
  213. }
  214. }
  215. .long-job-message {
  216. display: flex;
  217. flex-direction: column;
  218. strong {
  219. font-size: 12px;
  220. }
  221. span {
  222. display: -webkit-inline-box;
  223. text-overflow: ellipsis;
  224. white-space: normal;
  225. -webkit-box-orient: vertical;
  226. -webkit-line-clamp: 3;
  227. overflow: hidden;
  228. max-width: 200px;
  229. font-size: 12px;
  230. }
  231. }
  232. </style>