c4c93e3776
Closes #133
329 lines
8.8 KiB
JavaScript
329 lines
8.8 KiB
JavaScript
/*
|
||
* Copyright (C) 2020 Birte Kristina Friesel
|
||
*
|
||
* SPDX-License-Identifier: AGPL-3.0-or-later
|
||
*/
|
||
var j_departure = 0;
|
||
var j_duration = 0;
|
||
var j_arrival = 0;
|
||
var j_dest = '';
|
||
var j_stops = [];
|
||
var j_token = '';
|
||
|
||
function setTheme(theme) {
|
||
localStorage.setItem('theme', theme);
|
||
if (!otherTheme.hasOwnProperty(theme)) {
|
||
theme = window.matchMedia('(prefers-color-scheme: dark)').matches ? 'dark' : 'light';
|
||
}
|
||
addStyleSheet(theme, 'theme');
|
||
}
|
||
|
||
function upd_journey_data() {
|
||
$('.countdown').each(function() {
|
||
const journey_token = $(this).data('token');
|
||
if (journey_token) {
|
||
j_token = journey_token;
|
||
}
|
||
var journey_data = $(this).data('journey');
|
||
if (journey_data) {
|
||
journey_data = journey_data.split(';');
|
||
j_departure = parseInt(journey_data[0]);
|
||
j_arrival = parseInt(journey_data[1]);
|
||
j_duration = j_arrival - j_departure;
|
||
}
|
||
var journey_dest = $(this).data('dest');
|
||
if (journey_dest) {
|
||
j_dest = journey_dest;
|
||
}
|
||
var stops = $(this).data('route');
|
||
if (stops) {
|
||
stops = stops.split('|');
|
||
j_stops = [];
|
||
for (var stop_id in stops) {
|
||
var stopdata = stops[stop_id].split(';');
|
||
for (var i = 1; i < 5; i++) {
|
||
stopdata[i] = parseInt(stopdata[i]);
|
||
}
|
||
j_stops.push(stopdata);
|
||
}
|
||
}
|
||
});
|
||
}
|
||
function upd_countdown() {
|
||
const now = Date.now() / 1000;
|
||
if (j_departure > now) {
|
||
$('.countdown').text('Abfahrt in ' + Math.round((j_departure - now)/60) + ' Minuten');
|
||
} else if (j_arrival > 0) {
|
||
if (j_arrival > now) {
|
||
var diff = Math.round((j_arrival - now)/60);
|
||
if (diff >= 120) {
|
||
$('.countdown').text('Ankunft in ' + Math.floor(diff/60) + ' Stunden und ' + (diff%60) + ' Minuten');
|
||
} else if (diff >= 60) {
|
||
$('.countdown').text('Ankunft in 1 Stunde und ' + (diff%60) + ' Minuten');
|
||
} else {
|
||
$('.countdown').text('Ankunft in ' + diff + ' Minuten');
|
||
}
|
||
} else {
|
||
$('.countdown').text('Ziel erreicht');
|
||
}
|
||
}
|
||
}
|
||
function hhmm(epoch) {
|
||
var date = new Date(epoch * 1000);
|
||
var h = date.getHours();
|
||
var m = date.getMinutes();
|
||
return (h < 10 ? '0' + h : h) + ':' + (m < 10 ? '0' + m : m);
|
||
}
|
||
function odelay(sched, rt) {
|
||
if (sched < rt) {
|
||
return ' (+' + ((rt - sched) / 60) + ')';
|
||
}
|
||
else if (sched == rt) {
|
||
return '';
|
||
}
|
||
return ' (' + ((rt - sched) / 60) + ')';
|
||
}
|
||
|
||
function tvly_run(link, req, err_callback) {
|
||
var error_icon = '<i class="material-icons">error</i>';
|
||
var progressbar;
|
||
if (link.data('tr')) {
|
||
progressbar = $('<tr><td colspan="' + link.data('tr') + '"><div class="progress"><div class="indeterminate"></div></div></td></tr>');
|
||
}
|
||
else {
|
||
progressbar = $('<div class="progress"><div class="indeterminate"></div></div>');
|
||
}
|
||
link.hide();
|
||
link.after(progressbar);
|
||
$.post('/action', req, function(data) {
|
||
if (data.success) {
|
||
$(location).attr('href', data.redirect_to);
|
||
} else {
|
||
M.toast({html: error_icon + ' ' + data.error});
|
||
progressbar.remove();
|
||
if (err_callback) {
|
||
err_callback();
|
||
}
|
||
link.append(' ' + error_icon);
|
||
link.show();
|
||
}
|
||
});
|
||
}
|
||
function tvly_update() {
|
||
$.get('/ajax/status_card.html', function(data) {
|
||
$('.statuscol').html(data);
|
||
tvly_reg_handlers();
|
||
upd_journey_data();
|
||
setTimeout(tvly_update, 40000);
|
||
}).fail(function() {
|
||
$('.sync-failed-marker').css('display', 'block');
|
||
upd_countdown();
|
||
setTimeout(tvly_update, 5000);
|
||
});
|
||
}
|
||
function tvly_update_public() {
|
||
var user_name;
|
||
var profile_status = 0;
|
||
$('.publicstatuscol').each(function() {
|
||
user_name = $(this).data('user');
|
||
profile_status = $(this).data('profile');
|
||
});
|
||
$.get('/ajax/status/' + user_name + '.html', {token: j_token, profile: profile_status}, function(data) {
|
||
$('.publicstatuscol').html(data);
|
||
upd_journey_data();
|
||
setTimeout(tvly_update_public, 40000);
|
||
}).fail(function() {
|
||
$('.sync-failed-marker').css('display', 'block');
|
||
upd_countdown();
|
||
setTimeout(tvly_update_public, 5000);
|
||
});
|
||
}
|
||
function tvly_update_timeline() {
|
||
$.get('/timeline/in-transit', {ajax: 1}, function(data) {
|
||
$('.timeline-in-transit').html(data);
|
||
setTimeout(tvly_update_timeline, 60000);
|
||
}).fail(function() {
|
||
$('.sync-failed-marker').css('display', 'block');
|
||
setTimeout(tvly_update_timeline, 10000);
|
||
});
|
||
}
|
||
function tvly_journey_progress() {
|
||
var now = Date.now() / 1000;
|
||
var progress = 0;
|
||
if (j_duration > 0) {
|
||
progress = 1 - ((j_arrival - now) / j_duration);
|
||
if (progress < 0) {
|
||
progress = 0;
|
||
}
|
||
if (progress > 1) {
|
||
progress = 1;
|
||
}
|
||
$('.progress .determinate').css('width', (progress * 100) + '%');
|
||
|
||
for (stop in j_stops) {
|
||
var stop_name = j_stops[stop][0];
|
||
var sched_arr = j_stops[stop][1];
|
||
var rt_arr = j_stops[stop][2];
|
||
var sched_dep = j_stops[stop][3];
|
||
var rt_dep = j_stops[stop][4];
|
||
if (stop_name == j_dest) {
|
||
$('.next-stop').html('');
|
||
break;
|
||
}
|
||
if ((rt_arr != 0) && (rt_arr - now > 0)) {
|
||
$('.next-stop').html(stop_name + '<br/>' + hhmm(rt_arr) + odelay(sched_arr, rt_arr));
|
||
break;
|
||
}
|
||
if ((rt_dep != 0) && (rt_dep - now > 0)) {
|
||
if (rt_arr != 0) {
|
||
$('.next-stop').html(stop_name + '<br/>' + hhmm(rt_arr) + ' → ' + hhmm(rt_dep) + odelay(sched_dep, rt_dep));
|
||
} else {
|
||
$('.next-stop').html(stop_name + '<br/>' + hhmm(rt_dep) + odelay(sched_dep, rt_dep));
|
||
}
|
||
break;
|
||
}
|
||
}
|
||
setTimeout(tvly_journey_progress, 5000);
|
||
}
|
||
}
|
||
function tvly_reg_handlers() {
|
||
$('.action-checkin').click(function() {
|
||
var link = $(this);
|
||
var req = {
|
||
action: 'checkin',
|
||
station: link.data('station'),
|
||
train: link.data('train'),
|
||
dest: link.data('dest'),
|
||
ts: link.data('ts'),
|
||
};
|
||
tvly_run(link, req);
|
||
});
|
||
$('.action-checkout').click(function() {
|
||
var link = $(this);
|
||
var req = {
|
||
action: 'checkout',
|
||
station: link.data('station'),
|
||
force: link.data('force'),
|
||
};
|
||
tvly_run(link, req, function() {
|
||
if (!link.data('force')) {
|
||
link.append(' – Ohne Echtzeitdaten auschecken?')
|
||
link.data('force', true);
|
||
}
|
||
});
|
||
});
|
||
$('.action-undo').click(function() {
|
||
var link = $(this);
|
||
var now = Date.now() / 1000;
|
||
var checkints = parseInt(link.data('checkints'));
|
||
var req = {
|
||
action: 'undo',
|
||
undo_id: link.data('id'),
|
||
};
|
||
var do_checkout = true;
|
||
if (now - checkints > 900) {
|
||
do_checkout = confirm("Checkin wirklich rückgängig machen? Er kann ggf. nicht wiederholt werden.");
|
||
}
|
||
if (do_checkout) {
|
||
tvly_run(link, req);
|
||
}
|
||
});
|
||
$('.action-cancelled-from').click(function() {
|
||
var link = $(this);
|
||
var req = {
|
||
action: 'cancelled_from',
|
||
station: link.data('station'),
|
||
ts: link.data('ts'),
|
||
train: link.data('train'),
|
||
};
|
||
tvly_run(link, req);
|
||
});
|
||
$('.action-cancelled-to').click(function() {
|
||
var link = $(this);
|
||
var req = {
|
||
action: 'cancelled_to',
|
||
station: link.data('station'),
|
||
force: true,
|
||
};
|
||
tvly_run(link, req);
|
||
});
|
||
$('.action-delete').click(function() {
|
||
var link = $(this);
|
||
var req = {
|
||
action: 'delete',
|
||
id: link.data('id'),
|
||
checkin: link.data('checkin'),
|
||
checkout: link.data('checkout'),
|
||
};
|
||
var really_delete = confirm("Diese Fahrt wirklich löschen? Der Eintrag wird sofort aus der Datenbank entfernt und kann nicht wiederhergestellt werden.");
|
||
if (really_delete) {
|
||
tvly_run(link, req);
|
||
}
|
||
});
|
||
$('.action-share').click(function() {
|
||
var text = $(this).data('text');
|
||
var url = $(this).data('url');
|
||
if (navigator.share) {
|
||
shareObj = {
|
||
text: text
|
||
};
|
||
if (url) {
|
||
shareObj['url'] = url;
|
||
}
|
||
navigator.share(shareObj);
|
||
} else {
|
||
var el = document.createElement('textarea');
|
||
if (url) {
|
||
text += ' ' + url;
|
||
}
|
||
el.value = text;
|
||
el.setAttribute('readonly', '');
|
||
el.style.position = 'absolute';
|
||
el.style.left = '-9999px';
|
||
document.body.appendChild(el);
|
||
el.select();
|
||
el.setSelectionRange(0, 99999);
|
||
document.execCommand('copy');
|
||
document.body.removeChild(el);
|
||
M.toast({html: 'Text kopiert: „' + text + '“'});
|
||
}
|
||
});
|
||
}
|
||
$(document).ready(function() {
|
||
tvly_reg_handlers();
|
||
if ($('.statuscol .autorefresh').length) {
|
||
upd_journey_data();
|
||
setTimeout(tvly_update, 40000);
|
||
setTimeout(tvly_journey_progress, 5000);
|
||
}
|
||
if ($('.publicstatuscol .autorefresh').length) {
|
||
upd_journey_data();
|
||
setTimeout(tvly_update_public, 40000);
|
||
setTimeout(tvly_journey_progress, 5000);
|
||
}
|
||
if ($('.timeline-in-transit .autorefresh').length) {
|
||
setTimeout(tvly_update_timeline, 60000);
|
||
}
|
||
$('a[href]').click(function() {
|
||
$('nav .preloader-wrapper').addClass('active');
|
||
});
|
||
$('a[href="#now"]').keydown(function(event) {
|
||
// also trigger click handler on keyboard enter
|
||
if (event.keyCode == 13) {
|
||
event.preventDefault();
|
||
event.target.click();
|
||
}
|
||
});
|
||
$('a[href="#now"]').click(function(event) {
|
||
event.preventDefault();
|
||
$('nav .preloader-wrapper').removeClass('active');
|
||
now_el = $('#now')[0];
|
||
now_el.previousElementSibling.querySelector(".dep-time").focus();
|
||
now_el.scrollIntoView({behavior: "smooth", block: "center"});
|
||
});
|
||
const elems = document.querySelectorAll('.carousel');
|
||
const instances = M.Carousel.init(elems, {
|
||
fullWidth: true,
|
||
indicators: true}
|
||
);
|
||
});
|