Orders.vue 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321
  1. <template>
  2. <div class="min-h-screen bg-background flex flex-col">
  3. <Header />
  4. <main class="flex-grow pt-24 pb-20">
  5. <div class="container mx-auto px-4 max-w-5xl">
  6. <div class="mb-10">
  7. <RouterLink to="/" class="inline-flex items-center text-sm text-muted-foreground hover:text-primary transition-colors mb-6 group">
  8. <ArrowLeft class="w-4 h-4 mr-2 group-hover:-translate-x-1 transition-transform" />{{ t("auth.back") }}
  9. </RouterLink>
  10. <h1 class="font-display text-4xl font-bold bg-clip-text text-transparent bg-gradient-to-r from-foreground to-foreground/60">
  11. {{ t("nav.myOrders") }}
  12. </h1>
  13. <p class="text-muted-foreground mt-2">{{ t("orders.titleSubtitle") }}</p>
  14. </div>
  15. <div v-if="isLoading" class="flex flex-col items-center justify-center py-20 gap-4">
  16. <Loader2 class="w-10 h-10 animate-spin text-primary" />
  17. <p class="text-muted-foreground animate-pulse">{{ t("orders.loading") }}</p>
  18. </div>
  19. <div v-else-if="orders.length === 0"
  20. v-motion :initial="{ opacity: 0, y: 20 }" :enter="{ opacity: 1, y: 0 }"
  21. class="bg-card/40 backdrop-blur-xl border border-border/50 rounded-3xl p-20 text-center">
  22. <div class="w-20 h-20 bg-primary/10 rounded-full flex items-center justify-center mx-auto mb-6">
  23. <Package class="w-10 h-10 text-primary opacity-60" />
  24. </div>
  25. <h2 class="text-2xl font-bold mb-2">{{ t("orders.noOrders") }}</h2>
  26. <p class="text-muted-foreground mb-8 max-w-sm mx-auto">{{ t("orders.startProjectDesc") }}</p>
  27. <button @click="router.push('/#upload')" class="bg-primary hover:bg-primary/90 text-primary-foreground px-8 py-3 rounded-xl font-bold transition-all transform hover:scale-105">
  28. {{ t("orders.startProject") }}
  29. </button>
  30. </div>
  31. <div v-else class="space-y-4">
  32. <div
  33. v-for="(order, idx) in orders"
  34. :key="order.id"
  35. v-motion
  36. :initial="{ opacity: 0, x: -20 }"
  37. :enter="{ opacity: 1, x: 0, transition: { delay: idx * 50 } }"
  38. class="bg-card/40 backdrop-blur-xl border border-border/50 rounded-2xl p-6 hover:border-primary/30 transition-all group overflow-hidden relative"
  39. >
  40. <div class="absolute top-0 right-0 w-32 h-32 bg-primary/5 rounded-full blur-3xl -mr-16 -mt-16 group-hover:bg-primary/10 transition-colors" />
  41. <div class="flex flex-col md:flex-row md:items-center justify-between gap-6 relative z-10">
  42. <div class="flex items-center gap-5">
  43. <div class="w-12 h-12 bg-background/80 rounded-xl flex items-center justify-center border border-border/50">
  44. <Package class="w-6 h-6 text-primary" />
  45. </div>
  46. <div>
  47. <h3 class="font-bold text-lg leading-none mb-1">{{ t("common.orderId", { id: order.id }) }}</h3>
  48. <p class="text-xs text-muted-foreground">{{ formatDate(order.created_at) }}</p>
  49. </div>
  50. </div>
  51. <div class="flex flex-wrap items-center gap-4 md:gap-8">
  52. <div class="flex flex-col">
  53. <span class="text-[10px] uppercase tracking-wider text-muted-foreground font-bold mb-1">{{ t("orders.labels.quantity") }}</span>
  54. <div class="flex items-center gap-1.5 px-2.5 py-1 bg-primary/10 rounded-lg text-primary font-bold">
  55. <Hash class="w-3 h-3" /><span class="text-sm">{{ order.quantity || 1 }}</span>
  56. </div>
  57. </div>
  58. <div class="flex flex-col">
  59. <span class="text-[10px] uppercase tracking-wider text-muted-foreground font-bold mb-1">{{ t("orders.labels.status") }}</span>
  60. <StatusBadge :status="order.status" />
  61. </div>
  62. <div class="flex flex-col">
  63. <span class="text-[10px] uppercase tracking-wider text-muted-foreground font-bold mb-1">{{ t("orders.labels.estimate") }}</span>
  64. <span class="font-display font-bold text-lg">{{ order.total_price ? `${order.total_price} €` : t("common.pending") }}</span>
  65. </div>
  66. <div class="flex flex-col border-l border-border/50 pl-4">
  67. <span class="text-[10px] uppercase tracking-wider text-muted-foreground font-bold mb-1">{{ t("orders.labels.materialColor") }}</span>
  68. <div class="flex flex-col">
  69. <span class="text-xs font-bold uppercase text-primary">{{ order.material_name || "PLA" }}</span>
  70. <div class="flex items-center gap-1.5 mt-0.5">
  71. <div class="w-2.5 h-2.5 rounded-full border border-border/50" :style="order.color_name ? { backgroundColor: order.color_name.toLowerCase() } : { backgroundColor: '#ccc' }"></div>
  72. <span class="text-[9px] text-muted-foreground font-medium">{{ order.color_name || t("common.default") }}</span>
  73. </div>
  74. </div>
  75. </div>
  76. <div class="flex items-center gap-2">
  77. <a v-if="order.model_link && !order.model_link.startsWith('javascript:')"
  78. :href="order.model_link" target="_blank" rel="noopener noreferrer"
  79. class="p-2 hover:bg-secondary rounded-lg transition-colors">
  80. <ExternalLink class="w-4 h-4" />
  81. </a>
  82. <!-- Removed obsolete routing link -->
  83. <Button variant="ghost" size="sm" @click="toggleChat(order.id)" class="relative">
  84. <span v-if="order.unread_count > 0" class="absolute -top-2 -right-2 bg-red-500 text-white text-[10px] w-5 h-5 flex items-center justify-center rounded-full shadow-sm animate-pulse z-10">{{ order.unread_count }}</span>
  85. <MessageCircle class="w-4 h-4" :class="openChatId === order.id ? 'text-primary' : ''" />
  86. <span class="ml-1 text-xs">{{ t("chat.open") }}</span>
  87. </Button>
  88. </div>
  89. </div>
  90. </div>
  91. <div v-if="order.notes" class="mt-8 p-4 bg-primary/5 border border-primary/10 rounded-2xl relative z-10">
  92. <div class="flex items-center gap-2 mb-2">
  93. <FileText class="w-3.5 h-3.5 text-primary" />
  94. <span class="text-[10px] font-bold uppercase tracking-widest text-muted-foreground">{{ t("orders.labels.myNotes") }}</span>
  95. </div>
  96. <p class="text-xs text-muted-foreground italic leading-relaxed">"{{ order.notes }}"</p>
  97. </div>
  98. <!-- Items Specification -->
  99. <div v-if="order.items && order.items.length > 0" class="mt-8 p-4 bg-secondary/20 border border-border/50 rounded-2xl relative z-10">
  100. <div class="flex items-center gap-2 mb-4">
  101. <Hash class="w-3.5 h-3.5 text-primary" />
  102. <span class="text-[10px] font-bold uppercase tracking-widest text-muted-foreground">{{ t("orders.labels.specification") }}</span>
  103. </div>
  104. <div class="space-y-2">
  105. <div v-for="item in order.items" :key="item.id" class="flex justify-between items-center text-xs">
  106. <span class="text-muted-foreground">{{ item.description }} <span v-if="item.quantity > 1" class="text-primary font-bold">x{{ item.quantity }}</span></span>
  107. <span class="font-bold">{{ item.total_price }} €</span>
  108. </div>
  109. <div class="pt-2 mt-2 border-t border-border/50 flex justify-between items-center font-bold">
  110. <span class="text-[10px] uppercase tracking-wider">{{ t("orders.labels.total") }}</span>
  111. <span class="text-lg text-primary">{{ order.total_price }} €</span>
  112. </div>
  113. </div>
  114. </div>
  115. <!-- Pizza Tracker -->
  116. <div class="mt-8 pt-6 border-t border-border/50 relative z-10 px-2 sm:px-8">
  117. <OrderTracker :status="order.status" />
  118. </div>
  119. <!-- Files -->
  120. <div v-if="order.files && order.files.length > 0" class="mt-8 pt-6 border-t border-border/50 relative z-10">
  121. <div class="flex items-center gap-2 mb-4">
  122. <FileBox class="w-3.5 h-3.5 text-primary" />
  123. <span class="text-[10px] font-bold uppercase tracking-widest text-muted-foreground">{{ t("orders.labels.projectFiles") }}</span>
  124. </div>
  125. <div class="grid grid-cols-2 sm:grid-cols-4 gap-4">
  126. <template v-for="file in order.files" :key="file.id">
  127. <div v-if="file.id"
  128. class="bg-background/40 border border-border/50 rounded-2xl overflow-hidden group/file hover:border-primary/30 transition-all flex flex-col h-full">
  129. <div class="aspect-square bg-muted/20 flex items-center justify-center p-2 relative overflow-hidden">
  130. <img v-if="file.preview_path" :src="`${RESOURCES_BASE_URL}/${file.preview_path}`" class="w-full h-full object-contain" />
  131. <FileBox v-else class="w-8 h-8 text-muted-foreground/20" />
  132. <div class="absolute inset-0 bg-primary/20 opacity-0 group-hover/file:opacity-100 transition-opacity flex items-center justify-center">
  133. <a :href="`${RESOURCES_BASE_URL}/${file.file_path}`" target="_blank" class="bg-card w-8 h-8 rounded-full flex items-center justify-center shadow-lg"><Download class="w-4 h-4 text-primary" /></a>
  134. </div>
  135. </div>
  136. <div class="p-3">
  137. <p class="text-[10px] font-bold truncate">{{ file.filename }}</p>
  138. <div class="flex items-center gap-2 mt-1">
  139. <span v-if="file.quantity > 1" class="text-[8px] font-bold text-primary">x{{ file.quantity }}</span>
  140. </div>
  141. <div v-if="file.print_time || file.filament_g" class="flex flex-col gap-0.5 mt-2 pt-2 border-t border-border/10">
  142. <span v-if="file.print_time" class="text-[8px] text-primary/80">⏱️ {{ file.print_time }}</span>
  143. <span v-if="file.filament_g" class="text-[8px] text-primary/80">⚖️ {{ file.filament_g.toFixed(1) }}g</span>
  144. </div>
  145. </div>
  146. </div>
  147. </template>
  148. </div>
  149. </div>
  150. <div v-if="order.photos && order.photos.length > 0" class="mt-8 pt-6 border-t border-border/50 relative z-10">
  151. <div class="flex items-center gap-2 mb-4">
  152. <ImageIcon class="w-3.5 h-3.5 text-primary" />
  153. <span class="text-[10px] font-bold uppercase tracking-widest text-muted-foreground">{{ t("orders.labels.progressReport") }}</span>
  154. </div>
  155. <div class="flex flex-wrap gap-3">
  156. <div v-for="photo in order.photos" :key="photo.id"
  157. class="group/photo relative w-20 h-20 rounded-xl overflow-hidden border border-border/50 bg-background/50">
  158. <img :src="`${RESOURCES_BASE_URL}/${photo.file_path}`" class="w-full h-full object-cover" alt="Print Progress" />
  159. <a :href="`${RESOURCES_BASE_URL}/${photo.file_path}`" target="_blank"
  160. class="absolute inset-0 bg-black/40 flex items-center justify-center opacity-0 group-hover/photo:opacity-100 transition-opacity">
  161. <ExternalLink class="w-4 h-4 text-white" />
  162. </a>
  163. </div>
  164. </div>
  165. </div>
  166. <div v-if="openChatId === order.id" :id="`chat-${order.id}`" class="mt-6 pt-4 border-t border-border/50 relative z-10">
  167. <OrderChat :orderId="order.id" compact closable @close="openChatId = null" />
  168. </div>
  169. </div>
  170. </div>
  171. <!-- Pagination -->
  172. <div v-if="totalOrders > pageSize" class="flex flex-wrap items-center justify-center gap-2 mt-12 mb-8">
  173. <button v-for="p in Math.ceil(totalOrders / pageSize)" :key="p"
  174. @click="currentPage = p"
  175. v-motion :initial="{ opacity: 0, scale: 0.9 }" :enter="{ opacity: 1, scale: 1, transition: { delay: p * 50 } }"
  176. :class="['w-10 h-10 rounded-xl font-bold transition-all border',
  177. currentPage === p ? 'bg-primary text-primary-foreground border-primary shadow-glow' : 'bg-card/40 border-border/50 text-muted-foreground hover:border-primary/30']">
  178. {{ p }}
  179. </button>
  180. </div>
  181. </div>
  182. </main>
  183. <Footer />
  184. </div>
  185. </template>
  186. <script setup lang="ts">
  187. import { ref, onMounted, defineComponent, h, watch, nextTick } from "vue";
  188. import { RouterLink, useRouter, useRoute } from "vue-router";
  189. import { useI18n } from "vue-i18n";
  190. import { Package, Clock, ShieldCheck, Truck, XCircle, ArrowLeft, Loader2, ExternalLink, Hash, FileText, Image as ImageIcon, MessageCircle, FileBox, Download } from "lucide-vue-next";
  191. import Button from "@/components/ui/button.vue";
  192. import Header from "@/components/Header.vue";
  193. import Footer from "@/components/Footer.vue";
  194. import OrderChat from "@/components/OrderChat.vue";
  195. import OrderTracker from "@/components/OrderTracker.vue";
  196. import { getMyOrders, API_BASE_URL, RESOURCES_BASE_URL } from "@/lib/api";
  197. import { useAuthStore } from "@/stores/auth";
  198. import { toast } from "vue-sonner";
  199. const { t } = useI18n();
  200. const router = useRouter();
  201. const route = useRoute();
  202. const authStore = useAuthStore();
  203. const orders = ref<any[]>([]);
  204. const totalOrders = ref(0);
  205. const currentPage = ref(1);
  206. const pageSize = 10;
  207. const isLoading = ref(true);
  208. const openChatId = ref<number | null>(route.query.chat ? parseInt(route.query.chat.toString()) : null);
  209. async function toggleChat(orderId: number) {
  210. const isOpening = openChatId.value !== orderId;
  211. openChatId.value = isOpening ? orderId : null;
  212. if (isOpening) {
  213. // Clear unread count locally for instant UI feedback
  214. const order = orders.value.find(o => o.id === orderId);
  215. if (order) order.unread_count = 0;
  216. authStore.refreshUnreadCount();
  217. await nextTick();
  218. const el = document.getElementById(`chat-${orderId}`);
  219. if (el) {
  220. el.scrollIntoView({ behavior: 'smooth', block: 'center' });
  221. }
  222. }
  223. }
  224. watch(openChatId, (newId) => {
  225. if (route.query.chat?.toString() !== newId?.toString()) {
  226. router.replace({ query: { ...route.query, chat: newId || undefined } });
  227. }
  228. });
  229. // StatusBadge component defined inline
  230. const StatusBadge = defineComponent({
  231. props: { status: String },
  232. setup(props) {
  233. const styles: Record<string, string> = {
  234. pending: "bg-amber-500/10 text-amber-500 border-amber-500/20",
  235. processing: "bg-blue-500/10 text-blue-500 border-blue-500/20",
  236. shipped: "bg-indigo-500/10 text-indigo-500 border-indigo-500/20",
  237. completed: "bg-emerald-500/10 text-emerald-500 border-emerald-500/20",
  238. cancelled: "bg-destructive/10 text-destructive border-destructive/20",
  239. };
  240. const icons: Record<string, any> = { pending: Clock, processing: ShieldCheck, shipped: Truck, completed: Package, cancelled: XCircle };
  241. return () => {
  242. const s = props.status ?? "pending";
  243. const Icon = icons[s] || Clock;
  244. return h("span", { class: `inline-flex items-center gap-1.5 px-3 py-1 rounded-full text-xs font-semibold border ${styles[s] ?? styles.pending}` }, [
  245. h(Icon, { class: "w-3.5 h-3.5" }),
  246. h("span", null, t("statuses." + s)),
  247. ]);
  248. };
  249. },
  250. });
  251. function formatDate(dt: string) {
  252. return new Date(dt).toLocaleDateString(undefined, { year: "numeric", month: "long", day: "numeric" });
  253. }
  254. async function fetchOrders() {
  255. isLoading.value = true;
  256. try {
  257. const res = await getMyOrders(currentPage.value, pageSize);
  258. orders.value = res.orders;
  259. totalOrders.value = res.total;
  260. }
  261. catch (e) { console.error("Failed to fetch orders:", e); }
  262. finally { isLoading.value = false; }
  263. }
  264. onMounted(async () => {
  265. if (!localStorage.getItem("token")) { router.push("/auth"); return; }
  266. await fetchOrders();
  267. window.addEventListener("radionica:order_updated", handleRemoteUpdate);
  268. });
  269. import { onUnmounted } from "vue";
  270. onUnmounted(() => {
  271. window.removeEventListener("radionica:order_updated", handleRemoteUpdate);
  272. });
  273. async function handleRemoteUpdate(e: any) {
  274. const orderId = e.detail?.order_id;
  275. await fetchOrders();
  276. toast.success(t("chat.new_status", { id: orderId || "" }), {
  277. description: "Order data has been updated by administrator.",
  278. icon: h(Package, { class: "w-4 h-4 text-emerald-500" })
  279. });
  280. }
  281. watch(currentPage, () => {
  282. fetchOrders();
  283. window.scrollTo({ top: 0, behavior: 'smooth' });
  284. });
  285. watch(() => authStore.unreadMessagesCount, async (newVal, oldVal) => {
  286. if (newVal !== oldVal) {
  287. try {
  288. await fetchOrders();
  289. } catch (e) {
  290. console.error("Failed to auto-refresh orders:", e);
  291. }
  292. }
  293. });
  294. </script>