263 lines
14 KiB
HTML
Executable File
263 lines
14 KiB
HTML
Executable File
|
|
<!DOCTYPE html>
|
|
<html lang="hu">
|
|
<head>
|
|
<meta charset="UTF-8">
|
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
<title>Service Finder</title>
|
|
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.0/dist/css/bootstrap.min.css" rel="stylesheet">
|
|
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap-icons@1.11.1/font/bootstrap-icons.css">
|
|
<style>
|
|
body { background-color: #f0f2f5; font-family: 'Segoe UI', sans-serif; }
|
|
.main-container { max-width: 1000px; margin: 30px auto; }
|
|
|
|
/* Navigáció */
|
|
.nav-pills .nav-link { border-radius: 50px; padding: 10px 25px; font-weight: 600; color: #6c757d; margin-right: 10px; }
|
|
.nav-pills .nav-link.active { background-color: #0d6efd; color: white; }
|
|
|
|
/* Csempék */
|
|
.garage-card { border: none; border-radius: 12px; background: white; box-shadow: 0 2px 8px rgba(0,0,0,0.04); transition: transform 0.2s; cursor: pointer; height: 100%; position: relative; overflow: hidden; }
|
|
.garage-card:hover { transform: translateY(-3px); box-shadow: 0 5px 15px rgba(0,0,0,0.1); }
|
|
.card-top-strip { height: 6px; background: #0d6efd; width: 100%; position: absolute; top: 0; left: 0; }
|
|
.car-icon { font-size: 2rem; color: #6c757d; }
|
|
.plate-badge { background: #ffcc00; color: black; border: 1px solid #e0b000; font-family: 'Courier New', monospace; font-weight: bold; padding: 2px 6px; font-size: 0.9rem; border-radius: 4px; }
|
|
|
|
/* DETAIL VIEW (Adatlap) */
|
|
.detail-header { background: white; border-radius: 15px; padding: 25px; margin-bottom: 20px; box-shadow: 0 2px 10px rgba(0,0,0,0.03); }
|
|
.stat-box { background: #f8f9fa; border-radius: 10px; padding: 15px; text-align: center; height: 100%; }
|
|
.stat-value { font-size: 1.5rem; font-weight: bold; color: #212529; }
|
|
.stat-label { font-size: 0.85rem; color: #6c757d; text-transform: uppercase; }
|
|
|
|
/* Detail Tabs */
|
|
.detail-tabs .nav-link { border: none; color: #6c757d; font-weight: 600; padding-bottom: 15px; border-bottom: 3px solid transparent; }
|
|
.detail-tabs .nav-link.active { color: #0d6efd; border-bottom-color: #0d6efd; background: none; }
|
|
.tab-content-area { background: white; border-radius: 0 0 15px 15px; padding: 30px; box-shadow: 0 5px 15px rgba(0,0,0,0.03); margin-top: -1px; }
|
|
|
|
/* Csapat táblázat */
|
|
.avatar-circle { width: 40px; height: 40px; background: #e9ecef; border-radius: 50%; display: flex; align-items: center; justify-content: center; font-weight: bold; }
|
|
.modal-backdrop-custom { position: fixed; top: 0; left: 0; width: 100%; height: 100%; background: rgba(0,0,0,0.5); z-index: 1040; display: flex; align-items: center; justify-content: center; }
|
|
.modal-content-custom { background: white; padding: 30px; border-radius: 15px; width: 100%; max-width: 500px; box-shadow: 0 10px 30px rgba(0,0,0,0.2); }
|
|
</style>
|
|
</head>
|
|
<body>
|
|
|
|
<div id="app">
|
|
<nav class="navbar navbar-dark bg-primary mb-4 shadow-sm py-2">
|
|
<div class="container">
|
|
<span class="navbar-brand mb-0 h1 fs-5"><i class="bi bi-speedometer2 me-2"></i>Service Finder</span>
|
|
<div class="text-white small"><i class="bi bi-building me-1"></i> Demo Company Kft.</div>
|
|
</div>
|
|
</nav>
|
|
|
|
<div class="container main-container">
|
|
|
|
<div v-if="view === 'dashboard'">
|
|
<ul class="nav nav-pills mb-4">
|
|
<li class="nav-item"><a class="nav-link" :class="{active: activeTab === 'garage'}" href="#" @click="activeTab = 'garage'"><i class="bi bi-car-front-fill me-2"></i>Garázs</a></li>
|
|
<li class="nav-item"><a class="nav-link" :class="{active: activeTab === 'team'}" href="#" @click="activeTab = 'team'"><i class="bi bi-people-fill me-2"></i>Csapat</a></li>
|
|
</ul>
|
|
|
|
<div v-if="activeTab === 'garage'">
|
|
<div class="d-flex justify-content-between align-items-center mb-3">
|
|
<h4 class="fw-bold text-secondary mb-0">Járművek</h4>
|
|
<button class="btn btn-outline-primary btn-sm" @click="switchToWizard"><i class="bi bi-plus-lg"></i> Új rögzítése</button>
|
|
</div>
|
|
<div class="row g-3">
|
|
<div class="col-md-6 col-lg-4" v-for="car in myCars" :key="car.vehicle_id">
|
|
<div class="card garage-card p-3" @click="openVehicleDetail(car.vehicle_id)">
|
|
<div class="card-top-strip"></div>
|
|
<div class="d-flex justify-content-between align-items-start mb-2">
|
|
<div><h5 class="fw-bold mb-0 text-dark">{{ car.brand }}</h5><div class="text-muted small">{{ car.model }}</div></div>
|
|
<i class="bi bi-car-front car-icon"></i>
|
|
</div>
|
|
<div class="mt-2 d-flex justify-content-between align-items-end">
|
|
<div class="plate-badge">{{ car.plate }}</div>
|
|
<span class="badge bg-light text-dark border">{{ translateRole(car.role) }}</span>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div v-if="activeTab === 'team'">
|
|
<div class="d-flex justify-content-between align-items-center mb-3">
|
|
<h4 class="fw-bold text-secondary mb-0">Munkatársak</h4>
|
|
<button class="btn btn-primary btn-sm" @click="showInviteModal = true"><i class="bi bi-person-plus-fill me-2"></i>Új meghívása</button>
|
|
</div>
|
|
<div class="card border-0 shadow-sm rounded-4 p-4 text-center text-muted" v-if="team.length === 0">Nincs adat</div>
|
|
<div class="card border-0 shadow-sm rounded-4 overflow-hidden" v-else>
|
|
<table class="table table-hover align-middle mb-0">
|
|
<thead class="bg-light"><tr><th class="ps-4">Név</th><th>Szerepkör</th><th>Ország</th><th>Csatlakozott</th></tr></thead>
|
|
<tbody>
|
|
<tr v-for="member in team">
|
|
<td class="ps-4"><div class="fw-bold">Munkatárs</div><div class="small text-muted">{{member.email}}</div></td>
|
|
<td>{{translateRole(member.role)}}</td>
|
|
<td>{{member.country}}</td>
|
|
<td>{{member.joined_at ? member.joined_at.split('T')[0] : ''}}</td>
|
|
</tr>
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div v-if="view === 'detail' && selectedCar">
|
|
|
|
<button class="btn btn-link text-decoration-none mb-3 ps-0" @click="view = 'dashboard'">
|
|
<i class="bi bi-arrow-left me-1"></i> Vissza a Garázsba
|
|
</button>
|
|
|
|
<div class="detail-header d-flex justify-content-between align-items-center flex-wrap gap-3">
|
|
<div class="d-flex align-items-center">
|
|
<div class="bg-primary text-white rounded-circle d-flex align-items-center justify-content-center me-3" style="width: 60px; height: 60px;">
|
|
<i class="bi bi-car-front fs-2"></i>
|
|
</div>
|
|
<div>
|
|
<h2 class="fw-bold mb-0">{{ selectedCar.brand }} {{ selectedCar.model }}</h2>
|
|
<div class="d-flex align-items-center mt-1 gap-2">
|
|
<span class="plate-badge fs-6">{{ selectedCar.plate }}</span>
|
|
<span class="badge bg-secondary">{{ translateRole(selectedCar.role) }}</span>
|
|
<span class="text-muted small ms-2"><i class="bi bi-upc-scan me-1"></i>{{ selectedCar.vin }}</span>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="text-end">
|
|
<button class="btn btn-outline-danger me-2"><i class="bi bi-exclamation-triangle"></i> Hiba jelentése</button>
|
|
<button class="btn btn-success"><i class="bi bi-plus-lg"></i> Költség / Szerviz</button>
|
|
</div>
|
|
</div>
|
|
|
|
<ul class="nav nav-tabs detail-tabs mb-0">
|
|
<li class="nav-item"><a class="nav-link" :class="{active: detailTab === 'overview'}" href="#" @click="detailTab = 'overview'">Áttekintés</a></li>
|
|
<li class="nav-item"><a class="nav-link" :class="{active: detailTab === 'history'}" href="#" @click="detailTab = 'history'">Szervizkönyv</a></li>
|
|
<li class="nav-item"><a class="nav-link" :class="{active: detailTab === 'settings'}" href="#" @click="detailTab = 'settings'">Beállítások</a></li>
|
|
</ul>
|
|
|
|
<div class="tab-content-area">
|
|
|
|
<div v-if="detailTab === 'overview'">
|
|
<div class="row g-4">
|
|
<div class="col-md-3">
|
|
<div class="stat-box">
|
|
<div class="stat-value text-primary">{{ selectedCar.mileage.toLocaleString() }} km</div>
|
|
<div class="stat-label">Aktuális óraállás</div>
|
|
</div>
|
|
</div>
|
|
<div class="col-md-3">
|
|
<div class="stat-box">
|
|
<div class="stat-value text-success">OK</div>
|
|
<div class="stat-label">Műszaki állapot</div>
|
|
</div>
|
|
</div>
|
|
<div class="col-md-3">
|
|
<div class="stat-box">
|
|
<div class="stat-value">{{ formatCurrency(0, selectedCar.currency) }}</div>
|
|
<div class="stat-label">Idei költés</div>
|
|
</div>
|
|
</div>
|
|
<div class="col-md-3">
|
|
<div class="stat-box">
|
|
<div class="stat-value text-muted">{{ selectedCar.start_date }}</div>
|
|
<div class="stat-label">Flottába került</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<h5 class="fw-bold mt-5 mb-3">Legutóbbi aktivitás</h5>
|
|
<div class="alert alert-light border text-center text-muted py-4">
|
|
<i class="bi bi-clock-history fs-3 d-block mb-2"></i>
|
|
Még nincs rögzített esemény.
|
|
</div>
|
|
</div>
|
|
|
|
<div v-if="detailTab === 'history'">
|
|
<div class="text-center py-5 text-muted">
|
|
<i class="bi bi-tools display-4 mb-3"></i>
|
|
<h5>A szerviztörténet üres</h5>
|
|
<p>Rögzíts tankolást vagy szervizt a jobb felső gombbal!</p>
|
|
</div>
|
|
</div>
|
|
|
|
<div v-if="detailTab === 'settings'">
|
|
<h5 class="text-danger fw-bold">Veszélyzóna</h5>
|
|
<hr>
|
|
<p>Jármű eltávolítása a flottából vagy eladás.</p>
|
|
<button class="btn btn-outline-danger">Jármű archiválása</button>
|
|
</div>
|
|
|
|
</div>
|
|
</div>
|
|
|
|
<div v-if="showInviteModal" class="modal-backdrop-custom">
|
|
<div class="modal-content-custom">
|
|
<h4>Meghívás</h4>
|
|
<input class="form-control my-3" v-model="inviteForm.email" placeholder="Email cím">
|
|
<div class="d-flex justify-content-end gap-2">
|
|
<button class="btn btn-light" @click="showInviteModal = false">Mégse</button>
|
|
<button class="btn btn-primary" @click="sendInvite">Küldés</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div v-if="view === 'wizard'" class="wizard-card"><button @click="view='dashboard'">Mégse</button> <h3>Varázsló...</h3></div>
|
|
|
|
</div>
|
|
</div>
|
|
|
|
<script src="https://unpkg.com/vue@3/dist/vue.global.js"></script>
|
|
<script>
|
|
const { createApp } = Vue
|
|
|
|
createApp({
|
|
data() {
|
|
return {
|
|
view: 'dashboard', // dashboard | detail | wizard
|
|
activeTab: 'garage', // garage | team
|
|
detailTab: 'overview',// overview | history | settings
|
|
|
|
showInviteModal: false,
|
|
inviteForm: { email: '', role: 'DRIVER', access_level: 'LOG_ONLY' },
|
|
|
|
myCars: [],
|
|
team: [],
|
|
selectedCar: null // Ide töltjük be a részleteket
|
|
}
|
|
},
|
|
methods: {
|
|
translateRole(role) {
|
|
const map = { 'OWNER': 'Tulajdonos', 'DRIVER': 'Sofőr', 'FLEET_MANAGER': 'Flotta Menedzser' };
|
|
return map[role] || role;
|
|
},
|
|
formatCurrency(amount, currency) {
|
|
// Ez a "Varázsló", ami a böngésző nyelvétől függően formáz
|
|
try {
|
|
return new Intl.NumberFormat(navigator.language, { style: 'currency', currency: currency || 'HUF' }).format(amount);
|
|
} catch (e) { return amount + " " + currency; }
|
|
},
|
|
async fetchData() {
|
|
const res1 = await fetch('/api/my_vehicles');
|
|
this.myCars = await res1.json();
|
|
const res2 = await fetch('/api/fleet/members');
|
|
this.team = await res2.json();
|
|
},
|
|
async openVehicleDetail(id) {
|
|
// Lekérjük a részletes adatokat
|
|
try {
|
|
const res = await fetch('/api/vehicle/' + id);
|
|
if(res.ok) {
|
|
this.selectedCar = await res.json();
|
|
this.view = 'detail';
|
|
this.detailTab = 'overview';
|
|
} else { alert("Hiba az adatok betöltésekor"); }
|
|
} catch(e) { console.error(e); }
|
|
},
|
|
async sendInvite() { /* ... (előző kód) ... */ alert("Meghívó elküldve!"); this.showInviteModal = false; },
|
|
switchToWizard() { this.view = 'wizard'; }
|
|
},
|
|
mounted() {
|
|
this.fetchData();
|
|
}
|
|
}).mount('#app')
|
|
</script>
|
|
</body>
|
|
</html> |