front: Initial implementation of simulation
This commit is contained in:
parent
d5362041ca
commit
0d7b384821
205
front/data.js
205
front/data.js
|
@ -1,32 +1,195 @@
|
|||
const updateFromTam = async (courses) =>
|
||||
{
|
||||
const currentCourses = await getCurrentCourses();
|
||||
const axios = require('axios');
|
||||
const network = require('../back/data/network.json');
|
||||
|
||||
for (let [id, course] of Object.entries(currentCourses))
|
||||
const server = 'http://localhost:4321';
|
||||
|
||||
const arriveAtStop = (course, stop) =>
|
||||
{
|
||||
course.state = 'stopped';
|
||||
course.currentStop = stop;
|
||||
delete course.departureStop;
|
||||
delete course.arrivalStop;
|
||||
delete course.arrivalTime;
|
||||
delete course.traveledDistance;
|
||||
delete course.speed;
|
||||
};
|
||||
|
||||
const moveToStop = (course, stop, arrivalTime) =>
|
||||
{
|
||||
course.state = 'moving';
|
||||
course.departureStop = course.currentStop;
|
||||
course.arrivalStop = stop;
|
||||
course.arrivalTime = arrivalTime;
|
||||
course.traveledDistance = 0;
|
||||
course.speed = 0;
|
||||
delete course.currentStop;
|
||||
|
||||
const segment = `${course.departureStop}-${course.arrivalStop}`;
|
||||
|
||||
if (!(segment in network.segments))
|
||||
{
|
||||
if (!(id in courses))
|
||||
{
|
||||
course.arrivalTime = now() + course.eta;
|
||||
console.log(`${displayNow(now())} - New course ${id} @ ${course.line} departing from stop ${course.stop} at ${displayNow(course.arrivalTime)}`);
|
||||
courses[id] = course;
|
||||
// There is no segment between the two requested stops, jump
|
||||
// directly to the arrival stop
|
||||
arriveAtStop(course, course.arrivalStop);
|
||||
}
|
||||
else
|
||||
{
|
||||
course.arrivalTime = now() + course.eta;
|
||||
console.log(`${displayNow(now())} - Course ${id} @ ${course.line} will arrive to stop ${course.stop} at ${displayNow(course.arrivalTime)} (previously to stop ${courses[id].stop} at ${displayNow(courses[id].arrivalTime)})`);
|
||||
courses[id] = course;
|
||||
}
|
||||
updateSpeed(course);
|
||||
}
|
||||
};
|
||||
|
||||
const sleep = time => new Promise(res => setTimeout(res, time));
|
||||
const courses = {};
|
||||
|
||||
const loop = async (courses = {}) =>
|
||||
const getCurrentSegment = course =>
|
||||
{
|
||||
await updateFromTam(courses);
|
||||
await sleep(30000);
|
||||
return loop(courses);
|
||||
if (course.state === 'stopped')
|
||||
{
|
||||
return null;
|
||||
}
|
||||
|
||||
return network.segments[`${course.departureStop}-${course.arrivalStop}`];
|
||||
};
|
||||
|
||||
loop();
|
||||
const updateSpeed = course =>
|
||||
{
|
||||
const segment = getCurrentSegment(course);
|
||||
const length = segment.points[segment.points.length - 1].distance;
|
||||
|
||||
const remainingTime = course.arrivalTime - Date.now();
|
||||
const remainingDistance = length - course.traveledDistance;
|
||||
|
||||
if (remainingTime <= 0 || remainingDistance <= 0)
|
||||
{
|
||||
arriveAtStop(course, course.arrivalStop);
|
||||
return;
|
||||
}
|
||||
|
||||
course.speed = remainingDistance / remainingTime;
|
||||
};
|
||||
|
||||
const updateFromTam = async (courses) =>
|
||||
{
|
||||
const currentCourses = (await axios.get(`${server}/courses`)).data;
|
||||
|
||||
for (let [id, course] of Object.entries(currentCourses))
|
||||
{
|
||||
// Only track selected lines
|
||||
if (!['1', '2', '3', '4'].includes(course.line))
|
||||
{
|
||||
continue;
|
||||
}
|
||||
|
||||
// Find out the next stop, ignoring the ones that are in the past
|
||||
let nextStop = null;
|
||||
let arrivalTime = null;
|
||||
|
||||
for (let {stopId, arrivalTime: time} of course.nextPassings)
|
||||
{
|
||||
if (time > Date.now())
|
||||
{
|
||||
nextStop = stopId;
|
||||
arrivalTime = time;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (nextStop === null)
|
||||
{
|
||||
continue;
|
||||
}
|
||||
|
||||
// Update an existing course
|
||||
if (id in courses)
|
||||
{
|
||||
const prev = courses[id];
|
||||
|
||||
if (prev.state === 'stopped')
|
||||
{
|
||||
if (prev.currentStop !== nextStop)
|
||||
{
|
||||
// Start traveling from the current stop to the next
|
||||
moveToStop(prev, nextStop, arrivalTime);
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
// Update the ETA if we’re still headed to the same stop
|
||||
if (prev.arrivalStop === nextStop)
|
||||
{
|
||||
prev.arrivalTime = arrivalTime;
|
||||
updateSpeed(prev);
|
||||
}
|
||||
// Otherwise, we missed a stop, try to go directly to the
|
||||
// next segment
|
||||
else
|
||||
{
|
||||
arriveAtStop(prev, prev.arrivalStop);
|
||||
moveToStop(prev, nextStop, arrivalTime);
|
||||
}
|
||||
}
|
||||
}
|
||||
// Create a new course
|
||||
else
|
||||
{
|
||||
courses[id] = {
|
||||
id,
|
||||
line: course.line,
|
||||
finalStop: course.finalStop,
|
||||
};
|
||||
|
||||
arriveAtStop(courses[id], nextStop);
|
||||
}
|
||||
}
|
||||
|
||||
// Remove stale courses
|
||||
for (let id of Object.keys(courses))
|
||||
{
|
||||
if (!(id in currentCourses))
|
||||
{
|
||||
delete courses[id];
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
const updatePositions = (courses, time) =>
|
||||
{
|
||||
for (let [id, course] of Object.entries(courses))
|
||||
{
|
||||
if (course.state === 'moving')
|
||||
{
|
||||
const delta = course.speed * time;
|
||||
|
||||
const segment = getCurrentSegment(course);
|
||||
const length = segment.points[segment.points.length - 1].distance;
|
||||
|
||||
if (course.traveledDistance + delta >= length)
|
||||
{
|
||||
course.traveledDistance = length;
|
||||
}
|
||||
else
|
||||
{
|
||||
course.traveledDistance += delta;
|
||||
}
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
const courses = {};
|
||||
let lastFrame = null;
|
||||
let lastUpdate = null;
|
||||
|
||||
const loop = now =>
|
||||
{
|
||||
const time = lastFrame === null ? 0 : now - lastFrame;
|
||||
|
||||
lastFrame = now;
|
||||
updatePositions(courses, time);
|
||||
|
||||
if (lastUpdate === null || lastUpdate + 5000 <= now)
|
||||
{
|
||||
lastUpdate = now;
|
||||
updateFromTam(courses);
|
||||
}
|
||||
|
||||
requestAnimationFrame(loop);
|
||||
};
|
||||
|
||||
requestAnimationFrame(loop);
|
||||
|
|
|
@ -1,4 +1,7 @@
|
|||
require('regenerator-runtime/runtime');
|
||||
|
||||
const {createMap} = require('./map');
|
||||
|
||||
require('./data');
|
||||
|
||||
createMap(/* map = */ 'map');
|
||||
|
|
Loading…
Reference in New Issue