Users.vue 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482
  1. <template>
  2. <div>
  3. <page-metadata title="Admin | Users" />
  4. <div class="container">
  5. <h2>Data Requests</h2>
  6. <advanced-table
  7. :column-default="dataRequests.columnDefault"
  8. :columns="dataRequests.columns"
  9. :filters="dataRequests.filters"
  10. data-action="dataRequests.getData"
  11. name="admin-data-requests"
  12. max-width="1200"
  13. :query="false"
  14. >
  15. <template #column-options="slotProps">
  16. <div class="row-options">
  17. <quick-confirm
  18. placement="right"
  19. @confirm="resolveDataRequest(slotProps.item._id)"
  20. :disabled="slotProps.item.removed"
  21. >
  22. <button
  23. class="
  24. button
  25. is-success
  26. icon-with-button
  27. material-icons
  28. "
  29. content="Resolve Data Request"
  30. v-tippy
  31. >
  32. done_all
  33. </button>
  34. </quick-confirm>
  35. </div>
  36. </template>
  37. <template #column-type="slotProps">
  38. <span
  39. :title="
  40. slotProps.item.type
  41. ? 'Remove all associated data'
  42. : slotProps.item.type
  43. "
  44. >{{
  45. slotProps.item.type
  46. ? "Remove all associated data"
  47. : slotProps.item.type
  48. }}</span
  49. >
  50. </template>
  51. <template #column-userId="slotProps">
  52. <span :title="slotProps.item.userId">{{
  53. slotProps.item.userId
  54. }}</span>
  55. </template>
  56. <template #column-_id="slotProps">
  57. <span :title="slotProps.item._id">{{
  58. slotProps.item._id
  59. }}</span>
  60. </template>
  61. </advanced-table>
  62. <h1 id="page-title">Users</h1>
  63. <advanced-table
  64. :column-default="users.columnDefault"
  65. :columns="users.columns"
  66. :filters="users.filters"
  67. data-action="users.getData"
  68. name="admin-users"
  69. max-width="1200"
  70. >
  71. <template #column-options="slotProps">
  72. <div class="row-options">
  73. <button
  74. class="
  75. button
  76. is-primary
  77. icon-with-button
  78. material-icons
  79. "
  80. @click="edit(slotProps.item._id)"
  81. :disabled="slotProps.item.removed"
  82. content="Edit User"
  83. v-tippy
  84. >
  85. edit
  86. </button>
  87. </div>
  88. </template>
  89. <template #column-profilePicture="slotProps">
  90. <profile-picture
  91. :avatar="slotProps.item.avatar"
  92. :name="
  93. slotProps.item.name
  94. ? slotProps.item.name
  95. : slotProps.item.username
  96. "
  97. />
  98. </template>
  99. <template #column-name="slotProps">
  100. <span :title="slotProps.item.name">{{
  101. slotProps.item.name
  102. }}</span>
  103. </template>
  104. <template #column-username="slotProps">
  105. <span :title="slotProps.item.username">{{
  106. slotProps.item.username
  107. }}</span>
  108. </template>
  109. <template #column-_id="slotProps">
  110. <span :title="slotProps.item._id">{{
  111. slotProps.item._id
  112. }}</span>
  113. </template>
  114. <template #column-githubId="slotProps">
  115. <span
  116. v-if="slotProps.item.services.github"
  117. :title="slotProps.item.services.github.id"
  118. >{{ slotProps.item.services.github.id }}</span
  119. >
  120. </template>
  121. <!-- <template #column-hasPassword="slotProps">
  122. <span :title="slotProps.item.hasPassword">{{
  123. slotProps.item.hasPassword
  124. }}</span>
  125. </template> -->
  126. <template #column-role="slotProps">
  127. <span :title="slotProps.item.role">{{
  128. slotProps.item.role
  129. }}</span>
  130. </template>
  131. <template #column-emailAddress="slotProps">
  132. <span :title="slotProps.item.email.address">{{
  133. slotProps.item.email.address
  134. }}</span>
  135. </template>
  136. <template #column-emailVerified="slotProps">
  137. <span :title="slotProps.item.email.verified">{{
  138. slotProps.item.email.verified
  139. }}</span>
  140. </template>
  141. <template #column-songsRequested="slotProps">
  142. <span :title="slotProps.item.statistics.songsRequested">{{
  143. slotProps.item.statistics.songsRequested
  144. }}</span>
  145. </template>
  146. </advanced-table>
  147. </div>
  148. <edit-user
  149. v-if="modals.editUser"
  150. :user-id="editingUserId"
  151. sector="admin"
  152. />
  153. </div>
  154. </template>
  155. <script>
  156. import { mapState, mapActions, mapGetters } from "vuex";
  157. import { defineAsyncComponent } from "vue";
  158. import Toast from "toasters";
  159. import AdvancedTable from "@/components/AdvancedTable.vue";
  160. import ProfilePicture from "@/components/ProfilePicture.vue";
  161. import QuickConfirm from "@/components/QuickConfirm.vue";
  162. // import ws from "@/ws";
  163. export default {
  164. components: {
  165. EditUser: defineAsyncComponent(() =>
  166. import("@/components/modals/EditUser.vue")
  167. ),
  168. AdvancedTable,
  169. ProfilePicture,
  170. QuickConfirm
  171. },
  172. data() {
  173. return {
  174. editingUserId: "",
  175. dataRequests: {
  176. columnDefault: {
  177. sortable: true,
  178. hidable: true,
  179. defaultVisibility: "shown",
  180. draggable: true,
  181. resizable: true,
  182. minWidth: 150,
  183. maxWidth: 600
  184. },
  185. columns: [
  186. {
  187. name: "options",
  188. displayName: "Edit",
  189. properties: ["_id"],
  190. sortable: false,
  191. hidable: false,
  192. resizable: false,
  193. minWidth: 51,
  194. defaultWidth: 51
  195. },
  196. {
  197. name: "type",
  198. displayName: "Type",
  199. properties: ["type"],
  200. sortProperty: "type"
  201. },
  202. {
  203. name: "userId",
  204. displayName: "User ID",
  205. properties: ["userId"],
  206. sortProperty: "userId"
  207. },
  208. {
  209. name: "_id",
  210. displayName: "Request ID",
  211. properties: ["_id"],
  212. sortProperty: "_id"
  213. }
  214. ],
  215. filters: [
  216. {
  217. name: "_id",
  218. displayName: "Request ID",
  219. property: "_id",
  220. filterTypes: ["exact"],
  221. defaultFilterType: "exact"
  222. },
  223. {
  224. name: "userId",
  225. displayName: "User ID",
  226. property: "userId",
  227. filterTypes: ["contains", "exact", "regex"],
  228. defaultFilterType: "contains"
  229. },
  230. {
  231. name: "type",
  232. displayName: "Type",
  233. property: "type",
  234. filterTypes: ["contains", "exact", "regex"],
  235. defaultFilterType: "contains"
  236. }
  237. ]
  238. },
  239. users: {
  240. columnDefault: {
  241. sortable: true,
  242. hidable: true,
  243. defaultVisibility: "shown",
  244. draggable: true,
  245. resizable: true,
  246. minWidth: 150,
  247. maxWidth: 600
  248. },
  249. columns: [
  250. {
  251. name: "options",
  252. displayName: "Edit",
  253. properties: ["_id"],
  254. sortable: false,
  255. hidable: false,
  256. resizable: false,
  257. minWidth: 51,
  258. defaultWidth: 51
  259. },
  260. {
  261. name: "profilePicture",
  262. displayName: "Image",
  263. properties: ["avatar", "name", "username"],
  264. sortable: false,
  265. resizable: false,
  266. minWidth: 71,
  267. defaultWidth: 71
  268. },
  269. {
  270. name: "name",
  271. displayName: "Display Name",
  272. properties: ["name"],
  273. sortProperty: "name"
  274. },
  275. {
  276. name: "username",
  277. displayName: "Username",
  278. properties: ["username"],
  279. sortProperty: "username"
  280. },
  281. {
  282. name: "_id",
  283. displayName: "User ID",
  284. properties: ["_id"],
  285. sortProperty: "_id",
  286. minWidth: 230,
  287. defaultWidth: 230
  288. },
  289. {
  290. name: "githubId",
  291. displayName: "GitHub ID",
  292. properties: ["services.github.id"],
  293. sortProperty: "services.github.id",
  294. minWidth: 115,
  295. defaultWidth: 115
  296. },
  297. // {
  298. // name: "hasPassword",
  299. // displayName: "Has Password",
  300. // properties: ["hasPassword"],
  301. // sortProperty: "hasPassword"
  302. // }
  303. {
  304. name: "role",
  305. displayName: "Role",
  306. properties: ["role"],
  307. sortProperty: "role",
  308. minWidth: 90,
  309. defaultWidth: 90
  310. },
  311. {
  312. name: "emailAddress",
  313. displayName: "Email Address",
  314. properties: ["email.address"],
  315. sortProperty: "email.address",
  316. defaultVisibility: "hidden"
  317. },
  318. {
  319. name: "emailVerified",
  320. displayName: "Email Verified",
  321. properties: ["email.verified"],
  322. sortProperty: "email.verified",
  323. defaultVisibility: "hidden",
  324. minWidth: 140,
  325. defaultWidth: 140
  326. },
  327. {
  328. name: "songsRequested",
  329. displayName: "Songs Requested",
  330. properties: ["statistics.songsRequested"],
  331. sortProperty: "statistics.songsRequested",
  332. minWidth: 170,
  333. defaultWidth: 170
  334. }
  335. ],
  336. filters: [
  337. {
  338. name: "_id",
  339. displayName: "User ID",
  340. property: "_id",
  341. filterTypes: ["exact"],
  342. defaultFilterType: "exact"
  343. },
  344. {
  345. name: "name",
  346. displayName: "Display Name",
  347. property: "name",
  348. filterTypes: ["contains", "exact", "regex"],
  349. defaultFilterType: "contains"
  350. },
  351. {
  352. name: "username",
  353. displayName: "Username",
  354. property: "username",
  355. filterTypes: ["contains", "exact", "regex"],
  356. defaultFilterType: "contains"
  357. },
  358. {
  359. name: "githubId",
  360. displayName: "GitHub ID",
  361. property: "services.github.id",
  362. filterTypes: ["contains", "exact", "regex"],
  363. defaultFilterType: "contains"
  364. },
  365. // {
  366. // name: "hasPassword",
  367. // displayName: "Has Password",
  368. // property: "hasPassword",
  369. // filterTypes: ["contains", "exact", "regex"],
  370. // defaultFilterType: "contains"
  371. // },
  372. {
  373. name: "role",
  374. displayName: "Role",
  375. property: "role",
  376. filterTypes: ["array", "regex"],
  377. defaultFilterType: "array",
  378. filterValues: ["admin", "default"]
  379. },
  380. {
  381. name: "emailAddress",
  382. displayName: "Email Address",
  383. property: "email.address",
  384. filterTypes: ["contains", "exact", "regex"],
  385. defaultFilterType: "contains"
  386. },
  387. {
  388. name: "emailVerified",
  389. displayName: "Email Verified",
  390. property: "email.verified",
  391. filterTypes: ["contains", "exact", "regex"],
  392. defaultFilterType: "contains"
  393. },
  394. {
  395. name: "songsRequested",
  396. displayName: "Songs Requested",
  397. property: "statistics.songsRequested",
  398. filterTypes: [
  399. "numberLesser",
  400. "numberGreater",
  401. "numberEquals",
  402. "exact",
  403. "regex"
  404. ],
  405. defaultFilterType: "numberLesser"
  406. }
  407. ]
  408. }
  409. };
  410. },
  411. computed: {
  412. ...mapState("modalVisibility", {
  413. modals: state => state.modals
  414. }),
  415. ...mapGetters({
  416. socket: "websockets/getSocket"
  417. })
  418. },
  419. mounted() {
  420. if (this.$route.query.userId) this.edit(this.$route.query.userId);
  421. // ws.onConnect(this.init);
  422. // this.socket.on("event:admin.dataRequests.created", res =>
  423. // this.dataRequests.push(res.data.request)
  424. // );
  425. // this.socket.on("event:admin.dataRequests.resolved", res => {
  426. // this.dataRequests = this.dataRequests.filter(
  427. // request => request._id !== res.data.dataRequestId
  428. // );
  429. // });
  430. // this.socket.on("event:user.removed", res => {
  431. // this.users = this.users.filter(
  432. // user => user._id !== res.data.userId
  433. // );
  434. // });
  435. },
  436. methods: {
  437. edit(userId) {
  438. this.editingUserId = userId;
  439. this.openModal("editUser");
  440. },
  441. // init() {
  442. // this.socket.dispatch("apis.joinAdminRoom", "users", () => {});
  443. // },
  444. resolveDataRequest(id) {
  445. this.socket.dispatch("dataRequests.resolve", id, res => {
  446. if (res.status === "success") new Toast(res.message);
  447. });
  448. },
  449. ...mapActions("modalVisibility", ["openModal"])
  450. }
  451. };
  452. </script>
  453. <style lang="scss" scoped>
  454. #page-title {
  455. margin: 30px 0;
  456. }
  457. h2 {
  458. font-size: 30px;
  459. text-align: center;
  460. @media only screen and (min-width: 700px) {
  461. font-size: 35px;
  462. }
  463. }
  464. .profile-picture {
  465. max-width: 50px !important;
  466. max-height: 50px !important;
  467. }
  468. /deep/ .profile-picture.using-initials span {
  469. font-size: 20px; // 2/5th of .profile-picture height/width
  470. }
  471. </style>