Added call dropdown

This commit is contained in:
2025-09-24 16:38:56 -04:00
parent a9ee292b23
commit c57ceb7e47
6 changed files with 512 additions and 70 deletions

View File

@@ -4,6 +4,8 @@ ENV VITE_BASE_URL="http://localhost:9510"
ENV VITE_AUTO_URL="http://localhost:9514"
ENV VITE_MONEY_URL="http://localhost:9513"
ENV VITE_AUTHORIZE_URL="http://localhost:9516"
ENV VITE_VOIPMS_URL="http://localhost:9517"
ENV VITE_VOIPMS_TOKEN="my_secret_token"
ENV VITE_COMPANY_ID='1'
@@ -17,4 +19,4 @@ ENV PATH /app/node_modules/.bin:$PATH
RUN npm install
COPY . /app
COPY . /app

View File

@@ -4,6 +4,7 @@ ENV VITE_BASE_URL="http://192.168.1.204:9610"
ENV VITE_AUTO_URL="http://192.168.1.204:9614"
ENV VITE_MONEY_URL="http://192.168.1.204:9613"
ENV VITE_AUTHORIZE_URL="http://localhost:9516"
ENV VITE_VOIPMS_URL="http://localhost:9517"
ENV VITE_COMPANY_ID='1'
WORKDIR /app

View File

@@ -6,7 +6,7 @@ ENV VITE_BASE_URL="https://apioil.edwineames.com"
ENV VITE_AUTO_URL="https://apiauto.edwineames.com"
ENV VITE_MONEY_URL="https://apimoney.edwineames.com"
ENV VITE_AUTHORIZE_URL="https://apicard.edwineames.com"
ENV VITE_VOIPMS_URL="http://apiphone.edwineames.com:9516"
WORKDIR /app
COPY package.json ./

View File

@@ -1,74 +1,216 @@
<!-- headerauth.vue -->
<template>
<div class="navbar bg-primary border-b border-gray-700 sticky top-0 z-30">
<!-- Main header container, contains both rows -->
<header class="sticky top-0 z-50 bg-base-200 shadow-sm">
<!-- Navbar Start Section -->
<div class="navbar-start">
<!-- Hamburger Menu Toggle for Mobile -->
<label for="my-drawer-2" class="btn btn-ghost btn-circle drawer-button lg:hidden">
<svg xmlns="http://www.w3.org/2000/svg" class="h-5 w-5" fill="none" viewBox="0 0 24 24" stroke="currentColor"><path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M4 6h16M4 12h16M4 18h7" /></svg>
</label>
<!-- Logo -->
<router-link :to="{ name: 'home' }" class="btn btn-ghost normal-case text-xl">
Auburn Oil
</router-link>
</div>
<!-- Navbar Center Section (Desktop Search Bar) -->
<div class="navbar-center hidden lg:flex">
<!--
THIS IS THE ONLY CHANGE NEEDED ON THIS ENTIRE PAGE.
We are adding the @input event listener to trigger the search.
-->
<input
id="customer-search-input"
type="text"
placeholder="Search customers..."
v-model="searchStore.searchTerm"
class="input input-bordered"
@input="searchStore.debouncedSearch"
/>
</div>
<!-- Navbar End Section (Buttons and User Dropdown) -->
<div class="navbar-end gap-2">
<!-- Create Customer Button (Desktop Only) -->
<router-link :to="{ name: 'customerCreate' }" class="btn btn-accent btn-sm hidden lg:inline-flex">
Create Customer
</router-link>
<!-- User Dropdown -->
<div v-if="user.user_id" class="dropdown dropdown-end">
<label tabindex="0" class="btn btn-ghost btn-circle avatar">
<div class="w-10 rounded-full bg-neutral text-neutral-content flex items-center justify-center">
<span class="text-lg font-bold">{{ userInitials }}</span>
</div>
<!-- Row 1: The primary navbar -->
<div class="navbar px-4">
<!-- Navbar Start: Logo & Mobile Menu Toggle -->
<div class="navbar-start">
<label for="my-drawer-2" class="btn btn-ghost btn-circle drawer-button lg:hidden">
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-6 h-6">
<path stroke-linecap="round" stroke-linejoin="round" d="M3.75 6.75h16.5M3.75 12h16.5m-16.5 5.25h16.5" />
</svg>
</label>
<ul tabindex="0" class="menu menu-sm dropdown-content mt-3 z-[1] p-2 shadow bg-base-100 rounded-box w-52">
<li class="p-2 font-semibold">{{ user.user_name }}</li>
<div class="divider my-0"></div>
<li>
<router-link :to="{ name: 'employeeProfile', params: { id: user.user_id } }">
Profile
</router-link>
</li>
<li>
<router-link :to="{ name: 'changePassword' }">
Change Password
</router-link>
</li>
<li><a @click="logout">Logout</a></li>
</ul>
<router-link :to="{ name: 'home' }" class="btn btn-ghost normal-case text-xl font-bold">
Auburn Oil
</router-link>
</div>
<!-- This provides the loading indicator while we wait for the API call to finish. -->
<div v-else class="px-4">
{{user}}
<!--
Navbar End: Contains the Search Bar (on tablet+) and all action buttons.
Using flexbox to manage the space distribution.
-->
<div class="navbar-end w-full gap-4">
<!-- Search Bar Wrapper (for tablet and up) -->
<!-- This is the growing element. Hidden on mobile. -->
<div class="hidden md:flex flex-grow max-w-xl">
<div class="relative w-full">
<div class="absolute inset-y-0 left-0 flex items-center pl-3 pointer-events-none">
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-5 h-5 text-base-content/50">
<path stroke-linecap="round" stroke-linejoin="round" d="M21 21l-5.197-5.197m0 0A7.5 7.5 0 105.196 5.196a7.5 7.5 0 0010.607 10.607z" />
</svg>
</div>
<input
id="customer-search-input-desktop"
type="text"
placeholder="Search customers..."
v-model="searchStore.searchTerm"
class="input input-bordered w-full pl-10"
@input="searchStore.debouncedSearch"
/>
</div>
</div>
<!-- Action Buttons Wrapper -->
<!-- This group is set to not shrink, keeping it always visible. -->
<div class="flex-shrink-0 flex items-center gap-2">
<!-- VOIP Routing Dropdown (visible tablet+) -->
<div class="dropdown dropdown-end hidden md:flex">
<label tabindex="0" class="btn btn-ghost gap-2">
<svg xmlns="http://www.w.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-5 h-5 text-success">
<path stroke-linecap="round" stroke-linejoin="round" d="M2.25 6.75c0 8.284 6.716 15 15 15h2.25a2.25 2.25 0 002.25-2.25v-1.372c0-.516-.351-.966-.852-1.091l-4.423-1.106c-.44-.11-.902.055-1.173.417l-.97 1.293c-.282.376-.769.542-1.21.38a12.035 12.035 0 01-7.143-7.143c-.162-.441.004-.928.38-1.21l1.293-.97c.363-.271.527-.734.417-1.173L6.963 3.102a1.125 1.125 0 00-1.091-.852H4.5A2.25 2.25 0 002.25 6.75z" />
</svg>
<span class="font-semibold whitespace-nowrap">{{ currentPhone || 'Routing' }}</span>
</label>
<ul tabindex="0" class="dropdown-content z-[1] menu p-2 shadow bg-base-100 rounded-box w-52 mt-4">
<li v-for="option in routingOptions" :key="option.value">
<a @click.prevent="showConfirmRoute(option)" href="#">
<span class="font-mono">{{ option.label }}</span>
</a>
</li>
</ul>
</div>
<!-- Create Customer Button (visible tablet+) -->
<router-link :to="{ name: 'customerCreate' }" class="btn btn-success btn-sm hidden md:inline-flex">
New Customer
</router-link>
<!-- User Dropdown (Always visible) -->
<div v-if="user.user_id" class="dropdown dropdown-end">
<label tabindex="0" class="btn btn-ghost btn-circle avatar">
<div class="w-10 rounded-full bg-neutral text-neutral-content flex items-center justify-center">
<span class="text-lg font-bold">{{ userInitials }}</span>
</div>
</label>
<ul tabindex="0" class="menu menu-sm dropdown-content mt-3 z-[1] p-2 shadow bg-base-100 rounded-box w-52">
<li class="p-2 font-semibold">{{ user.user_name }}</li>
<div class="divider my-0"></div>
<li><router-link :to="{ name: 'employeeProfile', params: { id: user.user_id } }">Profile</router-link></li>
<li><router-link :to="{ name: 'changePassword' }">Change Password</router-link></li>
<li><a @click="logout">Logout</a></li>
</ul>
</div>
<div v-else class="skeleton w-10 h-10 rounded-full"></div>
</div>
</div>
</div>
<!-- Row 2: MOBILE SEARCH BAR -->
<!-- This full-width search bar only appears on screens smaller than md -->
<div class="w-full px-4 pb-3 md:hidden">
<div class="relative w-full">
<div class="absolute inset-y-0 left-0 flex items-center pl-3 pointer-events-none">
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-5 h-5 text-base-content/50">
<path stroke-linecap="round" stroke-linejoin="round" d="M21 21l-5.197-5.197m0 0A7.5 7.5 0 105.196 5.196a7.5 7.5 0 0010.607 10.607z" />
</svg>
</div>
<input
id="customer-search-input-mobile"
type="text"
placeholder="Search customers..."
v-model="searchStore.searchTerm"
class="input input-bordered w-full pl-10"
@input="searchStore.debouncedSearch"
/>
</div>
</div>
</header>
<!-- MODAL SECTION REMAINS UNCHANGED -->
<!-- ... (paste your existing modals here) ... -->
<div class="modal" :class="{ 'modal-open': isRouteModalVisible }">
<div class="modal-box">
<div v-if="routeModalMode === 'confirm'">
<h3 class="font-bold text-lg">Confirm Routing Change</h3>
<p class="py-4">FROM: {{ currentPhone }} TO: {{ selectedOption?.label }}</p>
<div class="modal-action">
<button @click="proceedRoute" class="btn btn-success">Confirm</button>
<button @click="closeRouteModal" class="btn btn-error">Cancel</button>
</div>
</div>
<div v-else-if="routeModalMode === 'loading'">
<h3 class="font-bold text-lg">Switching Route</h3>
<div class="py-4 text-center">
<p class="text-lg mb-3">Switching phone routing from {{ currentPhone }} to {{ selectedOption?.label }}...</p>
<div class="loading loading-spinner loading-lg text-primary mb-3"></div>
<p class="text-sm text-gray-600">Please wait while we update the routing.</p>
</div>
</div>
<div v-else-if="routeModalMode === 'result'">
<h3 class="font-bold text-lg">Route Change Result</h3>
<div class="py-4">
<div v-if="routeResponse && !routeResponse.error" class="alert alert-success mb-4">
<svg xmlns="http://www.w3.org/2000/svg" class="stroke-current shrink-0 h-6 w-6" fill="none" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z" />
</svg>
<span>Routing change completed successfully.</span>
</div>
<div v-else-if="routeResponse && routeResponse.error" class="alert alert-error mb-4">
<svg xmlns="http://www.w3.org/2000/svg" class="stroke-current shrink-0 h-6 w-6" fill="none" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M10 14l2-2m0 0l2-2m-2 2l-2-2m2 2l2 2m7-2a9 9 0 11-18 0 9 9 0 0118 0z" />
</svg>
<span>Error occurred during routing change.</span>
</div>
<div v-if="routeResponse.message || routeResponse.status" class="bg-base-200 p-3 rounded">
<p>{{ routeResponse.message || routeResponse.status }}</p>
</div>
<div v-else class="bg-base-200 p-3 rounded">
<p>Response received successfully.</p>
</div>
</div>
<div class="modal-action">
<button @click="closeRouteModal" class="btn">Close</button>
</div>
</div>
</div>
</div>
<div class="modal" :class="{ 'modal-open': isTestModalVisible }">
<div class="modal-box">
<h3 class="font-bold text-lg">DID Test Result</h3>
<div class="py-4">
<div v-if="isTestLoading" class="text-center">
<span class="text-lg mb-3">Testing DID connection...</span>
<div class="loading loading-spinner loading-lg text-primary mb-3"></div>
<p class="text-sm text-gray-600">Please wait while we fetch DID information.</p>
</div>
<div v-else-if="testResponse">
<div v-if="testResponse.status === 'success' && testResponse.dids && testResponse.dids.length > 0" class="space-y-4">
<div class="alert alert-success">
<svg xmlns="http://www.w3.org/2000/svg" class="stroke-current shrink-0 h-6 w-6" fill="none" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z" />
</svg>
<span>DID Test Successful! Connection is working.</span>
</div>
<div v-for="did in testResponse.dids" :key="did.did" class="border rounded-lg p-3 bg-base-100">
<h4 class="font-semibold mb-2">DID: {{ did.did }}</h4>
<div class="grid grid-cols-2 gap-2 text-sm">
<div><strong>Description:</strong> {{ did.description }}</div>
<div><strong>Routing:</strong> {{ did.routing }}</div>
<div><strong>Call Recording:</strong> {{ did.record_calls === '1' ? 'Enabled' : 'Disabled' }}</div>
<div><strong>Voicemail:</strong> {{ did.voicemail === '1' ? 'Enabled' : 'Disabled' }}</div>
</div>
</div>
</div>
<div v-else-if="testResponse.error" class="text-center text-error">
<svg xmlns="http://www.w3.org/2000/svg" class="stroke-current shrink-0 h-6 w-6 mx-auto mb-2" fill="none" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M10 14l2-2m0 0l2-2m-2 2l-2-2m2 2l2 2m7-2a9 9 0 11-18 0 9 9 0 0118 0z" />
</svg>
<p class="font-semibold">API Error</p>
<p>{{ testResponse.error }}</p>
</div>
<div v-else class="bg-base-200 p-3 rounded text-sm overflow-auto max-h-40">
<pre>{{ JSON.stringify(testResponse, null, 2) }}</pre>
</div>
</div>
<div v-else class="text-center">
<p>No response received.</p>
</div>
</div>
<div class="modal-action" v-if="!isTestLoading">
<button @click="isTestModalVisible = false; testResponse = null;" class="btn">Close</button>
</div>
</div>
</div>
</template>
<script lang="ts">
// NO CHANGES to the script block were made. All your logic remains intact.
import { defineComponent } from 'vue'
import axios from 'axios'
import authHeader from '../../services/auth.header'
@@ -81,12 +223,31 @@ interface User {
user_id: number;
}
interface RoutingOption {
value: string;
label: string;
}
export default defineComponent({
data() {
return {
// Initialize with empty objects to prevent template errors
user: {} as User,
currentPhone: '',
routingOptions: [
{ value: 'sip', label: '407323_auburnoil' },
{ value: 'cellphone1', label: 'Ed Cell' },
{ value: 'cellphone2', label: 'Aneta Cell' },
{ value: 'test_did', label: 'Test DID' }
] as RoutingOption[],
selectedOption: null as RoutingOption | null,
isRouteModalVisible: false,
routeModalMode: 'confirm',
routeResponse: null as any,
isTestModalVisible: false,
isTestLoading: false,
testResponse: null as any
}
},
@@ -108,7 +269,10 @@ export default defineComponent({
this.userStatus();
},
mounted() {
console.log('VITE_VOIPMS_URL:', import.meta.env.VITE_VOIPMS_URL);
console.log('VITE_VOIPMS_TOKEN:', import.meta.env.VITE_VOIPMS_TOKEN);
this.updatestatus();
this.fetchCurrentPhone();
},
methods: {
userStatus() {
@@ -118,6 +282,7 @@ export default defineComponent({
url: path,
withCredentials: true,
headers: authHeader(),
})
.then((response: any) => {
console.log(this.user)
@@ -148,6 +313,93 @@ export default defineComponent({
authStore.clearAuth();
// Redirect to login
this.$router.push({ name: 'login' });
},
fetchCurrentPhone() {
const path = import.meta.env.VITE_BASE_URL + '/admin/voip_routing';
axios({
method: 'get',
url: path,
headers: authHeader(),
})
.then((response: any) => {
if (response.data.current_phone) {
this.currentPhone = response.data.current_phone;
}
})
.catch((error: any) => {
console.error('Failed to fetch current routing:', error);
});
},
routeTo(route: string): Promise<any> {
console.log('Routing to:', route);
const path = `${import.meta.env.VITE_VOIPMS_URL}/route/${route}`;
console.log('API path:', path);
return axios({
method: 'post',
url: path,
headers: {
'Authorization': `Bearer ${import.meta.env.VITE_VOIPMS_TOKEN}`,
},
})
.then((response: any) => {
console.log('Success routing to:', route);
this.routeResponse = response.data;
// Find the corresponding label
const option = this.routingOptions.find(opt => opt.value === route);
if (option) {
this.currentPhone = option.label;
console.log('Updated currentPhone to:', this.currentPhone);
}
return response.data;
});
},
showConfirmRoute(option: RoutingOption) {
this.selectedOption = option;
if (option.value === 'test_did') {
this.testDid();
} else {
this.isRouteModalVisible = true;
this.routeModalMode = 'confirm';
}
},
proceedRoute() {
if (this.selectedOption && this.selectedOption.value !== 'test_did') {
this.routeModalMode = 'loading';
this.routeTo(this.selectedOption.value)
.then(() => {
this.routeModalMode = 'result';
})
.catch((error: any) => {
this.routeResponse = { error: error.message };
this.routeModalMode = 'result';
});
}
},
closeRouteModal() {
this.isRouteModalVisible = false;
this.routeModalMode = 'confirm';
this.routeResponse = null;
this.selectedOption = null;
},
testDid() {
this.isTestModalVisible = true;
this.isTestLoading = true;
const path = `${import.meta.env.VITE_VOIPMS_URL}/test/did`;
axios({
method: 'get',
url: path,
headers: {
'Authorization': `Bearer ${import.meta.env.VITE_VOIPMS_TOKEN}`,
},
})
.then((response: any) => {
this.testResponse = response.data;
this.isTestLoading = false;
})
.catch((error: any) => {
this.testResponse = { status: 'error', message: error.message };
this.isTestLoading = false;
});
}
}
});

View File

@@ -0,0 +1,175 @@
<!-- src/pages/admin/oilprice.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>
<li>Set Oil Pricing</li>
</ul>
</div>
<h1 class="text-3xl font-bold mt-4">
Set Today's Oil Pricing
</h1>
<!-- Main Form Card -->
<div class="bg-neutral rounded-lg p-6 mt-6">
<form @submit.prevent="onSubmit" class="space-y-6">
<!-- SECTION 1: Base Pricing -->
<div>
<h2 class="text-lg font-bold">Base Pricing</h2>
<p class="text-xs text-gray-400">Set the core price per gallon for different groups.</p>
<div class="divider mt-2 mb-4"></div>
<div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4">
<!-- Price from Supplier -->
<div class="form-control">
<label class="label"><span class="label-text">Price from Supplier</span></label>
<label class="input-group input-group-sm">
<span>$</span>
<input v-model.number="OilForm.price_from_supplier" type="number" step="0.01" placeholder="3.50" class="input input-bordered input-sm w-full" />
</label>
</div>
<!-- Price for Customer -->
<div class="form-control">
<label class="label"><span class="label-text">Price for Customer</span></label>
<label class="input-group input-group-sm">
<span>$</span>
<input v-model.number="OilForm.price_for_customer" type="number" step="0.01" placeholder="4.50" class="input input-bordered input-sm w-full" />
</label>
</div>
<!-- Price for Employee -->
<div class="form-control">
<label class="label"><span class="label-text">Price for Employee</span></label>
<label class="input-group input-group-sm">
<span>$</span>
<input v-model.number="OilForm.price_for_employee" type="number" step="0.01" placeholder="4.00" class="input input-bordered input-sm w-full" />
</label>
</div>
</div>
</div>
<!-- SECTION 2: Service Fees -->
<div>
<h2 class="text-lg font-bold">Service Fees</h2>
<p class="text-xs text-gray-400">Set the flat fees for special delivery services.</p>
<div class="divider mt-2 mb-4"></div>
<div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4">
<!-- Price Same Day -->
<div class="form-control">
<label class="label"><span class="label-text">Same Day Fee</span></label>
<label class="input-group input-group-sm">
<span>$</span>
<input v-model.number="OilForm.price_same_day" type="number" step="1.00" placeholder="50.00" class="input input-bordered input-sm w-full" />
</label>
</div>
<!-- Price Prime -->
<div class="form-control">
<label class="label"><span class="label-text">Prime Fee</span></label>
<label class="input-group input-group-sm">
<span>$</span>
<input v-model.number="OilForm.price_prime" type="number" step="1.00" placeholder="75.00" class="input input-bordered input-sm w-full" />
</label>
</div>
<!-- Price Emergency -->
<div class="form-control">
<label class="label"><span class="label-text">Emergency Fee</span></label>
<label class="input-group input-group-sm">
<span>$</span>
<input v-model.number="OilForm.price_emergency" type="number" step="1.00" placeholder="150.00" class="input input-bordered input-sm w-full" />
</label>
</div>
</div>
</div>
<!-- SUBMIT BUTTON -->
<div class="pt-4">
<button type="submit" class="btn btn-primary btn-sm">Update Pricing</button>
</div>
</form>
</div>
</div>
</div>
<Footer />
</template>
<script lang="ts">
import { defineComponent } from 'vue'
import axios from 'axios'
import authHeader from '../../services/auth.header'
import Footer from '../../layouts/footers/footer.vue'
import { notify } from "@kyvg/vue3-notification";
export default defineComponent({
name: 'auth',
components: {
Footer,
},
data() {
return {
user: null,
// --- REFACTORED: Simplified, flat form object ---
OilForm: {
price_from_supplier: 0,
price_for_customer: 0,
price_for_employee: 0,
price_same_day: 0,
price_prime: 0,
price_emergency: 0,
},
}
},
created() {
this.userStatus();
},
mounted() {
this.getCurrentPrices();
},
methods: {
userStatus() {
const path = import.meta.env.VITE_BASE_URL + '/auth/whoami';
axios.get(path, { withCredentials: true, headers: authHeader() })
.then((response: any) => {
if (response.data.ok) { this.user = response.data.user; }
})
.catch(() => { this.user = null; });
},
getCurrentPrices() {
const path = import.meta.env.VITE_BASE_URL + "/admin/oil/get";
axios.get(path, { withCredentials: true, headers: authHeader() })
.then((response: any) => {
if (response.data) {
// --- REFACTORED: Populate the flat form object ---
this.OilForm = response.data;
}
});
},
CreatePricing(payload: any) {
const path = import.meta.env.VITE_BASE_URL + "/admin/oil/create";
axios.post(path, payload, { withCredentials: true, headers: authHeader() })
.then((response: any) => {
if (response.data.ok) {
notify({
title: "Success",
text: "Prices have been updated!",
type: "success",
});
this.$router.push({ name: "home" });
} else {
notify({
title: "Error",
text: response.data.error || "An unknown error occurred.",
type: "error",
});
}
});
},
onSubmit() {
// --- REFACTORED: Submit the flat form object ---
this.CreatePricing(this.OilForm);
},
},
});
</script>

View File

@@ -28,11 +28,23 @@
<thead>
<tr>
<!-- SORTABLE HEADERS -->
<th @click="sortBy('tank_level_percent')" class="cursor-pointer hover:text-white">Tank Level</th>
<th @click="sortBy('days_since_last_fill')" class="cursor-pointer hover:text-white">Days Since Fill</th>
<th @click="sortBy('customer_full_name')" class="cursor-pointer hover:text-white">Name</th>
<th @click="sortBy('tank_level_percent')" :class="sortKey === 'tank_level_percent' ? 'cursor-pointer hover:text-white bg-orange-500' : 'cursor-pointer hover:text-white'">
Tank Level
<span v-if="sortKey === 'tank_level_percent'">{{ sortAsc ? '' : '' }}</span>
</th>
<th @click="sortBy('days_since_last_fill')" :class="sortKey === 'days_since_last_fill' ? 'cursor-pointer hover:text-white bg-orange-500' : 'cursor-pointer hover:text-white'">
Days Since Fill
<span v-if="sortKey === 'days_since_last_fill'">{{ sortAsc ? '' : '' }}</span>
</th>
<th @click="sortBy('customer_full_name')" :class="sortKey === 'customer_full_name' ? 'cursor-pointer hover:text-white bg-orange-500' : 'cursor-pointer hover:text-white'">
Name
<span v-if="sortKey === 'customer_full_name'">{{ sortAsc ? '' : '' }}</span>
</th>
<th @click="sortBy('house_factor')" :class="sortKey === 'house_factor' ? 'cursor-pointer hover:text-white bg-orange-500' : 'cursor-pointer hover:text-white'">
Usage Factor
<span v-if="sortKey === 'house_factor'">{{ sortAsc ? '' : '' }}</span>
</th>
<th>Address</th>
<th @click="sortBy('house_factor')" class="cursor-pointer hover:text-white">Usage Factor</th>
<th class="text-right">Actions</th>
</tr>
</thead>
@@ -60,8 +72,8 @@
{{ oil.customer_full_name }}
</router-link>
</td>
<td>{{ oil.customer_address }}, {{ oil.customer_town }}</td>
<td>{{ oil.house_factor }}</td>
<td>{{ oil.customer_address }}, {{ oil.customer_town }}</td>
<td class="text-right">
<div class="flex items-center justify-end gap-2">
<router-link :to="{ name: 'customerEdit', params: { id: oil.customer_id } }" class="btn btn-sm btn-secondary">Edit Customer</router-link>
@@ -162,7 +174,7 @@ export default defineComponent({
user: null,
deliveries: [] as AutoDelivery[],
// --- NEW: Data properties for sorting ---
sortKey: 'estimated_gallons_left' as keyof AutoDelivery | 'tank_level_percent',
sortKey: 'tank_level_percent' as keyof AutoDelivery | 'tank_level_percent',
sortAsc: true,
}
},
@@ -249,4 +261,4 @@ export default defineComponent({
},
},
})
</script>
</script>