Working API CHARGING!
This commit is contained in:
		| @@ -140,7 +140,7 @@ | ||||
|           <CreditCards | ||||
|             :cards="credit_cards" | ||||
|             :count="credit_cards_count" | ||||
|             :user_id="customer.user_id || 0" | ||||
|             :user_id="customer.id" | ||||
|             :auth_net_profile_id="customer.auth_net_profile_id" | ||||
|             @edit-card="editCard" | ||||
|             @remove-card="removeCard" | ||||
| @@ -1025,7 +1025,7 @@ onSubmitSocial(commentText: string) { | ||||
|     }, | ||||
|     addCreditCard() { | ||||
|       // Redirect to add card page | ||||
|       this.$router.push({ name: 'cardadd', params: { customerId: this.customer.id } }); | ||||
|       this.$router.push({ name: 'cardadd', params: { id: this.customer.id } }); | ||||
|     }, | ||||
|     showDeleteAccountModal() { | ||||
|       this.isDeleteAccountModalVisible = true; | ||||
|   | ||||
| @@ -44,7 +44,7 @@ | ||||
|               <tr v-for="oil in deliveries" :key="oil.id" class="hover:bg-blue-600 hover:text-white"> | ||||
|                 <td>{{ oil.id }}</td> | ||||
|                 <td> | ||||
|                   <router-link :to="{ name: 'customerProfile', params: { id: oil.customer_id } }" class="link link-hover"> | ||||
|                   <router-link :to="{ name: 'customerProfile', params: { id: oil.customer_id } }" class="link link-hover hover:text-green-500"> | ||||
|                     {{ oil.customer_name }} | ||||
|                   </router-link> | ||||
|                 </td> | ||||
| @@ -230,16 +230,19 @@ export default defineComponent({ | ||||
|     }, | ||||
|  | ||||
|  | ||||
|     get_oil_orders(page: any) { | ||||
|       let path = import.meta.env.VITE_BASE_URL + '/delivery/outfordelivery/' + page; | ||||
|       axios({ | ||||
|         method: 'get', | ||||
|         url: path, | ||||
|         headers: authHeader(), | ||||
|       }).then((response: any) => { | ||||
|         this.deliveries = response.data | ||||
|       }) | ||||
|     }, | ||||
|       mod: (date: any) => new Date (date).getTime(), | ||||
|       get_oil_orders(page: any) { | ||||
|         let path = import.meta.env.VITE_BASE_URL + '/delivery/outfordelivery/' + page; | ||||
|         axios({ | ||||
|           method: 'get', | ||||
|           url: path, | ||||
|           headers: authHeader(), | ||||
|         }).then((response: any) => { | ||||
|           this.deliveries = response.data | ||||
|           // Sort deliveries by Delivery # (id) in descending order | ||||
|           this.deliveries.sort((a, b) => b.id - a.id); | ||||
|         }) | ||||
|       }, | ||||
|  | ||||
|     deleteCall(delivery_id: any) { | ||||
|       let path = import.meta.env.VITE_BASE_URL + '/delivery/delete/' + delivery_id; | ||||
| @@ -315,4 +318,4 @@ export default defineComponent({ | ||||
| }) | ||||
| </script> | ||||
|  | ||||
| <style scoped></style> | ||||
| <style scoped></style> | ||||
|   | ||||
| @@ -117,7 +117,7 @@ | ||||
|               </router-link> | ||||
|               <button | ||||
|                 @click="handlePreauthorize" | ||||
|                 class="btn btn-warning" | ||||
|                 class="btn btn-success" | ||||
|                 :disabled="loading || !chargeAmount" | ||||
|               > | ||||
|                 <span v-if="loading && action === 'preauthorize'" class="loading loading-spinner loading-sm"></span> | ||||
| @@ -125,7 +125,7 @@ | ||||
|               </button> | ||||
|               <button | ||||
|                 @click="handleChargeNow" | ||||
|                 class="btn btn-primary" | ||||
|                 class="btn btn-warning text-black" | ||||
|                 :disabled="loading || !chargeAmount" | ||||
|               > | ||||
|                 <span v-if="loading && action === 'charge'" class="loading loading-spinner loading-sm"></span> | ||||
| @@ -137,6 +137,29 @@ | ||||
|       </div> | ||||
|     </div> | ||||
|   </div> | ||||
|  | ||||
|   <!-- Charge Confirmation Modal --> | ||||
|   <div class="modal" :class="{ 'modal-open': isChargeConfirmationModalVisible }"> | ||||
|     <div class="modal-box"> | ||||
|       <h3 class="font-bold text-lg text-warning">⚠️ Warning: Charge Now</h3> | ||||
|       <p class="py-4"> | ||||
|         You are about to <strong>immediately charge</strong> this customer's card | ||||
|         for <strong>${{ chargeAmount.toFixed(2) }}</strong>. | ||||
|         <br><br> | ||||
|         This action is <strong>not reversible</strong> and will debit the customer's account immediately. | ||||
|         <br><br> | ||||
|         Are you sure you want to proceed with the charge? | ||||
|       </p> | ||||
|       <div class="modal-action"> | ||||
|         <button @click="proceedWithCharge" class="btn btn-warning"> | ||||
|           Yes, Charge Now | ||||
|         </button> | ||||
|         <button @click="cancelCharge" class="btn btn-ghost"> | ||||
|           Cancel | ||||
|         </button> | ||||
|       </div> | ||||
|     </div> | ||||
|   </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| @@ -157,6 +180,7 @@ export default defineComponent({ | ||||
|       action: '', // 'preauthorize' or 'charge' | ||||
|       error: '', | ||||
|       success: '', | ||||
|       isChargeConfirmationModalVisible: false, | ||||
|       user: { | ||||
|         user_id: 0, | ||||
|       }, | ||||
| @@ -517,9 +541,26 @@ export default defineComponent({ | ||||
|     }, | ||||
|  | ||||
|     async handleChargeNow() { | ||||
|       if (!this.selectedCard) { | ||||
|         this.error = 'No credit card found for this customer' | ||||
|         return | ||||
|       } | ||||
|       if (!this.chargeAmount || this.chargeAmount <= 0) { | ||||
|         this.error = 'Please enter a valid charge amount' | ||||
|         return | ||||
|       } | ||||
|       this.isChargeConfirmationModalVisible = true | ||||
|     }, | ||||
|  | ||||
|     async proceedWithCharge() { | ||||
|       this.isChargeConfirmationModalVisible = false | ||||
|       await this.processPayment('charge') | ||||
|     }, | ||||
|  | ||||
|     cancelCharge() { | ||||
|       this.isChargeConfirmationModalVisible = false | ||||
|     }, | ||||
|  | ||||
|     async processPayment(actionType: string) { | ||||
|       if (!this.selectedCard) { | ||||
|         this.error = 'No credit card found for this customer' | ||||
|   | ||||
| @@ -3,6 +3,8 @@ | ||||
| import PayOil from './oil/pay_oil.vue'; | ||||
| import AuthorizePreauthCharge from './oil/authorize_preauthcharge.vue'; | ||||
| import CaptureAuthorize from './oil/capture_authorize.vue'; | ||||
| import PayService from './service/pay_service.vue'; | ||||
| import AuthorizeServicePreauthCharge from './service/authorize_preauthcharge.vue'; | ||||
| import ChargeServiceAuthorize from './service/capture_authorize.vue'; | ||||
|  | ||||
| const payRoutes = [ | ||||
| @@ -17,11 +19,25 @@ const payRoutes = [ | ||||
|         name: 'authorizePreauthCharge', | ||||
|         component: AuthorizePreauthCharge, | ||||
|     }, | ||||
|     // This is for oil delivery  | ||||
|     { | ||||
|         path: '/pay/capture/authorize/:id', | ||||
|         name: 'captureAuthorize', | ||||
|         component: CaptureAuthorize, | ||||
|     }, | ||||
|  | ||||
|      | ||||
|     { | ||||
|         path: '/pay/service/:id', | ||||
|         name: 'payService', | ||||
|         component: PayService, | ||||
|     }, | ||||
|     { | ||||
|         path: '/pay/service/authorize/:id', | ||||
|         name: 'authorizeServicePreauthCharge', | ||||
|         component: AuthorizeServicePreauthCharge, | ||||
|     }, | ||||
|     // this is for service  | ||||
|     { | ||||
|         path: '/pay/service/capture/authorize/:id', | ||||
|         name: 'chargeServiceAuthorize', | ||||
|   | ||||
							
								
								
									
										536
									
								
								src/pages/pay/service/authorize_preauthcharge.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										536
									
								
								src/pages/pay/service/authorize_preauthcharge.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,536 @@ | ||||
| <!-- src/pages/pay/oil/authorize_preauthcharge.vue --> | ||||
| <template> | ||||
|   <div class="flex"> | ||||
|  | ||||
|  | ||||
|     <!-- Main Content --> | ||||
|     <div class="flex-1 px-8 py-6"> | ||||
|  | ||||
|       <!-- Breadcrumbs & Header --> | ||||
|       <div class="text-sm breadcrumbs mb-6"> | ||||
|         <ul> | ||||
|           <li><router-link :to="{ name: 'home' }">Home</router-link></li> | ||||
|           <li><router-link :to="{ name: 'payService', params: { id: serviceId } }">Payment Confirmation</router-link></li> | ||||
|           <li>Payment Authorization</li> | ||||
|         </ul> | ||||
|       </div> | ||||
|  | ||||
|       <!-- 2x2 Grid Layout --> | ||||
|       <div class="max-w-6xl"> | ||||
|         <h1 class="text-3xl font-bold mb-8">Payment Authorization Authorize.net</h1> | ||||
|  | ||||
|         <!-- Top Row: Charge Breakdown and Payment Method --> | ||||
|         <div class="grid grid-cols-1 lg:grid-cols-2 gap-6 mb-8"> | ||||
|           <!-- Service Details --> | ||||
|           <div class="bg-base-100 rounded-lg p-6"> | ||||
|             <h3 class="text-lg font-semibold mb-4">Service Details</h3> | ||||
|             <div class="space-y-3"> | ||||
|               <div class="flex justify-between"> | ||||
|                 <span>Service Type:</span> | ||||
|                 <span class="badge" :class="getServiceTypeColor(service.type_service_call)"> | ||||
|                   {{ getServiceTypeName(service.type_service_call) }} | ||||
|                 </span> | ||||
|               </div> | ||||
|               <div class="flex justify-between"> | ||||
|                 <span>Scheduled Date:</span> | ||||
|                 <span>{{ service.scheduled_date ? formatScheduledDate(service.scheduled_date) : 'Not scheduled' }}</span> | ||||
|               </div> | ||||
|               <div class="flex flex-col space-y-2"> | ||||
|                 <span><strong>Description:</strong></span> | ||||
|                 <div class="bg-base-200 p-2 rounded text-sm">{{ service.description || 'No description provided' }}</div> | ||||
|               </div> | ||||
|               <hr class="my-3"> | ||||
|               <div class="flex justify-between font-bold text-lg"> | ||||
|                 <span>Service Cost:</span> | ||||
|                 <span>${{ service.service_cost || '0.00' }}</span> | ||||
|               </div> | ||||
|             </div> <!-- close space-y-3 --> | ||||
|           </div> <!-- close bg-base-100 --> | ||||
|  | ||||
|           <!-- Credit Card Selection --> | ||||
|           <div class="bg-base-100 rounded-lg p-6"> | ||||
|             <h3 class="text-lg font-semibold mb-4">Payment Method</h3> | ||||
|             <div v-if="credit_cards.length > 1" class="space-y-2"> | ||||
|               <div v-for="card in credit_cards" :key="card.id" | ||||
|                    @click="selectCard(card.id)" | ||||
|                    :class="[ | ||||
|                      'p-3 rounded-md cursor-pointer border-2 transition-all duration-200', | ||||
|                      selectedCardId === card.id || (!selectedCardId && selectedCard && selectedCard.id === card.id) | ||||
|                        ? 'bg-primary text-primary-content border-primary' | ||||
|                        : 'bg-base-200 hover:bg-base-300 border-transparent' | ||||
|                    ]"> | ||||
|                 <div class="flex justify-between items-center mb-1"> | ||||
|                   <div class="font-semibold">{{ card.type_of_card }}</div> | ||||
|                   <div class="flex gap-2"> | ||||
|                     <div v-if="card.main_card" class="badge badge-outline badge-sm">Primary</div> | ||||
|                     <div v-if="selectedCardId === card.id || (!selectedCardId && selectedCard && selectedCard.id === card.id)" | ||||
|                          class="w-3 h-3 bg-current rounded-full"></div> | ||||
|                   </div> | ||||
|                 </div> | ||||
|                 <div class="font-mono text-sm"> | ||||
|                   <div> **** **** **** {{ card.last_four_digits }}</div> | ||||
|                   <div>{{ card.name_on_card }}</div> | ||||
|                   <div>Expires: {{ card.expiration_month }}/{{ card.expiration_year }}</div> | ||||
|                 </div> | ||||
|               </div> | ||||
|             </div> | ||||
|             <div v-else-if="credit_cards.length === 1" class="bg-base-200 p-4 rounded-md"> | ||||
|               <div class="flex justify-between items-center mb-2"> | ||||
|                 <div class="font-semibold">{{ credit_cards[0].type_of_card }}</div> | ||||
|                 <div v-if="credit_cards[0].main_card" class="badge badge-primary">Primary</div> | ||||
|               </div> | ||||
|               <div class="font-mono text-sm"> | ||||
|                 <div> **** **** **** {{ credit_cards[0].last_four_digits }}</div> | ||||
|                 <div>{{ credit_cards[0].name_on_card }}</div> | ||||
|                 <div>Expires: {{ credit_cards[0].expiration_month }}/{{ credit_cards[0].expiration_year }}</div> | ||||
|               </div> | ||||
|             </div> | ||||
|             <div v-else class="text-gray-500 p-4"> | ||||
|               No payment methods available | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|  | ||||
|         <!-- Error/Success Messages --> | ||||
|         <div class="mb-6"> | ||||
|           <div v-if="error" class="alert alert-error"> | ||||
|             <span>{{ error }}</span> | ||||
|           </div> | ||||
|           <div v-if="success" class="alert alert-success"> | ||||
|             <span>{{ success }}</span> | ||||
|           </div> | ||||
|         </div> | ||||
|  | ||||
|         <!-- Bottom Section: Action Buttons --> | ||||
|         <div class="bg-base-100 rounded-lg p-6"> | ||||
|           <div class="flex flex-col lg:flex-row lg:items-center gap-6"> | ||||
|             <!-- Charge Amount Input - Always visible --> | ||||
|             <div class="flex-1"> | ||||
|               <h3 class="text-lg font-semibold mb-2">Charge/Preauthorize Amount</h3> | ||||
|               <div class="form-control"> | ||||
|                 <div class="relative"> | ||||
|                   <span class="absolute left-3 top-1/2 transform -translate-y-1/2 text-sm z-10">$</span> | ||||
|                   <input | ||||
|                     id="chargeAmountInput" | ||||
|                     v-model="chargeAmount" | ||||
|                     type="number" | ||||
|                     step="0.01" | ||||
|                     class="input input-bordered input-sm w-full pl-6" | ||||
|                     placeholder="0.00" | ||||
|                     :disabled="loading" | ||||
|                     min="0.01" | ||||
|                     autofocus | ||||
|                   /> | ||||
|                 </div> | ||||
|               </div> | ||||
|             </div> | ||||
|  | ||||
|             <!-- Action Buttons --> | ||||
|             <div class="flex gap-3 mt-4"> | ||||
|               <router-link :to="{ name: 'payService', params: { id: serviceId } }"> | ||||
|                 <button class="btn btn-ghost">Cancel</button> | ||||
|               </router-link> | ||||
|               <button | ||||
|                 @click="handlePreauthorize" | ||||
|                 class="btn btn-success" | ||||
|                 :disabled="loading || chargeAmount <= 0 || isNaN(Number(chargeAmount))" | ||||
|               > | ||||
|                 <span v-if="loading && action === 'preauthorize'" class="loading loading-spinner loading-sm"></span> | ||||
|                 Preauthorize | ||||
|               </button> | ||||
|               <button | ||||
|                 @click="handleChargeNow" | ||||
|                 class="btn btn-warning" | ||||
|                 :disabled="loading || chargeAmount <= 0 || isNaN(Number(chargeAmount))" | ||||
|               > | ||||
|                 <span v-if="loading && action === 'charge'" class="loading loading-spinner loading-sm"></span> | ||||
|                  Charge | ||||
|               </button> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|       </div> | ||||
|     </div> | ||||
|   </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import { defineComponent, watch } from 'vue' | ||||
| import axios from 'axios' | ||||
| import authHeader from '../../../services/auth.header' | ||||
| import { notify } from "@kyvg/vue3-notification" | ||||
|  | ||||
| export default defineComponent({ | ||||
|   name: 'AuthorizeServicePreauthCharge', | ||||
|  | ||||
|   data() { | ||||
|     return { | ||||
|       serviceId: this.$route.params.id as string, | ||||
|       loaded: false, | ||||
|       chargeAmount: 0, | ||||
|       loading: false, | ||||
|       action: '', // 'preauthorize' or 'charge' | ||||
|       error: '', | ||||
|       success: '', | ||||
|       selectedCardId: null as number | null, // Track which card is selected | ||||
|       user: { | ||||
|         user_id: 0, | ||||
|       }, | ||||
|       service: { | ||||
|         id: 0, | ||||
|         scheduled_date: '', | ||||
|         customer_id: 0, | ||||
|         customer_name: '', | ||||
|         customer_address: '', | ||||
|         customer_town: '', | ||||
|         type_service_call: 0, | ||||
|         description: '', | ||||
|         service_cost: '', | ||||
|         payment_card_id: 0, | ||||
|       }, | ||||
|       credit_cards: [ | ||||
|         { | ||||
|           id: 0, | ||||
|           name_on_card: '', | ||||
|           main_card: false, | ||||
|           card_number: '', | ||||
|           expiration_month: '', | ||||
|           type_of_card: '', | ||||
|           last_four_digits: '', | ||||
|           expiration_year: '', | ||||
|           security_number: '', | ||||
|  | ||||
|         } | ||||
|       ], | ||||
|       customer: { | ||||
|         id: 0, | ||||
|         user_id: 0, | ||||
|         customer_first_name: '', | ||||
|         customer_last_name: '', | ||||
|         customer_town: '', | ||||
|         customer_address: '', | ||||
|         customer_state: 0, | ||||
|         customer_zip: '', | ||||
|         customer_apt: '', | ||||
|         customer_home_type: 0, | ||||
|         customer_phone_number: '', | ||||
|         account_number: '', | ||||
|       }, | ||||
|  | ||||
|     } | ||||
|   }, | ||||
|  | ||||
|   computed: { | ||||
|     selectedCard(): any { | ||||
|       // If user has selected a card manually, use that | ||||
|       if (this.selectedCardId) { | ||||
|         return this.credit_cards.find((card: any) => card.id === this.selectedCardId) | ||||
|       } | ||||
|       // Otherwise use automatic selection logic | ||||
|       // First try to find payment_card_id from service | ||||
|       if (this.service.payment_card_id && this.service.payment_card_id > 0) { | ||||
|         return this.credit_cards.find((card: any) => card.id === this.service.payment_card_id) | ||||
|       } | ||||
|       // Otherwise return the primary card (main_card = true) | ||||
|       return this.credit_cards.find((card: any) => card.main_card === true) || | ||||
|              this.credit_cards.find((card: any) => card.id > 0) || // Any card if no primary | ||||
|              null | ||||
|     } | ||||
|   }, | ||||
|  | ||||
|   mounted() { | ||||
|     this.loadData(this.serviceId) | ||||
|   }, | ||||
|  | ||||
|   created() { | ||||
|     this.watchRoute() | ||||
|   }, | ||||
|  | ||||
|   methods: { | ||||
|     watchRoute() { | ||||
|       watch( | ||||
|         () => this.$route.params.id, | ||||
|         (newId) => { | ||||
|           if (newId !== this.serviceId) { | ||||
|             this.resetState() | ||||
|             this.serviceId = newId as string | ||||
|             this.loadData(newId as string) | ||||
|           } | ||||
|         } | ||||
|       ) | ||||
|     }, | ||||
|  | ||||
|     resetState() { | ||||
|       this.loading = false | ||||
|       this.action = '' | ||||
|       this.error = '' | ||||
|       this.success = '' | ||||
|       this.chargeAmount = 0 | ||||
|       this.serviceId = this.$route.params.id as string | ||||
|     }, | ||||
|  | ||||
|     loadData(serviceId: string) { | ||||
|       this.userStatus() | ||||
|       this.getServiceOrder(serviceId) | ||||
|     }, | ||||
|  | ||||
|     userStatus() { | ||||
|       let path = import.meta.env.VITE_BASE_URL + '/auth/whoami'; | ||||
|       axios({ | ||||
|         method: 'get', | ||||
|         url: path, | ||||
|         withCredentials: true, | ||||
|         headers: authHeader(), | ||||
|       }) | ||||
|         .then((response: any) => { | ||||
|           if (response.data.ok) { | ||||
|             this.user = response.data.user; | ||||
|           } | ||||
|         }) | ||||
|     }, | ||||
|  | ||||
|     getServiceOrder(serviceId: any) { | ||||
|       let path = import.meta.env.VITE_BASE_URL + "/service/" + serviceId; | ||||
|       axios({ | ||||
|         method: "get", | ||||
|         url: path, | ||||
|         withCredentials: true, | ||||
|         headers: authHeader(), | ||||
|       }) | ||||
|         .then((response: any) => { | ||||
|           let serviceData; | ||||
|           if (response.data) { | ||||
|             // Handle different API response structures | ||||
|             if (response.data.service) { | ||||
|               // API returns {ok: true, service: {...}} structure | ||||
|               serviceData = response.data.service; | ||||
|             } else if (Array.isArray(response.data)) { | ||||
|               serviceData = response.data[0]; // Array response | ||||
|             } else { | ||||
|               serviceData = response.data; // Direct object response | ||||
|             } | ||||
|  | ||||
|             if (serviceData && serviceData.id) { | ||||
|               this.service = { | ||||
|                 id: serviceData.id, | ||||
|                 scheduled_date: serviceData.scheduled_date, | ||||
|                 customer_id: serviceData.customer_id, | ||||
|                 customer_name: serviceData.customer_name, | ||||
|                 customer_address: serviceData.customer_address, | ||||
|                 customer_town: serviceData.customer_town, | ||||
|                 type_service_call: serviceData.type_service_call, | ||||
|                 description: serviceData.description, | ||||
|                 service_cost: serviceData.service_cost, | ||||
|                 payment_card_id: serviceData.payment_card_id || 0, | ||||
|               }; | ||||
|  | ||||
|               // Fetch related data | ||||
|               this.getCustomer(this.service.customer_id); | ||||
|               this.getCreditCards(this.service.customer_id); | ||||
|             } else { | ||||
|               console.error("API Error: Invalid service data received:", serviceData); | ||||
|               notify({ | ||||
|                 title: "Error", | ||||
|                 text: "Invalid service data received", | ||||
|                 type: "error", | ||||
|               }); | ||||
|             } | ||||
|           } else { | ||||
|             console.error("API Error: No response data received"); | ||||
|             notify({ | ||||
|               title: "Error", | ||||
|               text: "Could not get service data", | ||||
|               type: "error", | ||||
|             }); | ||||
|           } | ||||
|         }) | ||||
|         .catch((error: any) => { | ||||
|           console.error("API Error in getServiceOrder:", error); | ||||
|           notify({ | ||||
|             title: "Error", | ||||
|             text: "Could not get service data", | ||||
|             type: "error", | ||||
|           }); | ||||
|         }); | ||||
|     }, | ||||
|  | ||||
|    | ||||
|  | ||||
|     getCreditCards(user_id: any) { | ||||
|       let path = import.meta.env.VITE_BASE_URL + '/payment/cards/' + user_id; | ||||
|       axios({ | ||||
|         method: 'get', | ||||
|         url: path, | ||||
|         headers: authHeader(), | ||||
|       }).then((response: any) => { | ||||
|         console.log('Credit cards loaded:', response.data?.length || 0, 'cards'); | ||||
|         this.credit_cards = response.data || []; | ||||
|       }).catch((error: any) => { | ||||
|         console.error('Failed to load credit cards:', error); | ||||
|         this.credit_cards = []; | ||||
|       }); | ||||
|     }, | ||||
|  | ||||
|     getCustomer(userid: any) { | ||||
|       let path = import.meta.env.VITE_BASE_URL + '/customer/' + userid; | ||||
|       axios({ | ||||
|         method: 'get', | ||||
|         url: path, | ||||
|         headers: authHeader(), | ||||
|       }).then((response: any) => { | ||||
|         this.customer = response.data | ||||
|       }) | ||||
|     }, | ||||
|  | ||||
|  | ||||
|  | ||||
|     async handlePreauthorize() { | ||||
|       await this.processPayment('preauthorize') | ||||
|     }, | ||||
|  | ||||
|     async handleChargeNow() { | ||||
|       await this.processPayment('charge') | ||||
|     }, | ||||
|  | ||||
|     async processPayment(actionType: string) { | ||||
|       if (!this.selectedCard) { | ||||
|         this.error = 'No credit card found for this customer' | ||||
|         return | ||||
|       } | ||||
|  | ||||
|       this.loading = true | ||||
|       this.action = actionType | ||||
|       this.error = '' | ||||
|       this.success = '' | ||||
|  | ||||
|       try { | ||||
|         // Step 2: If payment method is credit, perform the pre-authorization | ||||
|         if (actionType === 'preauthorize') { | ||||
|           if (!this.chargeAmount || this.chargeAmount <= 0) { | ||||
|             throw new Error("Pre-authorization amount must be greater than zero."); | ||||
|           } | ||||
|  | ||||
|           const authPayload = { | ||||
|             card_id: (this.selectedCard as any).id, | ||||
|             preauthorize_amount: this.chargeAmount.toFixed(2), | ||||
|             service_id: this.service.id, | ||||
|             delivery_id: null, // No delivery for services | ||||
|           }; | ||||
|  | ||||
|           const authPath = `${import.meta.env.VITE_AUTHORIZE_URL}/api/payments/authorize/saved-card/${this.customer.id}`; | ||||
|  | ||||
|           const response = await axios.post(authPath, authPayload, { withCredentials: true, headers: authHeader() }); | ||||
|  | ||||
|           // Update payment type to 11 after successful preauthorization | ||||
|           try { | ||||
|             await axios.put(`${import.meta.env.VITE_BASE_URL}/payment/authorize/service/${this.service.id}`, { | ||||
|   card_id: (this.selectedCard as any).id, | ||||
|   status: actionType === 'preauthorize' ? 1 : 3 | ||||
| }, { headers: authHeader() }); | ||||
|           } catch (updateError) { | ||||
|             console.error('Failed to update payment type after preauthorization:', updateError); | ||||
|           } | ||||
|  | ||||
|           // On successful authorization, show success and redirect | ||||
|           this.success = `Preauthorization successful! Transaction ID: ${response.data?.auth_net_transaction_id || 'N/A'}`; | ||||
|           setTimeout(() => { | ||||
|             this.$router.push({ name: "customerProfile", params: { id: this.customer.id } }); | ||||
|           }, 2000); | ||||
|         } else { // Handle 'charge' action | ||||
|           if (!this.chargeAmount || this.chargeAmount <= 0) { | ||||
|             throw new Error("Charge amount must be greater than zero."); | ||||
|           } | ||||
|  | ||||
|           // Create a payload that matches the backend's TransactionCreateByCardID schema | ||||
|           const chargePayload = { | ||||
|             card_id: (this.selectedCard as any).id, | ||||
|             charge_amount: this.chargeAmount.toFixed(2), | ||||
|             service_id: this.service.id, | ||||
|             delivery_id: null, // No delivery for services | ||||
|             // You can add other fields here if your schema requires them | ||||
|           }; | ||||
|  | ||||
|           // Use the correct endpoint for charging a saved card | ||||
|           const chargePath = `${import.meta.env.VITE_AUTHORIZE_URL}/api/payments/charge/saved-card/${this.customer.id}`; | ||||
|  | ||||
|           console.log('=== DEBUG: Charge payload ==='); | ||||
|           console.log('Calling endpoint:', chargePath); | ||||
|           console.log('Final payload being sent:', chargePayload); | ||||
|  | ||||
|           const response = await axios.post(chargePath, chargePayload, { withCredentials: true, headers: authHeader() }); | ||||
|  | ||||
|           // Update service cost to the charged amount using new dedicated API | ||||
|           try { | ||||
|             await axios.put( | ||||
|               `${import.meta.env.VITE_BASE_URL}/service/update-cost/${this.service.id}`, | ||||
|               { service_cost: this.chargeAmount }, | ||||
|               { headers: authHeader(), withCredentials: true } | ||||
|             ); | ||||
|             console.log(`✅ Updated service cost to ${this.chargeAmount} for service ${this.service.id}`); | ||||
|           } catch (costError) { | ||||
|             console.error('❌ Failed to update service cost:', costError); | ||||
|           } | ||||
|  | ||||
|           // Update payment status after successful charge | ||||
|           try { | ||||
|             await axios.put(`${import.meta.env.VITE_BASE_URL}/payment/capture/service/${this.service.id}`, { | ||||
|               card_id: (this.selectedCard as any).id, | ||||
|               status: 3  // Approved status | ||||
|             }, { | ||||
|               headers: authHeader(), | ||||
|               withCredentials: true | ||||
|             }); | ||||
|           } catch (updateError) { | ||||
|             console.error('Failed to update payment status after charge:', updateError); | ||||
|           } | ||||
|  | ||||
|           // Status codes: 0 = APPROVED, 1 = DECLINED (based on backend TransactionStatus enum) | ||||
|           if (response.data && response.data.status === 0) { // 0 = APPROVED | ||||
|             this.success = `Charge successful! Transaction ID: ${response.data.auth_net_transaction_id || 'N/A'}`; | ||||
|             setTimeout(() => { | ||||
|               this.$router.push({ name: "customerProfile", params: { id: this.customer.id } }); | ||||
|             }, 2000); | ||||
|           } else { | ||||
|             // The error message from your backend will be more specific now | ||||
|             throw new Error(`Payment charge failed: ${response.data?.rejection_reason || 'Unknown error'}`); | ||||
|           } | ||||
|         } | ||||
|       } catch (error: any) { | ||||
|         console.log(error) | ||||
|         this.error = error.response?.data?.detail || `Failed to ${actionType} payment` | ||||
|         notify({ | ||||
|           title: "Error", | ||||
|           text: this.error, | ||||
|           type: "error", | ||||
|         }) | ||||
|       } finally { | ||||
|         this.loading = false | ||||
|         this.action = '' | ||||
|       } | ||||
|     }, | ||||
|  | ||||
|     getServiceTypeName(typeId: number): string { | ||||
|       const typeMap: { [key: number]: string } = { 0: 'Tune-up', 1: 'No Heat', 2: 'Fix', 3: 'Tank Install', 4: 'Other' }; | ||||
|       return typeMap[typeId] || 'Unknown'; | ||||
|     }, | ||||
|  | ||||
|     getServiceTypeColor(typeId: number): string { | ||||
|       const colorMap: { [key: number]: string } = { 0: 'primary', 1: 'error', 2: 'warning', 3: 'info', 4: 'neutral' }; | ||||
|       return `badge-${colorMap[typeId] || 'neutral'}`; | ||||
|     }, | ||||
|  | ||||
|     selectCard(cardId: number) { | ||||
|       this.selectedCardId = cardId; | ||||
|     }, | ||||
|  | ||||
|  | ||||
|  | ||||
|     formatScheduledDate(dateString: string): string { | ||||
|       if (!dateString) return 'Not scheduled'; | ||||
|       return dateString; // Could format with dayjs if needed | ||||
|     } | ||||
|   }, | ||||
| }) | ||||
| </script> | ||||
|  | ||||
| <style scoped></style> | ||||
| @@ -11,14 +11,14 @@ | ||||
|               {{ service.customer_name }} | ||||
|             </router-link> | ||||
|           </li> | ||||
|           <li>Charge Service #{{ service.id }}</li> | ||||
|           <li>Capture Service #{{ service.id }}</li> | ||||
|         </ul> | ||||
|       </div> | ||||
|  | ||||
|       <!-- Page Header --> | ||||
|       <div class="flex flex-wrap items-center justify-between gap-2 mt-4"> | ||||
|         <h1 class="text-3xl font-bold"> | ||||
|           Charge Service #{{ service?.id }} | ||||
|           Capture Charge for Service #{{ service?.id }} | ||||
|         </h1> | ||||
|         <router-link v-if="service" :to="{ name: 'customerProfile', params: { id: service.customer_id } }"> | ||||
|           <button class="btn btn-sm btn-secondary">Back to Customer</button> | ||||
| @@ -60,61 +60,102 @@ | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|  | ||||
|           <!-- Transaction History Card --> | ||||
|           <div class="p-5 rounded-lg bg-neutral"> | ||||
|             <h3 class="mb-4 text-xl font-bold">Transaction History</h3> | ||||
|             <div v-if="serviceTransactions.length === 0" class="text-sm opacity-80"> | ||||
|               <p>No transactions found for this service.</p> | ||||
|             </div> | ||||
|             <div v-else class="space-y-3"> | ||||
|               <div | ||||
|                 v-for="transaction in serviceTransactions" | ||||
|                 :key="transaction.id" | ||||
|                 class="p-3 bg-base-100 rounded-lg border border-base-300" | ||||
|               > | ||||
|                 <div class="flex justify-between items-start mb-2"> | ||||
|                   <div class="text-sm"> | ||||
|                     <div class="font-bold">Transaction ID: {{ transaction.id }}</div> | ||||
|                     <div class="opacity-70">{{ transaction.created_at ? formatDateTime(transaction.created_at) : 'N/A' }}</div> | ||||
|                   </div> | ||||
|                   <div class="badge" | ||||
|                        :class="{ | ||||
|                          'badge-success': transaction.status === 0, | ||||
|                          'badge-warning': transaction.status === 1, | ||||
|                          'badge-error': transaction.status === 2, | ||||
|                          'badge-info': transaction.status === 3, | ||||
|                        }"> | ||||
|                     {{ getTransactionStatus(transaction.status) }} | ||||
|                   </div> | ||||
|                 </div> | ||||
|  | ||||
|                 <div class="grid grid-cols-2 gap-4 text-sm"> | ||||
|                   <div> | ||||
|                     <div class="font-bold">Preauth Amount</div> | ||||
|                     <div class="font-mono">${{ formatCurrency(transaction.preauthorize_amount) }}</div> | ||||
|                   </div> | ||||
|                   <div> | ||||
|                     <div class="font-bold">Charge Amount</div> | ||||
|                     <div class="font-mono">${{ formatCurrency(transaction.charge_amount) }}</div> | ||||
|                   </div> | ||||
|                   <div> | ||||
|                     <div class="font-bold">Transaction Type</div> | ||||
|                     <div>{{ getTransactionType(transaction.transaction_type) }}</div> | ||||
|                   </div> | ||||
|                   <div> | ||||
|                     <div class="font-bold">Auth Transaction ID</div> | ||||
|                     <div class="font-mono text-xs">{{ transaction.auth_net_transaction_id || 'N/A' }}</div> | ||||
|                   </div> | ||||
|                 </div> | ||||
|  | ||||
|                 <div v-if="transaction.rejection_reason" class="mt-2 p-2 bg-error/10 border border-error/20 rounded text-sm"> | ||||
|                   <div class="font-bold text-error">Rejection Reason:</div> | ||||
|                   <div class="text-error">{{ transaction.rejection_reason }}</div> | ||||
|                 </div> | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|  | ||||
|         <!-- RIGHT COLUMN: Cards and Payment Form --> | ||||
|         <div class="space-y-6"> | ||||
|           <!-- Credit Cards Display --> | ||||
|           <div class="p-5 rounded-lg bg-neutral"> | ||||
|             <div class="flex items-center justify-between mb-4"> | ||||
|               <h3 class="text-xl font-bold">Credit Cards</h3> | ||||
|               <router-link :to="{ name: 'cardadd', params: { id: service.customer_id } }"> | ||||
|                 <button class="btn btn-xs btn-outline btn-success">Add New</button> | ||||
|               </router-link> | ||||
|             <h3 class="text-xl font-bold mb-4">Pre-authorized Card</h3> | ||||
|             <div v-if="!preAuthCard" class="mt-2 text-sm opacity-70"> | ||||
|               <p class="font-semibold text-warning">No pre-authorized card found.</p> | ||||
|             </div> | ||||
|             <div v-if="userCards.length === 0" class="mt-2 text-sm opacity-70"> | ||||
|               <p class="font-semibold text-warning">No cards on file.</p> | ||||
|             </div> | ||||
|             <div class="mt-4 space-y-3"> | ||||
|               <div | ||||
|                 v-for="card in userCards" | ||||
|                 :key="card.id" | ||||
|                 class="p-2 border rounded-lg cursor-pointer transition-colors" | ||||
|                 :class="{ | ||||
|                   'bg-blue-500 text-white border-blue-500': selectedCard?.id === card.id, | ||||
|                   'bg-primary/10 border-primary': selectedCard?.id !== card.id && card.main_card, | ||||
|                   'bg-base-200 border-base-300': selectedCard?.id !== card.id && !card.main_card, | ||||
|                 }" | ||||
|                 @click="selectCard(card)" | ||||
|               > | ||||
|             <div class="mt-4"> | ||||
|               <div class="p-2 border rounded-lg bg-base-200 border-base-300" v-if="preAuthCard"> | ||||
|                 <div class="flex items-start justify-between"> | ||||
|                   <div> | ||||
|                     <div class="text-sm font-bold">{{ card.name_on_card }}</div> | ||||
|                     <div class="text-xs opacity-70">{{ card.type_of_card }}</div> | ||||
|                     <div class="text-sm font-bold">{{ preAuthCard.name_on_card }}</div> | ||||
|                     <div class="text-xs opacity-70">{{ preAuthCard.type_of_card }}</div> | ||||
|                   </div> | ||||
|                   <div v-if="card.main_card" class="badge badge-primary badge-sm">Primary</div> | ||||
|                   <div v-if="preAuthCard.main_card" class="badge badge-primary badge-sm">Primary</div> | ||||
|                 </div> | ||||
|                 <div class="mt-1 font-mono text-sm tracking-wider"> | ||||
|                   <p>{{ card.card_number }}</p> | ||||
|                   <p>Exp: {{ formattedExpiration(card) }}</p> | ||||
|             <p>{{ card.security_number }}</p> | ||||
|                   <p>{{ preAuthCard.card_number }}</p> | ||||
|                   <p>Exp: {{ formattedExpiration(preAuthCard) }}</p> | ||||
|                   <p>{{ preAuthCard.security_number }}</p> | ||||
|                 </div> | ||||
|               </div> | ||||
|             </div> | ||||
|  | ||||
|             <!-- Payment Form - always shown --> | ||||
|             <div class="pt-4 mt-6 space-y-4 border-t border-base-300"> | ||||
|               <div v-if="preAuthAmount && preAuthAmount > 0" class="mb-2 text-sm text-orange-600">Preauthorization amount: ${{ preAuthAmount.toFixed(2) }}</div> | ||||
|               <div class="form-control"> | ||||
|                 <label class="label"> | ||||
|                   <span class="font-bold label-text">Charge Amount</span> | ||||
|                 </label> | ||||
|                 <input | ||||
|                   v-model="chargeAmount" | ||||
|                   class="w-full input input-bordered input-sm" | ||||
|                   type="number" | ||||
|                   step="0.01" | ||||
|                   placeholder="0.00" | ||||
|                 /> | ||||
|                 <div class="relative"> | ||||
|                   <span class="absolute left-3 top-1/2 transform -translate-y-1/2 text-sm z-10">$</span> | ||||
|                   <input | ||||
|                     v-model="chargeAmount" | ||||
|                     class="w-full input input-bordered input-sm pl-6" | ||||
|                     type="number" | ||||
|                     step="0.01" | ||||
|                     placeholder="0.00" | ||||
|                   /> | ||||
|                 </div> | ||||
|               </div> | ||||
|  | ||||
|               <div class="flex gap-4"> | ||||
| @@ -124,7 +165,7 @@ | ||||
|                   @click="chargeService" | ||||
|                 > | ||||
|                   <span v-if="isSubmitting" class="loading loading-spinner loading-sm"></span> | ||||
|                   Charge Service | ||||
|                   Capture Charge | ||||
|                 </button> | ||||
|                 <button class="btn btn-ghost" :disabled="isSubmitting" @click="cancelCharge"> | ||||
|                   Cancel | ||||
| @@ -165,6 +206,20 @@ interface UserCard { | ||||
|   security_number: string; | ||||
| } | ||||
|  | ||||
| interface ServiceTransaction { | ||||
|   id: number; | ||||
|   preauthorize_amount: number; | ||||
|   charge_amount: number; | ||||
|   transaction_type: number; | ||||
|   status: number; | ||||
|   created_at: string; | ||||
|   auth_net_transaction_id?: string; | ||||
|   rejection_reason?: string; | ||||
|   delivery_id?: number; | ||||
|   service_id?: number; | ||||
|   card_id?: number; | ||||
| } | ||||
|  | ||||
| interface Service { | ||||
|   id: number; | ||||
|   customer_id: number; | ||||
| @@ -178,6 +233,7 @@ interface Service { | ||||
|   scheduled_date: string; | ||||
|   description: string; | ||||
|   service_cost: number; | ||||
|   payment_card_id?: number; | ||||
| } | ||||
|  | ||||
| // --- Component State --- | ||||
| @@ -186,11 +242,12 @@ const router = useRouter(); | ||||
|  | ||||
| const isSubmitting = ref(false); | ||||
| const service = ref<Service | null>(null); | ||||
| const userCards = ref<UserCard[]>([]); | ||||
| const preAuthCard = ref<UserCard | null>(null); | ||||
| const selectedCard = ref<UserCard | null>(null); | ||||
| const chargeAmount = ref<number>(0); | ||||
| const transaction = ref(null as any); | ||||
| const preAuthAmount = ref<number>(0); | ||||
| const serviceTransactions = ref<ServiceTransaction[]>([]); | ||||
|  | ||||
| // --- Computed Properties for Cleaner Template --- | ||||
| const stateName = computed(() => { | ||||
| @@ -208,11 +265,50 @@ const serviceTypeName = computed(() => { | ||||
|   return service.value ? typeMap[service.value.type_service_call] || 'Unknown' : ''; | ||||
| }); | ||||
|  | ||||
| // --- Additional Utility Functions --- | ||||
| const formattedExpiration = (card: UserCard) => { | ||||
|   const month = String(card.expiration_month).padStart(2, '0'); | ||||
|   return `${month} / ${card.expiration_year}`; | ||||
| }; | ||||
|  | ||||
| const formatCurrency = (value: number): string => { | ||||
|   if (value === null || value === undefined || isNaN(value)) return '$0.00'; | ||||
|   return value.toFixed(2); | ||||
| }; | ||||
|  | ||||
| const formatDateTime = (dateString: string): string => { | ||||
|   if (!dateString) return 'N/A'; | ||||
|   try { | ||||
|     const date = new Date(dateString); | ||||
|     return date.toLocaleDateString() + ' ' + date.toLocaleTimeString([], { hour: '2-digit', minute: '2-digit' }); | ||||
|   } catch { | ||||
|     return 'N/A'; | ||||
|   } | ||||
| }; | ||||
|  | ||||
| const getTransactionStatus = (status: number): string => { | ||||
|   const statuses: { [key: number]: string } = { | ||||
|     0: 'Approved', | ||||
|     1: 'Pending', | ||||
|     2: 'Declined', | ||||
|     3: 'Captured', | ||||
|     4: 'Voided', | ||||
|     5: 'Settled', | ||||
|   }; | ||||
|   return statuses[status] || 'Unknown'; | ||||
| }; | ||||
|  | ||||
| const getTransactionType = (type: number): string => { | ||||
|   const types: { [key: number]: string } = { | ||||
|     0: 'Charge', | ||||
|     1: 'Pre-authorization', | ||||
|     2: 'Void', | ||||
|     3: 'Refund', | ||||
|     4: 'Capture', | ||||
|   }; | ||||
|   return types[type] || 'Unknown'; | ||||
| }; | ||||
|  | ||||
|  | ||||
| // --- Methods --- | ||||
|  | ||||
| @@ -229,6 +325,32 @@ const selectCard = (card: UserCard) => { | ||||
|   } | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Dedicated function to update the service cost using the new dedicated API | ||||
|  */ | ||||
| const updateServiceCost = async (serviceId: number, newCost: number): Promise<boolean> => { | ||||
|   try { | ||||
|     console.log(`🎯 UPDATING SERVICE COST: Service ${serviceId} → $${newCost}`); | ||||
|  | ||||
|     const response = await axios.put( | ||||
|       `${import.meta.env.VITE_BASE_URL}/service/update-cost/${serviceId}`, | ||||
|       { service_cost: newCost }, | ||||
|       { withCredentials: true, headers: authHeader() } | ||||
|     ); | ||||
|  | ||||
|     if (response.data.ok) { | ||||
|       console.log(`✅ SERVICE COST UPDATED SUCCESSFULLY: ${response.data.message}`); | ||||
|       return true; | ||||
|     } else { | ||||
|       console.error(`❌ SERVICE COST UPDATE FAILED:`, response.data); | ||||
|       return false; | ||||
|     } | ||||
|   } catch (error: any) { | ||||
|     console.error(`💥 ERROR UPDATING SERVICE COST:`, error.response?.data || error.message); | ||||
|     return false; | ||||
|   } | ||||
| }; | ||||
|  | ||||
| const chargeService = async () => { | ||||
|   if (!selectedCard.value || !chargeAmount.value || chargeAmount.value <= 0) { | ||||
|     notify({ title: "Error", text: "Please select a card and enter a valid amount.", type: "error" }); | ||||
| @@ -237,48 +359,95 @@ const chargeService = async () => { | ||||
|  | ||||
|   isSubmitting.value = true; | ||||
|   try { | ||||
|     const card = selectedCard.value; | ||||
|     const expMonth = String(card.expiration_month).padStart(2, '0'); | ||||
|     const expYear = String(card.expiration_year).toString().slice(-2); | ||||
|     // Check if we have an auth transaction ID from transaction history | ||||
|     const authTransaction = serviceTransactions.value?.find(t => t.auth_net_transaction_id); | ||||
|  | ||||
|     const payload = { | ||||
|     if (!authTransaction?.auth_net_transaction_id) { | ||||
|       // No auth transaction found, try the original charge flow | ||||
|       const card = selectedCard.value; | ||||
|       const expMonth = String(card.expiration_month).padStart(2, '0'); | ||||
|       const expYear = String(card.expiration_year).toString().slice(-2); | ||||
|  | ||||
|       const payload = { | ||||
|         charge_amount: chargeAmount.value, | ||||
|         service_id: service.value!.id, | ||||
|         delivery_id: null, | ||||
|         transaction_type: 0, | ||||
|         card_id: card.id, | ||||
|         card_number: card.card_number, | ||||
|         expiration_date: `${expMonth}${expYear}`, | ||||
|         cvv: card.security_number, | ||||
|       }; | ||||
|  | ||||
|       const response = await axios.post( | ||||
|         `${import.meta.env.VITE_AUTHORIZE_URL}/api/charge/${service.value!.customer_id}`, | ||||
|         payload, | ||||
|         { withCredentials: true, headers: authHeader() } | ||||
|       ); | ||||
|  | ||||
|       if (response.data?.status === 0) { | ||||
|         // Update service cost to the charged amount using the new dedicated function | ||||
|         const costUpdateSuccess = await updateServiceCost(service.value!.id, chargeAmount.value); | ||||
|  | ||||
|         // Update payment status to 3 for success | ||||
|         await axios.put( | ||||
|           `${import.meta.env.VITE_BASE_URL}/payment/capture/service/${service.value!.id}`, | ||||
|           { | ||||
|             card_id: selectedCard.value!.id, | ||||
|             status: 3 | ||||
|           }, | ||||
|           { withCredentials: true, headers: authHeader() } | ||||
|         ); | ||||
|  | ||||
|         notify({ title: "Success", text: "Service charged successfully!", type: "success" }); | ||||
|         router.push({ name: 'customerProfile', params: { id: service.value!.customer_id } }); | ||||
|         return; | ||||
|       } else { | ||||
|         const reason = response.data?.rejection_reason || "The charge was declined."; | ||||
|         notify({ title: "Charge Declined", text: reason, type: "error" }); | ||||
|         return; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     // We have an auth transaction, so capture it | ||||
|     const capturePayload = { | ||||
|       charge_amount: chargeAmount.value, | ||||
|       service_id: service.value!.id, | ||||
|       delivery_id: null, | ||||
|       transaction_type: 0, | ||||
|       card_id: card.id, | ||||
|       card_number: card.card_number, | ||||
|       expiration_date: `${expMonth}${expYear}`, | ||||
|       cvv: card.security_number, | ||||
|       auth_net_transaction_id: authTransaction.auth_net_transaction_id | ||||
|     }; | ||||
|     console.log(payload) | ||||
|     console.log(card) | ||||
|     const response = await axios.post( | ||||
|       `${import.meta.env.VITE_AUTHORIZE_URL}/api/charge/${service.value!.customer_id}`, | ||||
|       payload, | ||||
|  | ||||
|     const captureResponse = await axios.post( | ||||
|       `${import.meta.env.VITE_AUTHORIZE_URL}/api/capture/`, | ||||
|       capturePayload, | ||||
|       { withCredentials: true, headers: authHeader() } | ||||
|     ); | ||||
|  | ||||
|     if (response.data?.status === 0) { | ||||
|       // Payment approved: now update the service cost in the database | ||||
|     if (captureResponse.data?.status === 0) { | ||||
|       // Update service cost to the captured amount using the new dedicated function | ||||
|       const costUpdateSuccess = await updateServiceCost(service.value!.id, chargeAmount.value); | ||||
|  | ||||
|       // Update service payment status | ||||
|       await axios.put( | ||||
|         `${import.meta.env.VITE_BASE_URL}/service/update/${service.value!.id}`, | ||||
|         `${import.meta.env.VITE_BASE_URL}/payment/capture/service/${service.value!.id}`, | ||||
|         { | ||||
|           service_cost: chargeAmount.value | ||||
|           card_id: selectedCard.value!.id, | ||||
|           status: 3 | ||||
|         }, | ||||
|         { withCredentials: true, headers: authHeader() } | ||||
|       ); | ||||
|  | ||||
|       notify({ title: "Success", text: "Service charged successfully!", type: "success" }); | ||||
|       notify({ title: "Success", text: "Payment captured successfully!", type: "success" }); | ||||
|       router.push({ name: 'customerProfile', params: { id: service.value!.customer_id } }); | ||||
|     } else if (captureResponse.data?.status === 1) { | ||||
|       const reason = captureResponse.data.rejection_reason || "The payment was declined by the gateway."; | ||||
|       notify({ title: "Payment Declined", text: reason, type: "warn" }); | ||||
|     } else { | ||||
|       const reason = response.data?.rejection_reason || "The charge was declined."; | ||||
|       notify({ title: "Charge Declined", text: reason, type: "error" }); | ||||
|       throw new Error("Invalid response from server during capture."); | ||||
|     } | ||||
|  | ||||
|   } catch (error: any) { | ||||
|     const detail = error.response?.data?.detail || "Failed to charge service due to a server error."; | ||||
|     const detail = error.response?.data?.detail || "Failed to process payment due to a server error."; | ||||
|     notify({ title: "Error", text: detail, type: "error" }); | ||||
|     console.error("Charge Service Error:", error); | ||||
|     console.error("Charge/Capture Service Error:", error); | ||||
|   } finally { | ||||
|     isSubmitting.value = false; | ||||
|   } | ||||
| @@ -290,21 +459,25 @@ const cancelCharge = () => { | ||||
|   } | ||||
| }; | ||||
|  | ||||
| const getTransaction = () => { | ||||
| const getTransaction = async () => { | ||||
|   const serviceId = route.params.id; | ||||
|   const path = `${import.meta.env.VITE_AUTHORIZE_URL}/api/transaction/service/${serviceId}`; | ||||
|   axios.get(path, { withCredentials: true, headers: authHeader() }) | ||||
|     .then((response: any) => { | ||||
|       transaction.value = response.data; | ||||
|       preAuthAmount.value = parseFloat(response.data.preauthorize_amount || service.value?.service_cost || 0); | ||||
|       if (response.data.status !== 0) { // Not approved | ||||
|         preAuthAmount.value = 0; | ||||
|       } | ||||
|     }) | ||||
|     .catch((error: any) => { | ||||
|       console.error("No pre-authorized transaction found for service:", error); | ||||
|       preAuthAmount.value = service.value?.service_cost || 0; // fallback to service cost | ||||
|     }); | ||||
|   try { | ||||
|     const response = await axios.get(path, { withCredentials: true, headers: authHeader() }); | ||||
|     transaction.value = response.data; | ||||
|     preAuthAmount.value = parseFloat(response.data.preauthorize_amount || service.value?.service_cost || 0); | ||||
|     if (response.data.status !== 0) { | ||||
|       preAuthAmount.value = 0; | ||||
|     } else if (response.data.card_id) { | ||||
|       const cardResponse = await axios.get(`${import.meta.env.VITE_BASE_URL}/payment/card/${response.data.card_id}`, { withCredentials: true, headers: authHeader() }); | ||||
|       preAuthCard.value = cardResponse.data; | ||||
|       selectedCard.value = cardResponse.data; | ||||
|       chargeAmount.value = preAuthAmount.value; | ||||
|     } | ||||
|   } catch (error: any) { | ||||
|     console.error("No pre-authorized transaction found for service:", error); | ||||
|     preAuthAmount.value = service.value?.service_cost || 0; | ||||
|   } | ||||
| }; | ||||
|  | ||||
| // --- Lifecycle Hook --- | ||||
| @@ -325,13 +498,38 @@ onMounted(async () => { | ||||
|       service.value = serviceResponse.data.service; | ||||
|       chargeAmount.value = service.value?.service_cost || 0; | ||||
|  | ||||
|       // Fetch Customer Cards | ||||
|       const cardsPath = `${import.meta.env.VITE_BASE_URL}/payment/cards/${service.value!.customer_id}`; | ||||
|       const cardsResponse = await axios.get(cardsPath, { withCredentials: true, headers: authHeader() }); | ||||
|       userCards.value = cardsResponse.data; | ||||
|       // Fetch transaction history for this service | ||||
|       try { | ||||
|         const transactionPath = `${import.meta.env.VITE_BASE_URL}/payment/transactions/service/${serviceId}`; | ||||
|         const transactionResponse = await axios.get(transactionPath, { withCredentials: true, headers: authHeader() }); | ||||
|         serviceTransactions.value = Array.isArray(transactionResponse.data) ? transactionResponse.data : []; | ||||
|       } catch (error) { | ||||
|         console.error("Failed to fetch transaction history:", error); | ||||
|         serviceTransactions.value = []; | ||||
|       } | ||||
|  | ||||
|       // Fetch pre-auth transaction | ||||
|       getTransaction(); | ||||
|       await getTransaction(); | ||||
|  | ||||
|       // If no card from transaction but service has payment_card_id, fetch directly | ||||
|       if (!preAuthCard.value && service.value?.payment_card_id) { | ||||
|         try { | ||||
|           const cardResponse = await axios.get(`${import.meta.env.VITE_BASE_URL}/payment/card/${service.value.payment_card_id}`, { withCredentials: true, headers: authHeader() }); | ||||
|           preAuthCard.value = cardResponse.data; | ||||
|           selectedCard.value = cardResponse.data; | ||||
|           // Keep the chargeAmount as service_cost, since there's no preauth amount | ||||
|         } catch (error) { | ||||
|           console.error("Failed to fetch card from service payment_card_id:", error); | ||||
|         } | ||||
|  | ||||
|         // Check if we have transactions with preauth amounts to set the global preAuthAmount | ||||
|         if (serviceTransactions.value.length > 0 && preAuthAmount.value === (service.value?.service_cost || 0)) { | ||||
|           const firstTransaction = serviceTransactions.value.find(t => t.preauthorize_amount && t.preauthorize_amount > 0); | ||||
|           if (firstTransaction) { | ||||
|             preAuthAmount.value = firstTransaction.preauthorize_amount; | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|     } else { | ||||
|       throw new Error(serviceResponse.data?.error || "Failed to fetch service data."); | ||||
|     } | ||||
|   | ||||
							
								
								
									
										807
									
								
								src/pages/pay/service/pay_service.vue
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										807
									
								
								src/pages/pay/service/pay_service.vue
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,807 @@ | ||||
| <!-- src/pages/pay/pay_oil.vue --> | ||||
| <template> | ||||
|   <div class="flex"> | ||||
|   | ||||
|     <div class="w-full px-4 md:px-10 py-4"> | ||||
|       <!-- Breadcrumbs & Title --> | ||||
|       <div class="text-sm breadcrumbs"> | ||||
|         <ul> | ||||
|           <li><router-link :to="{ name: 'home' }">Home</router-link></li> | ||||
|           <!-- Add a link to the customer's profile if the data is available --> | ||||
|           <li v-if="customer && customer.id"> | ||||
|             <router-link :to="{ name: 'customerProfile', params: { id: customer.id } }"> | ||||
|               {{ customer.customer_first_name }} {{ customer.customer_last_name }} | ||||
|             </router-link> | ||||
|           </li> | ||||
|           <li>Confirm Service Payment</li> | ||||
|         </ul> | ||||
|       </div> | ||||
|       <h1 class="text-3xl font-bold mt-4 border-b border-gray-600 pb-2"> | ||||
|         Confirm Service Payment #{{ service.id }} | ||||
|       </h1> | ||||
|  | ||||
|       <!-- Main Content Grid --> | ||||
|       <div class="grid grid-cols-1 lg:grid-cols-2 gap-6 my-6"> | ||||
|          | ||||
|         <!-- LEFT COLUMN: Customer and Delivery Details --> | ||||
|         <div class="space-y-6"> | ||||
|            | ||||
|           <!-- Customer Info Card --> | ||||
|           <div class="bg-neutral rounded-lg p-5"> | ||||
|             <div class="flex justify-between items-center mb-4"> | ||||
|               <div> | ||||
|                 <div class="text-xl font-bold">{{ customer.customer_first_name }} {{ customer.customer_last_name }}</div> | ||||
|                 <div class="text-sm text-gray-400">Account: {{ customer.account_number }}</div> | ||||
|               </div> | ||||
|               <router-link v-if="customer && customer.id" :to="{ name: 'customerProfile', params: { id: customer.id } }" class="btn btn-secondary btn-sm"> | ||||
|                 View Profile | ||||
|               </router-link> | ||||
|             </div> | ||||
|             <div class="space-y-1"> | ||||
|               <div>{{ customer.customer_address }}</div> | ||||
|               <div v-if="customer.customer_apt && customer.customer_apt !== 'None'">Apt: {{ customer.customer_apt }}</div> | ||||
|               <div>{{ customer.customer_town }}, {{ customer.customer_state === 0 ? 'MA' : 'RI' }} {{ customer.customer_zip }}</div> | ||||
|               <div class="mt-2">{{ customer.customer_phone_number }}</div> | ||||
|             </div> | ||||
|           </div> | ||||
|  | ||||
|           <!-- Service Details Card --> | ||||
|           <div class="bg-neutral rounded-lg p-5"> | ||||
|             <h3 class="text-xl font-bold mb-4">Service Details</h3> | ||||
|             <div class="space-y-3"> | ||||
|               <div> | ||||
|                 <div class="font-bold text-sm">Service Type</div> | ||||
|                 <div class="badge" :class="getServiceTypeColor(service.type_service_call)"> | ||||
|                   {{ getServiceTypeName(service.type_service_call) }} | ||||
|                 </div> | ||||
|               </div> | ||||
|               <div> | ||||
|                 <div class="font-bold text-sm">Scheduled Date</div> | ||||
|                 <div>{{ service.scheduled_date ? formatScheduledDate(service.scheduled_date) : 'Not scheduled' }}</div> | ||||
|               </div> | ||||
|               <div> | ||||
|                 <div class="font-bold text-sm">Description</div> | ||||
|                 <div class="text-sm">{{ service.description || 'No description provided' }}</div> | ||||
|               </div> | ||||
|               <div> | ||||
|                 <div class="font-bold text-sm">Total Cost</div> | ||||
|                 <div>${{ service.service_cost || '0.00' }}</div> | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|  | ||||
|         <!-- RIGHT COLUMN: Payment and Pricing Details --> | ||||
|         <div class="space-y-6"> | ||||
|           <!-- Authorize.net Account Status Box --> | ||||
|           <div v-if="customer.id" class="bg-base-100 rounded-lg p-4 border"> | ||||
|             <div class="flex flex-col xl:flex-row xl:items-center xl:justify-between gap-3"> | ||||
|               <div class="flex items-center gap-3 min-w-0 flex-1"> | ||||
|                 <svg class="w-5 h-5 text-blue-600 flex-shrink-0" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|                   <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M3 10h18M7 15h1m4 0h1m-7 4h12a3 3 0 003-3V8a3 3 0 00-3-3H6a3 3 0 00-3 3v4a3 3 0 003 3z"/> | ||||
|                 </svg> | ||||
|                 <span v-if="isLoadingAuthorize" class="text-sm font-medium"> | ||||
|                   <span class="loading loading-dots loading-xs mr-2"></span> | ||||
|                   Loading... | ||||
|                 </span> | ||||
|                 <span v-else-if="authorizeCheck.valid_for_charging" class="text-sm font-medium"> | ||||
|                   Authorize Account ID: {{ customer.auth_net_profile_id }} | ||||
|                 </span> | ||||
|                 <span v-else class="text-sm font-medium text-red-600"> | ||||
|                   {{ getAccountStatusMessage() }} | ||||
|                 </span> | ||||
|               </div> | ||||
|               <div class="flex gap-2 flex-shrink-0" v-if="!isLoadingAuthorize"> | ||||
|                 <!-- CREATE ACCOUNT SECTION - Only show when account doesn't exist --> | ||||
|                 <div v-if="!authorizeCheck.valid_for_charging" class="flex gap-2"> | ||||
|                   <button | ||||
|                     v-if="credit_cards_count === 0" | ||||
|                     @click="addCreditCard" | ||||
|                     class="btn btn-primary btn-sm" | ||||
|                   > | ||||
|                     Add Card | ||||
|                   </button> | ||||
|                   <button | ||||
|                     @click="createAuthorizeAccount" | ||||
|                     :class="['btn btn-sm', credit_cards_count === 0 ? 'btn-disabled' : 'btn-primary']" | ||||
|                     :disabled="credit_cards_count === 0" | ||||
|                     v-if="credit_cards_count > 0" | ||||
|                   > | ||||
|                     Create Account | ||||
|                   </button> | ||||
|                   <button | ||||
|                     v-else | ||||
|                     @click="addCreditCard" | ||||
|                     class="btn btn-secondary btn-sm" | ||||
|                   > | ||||
|                     Add Card First | ||||
|                   </button> | ||||
|                 </div> | ||||
|  | ||||
|                 <!-- DELETE ACCOUNT SECTION - Only show when account exists --> | ||||
|                 <div v-if="authorizeCheck.valid_for_charging" class="flex gap-2"> | ||||
|                   <button | ||||
|                     @click="showDeleteAccountModal" | ||||
|                     class="btn btn-error btn-sm" | ||||
|                   > | ||||
|                     Delete Account | ||||
|                   </button> | ||||
|                 </div> | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|  | ||||
|           <!-- Service Payment Card --> | ||||
|           <div class="bg-neutral rounded-lg p-5"> | ||||
|             <h3 class="text-xl font-bold mb-4">Service Payment</h3> | ||||
|             <div class="space-y-4"> | ||||
|               <!-- Payment Method Selection --> | ||||
|               <div> | ||||
|                 <div class="font-bold text-sm mb-2">Select Payment Method</div> | ||||
|                 <div class="space-y-2"> | ||||
|                   <!-- Show the selected card if payment is by credit --> | ||||
|                   <div v-for="card in credit_cards" :key="card.id"> | ||||
|                     <div v-if="card.id === service.payment_card_id" class="bg-base-100 p-3 rounded-md text-sm"> | ||||
|                       <div class="font-mono font-semibold">{{ card.type_of_card }} ending in {{ card.last_four_digits }}</div> | ||||
|                       <div>{{ card.name_on_card }}</div> | ||||
|                       <div>Expires: {{ card.expiration_month }}/{{ card.expiration_year }}</div> | ||||
|                     </div> | ||||
|                   </div> | ||||
|                 </div> | ||||
|               </div> | ||||
|  | ||||
|               <!-- Service Cost --> | ||||
|               <div class="pt-2"> | ||||
|                 <div class="divide-y divide-gray-300"> | ||||
|                   <div class="flex justify-between items-center py-3"> | ||||
|                     <span class="text-lg font-bold">Service Total</span> | ||||
|                     <span class="text-2xl font-bold text-accent"> | ||||
|                       ${{ service.service_cost || '0.00' }} | ||||
|                     </span> | ||||
|                   </div> | ||||
|                 </div> | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|            | ||||
|           <!-- Actions Card --> | ||||
|           <div class="bg-neutral rounded-lg p-5"> | ||||
|              <div class="flex flex-wrap gap-4 justify-between items-center"> | ||||
|                 <!-- Pay Authorize Button --> | ||||
|                 <button class="btn btn-success" :class="{ 'btn-disabled': !authorizeCheck.valid_for_charging }" :disabled="!authorizeCheck.valid_for_charging" @click="$router.push({ name: 'authorizeServicePreauthCharge', params: { id: $route.params.id } })"> | ||||
|                   Pay Authorize.net | ||||
|                 </button> | ||||
|                 <!-- A single confirm button is cleaner --> | ||||
|                 <button class="btn btn-warning" @click="processServicePayment(1)"> | ||||
|                   Pay Tiger | ||||
|                 </button> | ||||
|                 <!-- Edit Service button removed due to no route defined --> | ||||
|             </div> | ||||
|           </div> | ||||
|  | ||||
|         </div> | ||||
|       </div> | ||||
|     </div> | ||||
|   </div> | ||||
|  | ||||
|   <!-- Delete Account Confirmation Modal --> | ||||
|   <div class="modal" :class="{ 'modal-open': isDeleteAccountModalVisible }"> | ||||
|     <div class="modal-box"> | ||||
|       <h3 class="font-bold text-lg">Confirm Account Deletion</h3> | ||||
|       <p class="py-4">This will permanently delete the Authorize.net account and remove all payment profiles. This action cannot be undone.</p> | ||||
|       <div class="modal-action"> | ||||
|         <button @click="deleteAccount" class="btn btn-error">Delete Account</button> | ||||
|         <button @click="isDeleteAccountModalVisible = false" class="btn">Cancel</button> | ||||
|       </div> | ||||
|     </div> | ||||
|   </div> | ||||
|  | ||||
|   <!-- Create Account Progress Modal --> | ||||
|   <div class="modal" :class="{ 'modal-open': isCreateAccountModalVisible }"> | ||||
|     <div class="modal-box"> | ||||
|       <h3 class="font-bold text-lg">Creating Authorize.net Account</h3> | ||||
|       <div class="py-4 flex flex-col items-center"> | ||||
|         <div v-if="isCreatingAccount" class="text-center"> | ||||
|           <span class="text-lg mb-3">Setting up your payment account...</span> | ||||
|           <div class="loading loading-spinner loading-lg text-primary mb-3"></div> | ||||
|           <p class="text-sm text-gray-600">Please wait while we create your Authorize.net customer profile.</p> | ||||
|         </div> | ||||
|         <div v-else class="text-center"> | ||||
|           <div class="text-success mb-3"> | ||||
|             <svg class="w-12 h-12 mx-auto mb-2" fill="currentColor" viewBox="0 0 20 20"> | ||||
|               <path fill-rule="evenodd" d="M10 18a8 8 0 100-16 8 8 0 000 16zm3.707-9.293a1 1 0 00-1.414-1.414L9 10.586 7.707 9.293a1 1 0 00-1.414 1.414l2 2a1 1 0 001.414 0l4-4z" clip-rule="evenodd" /> | ||||
|             </svg> | ||||
|           </div> | ||||
|           <p class="text-lg font-semibold mb-2">Account Created Successfully!</p> | ||||
|           <div class="bg-base-200 p-3 rounded-lg mb-4"> | ||||
|             <p class="text-sm mb-1">Authorize.net Profile ID:</p> | ||||
|             <p class="font-mono font-bold text-success">{{ createdProfileId }}</p> | ||||
|           </div> | ||||
|           <p class="text-sm text-gray-600">Your payment account is now ready for transactions.</p> | ||||
|         </div> | ||||
|       </div> | ||||
|     </div> | ||||
|   </div> | ||||
|  | ||||
|   <!-- Duplicate Account Error Modal --> | ||||
|   <div class="modal" :class="{ 'modal-open': isDuplicateErrorModalVisible }"> | ||||
|     <div class="modal-box"> | ||||
|       <h3 class="font-bold text-lg text-error">⚠️ Duplicate Account Detected</h3> | ||||
|       <div class="py-4 space-y-4"> | ||||
|         <div class="text-center"> | ||||
|           <svg class="w-16 h-16 mx-auto mb-4 text-warning" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|             <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" | ||||
|                   d="M12 9v2m0 4h.01m-6.938 4h13.856c1.54 0 2.502-1.667 1.732-2.5L13.732 4c-.77-.833-1.964-.833-2.732 0L3.732 16.5c-.77.833.192 2.5 1.732 2.5z"/> | ||||
|           </svg> | ||||
|           <p class="text-lg font-semibold">Duplicate Account in Authorize.net</p> | ||||
|           <p class="text-sm text-gray-600 mt-2"> | ||||
|             A duplicate account was found in your Authorize.net merchant account. | ||||
|           </p> | ||||
|           <p class="text-sm text-gray-600 mt-2"> | ||||
|             Customer ID: <strong>{{ customer.id }}</strong> | ||||
|           </p> | ||||
|         </div> | ||||
|  | ||||
|         <div class="bg-base-200 p-4 rounded-lg"> | ||||
|           <h4 class="font-semibold mb-2 text-warning">Action Required:</h4> | ||||
|           <ul class="list-disc list-inside text-sm space-y-1"> | ||||
|             <li>Manually check your Authorize.net merchant dashboard</li> | ||||
|             <li>Review existing customer profiles</li> | ||||
|             <li>Contact support for linkage if needed</li> | ||||
|           </ul> | ||||
|           <p class="text-xs text-gray-500 mt-2"> | ||||
|             Inconsistency between your system and Authorize.net detected. | ||||
|           </p> | ||||
|         </div> | ||||
|  | ||||
|         <div class="text-center pt-2"> | ||||
|           <p class="text-xs text-gray-500"> | ||||
|             This profile may have been created previously and needs manual linking. | ||||
|           </p> | ||||
|         </div> | ||||
|       </div> | ||||
|       <div class="modal-action"> | ||||
|         <button class="btn btn-primary" @click="hideDuplicateErrorModal()"> | ||||
|           Acknowledge | ||||
|         </button> | ||||
|       </div> | ||||
|     </div> | ||||
|   </div> | ||||
|  | ||||
|   <Footer /> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import { defineComponent } from 'vue' | ||||
| import axios from 'axios' | ||||
| import authHeader from '../../../services/auth.header' | ||||
| import Header from '../../../layouts/headers/headerauth.vue' | ||||
| import SideBar from '../../../layouts/sidebar/sidebar.vue' | ||||
| import Footer from '../../../layouts/footers/footer.vue' | ||||
|  | ||||
| import useValidate from "@vuelidate/core"; | ||||
| import { notify } from "@kyvg/vue3-notification" | ||||
| import {  required } from "@vuelidate/validators"; | ||||
|  | ||||
| export default defineComponent({ | ||||
|   name: 'PayService', | ||||
|  | ||||
|   components: { | ||||
|     Header, | ||||
|     SideBar, | ||||
|     Footer, | ||||
|   }, | ||||
|  | ||||
|   data() { | ||||
|     return { | ||||
|       v$: useValidate(), | ||||
|       loaded: false, | ||||
|       user: { | ||||
|         user_id: 0, | ||||
|       }, | ||||
|       service: { | ||||
|         id: 0, | ||||
|         scheduled_date: '', | ||||
|         customer_id: 0, | ||||
|         customer_name: '', | ||||
|         customer_address: '', | ||||
|         customer_town: '', | ||||
|         type_service_call: 0, | ||||
|         description: '', | ||||
|         service_cost: '', | ||||
|         payment_card_id: 0, | ||||
|       }, | ||||
|       serviceParts: null as any, | ||||
|       credit_cards: [ | ||||
|         { | ||||
|           id: 0, | ||||
|           name_on_card: '', | ||||
|           main_card: false, | ||||
|           card_number: '', | ||||
|           expiration_month: '', | ||||
|           type_of_card: '', | ||||
|           last_four_digits: '', | ||||
|           expiration_year: '', | ||||
|           security_number: '', | ||||
|  | ||||
|         } | ||||
|       ], | ||||
|  | ||||
|       stripe: null, | ||||
|       customer: { | ||||
|         id: 0, | ||||
|         user_id: 0, | ||||
|         customer_first_name: '', | ||||
|         customer_last_name: '', | ||||
|         customer_town: '', | ||||
|         customer_address: '', | ||||
|         customer_state: 0, | ||||
|         customer_zip: '', | ||||
|         customer_apt: '', | ||||
|         customer_home_type: 0, | ||||
|         customer_phone_number: '', | ||||
|         account_number: '', | ||||
|         auth_net_profile_id: null, | ||||
|       }, | ||||
|       pricing: { | ||||
|         price_from_supplier: 0, | ||||
|         price_for_customer: 0, | ||||
|         price_for_employee: 0, | ||||
|         price_same_day: 0, | ||||
|         price_prime: 0, | ||||
|         price_emergency: 0, | ||||
|         date: "", | ||||
|       }, | ||||
|       promo_active: false, | ||||
|       promo: { | ||||
|         name_of_promotion: '', | ||||
|         description: '', | ||||
|         money_off_delivery: 0, | ||||
|         text_on_ticket: '' | ||||
|       }, | ||||
|       priceprime: 0, | ||||
|       pricesameday: 0, | ||||
|       priceemergency: 0, | ||||
|       total_amount: 0, | ||||
|       discount: 0, | ||||
|       total_amount_after_discount: 0, | ||||
|       credit_cards_count: 0, | ||||
|       isLoadingAuthorize: true, | ||||
|       authorizeCheck: { profile_exists: false, has_payment_methods: false, missing_components: [] as string[], valid_for_charging: false }, | ||||
|       isDeleteAccountModalVisible: false, | ||||
|       isCreateAccountModalVisible: false, | ||||
|       isCreatingAccount: false, | ||||
|       createdProfileId: '', | ||||
|       isDuplicateErrorModalVisible: false, | ||||
|     } | ||||
|   }, | ||||
|   validations() { | ||||
|     return { | ||||
|       CreateServiceOrderForm: { | ||||
|         basicInfo: { | ||||
|           description: { required }, | ||||
|           service_cost: { required }, | ||||
|         }, | ||||
|       }, | ||||
|     }; | ||||
|   }, | ||||
|   created() { | ||||
|     this.userStatus() | ||||
|  | ||||
|   }, | ||||
|   watch: { | ||||
|     $route() { | ||||
|  | ||||
|     }, | ||||
|   }, | ||||
|   mounted() { | ||||
|     this.getServiceOrder(this.$route.params.id) | ||||
|     this.getServicePartsForCustomer(); | ||||
|  | ||||
|   }, | ||||
|  | ||||
|   methods: { | ||||
|     userStatus() { | ||||
|       let path = import.meta.env.VITE_BASE_URL + '/auth/whoami'; | ||||
|       axios({ | ||||
|         method: 'get', | ||||
|         url: path, | ||||
|         withCredentials: true, | ||||
|         headers: authHeader(), | ||||
|       }) | ||||
|         .then((response: any) => { | ||||
|           if (response.data.ok) { | ||||
|             this.user = response.data.user; | ||||
|           } | ||||
|         }) | ||||
|  | ||||
|     }, | ||||
|     getServiceOrder(service_id: any) { | ||||
|       let path = import.meta.env.VITE_BASE_URL + "/service/" + service_id; | ||||
|       axios({ | ||||
|         method: "get", | ||||
|         url: path, | ||||
|         withCredentials: true, | ||||
|         headers: authHeader(), | ||||
|       }) | ||||
|         .then((response: any) => { | ||||
|           let serviceData; | ||||
|           if (response.data) { | ||||
|             // Handle different API response structures | ||||
|             if (response.data.service) { | ||||
|               // API returns {ok: true, service: {...}} structure | ||||
|               serviceData = response.data.service; | ||||
|             } else if (Array.isArray(response.data)) { | ||||
|               serviceData = response.data[0]; // Array response | ||||
|             } else { | ||||
|               serviceData = response.data; // Direct object response | ||||
|             } | ||||
|  | ||||
|             if (serviceData && serviceData.id) { | ||||
|               this.service = { | ||||
|                 id: serviceData.id, | ||||
|                 scheduled_date: serviceData.scheduled_date, | ||||
|                 customer_id: serviceData.customer_id, | ||||
|                 customer_name: serviceData.customer_name, | ||||
|                 customer_address: serviceData.customer_address, | ||||
|                 customer_town: serviceData.customer_town, | ||||
|                 type_service_call: serviceData.type_service_call, | ||||
|                 description: serviceData.description, | ||||
|                 service_cost: serviceData.service_cost, | ||||
|                 payment_card_id: serviceData.payment_card_id || 0, | ||||
|               }; | ||||
|  | ||||
|               // Fetch related data | ||||
|               this.getCustomer(this.service.customer_id); | ||||
|               this.getCreditCards(this.service.customer_id); | ||||
|               this.getCreditCardsCount(this.service.customer_id); | ||||
|               this.getServicePartsForCustomer(); | ||||
|             } else { | ||||
|               console.error("API Error: Invalid service data received:", serviceData); | ||||
|               notify({ | ||||
|                 title: "Error", | ||||
|                 text: "Invalid service data received", | ||||
|                 type: "error", | ||||
|               }); | ||||
|             } | ||||
|           } else { | ||||
|             console.error("API Error: No response data received"); | ||||
|             notify({ | ||||
|               title: "Error", | ||||
|               text: "Could not get service data", | ||||
|               type: "error", | ||||
|             }); | ||||
|           } | ||||
|         }) | ||||
|         .catch((error: any) => { | ||||
|           console.error("API Error in getServiceOrder:", error); | ||||
|           console.error("Error details:", error.response?.data || error.message); | ||||
|           notify({ | ||||
|             title: "Error", | ||||
|             text: "Could not get service data", | ||||
|             type: "error", | ||||
|           }); | ||||
|         }); | ||||
|     }, | ||||
|     getServicePartsForCustomer() { | ||||
|       if (!this.service.customer_id) return; | ||||
|  | ||||
|       let path = `${import.meta.env.VITE_BASE_URL}/service/parts/customer/${this.service.customer_id}`; | ||||
|       axios.get(path, { headers: authHeader() }) | ||||
|         .then((response: any) => { | ||||
|           this.serviceParts = response.data; | ||||
|         }) | ||||
|         .catch((error: any) => { | ||||
|           console.error("Failed to fetch service parts:", error); | ||||
|           this.serviceParts = null; | ||||
|         }); | ||||
|     }, | ||||
|     getCreditCards(user_id: any) { | ||||
|       let path = import.meta.env.VITE_BASE_URL + '/payment/cards/' + user_id; | ||||
|       axios({ | ||||
|         method: 'get', | ||||
|         url: path, | ||||
|         headers: authHeader(), | ||||
|       }).then((response: any) => { | ||||
|  | ||||
|         this.credit_cards = response.data | ||||
|       }) | ||||
|     }, | ||||
|     getCreditCardsCount(user_id: any) { | ||||
|       let path = import.meta.env.VITE_BASE_URL + '/payment/cards/onfile/' + user_id; | ||||
|       axios({ | ||||
|         method: 'get', | ||||
|         url: path, | ||||
|         headers: authHeader(), | ||||
|       }).then((response: any) => { | ||||
|         this.credit_cards_count = response.data.cards | ||||
|       }) | ||||
|     }, | ||||
|     getCustomer(userid: any) { | ||||
|       let path = import.meta.env.VITE_BASE_URL + '/customer/' + userid; | ||||
|       axios({ | ||||
|         method: 'get', | ||||
|         url: path, | ||||
|         headers: authHeader(), | ||||
|       }).then((response: any) => { | ||||
|         this.customer = response.data | ||||
|         this.checkAuthorizeAccount(); | ||||
|       }) | ||||
|     }, | ||||
|     processServicePayment(payment_type: number) { | ||||
|       let path = import.meta.env.VITE_BASE_URL + "/payment/service/payment/" + this.service.id + '/' + payment_type; | ||||
|       axios({ | ||||
|         method: "PUT", | ||||
|         url: path, | ||||
|       }) | ||||
|         .then((response: any) => { | ||||
|           if (response.data.ok) { | ||||
|             if (payment_type == 0) { | ||||
|               notify({ | ||||
|                 title: "Success", | ||||
|                 text: "Service marked as cash payment", | ||||
|                 type: "success", | ||||
|               }); | ||||
|             } | ||||
|             if (payment_type == 1) { | ||||
|               notify({ | ||||
|                 title: "Success", | ||||
|                 text: "Service marked as credit card payment", | ||||
|                 type: "success", | ||||
|               }); | ||||
|             } | ||||
|             if (payment_type == 3) { | ||||
|               notify({ | ||||
|                 title: "Success", | ||||
|                 text: "Service marked as check payment", | ||||
|                 type: "success", | ||||
|               }); | ||||
|             } | ||||
|             if (payment_type == 11) { | ||||
|               notify({ | ||||
|                 title: "Success", | ||||
|                 text: "Service payment processed via Authorize.net", | ||||
|                 type: "success", | ||||
|               }); | ||||
|             } | ||||
|             this.$router.push({ name: "ServiceHome" }); | ||||
|           } | ||||
|         }) | ||||
|         .catch(() => { | ||||
|           notify({ | ||||
|             title: "Error", | ||||
|             text: "Could not process service payment", | ||||
|             type: "error", | ||||
|           }); | ||||
|         }); | ||||
|     }, | ||||
|     async checkAuthorizeAccount() { | ||||
|       if (!this.customer.id) return; | ||||
|  | ||||
|       this.isLoadingAuthorize = true; | ||||
|  | ||||
|       try { | ||||
|         const path = `${import.meta.env.VITE_AUTHORIZE_URL}/user/check-authorize-account/${this.customer.id}`; | ||||
|         const response = await axios.get(path, { headers: authHeader() }); | ||||
|         this.authorizeCheck = response.data; | ||||
|  | ||||
|         // Check if the API returned an error in the response body | ||||
|         if (this.authorizeCheck.missing_components && this.authorizeCheck.missing_components.includes('api_error')) { | ||||
|           console.log("API error detected in response, calling cleanup for customer:", this.customer.id); | ||||
|           this.cleanupAuthorizeData(); | ||||
|           return; // Don't set loading to false yet, let cleanup handle it | ||||
|         } | ||||
|       } catch (error) { | ||||
|         console.error("Failed to check authorize account:", error); | ||||
|         notify({ title: "Error", text: "Could not check payment account status.", type: "error" }); | ||||
|         // Set default error state | ||||
|         this.authorizeCheck = { | ||||
|           profile_exists: false, | ||||
|           has_payment_methods: false, | ||||
|           missing_components: ['api_error'], | ||||
|           valid_for_charging: false | ||||
|         }; | ||||
|         // Automatically cleanup the local Authorize.Net data on API error | ||||
|         console.log("Calling cleanupAuthorizedData for customer:", this.customer.id); | ||||
|         this.cleanupAuthorizeData(); | ||||
|       } finally { | ||||
|         this.isLoadingAuthorize = false; | ||||
|       } | ||||
|     }, | ||||
|     async createAuthorizeAccount() { | ||||
|       // Show the creating account modal | ||||
|       this.isCreatingAccount = true; | ||||
|       this.isCreateAccountModalVisible = true; | ||||
|  | ||||
|       try { | ||||
|         const path = `${import.meta.env.VITE_AUTHORIZE_URL}/user/create-account/${this.customer.id}`; | ||||
|         const response = await axios.post(path, {}, { headers: authHeader() }); | ||||
|  | ||||
|         if (response.data.success) { | ||||
|           // Update local state | ||||
|           this.customer.auth_net_profile_id = response.data.profile_id; | ||||
|           this.authorizeCheck.valid_for_charging = true; | ||||
|           this.authorizeCheck.profile_exists = true; | ||||
|           this.authorizeCheck.has_payment_methods = true; | ||||
|           this.authorizeCheck.missing_components = []; | ||||
|           this.createdProfileId = response.data.profile_id; | ||||
|  | ||||
|           // Refresh credit cards to get updated payment profile IDs | ||||
|           await this.getCreditCards(this.customer.id); | ||||
|  | ||||
|           // Switch modal to success view and close after delay | ||||
|           setTimeout(() => { | ||||
|             this.isCreatingAccount = false; | ||||
|             setTimeout(() => { | ||||
|               this.isCreateAccountModalVisible = false; | ||||
|               this.createdProfileId = ''; | ||||
|  | ||||
|               notify({ | ||||
|                 title: "Success", | ||||
|                 text: "Authorize.net account created successfully!", | ||||
|                 type: "success" | ||||
|               }); | ||||
|             }, 3000); // Show success message for 3 seconds | ||||
|           }, 1000); // Brief delay to show success animation | ||||
|  | ||||
|         } else { | ||||
|           // Hide modal on error | ||||
|           this.isCreateAccountModalVisible = false; | ||||
|  | ||||
|           // Check for E00039 duplicate error | ||||
|           const errorMessage = response.data.message || response.data.error_detail || "Failed to create Authorize.net account"; | ||||
|  | ||||
|           if (response.data.is_duplicate || errorMessage.includes("E00039")) { | ||||
|             // Show duplicate account popup | ||||
|             setTimeout(() => { | ||||
|               this.showDuplicateErrorModal(); | ||||
|             }, 300); | ||||
|             return; | ||||
|           } else { | ||||
|             // Normal error notification | ||||
|             notify({ | ||||
|               title: "Error", | ||||
|               text: errorMessage, | ||||
|               type: "error" | ||||
|             }); | ||||
|           } | ||||
|         } | ||||
|       } catch (error: any) { | ||||
|         console.error("Failed to create account:", error); | ||||
|         this.isCreateAccountModalVisible = false; | ||||
|         this.isCreatingAccount = false; | ||||
|  | ||||
|         // Check for E00039 duplicate error | ||||
|         const errorMessage = error.response?.data?.error_detail || | ||||
|                            error.response?.data?.detail || | ||||
|                            error.response?.data?.message || | ||||
|                            error.message || "Failed to create Authorize.net account"; | ||||
|  | ||||
|         if (error.response?.data?.is_duplicate || errorMessage.includes("E00039")) { | ||||
|           // Show duplicate account popup | ||||
|           setTimeout(() => { | ||||
|             this.showDuplicateErrorModal(); | ||||
|           }, 300); | ||||
|           return; | ||||
|         } | ||||
|  | ||||
|         // Normal error notification | ||||
|         notify({ | ||||
|           title: "Error", | ||||
|           text: errorMessage, | ||||
|           type: "error" | ||||
|         }); | ||||
|       } | ||||
|     }, | ||||
|     showDeleteAccountModal() { | ||||
|       this.isDeleteAccountModalVisible = true; | ||||
|     }, | ||||
|     showDuplicateErrorModal() { | ||||
|       this.isDuplicateErrorModalVisible = true; | ||||
|     }, | ||||
|     hideDuplicateErrorModal() { | ||||
|       this.isDuplicateErrorModalVisible = false; | ||||
|     }, | ||||
|     addCreditCard() { | ||||
|       // Redirect to add card page | ||||
|       this.$router.push({ name: 'cardadd', params: { customerId: this.customer.id } }); | ||||
|     }, | ||||
|     async deleteAccount() { | ||||
|       this.isDeleteAccountModalVisible = false; | ||||
|  | ||||
|       try { | ||||
|         const path = `${import.meta.env.VITE_AUTHORIZE_URL}/user/delete-account/${this.customer.id}`; | ||||
|         const response = await axios.delete(path, { headers: authHeader() }); | ||||
|  | ||||
|         if (response.data.success) { | ||||
|           // Update local state | ||||
|           this.customer.auth_net_profile_id = null; | ||||
|           this.authorizeCheck.valid_for_charging = false; | ||||
|           this.authorizeCheck.profile_exists = false; | ||||
|           this.authorizeCheck.has_payment_methods = false; | ||||
|  | ||||
|           // Refresh credit cards list (IDs should now be null) | ||||
|           this.getCreditCards(this.customer.id); | ||||
|  | ||||
|           notify({ | ||||
|             title: "Success", | ||||
|             text: "Authorize.net account deleted successfully", | ||||
|             type: "success" | ||||
|           }); | ||||
|         } else { | ||||
|           notify({ | ||||
|             title: "Warning", | ||||
|             text: response.data.message || "Account deletion completed with warnings", | ||||
|             type: "warning" | ||||
|           }); | ||||
|         } | ||||
|       } catch (error: any) { | ||||
|         console.error("Failed to delete account:", error); | ||||
|         notify({ | ||||
|           title: "Error", | ||||
|           text: "Failed to delete Authorize.net account", | ||||
|           type: "error" | ||||
|         }); | ||||
|       } | ||||
|     }, | ||||
|     async cleanupAuthorizeData() { | ||||
|       try { | ||||
|         const path = `${import.meta.env.VITE_BASE_URL}/payment/authorize/cleanup/${this.customer.id}`; | ||||
|         const response = await axios.post(path, {}, { headers: authHeader() }); | ||||
|  | ||||
|         if (response.data.ok) { | ||||
|           // Update local state to reflect cleanup | ||||
|           this.customer.auth_net_profile_id = null; | ||||
|           this.authorizeCheck.valid_for_charging = false; | ||||
|           this.authorizeCheck.profile_exists = false; | ||||
|           this.authorizeCheck.has_payment_methods = false; | ||||
|  | ||||
|           // Refresh credit cards to reflect null payment profile IDs | ||||
|           this.getCreditCards(this.customer.id); | ||||
|  | ||||
|           console.log("Successfully cleaned up Authorize.Net data:", response.data.message); | ||||
|         } else { | ||||
|           console.error("Failed to cleanup Authorize.Net data:", response.data.error); | ||||
|         } | ||||
|       } catch (error) { | ||||
|         console.error("Error during cleanup:", error); | ||||
|       } | ||||
|     }, | ||||
|     getAccountStatusMessage(): string { | ||||
|       if (!this.authorizeCheck || !this.authorizeCheck.missing_components) { | ||||
|         return 'Account setup incomplete'; | ||||
|       } | ||||
|  | ||||
|       const missing = this.authorizeCheck.missing_components; | ||||
|       if (missing.includes('customer_not_found')) { | ||||
|         return 'Customer not found in Authorize.net'; | ||||
|       } else if (missing.includes('authorize_net_profile')) { | ||||
|         return 'No Authorize.net profile configured'; | ||||
|       } else if (missing.includes('authorize_net_profile_invalid')) { | ||||
|         return 'Authorize.net profile is invalid'; | ||||
|       } else if (missing.includes('payment_method')) { | ||||
|         return 'No payment methods configured'; | ||||
|       } else if (missing.includes('api_error')) { | ||||
|         return 'Error checking account status'; | ||||
|       } else { | ||||
|         return 'Account requires setup'; | ||||
|       } | ||||
|     }, | ||||
|     getServiceTypeName(typeId: number): string { | ||||
|       const typeMap: { [key: number]: string } = { 0: 'Tune-up', 1: 'No Heat', 2: 'Fix', 3: 'Tank Install', 4: 'Other' }; | ||||
|       return typeMap[typeId] || 'Unknown'; | ||||
|     }, | ||||
|     getServiceTypeColor(typeId: number): string { | ||||
|       const colorMap: { [key: number]: string } = { 0: 'primary', 1: 'error', 2: 'warning', 3: 'info', 4: 'neutral' }; | ||||
|       return `badge-${colorMap[typeId] || 'neutral'}`; | ||||
|     }, | ||||
|     formatScheduledDate(dateString: string): string { | ||||
|       if (!dateString) return 'Not scheduled'; | ||||
|       return dateString; // Could format with dayjs if needed | ||||
|     } | ||||
|  | ||||
|   }, | ||||
| }) | ||||
| </script> | ||||
|  | ||||
| <style scoped></style> | ||||
| @@ -51,13 +51,20 @@ | ||||
|               <tbody> | ||||
|                 <!-- Removed @click from tr to avoid conflicting actions --> | ||||
|                 <tr v-for="service in services" :key="service.id" class=" hover:bg-blue-600"> | ||||
|                   <td class="align-top">{{ service.id }}</td> | ||||
|                   <td class="align-top"> | ||||
|                   <td class="align-top text-white">{{ service.id }}</td> | ||||
|                   <td class="align-top text-white"> | ||||
|                     <div>{{ formatDate(service.scheduled_date) }}</div> | ||||
|                     <div class="text-xs opacity-70">{{ formatTime(service.scheduled_date) }}</div> | ||||
|                   </td> | ||||
|                   <td class="align-top">{{ service.customer_name }}</td> | ||||
|                   <td class="align-top">{{ service.customer_address }}, {{ service.customer_town }}</td> | ||||
|                   <td class="align-top"> | ||||
|                     <router-link | ||||
|                       :to="{ name: 'customerProfile', params: { id: service.customer_id } }" | ||||
|                       class="text-white hover:text-green-500 hover:underline" | ||||
|                     > | ||||
|                       {{ service.customer_name }} | ||||
|                     </router-link> | ||||
|                   </td> | ||||
|                   <td class="align-top text-white">{{ service.customer_address }}, {{ service.customer_town }}</td> | ||||
|                   <td class="align-top"> | ||||
|                    <span  | ||||
|     class="badge badge-sm text-white"  | ||||
| @@ -66,7 +73,7 @@ | ||||
|                       {{ getServiceTypeName(service.type_service_call) }} | ||||
|                     </span> | ||||
|                   </td> | ||||
|                   <td class="whitespace-normal text-sm align-top"> | ||||
|                   <td class="whitespace-normal text-sm align-top text-white"> | ||||
|                     <!-- TRUNCATION LOGIC FOR DESKTOP --> | ||||
|                     <div v-if="!isLongDescription(service.description) || isExpanded(service.id)"> | ||||
|                       {{ service.description }} | ||||
| @@ -77,10 +84,11 @@ | ||||
|                       <a @click.prevent="toggleExpand(service.id)" href="#" class="link link-info link-hover text-xs ml-1 whitespace-nowrap">Read more</a> | ||||
|                     </div> | ||||
|                   </td> | ||||
|                   <td class="text-right font-mono align-top">{{ formatCurrency(service.service_cost) }}</td> | ||||
|                   <td class="text-right font-mono align-top text-white">{{ formatCurrency(service.service_cost) }}</td> | ||||
|                   <td class="text-right align-top space-x-2"> | ||||
|                     <button @click="openEditModal(service)" class="btn btn-sm btn-primary">View</button> | ||||
|                     <router-link v-if="service.service_cost !== undefined && service.service_cost !== '' && Number(service.service_cost) === 0" :to="{ name: 'chargeServiceAuthorize', params: { id: service.id } }" class="btn btn-sm btn-success">Charge</router-link> | ||||
|                     <router-link v-if="shouldShowChargeButton(service)" :to="{ name: 'payService', params: { id: service.id } }" class="btn btn-sm btn-success">Charge</router-link> | ||||
|                     <router-link v-if="shouldShowCaptureButton(service)" :to="{ name: 'chargeServiceAuthorize', params: { id: service.id } }" class="btn btn-sm btn-warning">Capture</router-link> | ||||
|                   </td> | ||||
|                 </tr> | ||||
|               </tbody> | ||||
| @@ -93,7 +101,12 @@ | ||||
|               <div class="card-body p-4"> | ||||
|                 <div class="flex justify-between items-start"> | ||||
|                   <div> | ||||
|                     <h2 class="card-title text-base">{{ service.customer_name }}</h2> | ||||
|                     <router-link | ||||
|                       :to="{ name: 'customerProfile', params: { id: service.customer_id } }" | ||||
|                       class="card-title text-base text-white hover:text-green-500 hover:underline" | ||||
|                     > | ||||
|                       {{ service.customer_name }} | ||||
|                     </router-link> | ||||
|                     <p class="text-xs text-gray-500">ID: {{ service.id }}</p> | ||||
|                     <p class="text-xs text-gray-400">{{ service.customer_address }}, {{ service.customer_town }}</p> | ||||
|                   </div> | ||||
| @@ -122,7 +135,8 @@ | ||||
|                  | ||||
|                 <div class="card-actions justify-end mt-2 space-x-2"> | ||||
|                   <button @click="openEditModal(service)" class="btn btn-sm btn-primary">View</button> | ||||
|                   <router-link v-if="service.service_cost !== undefined && service.service_cost !== '' && Number(service.service_cost) === 0" :to="{ name: 'chargeServiceAuthorize', params: { id: service.id } }" class="btn btn-sm btn-success">Charge</router-link> | ||||
|                   <router-link v-if="shouldShowChargeButton(service)" :to="{ name: 'payService', params: { id: service.id } }" class="btn btn-sm btn-success">Charge</router-link> | ||||
|                   <router-link v-if="shouldShowCaptureButton(service)" :to="{ name: 'chargeServiceAuthorize', params: { id: service.id } }" class="btn btn-sm btn-warning">Capture</router-link> | ||||
|                 </div> | ||||
|               </div> | ||||
|             </div> | ||||
| @@ -160,6 +174,7 @@ interface ServiceCall { | ||||
|   type_service_call: number; | ||||
|   description: string; | ||||
|   service_cost: string; | ||||
|   payment_status?: number; | ||||
| } | ||||
|  | ||||
| export default defineComponent({ | ||||
| @@ -189,7 +204,7 @@ export default defineComponent({ | ||||
|           headers: authHeader(), | ||||
|           withCredentials: true, | ||||
|         }); | ||||
|         this.services = response.data; | ||||
|         this.services = response.data.sort((a: ServiceCall, b: ServiceCall) => b.id - a.id); | ||||
|       } catch (error) { | ||||
|         console.error("Failed to fetch upcoming service calls:", error); | ||||
|       } finally { | ||||
| @@ -313,6 +328,14 @@ export default defineComponent({ | ||||
|         4: 'black', | ||||
|       }; | ||||
|       return colorMap[typeId] || 'gray'; | ||||
|     }, | ||||
|  | ||||
|     shouldShowChargeButton(service: any): boolean { | ||||
|       return service.payment_status === null || service.payment_status === undefined; | ||||
|     }, | ||||
|  | ||||
|     shouldShowCaptureButton(service: any): boolean { | ||||
|       return service.payment_status === 1; | ||||
|     } | ||||
|   }, | ||||
| }) | ||||
|   | ||||
| @@ -50,13 +50,20 @@ | ||||
|               </thead> | ||||
|               <tbody> | ||||
|                 <tr v-for="service in services" :key="service.id" class="hover:bg-blue-600"> | ||||
|                   <td class="align-top">{{ service.id }}</td> | ||||
|                   <td class="align-top"> | ||||
|                   <td class="align-top text-white">{{ service.id }}</td> | ||||
|                   <td class="align-top text-white"> | ||||
|                     <div>{{ formatDate(service.scheduled_date) }}</div> | ||||
|                     <div class="text-xs opacity-70">{{ formatTime(service.scheduled_date) }}</div> | ||||
|                   </td> | ||||
|                   <td class="align-top">{{ service.customer_name }}</td> | ||||
|                   <td class="align-top">{{ service.customer_address }}, {{ service.customer_town }}</td> | ||||
|                   <td class="align-top"> | ||||
|                     <router-link | ||||
|                       :to="{ name: 'customerProfile', params: { id: service.customer_id } }" | ||||
|                       class="text-white hover:text-green-500 hover:underline" | ||||
|                     > | ||||
|                       {{ service.customer_name }} | ||||
|                     </router-link> | ||||
|                   </td> | ||||
|                   <td class="align-top text-white">{{ service.customer_address }}, {{ service.customer_town }}</td> | ||||
|                    | ||||
|                   <!--  | ||||
|                     FIX IS HERE: Replaced the colored text with a styled badge. | ||||
| @@ -73,7 +80,7 @@ | ||||
|                     </span> | ||||
|                   </td> | ||||
|  | ||||
|                   <td class="whitespace-normal text-sm align-top"> | ||||
|                   <td class="whitespace-normal text-sm align-top text-white"> | ||||
|                     <div v-if="!isLongDescription(service.description) || isExpanded(service.id)"> | ||||
|                       {{ service.description }} | ||||
|                       <a v-if="isLongDescription(service.description)" @click.prevent="toggleExpand(service.id)" href="#" class="link link-info link-hover text-xs ml-1 whitespace-nowrap">Show less</a> | ||||
| @@ -83,10 +90,11 @@ | ||||
|                       <a @click.prevent="toggleExpand(service.id)" href="#" class="link link-info link-hover text-xs ml-1 whitespace-nowrap">Read more</a> | ||||
|                     </div> | ||||
|                   </td> | ||||
|                   <td class="text-right font-mono align-top">{{ formatCurrency(service.service_cost) }}</td> | ||||
|                   <td class="text-right font-mono align-top text-white">{{ formatCurrency(service.service_cost) }}</td> | ||||
|                   <td class="text-right align-top space-x-2"> | ||||
|                     <button @click="openEditModal(service)" class="btn btn-sm btn-primary">View</button> | ||||
|                     <router-link v-if="service.service_cost !== undefined && service.service_cost !== '' && Number(service.service_cost) === 0" :to="{ name: 'chargeServiceAuthorize', params: { id: service.id } }" class="btn btn-sm btn-success">Charge</router-link> | ||||
|                     <router-link v-if="shouldShowChargeButton(service)" :to="{ name: 'payService', params: { id: service.id } }" class="btn btn-sm btn-success">Charge</router-link> | ||||
|                     <router-link v-if="shouldShowCaptureButton(service)" :to="{ name: 'chargeServiceAuthorize', params: { id: service.id } }" class="btn btn-sm btn-warning">Capture</router-link> | ||||
|                   </td> | ||||
|                 </tr> | ||||
|               </tbody> | ||||
| @@ -99,7 +107,12 @@ | ||||
|               <div class="card-body p-4"> | ||||
|                 <div class="flex justify-between items-start"> | ||||
|                   <div> | ||||
|                     <h2 class="card-title text-base">{{ service.customer_name }}</h2> | ||||
|                     <router-link | ||||
|                       :to="{ name: 'customerProfile', params: { id: service.customer_id } }" | ||||
|                       class="card-title text-base text-white hover:text-green-500 hover:underline" | ||||
|                     > | ||||
|                       {{ service.customer_name }} | ||||
|                     </router-link> | ||||
|                     <p class="text-xs text-gray-500">ID: {{ service.id }}</p> | ||||
|                     <p class="text-xs text-gray-400">{{ service.customer_address }}, {{ service.customer_town }}</p> | ||||
|                   </div> | ||||
| @@ -128,7 +141,8 @@ | ||||
|                  | ||||
|                 <div class="card-actions justify-end mt-2 space-x-2"> | ||||
|                   <button @click="openEditModal(service)" class="btn btn-sm btn-primary">View</button> | ||||
|                   <router-link v-if="service.service_cost !== undefined && service.service_cost !== '' && Number(service.service_cost) === 0" :to="{ name: 'chargeServiceAuthorize', params: { id: service.id } }" class="btn btn-sm btn-success">Charge</router-link> | ||||
|                   <router-link v-if="shouldShowChargeButton(service)" :to="{ name: 'payService', params: { id: service.id } }" class="btn btn-sm btn-success">Charge</router-link> | ||||
|                   <router-link v-if="shouldShowCaptureButton(service)" :to="{ name: 'chargeServiceAuthorize', params: { id: service.id } }" class="btn btn-sm btn-warning">Capture</router-link> | ||||
|                 </div> | ||||
|               </div> | ||||
|             </div> | ||||
| @@ -167,6 +181,7 @@ interface ServiceCall { | ||||
|   type_service_call: number; | ||||
|   description: string; | ||||
|   service_cost: string; | ||||
|   payment_status?: number; | ||||
| } | ||||
|  | ||||
| export default defineComponent({ | ||||
| @@ -219,7 +234,7 @@ export default defineComponent({ | ||||
|           headers: authHeader(), | ||||
|           withCredentials: true, | ||||
|         }); | ||||
|         this.services = response.data; | ||||
|         this.services = response.data.sort((a: ServiceCall, b: ServiceCall) => b.id - a.id); | ||||
|       } catch (error) { | ||||
|         console.error("Failed to fetch past service calls:", error); | ||||
|       } finally { | ||||
| @@ -326,6 +341,14 @@ export default defineComponent({ | ||||
|         4: 'black', | ||||
|       }; | ||||
|       return colorMap[typeId] || 'gray'; | ||||
|     }, | ||||
|  | ||||
|     shouldShowChargeButton(service: any): boolean { | ||||
|       return service.payment_status === null || service.payment_status === undefined; | ||||
|     }, | ||||
|  | ||||
|     shouldShowCaptureButton(service: any): boolean { | ||||
|       return service.payment_status === 1; | ||||
|     } | ||||
|   }, | ||||
| }) | ||||
|   | ||||
| @@ -55,7 +55,14 @@ | ||||
|                     <div>{{ formatDate(service.scheduled_date) }}</div> | ||||
|                     <div class="text-xs opacity-70">{{ formatTime(service.scheduled_date) }}</div> | ||||
|                   </td> | ||||
|                   <td class="align-top">{{ service.customer_name }}</td> | ||||
|                   <td class="align-top"> | ||||
|                     <router-link | ||||
|                       :to="{ name: 'customerProfile', params: { id: service.customer_id } }" | ||||
|                       class="text-white hover:text-green-500 hover:underline" | ||||
|                     > | ||||
|                       {{ service.customer_name }} | ||||
|                     </router-link> | ||||
|                   </td> | ||||
|                   <td class="align-top">{{ service.customer_address }}, {{ service.customer_town }}</td> | ||||
|  | ||||
|                   <!-- | ||||
| @@ -86,7 +93,8 @@ | ||||
|                   <td class="text-right font-mono align-top">{{ formatCurrency(service.service_cost) }}</td> | ||||
|                   <td class="text-right align-top space-x-2"> | ||||
|                     <button @click="openEditModal(service)" class="btn btn-sm btn-primary">View</button> | ||||
|                     <router-link v-if="service.service_cost !== undefined && service.service_cost !== '' && Number(service.service_cost) === 0" :to="{ name: 'chargeServiceAuthorize', params: { id: service.id } }" class="btn btn-sm btn-success">Charge</router-link> | ||||
|                     <router-link v-if="shouldShowChargeButton(service)" :to="{ name: 'payService', params: { id: service.id } }" class="btn btn-sm btn-success">Charge</router-link> | ||||
|                     <router-link v-if="shouldShowCaptureButton(service)" :to="{ name: 'chargeServiceAuthorize', params: { id: service.id } }" class="btn btn-sm btn-warning">Capture</router-link> | ||||
|                   </td> | ||||
|                 </tr> | ||||
|               </tbody> | ||||
| @@ -99,7 +107,12 @@ | ||||
|               <div class="card-body p-4"> | ||||
|                 <div class="flex justify-between items-start"> | ||||
|                   <div> | ||||
|                     <h2 class="card-title text-base">{{ service.customer_name }}</h2> | ||||
|                     <router-link | ||||
|                       :to="{ name: 'customerProfile', params: { id: service.customer_id } }" | ||||
|                       class="card-title text-base text-white hover:text-green-500 hover:underline" | ||||
|                     > | ||||
|                       {{ service.customer_name }} | ||||
|                     </router-link> | ||||
|                     <p class="text-xs text-gray-500">ID: {{ service.id }}</p> | ||||
|                     <p class="text-xs text-gray-400">{{ service.customer_address }}, {{ service.customer_town }}</p> | ||||
|                   </div> | ||||
| @@ -128,7 +141,8 @@ | ||||
|  | ||||
|                 <div class="card-actions justify-end mt-2 space-x-2"> | ||||
|                   <button @click="openEditModal(service)" class="btn btn-sm btn-primary">View</button> | ||||
|                     <router-link v-if="service.service_cost !== undefined && service.service_cost !== '' && Number(service.service_cost) === 0" :to="{ name: 'chargeServiceAuthorize', params: { id: service.id } }" class="btn btn-sm btn-success">Charge</router-link> | ||||
|                   <router-link v-if="shouldShowChargeButton(service)" :to="{ name: 'payService', params: { id: service.id } }" class="btn btn-sm btn-success">Charge</router-link> | ||||
|                     <router-link v-if="shouldShowCaptureButton(service)" :to="{ name: 'chargeServiceAuthorize', params: { id: service.id } }" class="btn btn-sm btn-warning">Capture</router-link> | ||||
|                 </div> | ||||
|               </div> | ||||
|             </div> | ||||
| @@ -167,6 +181,7 @@ interface ServiceCall { | ||||
|   type_service_call: number; | ||||
|   description: string; | ||||
|   service_cost: string; | ||||
|   payment_status?: number; | ||||
| } | ||||
|  | ||||
| export default defineComponent({ | ||||
| @@ -219,7 +234,7 @@ export default defineComponent({ | ||||
|           headers: authHeader(), | ||||
|           withCredentials: true, | ||||
|         }); | ||||
|         this.services = response.data; | ||||
|         this.services = response.data.sort((a: ServiceCall, b: ServiceCall) => b.id - a.id); | ||||
|       } catch (error) { | ||||
|         console.error("Failed to fetch today's service calls:", error); | ||||
|       } finally { | ||||
| @@ -326,6 +341,14 @@ export default defineComponent({ | ||||
|         4: 'black', | ||||
|       }; | ||||
|       return colorMap[typeId] || 'gray'; | ||||
|     }, | ||||
|  | ||||
|     shouldShowChargeButton(service: any): boolean { | ||||
|       return service.payment_status === null || service.payment_status === undefined; | ||||
|     }, | ||||
|  | ||||
|     shouldShowCaptureButton(service: any): boolean { | ||||
|       return service.payment_status === 1; | ||||
|     } | ||||
|   }, | ||||
| }) | ||||
|   | ||||
		Reference in New Issue
	
	Block a user