nidus-sync/ts/view/Home.vue

238 lines
6.1 KiB
Vue
Raw Normal View History

<template>
2026-03-21 23:42:23 +00:00
<!-- Dashboard Header -->
<div class="row mb-4">
<div class="col-md-6">
<h1>{{ dashboard.organization.name }} Dashboard</h1>
<p class="text-muted">
Overview of mosquito control activities in your district
</p>
</div>
<div
class="col-md-6 text-md-end d-flex align-items-center justify-content-md-end"
>
<p v-if="dashboard.isSyncOngoing" class="last-refreshed mb-0">
<i class="fas fa-sync-alt me-2 syncing"></i>Syncing now...
</p>
<p v-else class="last-refreshed mb-0">
<i class="fas fa-sync-alt me-2"></i>Last updated:
<span id="last-refreshed-time">{{
formatTimeRelative(dashboard.lastSync)
}}</span>
<button
class="btn btn-sm btn-outline-primary ms-3"
@click="refreshData"
>
Refresh Data
</button>
</p>
</div>
</div>
<!-- Key Metrics -->
<div class="row g-4">
<!-- Last Refreshed -->
<div class="col-md-3">
<div class="card stats-card h-100">
<div class="card-body text-center">
<div class="metric-icon bg-success text-white">
<i class="fas fa-clock"></i>
</div>
<h5 class="card-title">Last Data Refresh</h5>
<p class="metric-value">
{{ formatTimeRelative(dashboard.lastSync) }}
</p>
<!-- <p class="card-text text-muted">Last sync: 12:45 PM</p> -->
</div>
</div>
</div>
<!-- Service Requests -->
<div class="col-md-3">
<div class="card stats-card h-100">
<div class="card-body text-center">
<div class="metric-icon bg-warning text-white">
<i class="fas fa-ticket-alt"></i>
</div>
<h5 class="card-title">Service Requests</h5>
<p v-if="dashboard.isSyncOngoing" class="metric-value">
{{ formatBigNumber(dashboard.counts.service_requests) }}...?
</p>
<p v-else class="metric-value">
{{ formatBigNumber(dashboard.counts.service_requests) }}
</p>
<!--<p class="card-text text-muted">
<span class="text-success">
<i class="fas fa-arrow-up"></i> 12%
</span> since last week
</p>-->
</div>
</div>
</div>
<!-- Mosquito Sources -->
<div class="col-md-3">
<div class="card stats-card h-100">
<div class="card-body text-center">
<div class="metric-icon bg-danger text-white">
<i class="fas fa-bug"></i>
</div>
<h5 class="card-title">Mosquito Sources</h5>
<p v-if="dashboard.isSyncOngoing" class="metric-value">
{{ formatBigNumber(dashboard.counts.mosquito_sources) }}..?
</p>
<p v-else class="metric-value">
{{ formatBigNumber(dashboard.counts.mosquito_sources) }}
</p>
<!-- <p class="card-text text-muted">
<span class="text-danger">
<i class="fas fa-arrow-up"></i> 8%
</span> since last month
</p> -->
</div>
</div>
</div>
<!-- Inspections -->
<div class="col-md-3">
<div class="card stats-card h-100">
<div class="card-body text-center">
<div class="metric-icon bg-info text-white">
<i class="fas fa-clipboard-check"></i>
</div>
<h5 class="card-title">Traps</h5>
<p v-if="dashboard.isSyncOngoing" class="metric-value">
{{ formatBigNumber(dashboard.counts.traps) }}...?
</p>
<p v-else class="metric-value">
{{ formatBigNumber(dashboard.counts.traps) }}
</p>
<!-- <p class="card-text text-muted">
<span class="text-success">
<i class="fas fa-arrow-up"></i> 15%
</span> since last week
</p> -->
</div>
</div>
</div>
</div>
<!-- Map Section -->
<h3 class="section-title">Mosquito Activity Heatmap</h3>
<div class="row">
<div class="col-12">
<p v-if="session.user && dashboard.serviceArea.min.x === 0.0">
2026-03-21 23:42:23 +00:00
No service area for this organization yet
</p>
<MapAggregate
2026-03-21 23:42:23 +00:00
v-else
:bounds="session.user?.organization.service_area"
:markers="[]"
:organizationId="dashboard.organization.id"
:tegola="session.urls?.tegola ?? ''"
2026-03-21 23:42:23 +00:00
/>
</div>
</div>
<!-- Recent Activity Section -->
<h3 class="section-title">Recent Activity</h3>
<div class="row">
<div class="col-12">
<div class="card">
<div class="card-body">
<div class="table-responsive">
<table class="table table-hover">
<thead>
<tr>
<th>Date</th>
<th>Type</th>
<th>Location</th>
<th>Status</th>
<th>Action</th>
</tr>
</thead>
</table>
</div>
</div>
</div>
</div>
</div>
</template>
<script setup lang="ts">
2026-03-21 23:42:23 +00:00
import { onMounted, reactive } from "vue";
import MapAggregate from "@/components/MapAggregate.vue";
import { useSessionStore } from "@/store/session";
2026-03-21 23:42:23 +00:00
const dashboard = reactive({
counts: {
service_requests: 0,
mosquito_sources: 0,
traps: 0,
},
organization: {
name: "",
id: 0,
2026-03-21 23:42:23 +00:00
},
isSyncOngoing: false,
lastSync: new Date(),
tegolaUrl: "",
serviceArea: {
min: { x: 0, y: 0 },
max: { x: 0, y: 0 },
},
recentRequests: [],
});
const session = useSessionStore();
2026-03-21 23:42:23 +00:00
onMounted(async () => {});
function formatBigNumber(n: number): string {
// Convert the number to a string
const numStr = n.toString();
// Add commas every three digits from the right
let result = "";
for (let i = 0; i < numStr.length; i++) {
if (i > 0 && (numStr.length - i) % 3 === 0) {
result += ",";
}
result += numStr[i];
}
return result;
}
function formatTimeRelative(t: Date): string {
const now = new Date();
const diffMs = now.getTime() - t.getTime();
const hours = diffMs / (1000 * 60 * 60);
if (hours > 0) {
if (hours < 1) {
const minutes = diffMs / (1000 * 60);
return `${Math.floor(minutes)} minutes ago`;
} else if (hours < 24) {
return `${Math.floor(hours)} hours ago`;
} else {
const days = hours / 24;
return `${Math.floor(days)} days ago`;
}
} else {
if (hours < -24) {
const days = hours / 24;
return `in ${Math.floor(-1 * days)} days`;
} else if (hours < -1) {
return `in ${Math.floor(-1 * hours)} hours`;
} else {
const minutes = diffMs / (1000 * 60);
if (minutes > -1) {
const seconds = diffMs / 1000;
return `in ${Math.floor(-1 * seconds)} seconds`;
}
return `in ${Math.floor(-1 * minutes)} minutes`;
}
}
}
function refreshData() {
console.log("fake refresh");
}
</script>