version 2.0 | refactor

changed structure of codebase. also include docker files for easy deployment
This commit is contained in:
2025-12-03 15:07:36 -07:00
parent 456d486d39
commit a709c80f38
21 changed files with 2532 additions and 81535 deletions

View File

@@ -1,24 +1,39 @@
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Traxer</title>
<script src="https://cdn.tailwindcss.com"></script>
<link
rel="stylesheet"
href="https://unpkg.com/leaflet@1.9.4/dist/leaflet.css"
crossorigin=""
/>
<script
src="https://unpkg.com/leaflet@1.9.4/dist/leaflet.js"
crossorigin=""
></script>
</head>
<body>
<div id="map" class="w-full h-screen"></div>
</body>
<script src="app.js"></script>
</html>
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Traxer</title>
<link rel="stylesheet" href="https://unpkg.com/leaflet@1.9.4/dist/leaflet.css" />
<link rel="stylesheet" href="style.css" />
</head>
<body>
<div id="map"></div>
<div
style="position:absolute; top:10px; right:10px; z-index:1000; background:white; padding:8px; border-radius:4px; box-shadow:0 0 10px rgba(0,0,0,0.3);">
<button id="toggleStops">Hide Stops</button>
</div>
<div class="controls">
<select id="routeSelect">
<option value="">All Trains & Buses</option>
<option value="701">Blue Line (701)</option>
<option value="703">Red Line (703)</option>
<option value="704">Green Line (704)</option>
<option value="720">S-Line (720)</option>
<option value="750">FrontRunner (750)</option>
</select>
<button id="refresh">Refresh Now</button>
<span class="status" id="status">Loading...</span>
</div>
<div id="vehicleSchedule" style="margin-top: 20px;">
</div>
<script src="https://unpkg.com/leaflet@1.9.4/dist/leaflet.js"></script>
<script src="index.js"> </script>
</body>
</html>

388
public/index.js Normal file
View File

@@ -0,0 +1,388 @@
const map = L.map("map").setView([40.7608, -111.8910], 12);
L.tileLayer("https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png", {
attribution: '© OpenStreetMap contributors'
}).addTo(map);
const trainIcon = L.divIcon({
html: '🚂',
className: "train-icon",
iconSize: [30, 30],
iconAnchor: [15, 15]
});
const busIcon = L.divIcon({
html: '🚌',
className: "train-icon",
iconSize: [28, 28],
iconAnchor: [14, 14]
});
const vehicleMarkers = L.layerGroup().addTo(map);
const routeShapes = L.layerGroup().addTo(map);
const stopMarkers = L.layerGroup().addTo(map);
const API = "http://localhost:1001/api";
const vehicleMarkersMap = {};
let stopsVisible = true;
let currentRoute = "";
async function loadStops(routeId = null) {
stopMarkers.clearLayers();
if (!stopsVisible) return;
try {
let url = `${API}/stop`;
if (routeId) url = `${API}/stop/${routeId}`;
const res = await fetch(url);
const stops = await res.json();
stops.forEach(stop => {
if (!stop.stop_lat || !stop.stop_lon) return;
const wheelchair = (stop.wheelchair_boarding === "1" || stop.wheelchair_boarding === 1) ? " ♿" : "";
const marker = L.circleMarker([stop.stop_lat, stop.stop_lon], {
radius: 6,
color: "#0077ff",
fillColor: "#0077ff",
fillOpacity: 0.8
})
.bindPopup(`<b>${stop.stop_name}${wheelchair}</b><br>ID: ${stop.stop_desc || stop.stop_id}`)
.addTo(stopMarkers);
marker.on('click', () => showStopSchedule(stop.stop_id, stop.stop_name));
});
} catch (err) {
console.error("Error loading stops:", err);
}
}
document.getElementById("toggleStops").addEventListener("click", () => {
stopsVisible = !stopsVisible;
if (stopsVisible) {
document.getElementById("toggleStops").textContent = "Hide Stops";
// Reload current stops (all or just the selected route)
if (currentRoute === "") {
loadStops();
} else {
loadStops(currentRoute);
}
} else {
document.getElementById("toggleStops").textContent = "Show Stops";
stopMarkers.clearLayers();
}
});
async function showStopSchedule(stopId, stopName) {
const scheduleDiv = document.getElementById("vehicleSchedule");
scheduleDiv.innerHTML = "Loading stop schedule...";
try {
const res = await fetch(`${API}/schedule/?station=${stopId}`);
const data = await res.json();
if (!data || data.length === 0) {
scheduleDiv.innerHTML = `<p>No upcoming arrivals for <strong>${stopName}</strong></p>`;
return;
}
const now = new Date();
const today = new Date(now.getFullYear(), now.getMonth(), now.getDate()); // midnight today
// Process and filter valid arrivals
const upcoming = data
.map(item => {
const arrivalRaw = item.stopTime?.arrival_time?.trim();
if (!arrivalRaw || !item.trip || !item.route) return null; // skip invalid
let [hours, minutes, seconds] = arrivalRaw.split(":").map(Number);
let arrivalDate = new Date(today);
if (hours >= 24) {
hours -= 24;
arrivalDate.setDate(arrivalDate.getDate() + 1);
}
arrivalDate.setHours(hours, minutes, seconds, 0);
return { ...item, arrivalDate };
})
.filter(item => item && item.arrivalDate > now) // remove past or invalid
.sort((a, b) => a.arrivalDate - b.arrivalDate);
// Deduplicate by trip_id + arrival_time
const seen = new Set();
const uniqueUpcoming = upcoming.filter(item => {
const key = `${item.trip.trip_id}_${item.arrivalDate.getTime()}`;
if (seen.has(key)) return false;
seen.add(key);
return true;
});
if (uniqueUpcoming.length === 0) {
scheduleDiv.innerHTML = `<p>No upcoming arrivals for <strong>${stopName}</strong></p>`;
return;
}
// Build table
let html = `<h3>Upcoming at ${stopName}</h3>`;
html += `<table style="width:100%; border-collapse:collapse; font-size:0.95em;">
<thead style="background:#333; color:white;">
<tr>
<th style="padding:8px;">Route</th>
<th style="padding:8px;">Destination</th>
<th style="padding:8px;">Arrival</th>
</tr>
</thead>
<tbody>`;
uniqueUpcoming.forEach(item => {
const { stopTime, trip, route, arrivalDate } = item;
const routeNum = route?.route_short_name || route?.route_id || "—";
const headsign = trip?.trip_headsign || "—";
const minutesUntil = Math.round((arrivalDate - now) / 60000);
const displayTime = arrivalDate.toLocaleTimeString([], { hour: '2-digit', minute: '2-digit' });
const timeDisplay = minutesUntil <= 30
? `<strong style="color:#d00;">${displayTime} (${minutesUntil === 0 ? "Due" : minutesUntil + " min"})</strong>`
: displayTime;
const rowColor = route?.route_color && route.route_color !== "000000"
? `#${route.route_color}`
: "#666";
html += `<tr style="border-bottom:1px solid #ddd;">
<td style="padding:6px; text-align:center; background:${rowColor}; color:${getContrastColor(rowColor)}; font-weight:bold;">
${routeNum}
</td>
<td style="padding:6px;">${headsign}</td>
<td style="padding:6px; text-align:center;">${timeDisplay}</td>
</tr>`;
});
html += `</tbody></table>`;
html += `<small style="color:#666; display:block; margin-top:8px;">
Updated ${now.toLocaleTimeString()}
</small>`;
scheduleDiv.innerHTML = html;
} catch (err) {
console.error("Error fetching stop schedule:", err);
scheduleDiv.innerHTML = "<p>Error loading schedule.</p>";
}
}
function getContrastColor(hex) {
if (!hex || hex.length < 6) return "#ffffff";
const r = parseInt(hex.substr(0, 2), 16);
const g = parseInt(hex.substr(2, 2), 16);
const b = parseInt(hex.substr(4, 2), 16);
const luminance = (0.299 * r + 0.587 * g + 0.114 * b) / 255;
return luminance > 0.5 ? "#000000" : "#ffffff";
}
async function updateVehicles() {
document.getElementById("status").textContent = "Updating...";
try {
let trains = [];
let buses = [];
if (!currentRoute || ["701", "703", "704", "720", "750"].includes(currentRoute)) {
const trainRes = currentRoute
? await fetch(`${API}/train/${currentRoute}`)
: await fetch(`${API}/train`);
trains = await trainRes.json();
}
if (!currentRoute) {
const busRes = await fetch(`${API}/bus`);
buses = await busRes.json();
}
const allVehicles = [...trains, ...buses];
const seenVehicleIds = new Set();
allVehicles.forEach(item => {
const v = item.vehicle;
if (!v?.position?.latitude) return;
const id = v.vehicle.id;
seenVehicleIds.add(id);
const lat = v.position.latitude;
const lon = v.position.longitude;
const route = item.route?.route_short_name || "???";
const isTrain = trains.includes(item);
const icon = isTrain ? trainIcon : busIcon;
const speed = v.position.speed ? (v.position.speed * 2.237).toFixed(0) + " mph" : "Stopped";
const popup = `
<b>${item.route?.route_long_name || route + " Service"}</b><br/>
Vehicle ID: ${id}<br/>
Route: ${route}<br/>
Speed: ${speed}<br/>
Updated: ${new Date(v.timestamp * 1000).toLocaleTimeString()}
`;
if (vehicleMarkersMap[id]) {
vehicleMarkersMap[id].setLatLng([lat, lon]);
vehicleMarkersMap[id].setPopupContent(popup);
} else {
const marker = L.marker([lat, lon], {
icon: icon,
rotationAngle: v.position.bearing || 0
}).bindPopup(popup)
.addTo(vehicleMarkers);
vehicleMarkersMap[id] = marker;
}
});
Object.keys(vehicleMarkersMap).forEach(id => {
if (!seenVehicleIds.has(id)) {
vehicleMarkers.removeLayer(vehicleMarkersMap[id]);
delete vehicleMarkersMap[id];
}
});
document.getElementById("status").textContent = `${allVehicles.length} vehicles • ${new Date().toLocaleTimeString()}`;
} catch (err) {
console.error("Update failed:", err);
document.getElementById("status").textContent = "Error";
}
}
async function loadRouteShape(routeId) {
if (!routeId) {
routeShapes.clearLayers();
return;
}
try {
const res = await fetch(`${API}/shape/${routeId}`);
const shapes = await res.json();
routeShapes.clearLayers();
const res2 = await fetch(`${API}/routes`);
const routes = await res2.json();
const route = routes.filter(r => r.route_short_name == routeId);
console.log(route[0])
const routeHexColor = `#${route[0].route_color}`;
shapes.forEach(shape => {
const points = shape.points.map(p => [p.lat, p.lon]);
L.polyline(points, { color: routeHexColor, weight: 5, opacity: 0.7 }).addTo(routeShapes);
});
if (shapes.length > 0) {
const bounds = L.latLngBounds(shapes[0].points.map(p => [p.lat, p.lon]));
map.fitBounds(bounds, { padding: [50, 50] });
}
} catch (e) { console.error("Shape load error:", e); }
}
document.getElementById("routeSelect").addEventListener("change", async (e) => {
currentRoute = e.target.value;
console.log(currentRoute == "")
if (currentRoute === "") {
loadAllRoutes();
loadStops();
}
else {
loadRouteShape(currentRoute);
loadStops(currentRoute);
}
updateVehicles();
});
function delay(ms) {
return new Promise(resolve => setTimeout(resolve, ms));
}
document.getElementById("refresh").addEventListener("click", updateVehicles);
async function loadAllRoutes() {
try {
const res = await fetch(`${API}/routes`);
const routes = await res.json();
routeShapes.clearLayers();
for (const route of routes) {
const routeId = route.route_short_name;
const routeHexColor = route.route_color ? `#${route.route_color}` : "#888";
const shapeRes = await fetch(`${API}/shape/${routeId}`);
const shapes = await shapeRes.json();
shapes.forEach(shape => {
const points = shape.points.map(p => [p.lat, p.lon]);
L.polyline(points, {
color: routeHexColor,
weight: 4,
opacity: 0.6
}).addTo(routeShapes);
});
//await delay(200);
}
const allPoints = [];
routeShapes.eachLayer(layer => {
allPoints.push(...layer.getLatLngs());
});
} catch (e) {
console.error("Error loading all routes:", e);
}
}
async function showVehicleSchedule(vehicleId) {
const scheduleDiv = document.getElementById("vehicleSchedule");
scheduleDiv.innerHTML = "Loading schedule...";
try {
const res = await fetch(`${API}/schedule/?vehicle=${vehicleId}`);
const schedule = await res.json();
if (!schedule.length) {
scheduleDiv.innerHTML = "<p>No schedule available.</p>";
return;
}
let html = `
<h3>Schedule for Vehicle ${vehicleId}</h3>
<table border="1" cellpadding="5" cellspacing="0">
<thead>
<tr>
<th>Stop</th>
<th>Arrival Time</th>
</tr>
</thead>
<tbody>
`;
schedule.forEach(stop => {
const time = new Date(stop.arrival_time * 1000).toLocaleTimeString();
html += `<tr><td>${stop.stop_name}</td><td>${time}</td></tr>`;
});
html += `</tbody></table>`;
scheduleDiv.innerHTML = html;
} catch (err) {
console.error("Error fetching schedule:", err);
scheduleDiv.innerHTML = "<p>Error loading schedule.</p>";
}
}
loadAllRoutes()
.then(() => loadStops())
.then(() => {
updateVehicles();
setInterval(updateVehicles, 10000); // Update every 10 seconds
})
.catch(err => {
console.error("Init failed:", err);
document.getElementById("status").textContent = "Failed to initialize";
});

63
public/style.css Normal file
View File

@@ -0,0 +1,63 @@
body {
margin: 0;
font-family: system-ui, sans-serif;
}
#map {
width: 100vw;
height: 80vh;
}
.controls {
position: absolute;
top: 10px;
left: 50%;
transform: translateX(-50%);
background: rgba(255, 255, 255, 0.95);
padding: 12px 20px;
border-radius: 12px;
box-shadow: 0 4px 20px rgba(0, 0, 0, 0.2);
z-index: 1000;
text-align: center;
display: flex;
gap: 12px;
align-items: center;
flex-wrap: wrap;
justify-content: center;
}
select,
button {
padding: 10px;
font-size: 15px;
border-radius: 8px;
border: 1px solid #ccc;
}
button {
background: #0066cc;
color: white;
border: none;
cursor: pointer;
}
button:hover {
background: #0052a3;
}
.status {
font-weight: bold;
color: #0066cc;
}
.legend {
position: absolute;
top: 10px;
right: 10px;
background: white;
padding: 12px;
border-radius: 8px;
box-shadow: 0 4px 20px rgba(0, 0, 0, 0.15);
z-index: 1000;
font-size: 13px;
}