main.ts 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231
  1. import axios from "axios"
  2. import { defineStore } from "pinia";
  3. import { api } from "@/api"
  4. import router from "@/router"
  5. import { getLocalToken, removeLocalToken, saveLocalToken } from "@/utils";
  6. import type { AppNotification } from '@/interfaces';
  7. import type { IUserProfile, IUserProfileCreate, IUserProfileUpdate, MainState } from '@/interfaces';
  8. const defaultState: MainState = {
  9. isLoggedIn: null,
  10. token: '',
  11. logInError: false,
  12. userProfile: null,
  13. dashboardMiniDrawer: false,
  14. dashboardShowDrawer: true,
  15. notifications: [],
  16. };
  17. export const useMainStore = defineStore("MainStoreId", {
  18. state: () => defaultState,
  19. getters: {
  20. readhasAdminAccess: (state) =>
  21. state.userProfile && state.userProfile.is_superuser && state.userProfile.is_active,
  22. readLoginError: (state) => state.logInError,
  23. readDashboardShowDrawer: (state) => state.dashboardShowDrawer,
  24. readDashboardMiniDrawer: (state) => state.dashboardMiniDrawer,
  25. readUserProfile: (state) => state.userProfile,
  26. readToken: (state) => state.token,
  27. readIsLoggedIn: (state) => state.isLoggedIn,
  28. readFirstNotification: (state) => state.notifications.length > 0 && state.notifications[0],
  29. },
  30. actions:{
  31. // setters
  32. setToken(payload: string) { this.token = payload; },
  33. setLoggedIn(payload: boolean) { this.isLoggedIn = payload; },
  34. setLogInError(payload: boolean) { this.logInError = payload; },
  35. setUserProfile(payload: IUserProfile) { this.userProfile = payload },
  36. setDashboardMiniDrawer(payload: boolean) { this.dashboardMiniDrawer = payload; },
  37. setDashboardShowDrawer(payload: boolean) { this.dashboardShowDrawer = payload; },
  38. addNotification(payload: AppNotification) { this.notifications.push(payload); },
  39. removeNotification(payload: AppNotification) {
  40. if (payload) {
  41. this.notifications = this.notifications.filter(
  42. (notification) => notification !== payload,
  43. );
  44. }
  45. },
  46. // actions
  47. async logIn(username:string, password:string) {
  48. try {
  49. const response = await api.logInGetToken(username, password);
  50. const token: string = response.data.access_token;
  51. if (token) {
  52. saveLocalToken(token);
  53. this.setToken(token);
  54. this.setLoggedIn(true);
  55. this.setLogInError(false);
  56. await this.getUserProfile();
  57. await this.routeLoggedIn();
  58. this.addNotification({content: "Logged in", color: "success" });
  59. } else {
  60. await this.logOut();
  61. }
  62. } catch (err) {
  63. this.setLogInError(true);
  64. await this.logOut();
  65. }
  66. },
  67. async getUserProfile() {
  68. try {
  69. const response = await api.getMe(this.token);
  70. if (response.data) {
  71. this.setUserProfile(response.data)
  72. }
  73. } catch (error) {
  74. await this.checkApiError(error);
  75. }
  76. },
  77. async updateUserProfile(payload: IUserProfileUpdate) {
  78. try {
  79. const loadingNotification = { content: "saving", showProgress: true };
  80. await this.addNotification(loadingNotification);
  81. const response = (
  82. await Promise.all([
  83. api.updateMe(this.token, payload),
  84. await new Promise<void>((resolve, _) => setTimeout(() => resolve(), 500)),
  85. ])
  86. )[0];
  87. this.setUserProfile(response.data);
  88. this.removeNotification(loadingNotification);
  89. this.addNotification({
  90. content: "Profile successfully updated",
  91. color: "success",
  92. });
  93. } catch (error) {
  94. await this.checkApiError(error);
  95. }
  96. },
  97. async checkLoggedIn() {
  98. if (!this.isLoggedIn) {
  99. let token = this.token;
  100. if (!token) {
  101. const localToken = getLocalToken();
  102. if (localToken) {
  103. this.setToken(localToken);
  104. token = localToken;
  105. }
  106. }
  107. if (token) {
  108. try {
  109. const response = await api.getMe(token);
  110. this.setLoggedIn(true);
  111. this.setUserProfile(response.data);
  112. } catch (error) {
  113. await this.removeLogIn();
  114. }
  115. } else {
  116. await this.removeLogIn();
  117. }
  118. }
  119. },
  120. async removeLogIn() {
  121. removeLocalToken();
  122. this.setToken("");
  123. this.setLoggedIn(false);
  124. },
  125. async logOut() {
  126. await this.removeLogIn();
  127. this.routeLogOut();
  128. },
  129. async userLogOut() {
  130. await this.logOut();
  131. this.addNotification({ content: "Logged out", color: "success" });
  132. },
  133. routeLogOut() {
  134. if (router.currentRoute.value.path !== "/login") {
  135. router.push("/login");
  136. }
  137. },
  138. async checkApiError(payload: unknown) {
  139. if (axios.isAxiosError(payload)) {
  140. if (payload.response?.status === 401) {
  141. await this.logOut();
  142. }
  143. }
  144. },
  145. routeLoggedIn() {
  146. if (router.currentRoute.value.path === "/login" || router.currentRoute.value.path === "/") {
  147. router.push("/main/dashboard");
  148. }
  149. },
  150. async dispatchRemoveNotification(payload: {notification: AppNotification; timeout: number },) {
  151. return new Promise((resolve, _) => {
  152. setTimeout(() => {
  153. this.removeNotification(payload.notification);
  154. resolve(true);
  155. }, payload.timeout);
  156. });
  157. },
  158. async register(payload: IUserProfileRegister) {
  159. const loadingNotification = {
  160. content: "Sgining up...",
  161. showProgress: true,
  162. };
  163. try {
  164. this.addNotification(loadingNotification);
  165. const response = (
  166. await Promise.all([
  167. api.registerUser(payload),
  168. await new Promise<void>((resolve, _) => setTimeout(() => resolve(), 500)),
  169. ])
  170. )[0];
  171. this.removeNotification(loadingNotification);
  172. this.addNotification({
  173. content: "successfully registered",
  174. color: "success",
  175. });
  176. } catch (error) {
  177. await this.checkApiError(error);
  178. }
  179. },
  180. async passwordRecovery(payload: {username: string }) {
  181. const loadingNotification = {
  182. content: "Sending password recovery email",
  183. showProgress: true,
  184. };
  185. try {
  186. this.addNotification(loadingNotification);
  187. await Promise.all([
  188. api.passwordRecovery(payload.username),
  189. await new Promise<void>((resolve, _) => setTimeout(() => resolve(), 500)),
  190. ]);
  191. this.removeNotification(loadingNotification);
  192. this.addNotification({
  193. content: "Password recovery email sent",
  194. color: "success",
  195. });
  196. await this.logOut();
  197. } catch (error) {
  198. this.removeNotification(loadingNotification);
  199. this.addNotification({ color: "error", content: "Incorrect username" });
  200. }
  201. },
  202. async resetPassword(payload: {password: string; token: string }) {
  203. const loadingNotification = { content: "Resetting password", showProgress: true };
  204. try {
  205. this.addNotification(loadingNotification);
  206. await Promise.all([
  207. api.resetPassword(payload.password, payload.token),
  208. await new Promise<void>((resolve, _) => setTimeout(() => resolve(), 500)),
  209. ]);
  210. this.removeNotification(loadingNotification);
  211. this.addNotification( {
  212. content: "Password successfully reset",
  213. color: "success",
  214. });
  215. await this.logOut();
  216. } catch (error) {
  217. this.removeNotification(loadingNotification);
  218. this.addNotification({
  219. color: "error",
  220. content: "Error resetting password",
  221. });
  222. }
  223. },
  224. }
  225. });