App.vue 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162
  1. <script setup lang="ts">
  2. import { RouterView } from "vue-router";
  3. import { useMainStore } from "@/stores/main";
  4. import { onMounted } from "vue";
  5. import LoadingView from "@/components/Loading.vue";
  6. import { storeToRefs } from "pinia";
  7. import NotificationsManager from "@/components/NotificationsManager.vue";
  8. //store
  9. const mainStore = useMainStore();
  10. const mainStoreRef = storeToRefs(mainStore);
  11. //variable
  12. const loggedIn = mainStoreRef.readIsLoggedIn;
  13. //getter
  14. //lifecycle
  15. onMounted(() => {
  16. let path = location.pathname;
  17. if (path !== "/qrcode" && path !== "/yt-views" && path !== "/orders") {
  18. mainStore.checkLoggedIn();
  19. }
  20. });
  21. </script>
  22. <template>
  23. <v-app>
  24. <LoadingView v-if="loggedIn === null" />
  25. <RouterView v-else />
  26. <NotificationsManager />
  27. </v-app>
  28. </template>
  29. <style lang="scss">
  30. :root {
  31. --main-color: #ea5413;
  32. }
  33. a {
  34. transition: 0.3s;
  35. color: var(--main-color);
  36. text-decoration: none;
  37. &:hover {
  38. opacity: 0.7;
  39. }
  40. }
  41. .v-input__details {
  42. padding: 3px 16px;
  43. }
  44. .login-form {
  45. margin: auto;
  46. max-width: 500px;
  47. position: relative;
  48. .v-input__details {
  49. .v-messages {
  50. color: #b00020 !important;
  51. text-align: end;
  52. }
  53. }
  54. .login-btn,
  55. .google-btn {
  56. width: 280px;
  57. display: flex;
  58. margin: 50px auto 0;
  59. padding: 25px 40px;
  60. font-size: 16px;
  61. }
  62. .v-alert {
  63. position: absolute;
  64. width: 100%;
  65. bottom: -110px;
  66. margin-bottom: 10px;
  67. padding: 10px !important;
  68. .v-alert__content {
  69. text-align: start;
  70. }
  71. .v-icon--size-default {
  72. padding: 0;
  73. }
  74. }
  75. .v-input__append {
  76. position: relative;
  77. margin-inline-start: 0 !important;
  78. .material-icons {
  79. position: absolute !important;
  80. right: 10px;
  81. }
  82. }
  83. }
  84. .banner-item {
  85. position: relative;
  86. img {
  87. width: 100%;
  88. height: 100vh;
  89. object-fit: cover;
  90. margin-bottom: -7px;
  91. animation: scale 3s ease-in-out;
  92. filter: brightness(60%); // 圖片遮罩
  93. @media (max-width: 959px) {
  94. height: 40vh;
  95. }
  96. }
  97. h2 {
  98. width: 100%;
  99. padding: 0 10px;
  100. position: absolute;
  101. top: 50%;
  102. left: 50%;
  103. transform: translate(-50%, -50%);
  104. color: #fff;
  105. font-size: 30px;
  106. font-weight: bold;
  107. text-align: center;
  108. letter-spacing: 1px;
  109. @media (max-width: 959px) {
  110. top: 55%;
  111. }
  112. @media (max-width: 575px) {
  113. top: 60%;
  114. font-size: 22px;
  115. }
  116. }
  117. }
  118. @keyframes scale {
  119. 0% {
  120. transform: scale(1.3);
  121. }
  122. 100% {
  123. transform: scale(1);
  124. }
  125. }
  126. .form-title {
  127. display: flex;
  128. flex-direction: column;
  129. align-items: center;
  130. h3 {
  131. font-size: 32px;
  132. font-weight: bold;
  133. letter-spacing: 3px;
  134. @media (max-width: 575px) {
  135. font-size: 28px;
  136. }
  137. }
  138. span {
  139. width: 60px;
  140. height: 3px;
  141. margin: 20px auto 35px;
  142. display: block;
  143. background: var(--main-color);
  144. }
  145. }
  146. </style>